source: trunk/minix/servers/fs/main.c@ 9

Last change on this file since 9 was 9, checked in by Mattia Monga, 13 years ago

Minix 3.1.2a

File size: 9.2 KB
Line 
1/* This file contains the main program of the File System. It consists of
2 * a loop that gets messages requesting work, carries out the work, and sends
3 * replies.
4 *
5 * The entry points into this file are:
6 * main: main program of the File System
7 * reply: send a reply to a process after the requested work is done
8 *
9 */
10
11struct super_block; /* proto.h needs to know this */
12
13#include "fs.h"
14#include <fcntl.h>
15#include <string.h>
16#include <stdio.h>
17#include <signal.h>
18#include <stdlib.h>
19#include <sys/ioc_memory.h>
20#include <sys/svrctl.h>
21#include <sys/select.h>
22#include <minix/callnr.h>
23#include <minix/com.h>
24#include <minix/keymap.h>
25#include <minix/const.h>
26#include <minix/endpoint.h>
27#include "buf.h"
28#include "file.h"
29#include "fproc.h"
30#include "inode.h"
31#include "param.h"
32#include "super.h"
33
34FORWARD _PROTOTYPE( void fs_init, (void) );
35FORWARD _PROTOTYPE( void get_work, (void) );
36FORWARD _PROTOTYPE( void init_root, (void) );
37
38/*===========================================================================*
39 * main *
40 *===========================================================================*/
41PUBLIC int main()
42{
43/* This is the main program of the file system. The main loop consists of
44 * three major activities: getting new work, processing the work, and sending
45 * the reply. This loop never terminates as long as the file system runs.
46 */
47 int error;
48
49 fs_init();
50
51
52 /* This is the main loop that gets work, processes it, and sends replies. */
53 while (TRUE) {
54 get_work(); /* sets who and call_nr */
55 fp = &fproc[who_p]; /* pointer to proc table struct */
56 super_user = (fp->fp_effuid == SU_UID ? TRUE : FALSE); /* su? */
57
58 /* Check for special control messages first. */
59 if (call_nr == PROC_EVENT) {
60 /* Assume FS got signal. Synchronize, but don't exit. */
61 do_sync();
62 } else if (call_nr == SYN_ALARM) {
63 /* Alarm timer expired. Used only for select(). Check it. */
64 fs_expire_timers(m_in.NOTIFY_TIMESTAMP);
65 } else if ((call_nr & NOTIFY_MESSAGE)) {
66 /* Device notifies us of an event. */
67 dev_status(&m_in);
68 } else {
69 /* Call the internal function that does the work. */
70 if (call_nr < 0 || call_nr >= NCALLS) {
71 error = ENOSYS;
72 /* Not supposed to happen. */
73 printf("FS, warning illegal %d system call by %d\n", call_nr, who_e);
74 } else if (fp->fp_pid == PID_FREE) {
75 error = ENOSYS;
76 printf("FS, bad process, who = %d, call_nr = %d, endpt1 = %d\n",
77 who_e, call_nr, m_in.endpt1);
78 } else {
79 error = (*call_vec[call_nr])();
80 }
81
82 /* Copy the results back to the user and send reply. */
83 if (error != SUSPEND) { reply(who_e, error); }
84 if (rdahed_inode != NIL_INODE) {
85 read_ahead(); /* do block read ahead */
86 }
87 }
88 }
89 return(OK); /* shouldn't come here */
90}
91
92/*===========================================================================*
93 * get_work *
94 *===========================================================================*/
95PRIVATE void get_work()
96{
97 /* Normally wait for new input. However, if 'reviving' is
98 * nonzero, a suspended process must be awakened.
99 */
100 register struct fproc *rp;
101
102 if (reviving != 0) {
103 /* Revive a suspended process. */
104 for (rp = &fproc[0]; rp < &fproc[NR_PROCS]; rp++)
105 if (rp->fp_pid != PID_FREE && rp->fp_revived == REVIVING) {
106 who_p = (int)(rp - fproc);
107 who_e = rp->fp_endpoint;
108 call_nr = rp->fp_fd & BYTE;
109 m_in.fd = (rp->fp_fd >>8) & BYTE;
110 m_in.buffer = rp->fp_buffer;
111 m_in.nbytes = rp->fp_nbytes;
112 rp->fp_suspended = NOT_SUSPENDED; /*no longer hanging*/
113 rp->fp_revived = NOT_REVIVING;
114 reviving--;
115 return;
116 }
117 panic(__FILE__,"get_work couldn't revive anyone", NO_NUM);
118 }
119
120 for(;;) {
121 /* Normal case. No one to revive. */
122 if (receive(ANY, &m_in) != OK)
123 panic(__FILE__,"fs receive error", NO_NUM);
124 who_e = m_in.m_source;
125 who_p = _ENDPOINT_P(who_e);
126 if(who_p < -NR_TASKS || who_p >= NR_PROCS)
127 panic(__FILE__,"receive process out of range", who_p);
128 if(who_p >= 0 && fproc[who_p].fp_endpoint == NONE) {
129 printf("FS: ignoring request from %d, endpointless slot %d (%d)\n",
130 m_in.m_source, who_p, m_in.m_type);
131 continue;
132 }
133 if(who_p >= 0 && fproc[who_p].fp_endpoint != who_e) {
134 printf("FS: receive endpoint inconsistent (%d, %d, %d).\n",
135 who_e, fproc[who_p].fp_endpoint, who_e);
136 panic(__FILE__, "FS: inconsistent endpoint ", NO_NUM);
137 continue;
138 }
139 call_nr = m_in.m_type;
140 return;
141 }
142}
143
144/*===========================================================================*
145 * buf_pool *
146 *===========================================================================*/
147PRIVATE void buf_pool(void)
148{
149/* Initialize the buffer pool. */
150
151 register struct buf *bp;
152
153 bufs_in_use = 0;
154 front = &buf[0];
155 rear = &buf[NR_BUFS - 1];
156
157 for (bp = &buf[0]; bp < &buf[NR_BUFS]; bp++) {
158 bp->b_blocknr = NO_BLOCK;
159 bp->b_dev = NO_DEV;
160 bp->b_next = bp + 1;
161 bp->b_prev = bp - 1;
162 }
163 buf[0].b_prev = NIL_BUF;
164 buf[NR_BUFS - 1].b_next = NIL_BUF;
165
166 for (bp = &buf[0]; bp < &buf[NR_BUFS]; bp++) bp->b_hash = bp->b_next;
167 buf_hash[0] = front;
168
169}
170
171/*===========================================================================*
172 * reply *
173 *===========================================================================*/
174PUBLIC void reply(whom, result)
175int whom; /* process to reply to */
176int result; /* result of the call (usually OK or error #) */
177{
178/* Send a reply to a user process. If the send fails, just ignore it. */
179 int s;
180 m_out.reply_type = result;
181 s = send(whom, &m_out);
182 if (s != OK) printf("FS: couldn't send reply %d to %d: %d\n",
183 result, whom, s);
184}
185
186/*===========================================================================*
187 * fs_init *
188 *===========================================================================*/
189PRIVATE void fs_init()
190{
191/* Initialize global variables, tables, etc. */
192 register struct inode *rip;
193 register struct fproc *rfp;
194 message mess;
195 int s;
196
197 /* Initialize the process table with help of the process manager messages.
198 * Expect one message for each system process with its slot number and pid.
199 * When no more processes follow, the magic process number NONE is sent.
200 * Then, stop and synchronize with the PM.
201 */
202 do {
203 if (OK != (s=receive(PM_PROC_NR, &mess)))
204 panic(__FILE__,"FS couldn't receive from PM", s);
205 if (NONE == mess.PR_ENDPT) break;
206
207 rfp = &fproc[mess.PR_SLOT];
208 rfp->fp_pid = mess.PR_PID;
209 rfp->fp_endpoint = mess.PR_ENDPT;
210 rfp->fp_realuid = (uid_t) SYS_UID;
211 rfp->fp_effuid = (uid_t) SYS_UID;
212 rfp->fp_realgid = (gid_t) SYS_GID;
213 rfp->fp_effgid = (gid_t) SYS_GID;
214 rfp->fp_umask = ~0;
215
216 } while (TRUE); /* continue until process NONE */
217 mess.m_type = OK; /* tell PM that we succeeded */
218 s=send(PM_PROC_NR, &mess); /* send synchronization message */
219
220 /* All process table entries have been set. Continue with FS initialization.
221 * Certain relations must hold for the file system to work at all. Some
222 * extra block_size requirements are checked at super-block-read-in time.
223 */
224 if (OPEN_MAX > 127) panic(__FILE__,"OPEN_MAX > 127", NO_NUM);
225 if (NR_BUFS < 6) panic(__FILE__,"NR_BUFS < 6", NO_NUM);
226 if (V1_INODE_SIZE != 32) panic(__FILE__,"V1 inode size != 32", NO_NUM);
227 if (V2_INODE_SIZE != 64) panic(__FILE__,"V2 inode size != 64", NO_NUM);
228 if (OPEN_MAX > 8 * sizeof(long))
229 panic(__FILE__,"Too few bits in fp_cloexec", NO_NUM);
230
231 /* The following initializations are needed to let dev_opcl succeed .*/
232 fp = (struct fproc *) NULL;
233 who_e = who_p = FS_PROC_NR;
234
235 buf_pool(); /* initialize buffer pool */
236 build_dmap(); /* build device table and map boot driver */
237 init_root(); /* init root device and load super block */
238 init_select(); /* init select() structures */
239
240 /* The root device can now be accessed; set process directories. */
241 for (rfp=&fproc[0]; rfp < &fproc[NR_PROCS]; rfp++) {
242 FD_ZERO(&(rfp->fp_filp_inuse));
243 if (rfp->fp_pid != PID_FREE) {
244 rip = get_inode(root_dev, ROOT_INODE);
245 dup_inode(rip);
246 rfp->fp_rootdir = rip;
247 rfp->fp_workdir = rip;
248 } else rfp->fp_endpoint = NONE;
249 }
250}
251
252/*===========================================================================*
253 * init_root *
254 *===========================================================================*/
255PRIVATE void init_root()
256{
257 int bad;
258 register struct super_block *sp;
259 register struct inode *rip = NIL_INODE;
260 int s;
261
262 /* Open the root device. */
263 root_dev = DEV_IMGRD;
264 if ((s=dev_open(root_dev, FS_PROC_NR, R_BIT|W_BIT)) != OK)
265 panic(__FILE__,"Cannot open root device", s);
266
267#if ENABLE_CACHE2
268 /* The RAM disk is a second level block cache while not otherwise used. */
269 init_cache2(ram_size);
270#endif
271
272 /* Initialize the super_block table. */
273 for (sp = &super_block[0]; sp < &super_block[NR_SUPERS]; sp++)
274 sp->s_dev = NO_DEV;
275
276 /* Read in super_block for the root file system. */
277 sp = &super_block[0];
278 sp->s_dev = root_dev;
279
280 /* Check super_block for consistency. */
281 bad = (read_super(sp) != OK);
282 if (!bad) {
283 rip = get_inode(root_dev, ROOT_INODE); /* inode for root dir */
284 if ( (rip->i_mode & I_TYPE) != I_DIRECTORY || rip->i_nlinks < 3) bad++;
285 }
286 if (bad) panic(__FILE__,"Invalid root file system", NO_NUM);
287
288 sp->s_imount = rip;
289 dup_inode(rip);
290 sp->s_isup = rip;
291 sp->s_rd_only = 0;
292 return;
293}
Note: See TracBrowser for help on using the repository browser.