BOOK.TXT
上传用户:datang2001
上传日期:2007-02-01
资源大小:53269k
文件大小:999k
源码类别:

操作系统开发

开发平台:

C/C++

  1. 13915                 vid_vid_copy(cons->c_start + scr_width, cons->c_start, chars);
  2. 13916         } else
  3. 13917         if (!wrap && cons->c_org + scr_size + scr_width >= cons->c_limit) {
  4. 13918                 vid_vid_copy(cons->c_org + scr_width, cons->c_start, chars);
  5. 13919                 cons->c_org = cons->c_start;
  6. 13920         } else {
  7. 13921                 cons->c_org = (cons->c_org + scr_width) & vid_mask;
  8. 13922         }
  9. 13923         new_line = (cons->c_org + chars) & vid_mask;
  10. 13924   } else {
  11. 13925         /* Scroll one line down in 3 ways: soft, avoid wrap, use origin. */
  12. 13926         if (softscroll) {
  13. 13927                 vid_vid_copy(cons->c_start, cons->c_start + scr_width, chars);
  14. 13928         } else
  15. 13929         if (!wrap && cons->c_org < cons->c_start + scr_width) {
  16. 13930                 new_org = cons->c_limit - scr_size;
  17. 13931                 vid_vid_copy(cons->c_org, new_org + scr_width, chars);
  18. 13932                 cons->c_org = new_org;
  19. 13933         } else {
  20. 13934                 cons->c_org = (cons->c_org - scr_width) & vid_mask;
  21. 13935         }
  22. 13936         new_line = cons->c_org;
  23. 13937   }
  24. 13938   /* Blank the new line at top or bottom. */
  25. 13939   blank_color = cons->c_blank;
  26. 13940   mem_vid_copy(BLANK_MEM, new_line, scr_width);
  27. 13941
  28. 13942   /* Set the new video origin. */
  29. 13943   if (cons == curcons) set_6845(VID_ORG, cons->c_org);
  30. 13944   flush(cons);
  31. 13945 }
  32. 13948 /*===========================================================================*
  33. 13949  *                              flush                                        *
  34. 13950  *===========================================================================*/
  35. 13951 PRIVATE void flush(cons)
  36. 13952 register console_t *cons;       /* pointer to console struct */
  37. 13953 {
  38. 13954 /* Send characters buffered in 'ramqueue' to screen memory, check the new
  39. 13955  * cursor position, compute the new hardware cursor position and set it.
  40. 13956  */
  41. 13957   unsigned cur;
  42. 13958   tty_t *tp = cons->c_tty;
  43. 13959
  44. 13960   /* Have the characters in 'ramqueue' transferred to the screen. */
  45. 13961   if (cons->c_rwords > 0) {
  46. 13962         mem_vid_copy(cons->c_ramqueue, cons->c_cur, cons->c_rwords);
  47. 13963         cons->c_rwords = 0;
  48. 13964
  49. 13965         /* TTY likes to know the current column and if echoing messed up. */
  50. 13966         tp->tty_position = cons->c_column;
  51. 13967         tp->tty_reprint = TRUE;
  52. 13968   }
  53. 13969
  54. 13970   /* Check and update the cursor position. */
  55. 13971   if (cons->c_column < 0) cons->c_column = 0;
  56. 13972   if (cons->c_column > scr_width) cons->c_column = scr_width;
  57. 13973   if (cons->c_row < 0) cons->c_row = 0;
  58. 13974   if (cons->c_row >= scr_lines) cons->c_row = scr_lines - 1;
  59. 13975   cur = cons->c_org + cons->c_row * scr_width + cons->c_column;
  60. 13976   if (cur != cons->c_cur) {
  61. 13977         if (cons == curcons) set_6845(CURSOR, cur);
  62. 13978         cons->c_cur = cur;
  63. 13979   }
  64. 13980 }
  65. 13983 /*===========================================================================*
  66. 13984  *                              parse_escape                                 *
  67. 13985  *===========================================================================*/
  68. 13986 PRIVATE void parse_escape(cons, c)
  69. 13987 register console_t *cons;       /* pointer to console struct */
  70. 13988 char c;                         /* next character in escape sequence */
  71. 13989 {
  72. 13990 /* The following ANSI escape sequences are currently supported.
  73. 13991  * If n and/or m are omitted, they default to 1. Omitted s defaults to 0.
  74. 13992  *   ESC [nA moves up n lines
  75. 13993  *   ESC [nB moves down n lines
  76. 13994  *   ESC [nC moves right n spaces
  77. 13995  *   ESC [nD moves left n spaces
  78. 13996  *   ESC [m;nH moves cursor to (m,n)
  79. 13997  *   ESC [sJ clears screen relative to cursor (0 to end, 1 from start, 2 all)
  80. 13998  *   ESC [sK clears line relative to cursor (0 to end, 1 from start, 2 all)
  81. 13999  *   ESC [nL inserts n lines at cursor
  82. 14000  *   ESC [nM deletes n lines at cursor
  83. 14001  *   ESC [nP deletes n chars at cursor
  84. 14002  *   ESC [n@ inserts n chars at cursor
  85. 14003  *   ESC [nm enables rendition n (0= normal, 1=bold, 4=underline, 5=blinking,
  86. 14004  *       7=reverse, 30..37 set foreground color, 40..47 set background color)
  87. 14005  *   ESC M scrolls the screen backwards if the cursor is on the top line
  88. 14006  */
  89. 14007
  90. 14008   switch (cons->c_esc_state) {
  91. 14009     case 1:                     /* ESC seen */
  92. 14010         cons->c_esc_intro = '';
  93. 14011         cons->c_esc_parmp = cons->c_esc_parmv;
  94. 14012         cons->c_esc_parmv[0] = cons->c_esc_parmv[1] = 0;
  95. 14013         switch (c) {
  96. 14014             case '[':   /* Control Sequence Introducer */
  97. 14015                 cons->c_esc_intro = c;
  98. 14016                 cons->c_esc_state = 2;
  99. 14017                 break;
  100. 14018             case 'M':   /* Reverse Index */
  101. 14019                 do_escape(cons, c);
  102. 14020                 break;
  103. 14021             default:
  104. 14022                 cons->c_esc_state = 0;
  105. 14023         }
  106. 14024         break;
  107. 14025
  108. 14026     case 2:                     /* ESC [ seen */
  109. 14027         if (c >= '0' && c <= '9') {
  110. 14028                 if (cons->c_esc_parmp < bufend(cons->c_esc_parmv))
  111. 14029                         *cons->c_esc_parmp = *cons->c_esc_parmp * 10 + (c-'0');
  112. 14030         } else
  113. 14031         if (c == ';') {
  114. 14032                 if (++cons->c_esc_parmp < bufend(cons->c_esc_parmv))
  115. 14033                         *cons->c_esc_parmp = 0;
  116. 14034         } else {
  117. 14035                 do_escape(cons, c);
  118. 14036         }
  119. 14037         break;
  120. 14038   }
  121. 14039 }
  122. 14042 /*===========================================================================*
  123. 14043  *                              do_escape                                    *
  124. 14044  *===========================================================================*/
  125. 14045 PRIVATE void do_escape(cons, c)
  126. 14046 register console_t *cons;       /* pointer to console struct */
  127. 14047 char c;                         /* next character in escape sequence */
  128. 14048 {
  129. 14049   int value, n;
  130. 14050   unsigned src, dst, count;
  131. 14051
  132. 14052   /* Some of these things hack on screen RAM, so it had better be up to date */
  133. 14053   flush(cons);
  134. 14054
  135. 14055   if (cons->c_esc_intro == '') {
  136. 14056         /* Handle a sequence beginning with just ESC */
  137. 14057         switch (c) {
  138. 14058             case 'M':           /* Reverse Index */
  139. 14059                 if (cons->c_row == 0) {
  140. 14060                         scroll_screen(cons, SCROLL_DOWN);
  141. 14061                 } else {
  142. 14062                         cons->c_row--;
  143. 14063                 }
  144. 14064                 flush(cons);
  145. 14065                 break;
  146. 14066
  147. 14067             default: break;
  148. 14068         }
  149. 14069   } else
  150. 14070   if (cons->c_esc_intro == '[') {
  151. 14071         /* Handle a sequence beginning with ESC [ and parameters */
  152. 14072         value = cons->c_esc_parmv[0];
  153. 14073         switch (c) {
  154. 14074             case 'A':           /* ESC [nA moves up n lines */
  155. 14075                 n = (value == 0 ? 1 : value);
  156. 14076                 cons->c_row -= n;
  157. 14077                 flush(cons);
  158. 14078                 break;
  159. 14079
  160. 14080             case 'B':           /* ESC [nB moves down n lines */
  161. 14081                 n = (value == 0 ? 1 : value);
  162. 14082                 cons->c_row += n;
  163. 14083                 flush(cons);
  164. 14084                 break;
  165. 14085
  166. 14086             case 'C':           /* ESC [nC moves right n spaces */
  167. 14087                 n = (value == 0 ? 1 : value);
  168. 14088                 cons->c_column += n;
  169. 14089                 flush(cons);
  170. 14090                 break;
  171. 14091
  172. 14092             case 'D':           /* ESC [nD moves left n spaces */
  173. 14093                 n = (value == 0 ? 1 : value);
  174. 14094                 cons->c_column -= n;
  175. 14095                 flush(cons);
  176. 14096                 break;
  177. 14097
  178. 14098             case 'H':           /* ESC [m;nH" moves cursor to (m,n) */
  179. 14099                 cons->c_row = cons->c_esc_parmv[0] - 1;
  180. 14100                 cons->c_column = cons->c_esc_parmv[1] - 1;
  181. 14101                 flush(cons);
  182. 14102                 break;
  183. 14103
  184. 14104             case 'J':           /* ESC [sJ clears in display */
  185. 14105                 switch (value) {
  186. 14106                     case 0:     /* Clear from cursor to end of screen */
  187. 14107                         count = scr_size - (cons->c_cur - cons->c_org);
  188. 14108                         dst = cons->c_cur;
  189. 14109                         break;
  190. 14110                     case 1:     /* Clear from start of screen to cursor */
  191. 14111                         count = cons->c_cur - cons->c_org;
  192. 14112                         dst = cons->c_org;
  193. 14113                         break;
  194. 14114                     case 2:     /* Clear entire screen */
  195. 14115                         count = scr_size;
  196. 14116                         dst = cons->c_org;
  197. 14117                         break;
  198. 14118                     default:    /* Do nothing */
  199. 14119                         count = 0;
  200. 14120                         dst = cons->c_org;
  201. 14121                 }
  202. 14122                 blank_color = cons->c_blank;
  203. 14123                 mem_vid_copy(BLANK_MEM, dst, count);
  204. 14124                 break;
  205. 14125
  206. 14126             case 'K':           /* ESC [sK clears line from cursor */
  207. 14127                 switch (value) {
  208. 14128                     case 0:     /* Clear from cursor to end of line */
  209. 14129                         count = scr_width - cons->c_column;
  210. 14130                         dst = cons->c_cur;
  211. 14131                         break;
  212. 14132                     case 1:     /* Clear from beginning of line to cursor */
  213. 14133                         count = cons->c_column;
  214. 14134                         dst = cons->c_cur - cons->c_column;
  215. 14135                         break;
  216. 14136                     case 2:     /* Clear entire line */
  217. 14137                         count = scr_width;
  218. 14138                         dst = cons->c_cur - cons->c_column;
  219. 14139                         break;
  220. 14140                     default:    /* Do nothing */
  221. 14141                         count = 0;
  222. 14142                         dst = cons->c_cur;
  223. 14143                 }
  224. 14144                 blank_color = cons->c_blank;
  225. 14145                 mem_vid_copy(BLANK_MEM, dst, count);
  226. 14146                 break;
  227. 14147
  228. 14148             case 'L':           /* ESC [nL inserts n lines at cursor */
  229. 14149                 n = value;
  230. 14150                 if (n < 1) n = 1;
  231. 14151                 if (n > (scr_lines - cons->c_row))
  232. 14152                         n = scr_lines - cons->c_row;
  233. 14153
  234. 14154                 src = cons->c_org + cons->c_row * scr_width;
  235. 14155                 dst = src + n * scr_width;
  236. 14156                 count = (scr_lines - cons->c_row - n) * scr_width;
  237. 14157                 vid_vid_copy(src, dst, count);
  238. 14158                 blank_color = cons->c_blank;
  239. 14159                 mem_vid_copy(BLANK_MEM, src, n * scr_width);
  240. 14160                 break;
  241. 14161
  242. 14162             case 'M':           /* ESC [nM deletes n lines at cursor */
  243. 14163                 n = value;
  244. 14164                 if (n < 1) n = 1;
  245. 14165                 if (n > (scr_lines - cons->c_row))
  246. 14166                         n = scr_lines - cons->c_row;
  247. 14167
  248. 14168                 dst = cons->c_org + cons->c_row * scr_width;
  249. 14169                 src = dst + n * scr_width;
  250. 14170                 count = (scr_lines - cons->c_row - n) * scr_width;
  251. 14171                 vid_vid_copy(src, dst, count);
  252. 14172                 blank_color = cons->c_blank;
  253. 14173                 mem_vid_copy(BLANK_MEM, dst + count, n * scr_width);
  254. 14174                 break;
  255. 14175
  256. 14176             case '@':           /* ESC [n@ inserts n chars at cursor */
  257. 14177                 n = value;
  258. 14178                 if (n < 1) n = 1;
  259. 14179                 if (n > (scr_width - cons->c_column))
  260. 14180                         n = scr_width - cons->c_column;
  261. 14181
  262. 14182                 src = cons->c_cur;
  263. 14183                 dst = src + n;
  264. 14184                 count = scr_width - cons->c_column - n;
  265. 14185                 vid_vid_copy(src, dst, count);
  266. 14186                 blank_color = cons->c_blank;
  267. 14187                 mem_vid_copy(BLANK_MEM, src, n);
  268. 14188                 break;
  269. 14189
  270. 14190             case 'P':           /* ESC [nP deletes n chars at cursor */
  271. 14191                 n = value;
  272. 14192                 if (n < 1) n = 1;
  273. 14193                 if (n > (scr_width - cons->c_column))
  274. 14194                         n = scr_width - cons->c_column;
  275. 14195
  276. 14196                 dst = cons->c_cur;
  277. 14197                 src = dst + n;
  278. 14198                 count = scr_width - cons->c_column - n;
  279. 14199                 vid_vid_copy(src, dst, count);
  280. 14200                 blank_color = cons->c_blank;
  281. 14201                 mem_vid_copy(BLANK_MEM, dst + count, n);
  282. 14202                 break;
  283. 14203
  284. 14204             case 'm':           /* ESC [nm enables rendition n */
  285. 14205                 switch (value) {
  286. 14206                     case 1:     /* BOLD  */
  287. 14207                         if (color) {
  288. 14208                                 /* Can't do bold, so use yellow */
  289. 14209                                 cons->c_attr = (cons->c_attr & 0xf0ff) | 0x0E00;
  290. 14210                         } else {
  291. 14211                                 /* Set intensity bit */
  292. 14212                                 cons->c_attr |= 0x0800;
  293. 14213                         }
  294. 14214                         break;
  295. 14215
  296. 14216                     case 4:     /* UNDERLINE */
  297. 14217                         if (color) {
  298. 14218                                 /* Use light green */
  299. 14219                                 cons->c_attr = (cons->c_attr & 0xf0ff) | 0x0A00;
  300. 14220                         } else {
  301. 14221                                 cons->c_attr = (cons->c_attr & 0x8900);
  302. 14222                         }
  303. 14223                         break;
  304. 14224
  305. 14225                     case 5:     /* BLINKING */
  306. 14226                         if (color) {
  307. 14227                                 /* Use magenta */
  308. 14228                                 cons->c_attr = (cons->c_attr & 0xf0ff) | 0x0500;
  309. 14229                         } else {
  310. 14230                                 /* Set the blink bit */
  311. 14231                                 cons->c_attr |= 0x8000;
  312. 14232                         }
  313. 14233                         break;
  314. 14234
  315. 14235                     case 7:     /* REVERSE */
  316. 14236                         if (color) {
  317. 14237                                 /* Swap fg and bg colors */
  318. 14238                                 cons->c_attr =
  319. 14239                                         ((cons->c_attr & 0xf000) >> 4) |
  320. 14240                                         ((cons->c_attr & 0x0f00) << 4);
  321. 14241                         } else
  322. 14242                         if ((cons->c_attr & 0x7000) == 0) {
  323. 14243                                 cons->c_attr = (cons->c_attr & 0x8800) | 0x7000;
  324. 14244                         } else {
  325. 14245                                 cons->c_attr = (cons->c_attr & 0x8800) | 0x0700;
  326. 14246                         }
  327. 14247                         break;
  328. 14248
  329. 14249                     default:    /* COLOR */
  330. 14250                         if (30 <= value && value <= 37) {
  331. 14251                                 cons->c_attr =
  332. 14252                                         (cons->c_attr & 0xf0ff) |
  333. 14253                                         (ansi_colors[(value - 30)] << 8);
  334. 14254                                 cons->c_blank =
  335. 14255                                         (cons->c_blank & 0xf0ff) |
  336. 14256                                         (ansi_colors[(value - 30)] << 8);
  337. 14257                         } else
  338. 14258                         if (40 <= value && value <= 47) {
  339. 14259                                 cons->c_attr =
  340. 14260                                         (cons->c_attr & 0x0fff) |
  341. 14261                                         (ansi_colors[(value - 40)] << 12);
  342. 14262                                 cons->c_blank =
  343. 14263                                         (cons->c_blank & 0x0fff) |
  344. 14264                                         (ansi_colors[(value - 40)] << 12);
  345. 14265                         } else {
  346. 14266                                 cons->c_attr = cons->c_blank;
  347. 14267                         }
  348. 14268                         break;
  349. 14269                 }
  350. 14270                 break;
  351. 14271         }
  352. 14272   }
  353. 14273   cons->c_esc_state = 0;
  354. 14274 }
  355. 14277 /*===========================================================================*
  356. 14278  *                              set_6845                                     *
  357. 14279  *===========================================================================*/
  358. 14280 PRIVATE void set_6845(reg, val)
  359. 14281 int reg;                        /* which register pair to set */
  360. 14282 unsigned val;                   /* 16-bit value to set it to */
  361. 14283 {
  362. 14284 /* Set a register pair inside the 6845.
  363. 14285  * Registers 12-13 tell the 6845 where in video ram to start
  364. 14286  * Registers 14-15 tell the 6845 where to put the cursor
  365. 14287  */
  366. 14288   lock();                       /* try to stop h/w loading in-between value */
  367. 14289   out_byte(vid_port + INDEX, reg);              /* set the index register */
  368. 14290   out_byte(vid_port + DATA, (val>>8) & BYTE);   /* output high byte */
  369. 14291   out_byte(vid_port + INDEX, reg + 1);          /* again */
  370. 14292   out_byte(vid_port + DATA, val&BYTE);          /* output low byte */
  371. 14293   unlock();
  372. 14294 }
  373. 14297 /*===========================================================================*
  374. 14298  *                              beep                                         *
  375. 14299  *===========================================================================*/
  376. 14300 PRIVATE void beep()
  377. 14301 {
  378. 14302 /* Making a beeping sound on the speaker (output for CRTL-G).
  379. 14303  * This routine works by turning on the bits 0 and 1 in port B of the 8255
  380. 14304  * chip that drives the speaker.
  381. 14305  */
  382. 14306
  383. 14307   message mess;
  384. 14308
  385. 14309   if (beeping) return;
  386. 14310   out_byte(TIMER_MODE, 0xB6);   /* set up timer channel 2 (square wave) */
  387. 14311   out_byte(TIMER2, BEEP_FREQ & BYTE);   /* load low-order bits of frequency */
  388. 14312   out_byte(TIMER2, (BEEP_FREQ >> 8) & BYTE);    /* now high-order bits */
  389. 14313   lock();                       /* guard PORT_B from keyboard intr handler */
  390. 14314   out_byte(PORT_B, in_byte(PORT_B) | 3);        /* turn on beep bits */
  391. 14315   unlock();
  392. 14316   beeping = TRUE;
  393. 14317
  394. 14318   mess.m_type = SET_ALARM;
  395. 14319   mess.CLOCK_PROC_NR = TTY;
  396. 14320   mess.DELTA_TICKS = B_TIME;
  397. 14321   mess.FUNC_TO_CALL = (sighandler_t) stop_beep;
  398. 14322   sendrec(CLOCK, &mess);
  399. 14323 }
  400. 14326 /*===========================================================================*
  401. 14327  *                              stop_beep                                    *
  402. 14328  *===========================================================================*/
  403. 14329 PRIVATE void stop_beep()
  404. 14330 {
  405. 14331 /* Turn off the beeper by turning off bits 0 and 1 in PORT_B. */
  406. 14332
  407. 14333   lock();                       /* guard PORT_B from keyboard intr handler */
  408. 14334   out_byte(PORT_B, in_byte(PORT_B) & ~3);
  409. 14335   beeping = FALSE;
  410. 14336   unlock();
  411. 14337 }
  412. 14340 /*===========================================================================*
  413. 14341  *                              scr_init                                     *
  414. 14342  *===========================================================================*/
  415. 14343 PUBLIC void scr_init(tp)
  416. 14344 tty_t *tp;
  417. 14345 {
  418. 14346 /* Initialize the screen driver. */
  419. 14347   console_t *cons;
  420. 14348   phys_bytes vid_base;
  421. 14349   u16_t bios_crtbase;
  422. 14350   int line;
  423. 14351   unsigned page_size;
  424. 14352
  425. 14353   /* Associate console and TTY. */
  426. 14354   line = tp - &tty_table[0];
  427. 14355   if (line >= nr_cons) return;
  428. 14356   cons = &cons_table[line];
  429. 14357   cons->c_tty = tp;
  430. 14358   tp->tty_priv = cons;
  431. 14359
  432. 14360   /* Initialize the keyboard driver. */
  433. 14361   kb_init(tp);
  434. 14362
  435. 14363   /* Output functions. */
  436. 14364   tp->tty_devwrite = cons_write;
  437. 14365   tp->tty_echo = cons_echo;
  438. 14366
  439. 14367   /* Get the BIOS parameters that tells the VDU I/O base register. */
  440. 14368   phys_copy(0x463L, vir2phys(&bios_crtbase), 2L);
  441. 14369
  442. 14370   vid_port = bios_crtbase;
  443. 14371
  444. 14372   if (color) {
  445. 14373         vid_base = COLOR_BASE;
  446. 14374         vid_size = COLOR_SIZE;
  447. 14375   } else {
  448. 14376         vid_base = MONO_BASE;
  449. 14377         vid_size = MONO_SIZE;
  450. 14378   }
  451. 14379   if (ega) vid_size = EGA_SIZE; /* for both EGA and VGA */
  452. 14380   wrap = !ega;
  453. 14381
  454. 14382   vid_seg = protected_mode ? VIDEO_SELECTOR : physb_to_hclick(vid_base);
  455. 14383   init_dataseg(&gdt[VIDEO_INDEX], vid_base, (phys_bytes) vid_size,
  456. 14384                                                         TASK_PRIVILEGE);
  457. 14385   vid_size >>= 1;               /* word count */
  458. 14386   vid_mask = vid_size - 1;
  459. 14387
  460. 14388   /* There can be as many consoles as video memory allows. */
  461. 14389   nr_cons = vid_size / scr_size;
  462. 14390   if (nr_cons > NR_CONS) nr_cons = NR_CONS;
  463. 14391   if (nr_cons > 1) wrap = 0;
  464. 14392   page_size = vid_size / nr_cons;
  465. 14393   cons->c_start = line * page_size;
  466. 14394   cons->c_limit = cons->c_start + page_size;
  467. 14395   cons->c_org = cons->c_start;
  468. 14396   cons->c_attr = cons->c_blank = BLANK_COLOR;
  469. 14397
  470. 14398   /* Clear the screen. */
  471. 14399   blank_color = BLANK_COLOR;
  472. 14400   mem_vid_copy(BLANK_MEM, cons->c_start, scr_size);
  473. 14401   select_console(0);
  474. 14402 }
  475. 14405 /*===========================================================================*
  476. 14406  *                              putk                                         *
  477. 14407  *===========================================================================*/
  478. 14408 PUBLIC void putk(c)
  479. 14409 int c;                          /* character to print */
  480. 14410 {
  481. 14411 /* This procedure is used by the version of printf() that is linked with
  482. 14412  * the kernel itself.  The one in the library sends a message to FS, which is
  483. 14413  * not what is needed for printing within the kernel.  This version just queues
  484. 14414  * the character and starts the output.
  485. 14415  */
  486. 14416
  487. 14417   if (c != 0) {
  488. 14418         if (c == 'n') putk('r');
  489. 14419         out_char(&cons_table[0], (int) c);
  490. 14420   } else {
  491. 14421         flush(&cons_table[0]);
  492. 14422   }
  493. 14423 }
  494. 14426 /*===========================================================================*
  495. 14427  *                              toggle_scroll                                *
  496. 14428  *===========================================================================*/
  497. 14429 PUBLIC void toggle_scroll()
  498. 14430 {
  499. 14431 /* Toggle between hardware and software scroll. */
  500. 14432
  501. 14433   cons_org0();
  502. 14434   softscroll = !softscroll;
  503. 14435   printf("%sware scrolling enabled.n", softscroll ? "Soft" : "Hard");
  504. 14436 }
  505. 14439 /*===========================================================================*
  506. 14440  *                              cons_stop                                    *
  507. 14441  *===========================================================================*/
  508. 14442 PUBLIC void cons_stop()
  509. 14443 {
  510. 14444 /* Prepare for halt or reboot. */
  511. 14445
  512. 14446   cons_org0();
  513. 14447   softscroll = 1;
  514. 14448   select_console(0);
  515. 14449   cons_table[0].c_attr = cons_table[0].c_blank = BLANK_COLOR;
  516. 14450 }
  517. 14453 /*===========================================================================*
  518. 14454  *                              cons_org0                                    *
  519. 14455  *===========================================================================*/
  520. 14456 PRIVATE void cons_org0()
  521. 14457 {
  522. 14458 /* Scroll video memory back to put the origin at 0. */
  523. 14459
  524. 14460   int cons_line;
  525. 14461   console_t *cons;
  526. 14462   unsigned n;
  527. 14463
  528. 14464   for (cons_line = 0; cons_line < nr_cons; cons_line++) {
  529. 14465         cons = &cons_table[cons_line];
  530. 14466         while (cons->c_org > cons->c_start) {
  531. 14467                 n = vid_size - scr_size;        /* amount of unused memory */
  532. 14468                 if (n > cons->c_org - cons->c_start)
  533. 14469                         n = cons->c_org - cons->c_start;
  534. 14470                 vid_vid_copy(cons->c_org, cons->c_org - n, scr_size);
  535. 14471                 cons->c_org -= n;
  536. 14472         }
  537. 14473         flush(cons);
  538. 14474   }
  539. 14475   select_console(current);
  540. 14476 }
  541. 14479 /*===========================================================================*
  542. 14480  *                              select_console                               *
  543. 14481  *===========================================================================*/
  544. 14482 PUBLIC void select_console(int cons_line)
  545. 14483 {
  546. 14484 /* Set the current console to console number 'cons_line'. */
  547. 14485
  548. 14486   if (cons_line < 0 || cons_line >= nr_cons) return;
  549. 14487   current = cons_line;
  550. 14488   curcons = &cons_table[cons_line];
  551. 14489   set_6845(VID_ORG, curcons->c_org);
  552. 14490   set_6845(CURSOR, curcons->c_cur);
  553. 14491 }
  554. 14494 /*===========================================================================*
  555. 14495  *                              con_loadfont                                 *
  556. 14496  *===========================================================================*/
  557. 14497 PUBLIC int con_loadfont(user_phys)
  558. 14498 phys_bytes user_phys;
  559. 14499 {
  560. 14500 /* Load a font into the EGA or VGA adapter. */
  561. 14501
  562. 14502   static struct sequence seq1[7] = {
  563. 14503         { GA_SEQUENCER_INDEX, 0x00, 0x01 },
  564. 14504         { GA_SEQUENCER_INDEX, 0x02, 0x04 },
  565. 14505         { GA_SEQUENCER_INDEX, 0x04, 0x07 },
  566. 14506         { GA_SEQUENCER_INDEX, 0x00, 0x03 },
  567. 14507         { GA_GRAPHICS_INDEX, 0x04, 0x02 },
  568. 14508         { GA_GRAPHICS_INDEX, 0x05, 0x00 },
  569. 14509         { GA_GRAPHICS_INDEX, 0x06, 0x00 },
  570. 14510   };
  571. 14511   static struct sequence seq2[7] = {
  572. 14512         { GA_SEQUENCER_INDEX, 0x00, 0x01 },
  573. 14513         { GA_SEQUENCER_INDEX, 0x02, 0x03 },
  574. 14514         { GA_SEQUENCER_INDEX, 0x04, 0x03 },
  575. 14515         { GA_SEQUENCER_INDEX, 0x00, 0x03 },
  576. 14516         { GA_GRAPHICS_INDEX, 0x04, 0x00 },
  577. 14517         { GA_GRAPHICS_INDEX, 0x05, 0x10 },
  578. 14518         { GA_GRAPHICS_INDEX, 0x06,    0 },
  579. 14519   };
  580. 14520
  581. 14521   seq2[6].value= color ? 0x0E : 0x0A;
  582. 14522
  583. 14523   if (!ega) return(ENOTTY);
  584. 14524
  585. 14525   lock();
  586. 14526   ga_program(seq1);     /* bring font memory into view */
  587. 14527
  588. 14528   phys_copy(user_phys, (phys_bytes)GA_VIDEO_ADDRESS, (phys_bytes)GA_FONT_SIZE);
  589. 14529
  590. 14530   ga_program(seq2);     /* restore */
  591. 14531   unlock();
  592. 14532
  593. 14533   return(OK);
  594. 14534 }
  595. 14537 /*===========================================================================*
  596. 14538  *                              ga_program                                   *
  597. 14539  *===========================================================================*/
  598. 14540 PRIVATE void ga_program(seq)
  599. 14541 struct sequence *seq;
  600. 14542 {
  601. 14543 /* support function for con_loadfont */
  602. 14544
  603. 14545   int len= 7;
  604. 14546   do {
  605. 14547         out_byte(seq->index, seq->port);
  606. 14548         out_byte(seq->index+1, seq->value);
  607. 14549         seq++;
  608. 14550   } while (--len > 0);
  609. 14551 }
  610. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  611. src/kernel/dmp.c    
  612. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  613. 14600 /* This file contains some dumping routines for debugging. */
  614. 14601
  615. 14602 #include "kernel.h"
  616. 14603 #include <minix/com.h>
  617. 14604 #include "proc.h"
  618. 14605
  619. 14606 char *vargv;
  620. 14607
  621. 14608 FORWARD _PROTOTYPE(char *proc_name, (int proc_nr));
  622. 14609
  623. 14610 /*===========================================================================*
  624. 14611  *                              p_dmp                                        *
  625. 14612  *===========================================================================*/
  626. 14613 PUBLIC void p_dmp()
  627. 14614 {
  628. 14615 /* Proc table dump */
  629. 14616
  630. 14617   register struct proc *rp;
  631. 14618   static struct proc *oldrp = BEG_PROC_ADDR;
  632. 14619   int n = 0;
  633. 14620   phys_clicks text, data, size;
  634. 14621   int proc_nr;
  635. 14622
  636. 14623   printf("n--pid --pc- ---sp- flag -user --sys-- -text- -data- -size- -recv- commandn");
  637. 14624
  638. 14625   for (rp = oldrp; rp < END_PROC_ADDR; rp++) {
  639. 14626         proc_nr = proc_number(rp);
  640. 14627         if (rp->p_flags & P_SLOT_FREE) continue;
  641. 14628         if (++n > 20) break;
  642. 14629         text = rp->p_map[T].mem_phys;
  643. 14630         data = rp->p_map[D].mem_phys;
  644. 14631         size = rp->p_map[T].mem_len
  645. 14632                 + ((rp->p_map[S].mem_phys + rp->p_map[S].mem_len) - data);
  646. 14633         printf("%5d %5lx %6lx %2x %7U %7U %5uK %5uK %5uK ",
  647. 14634                proc_nr < 0 ? proc_nr : rp->p_pid,
  648. 14635                (unsigned long) rp->p_reg.pc,
  649. 14636                (unsigned long) rp->p_reg.sp,
  650. 14637                rp->p_flags,
  651. 14638                rp->user_time, rp->sys_time,
  652. 14639                click_to_round_k(text), click_to_round_k(data),
  653. 14640                click_to_round_k(size));
  654. 14641         if (rp->p_flags & RECEIVING) {
  655. 14642                 printf("%-7.7s", proc_name(rp->p_getfrom));
  656. 14643         } else
  657. 14644         if (rp->p_flags & SENDING) {
  658. 14645                 printf("S:%-5.5s", proc_name(rp->p_sendto));
  659. 14646         } else
  660. 14647         if (rp->p_flags == 0) {
  661. 14648                 printf("       ");
  662. 14649         }
  663. 14650         printf("%sn", rp->p_name);
  664. 14651   }
  665. 14652   if (rp == END_PROC_ADDR) rp = BEG_PROC_ADDR; else printf("--more--r");
  666. 14653   oldrp = rp;
  667. 14654 }
  668. 14657 /*===========================================================================*
  669. 14658  *                              map_dmp                                      *
  670. 14659  *===========================================================================*/
  671. 14660 PUBLIC void map_dmp()
  672. 14661 {
  673. 14662   register struct proc *rp;
  674. 14663   static struct proc *oldrp = cproc_addr(HARDWARE);
  675. 14664   int n = 0;
  676. 14665   phys_clicks size;
  677. 14666
  678. 14667   printf("nPROC NAME-  -----TEXT-----  -----DATA-----  ----STACK-----  -SIZE-n");
  679. 14668   for (rp = oldrp; rp < END_PROC_ADDR; rp++) {
  680. 14669         if (rp->p_flags & P_SLOT_FREE) continue;
  681. 14670         if (++n > 20) break;
  682. 14671         size = rp->p_map[T].mem_len
  683. 14672                 + ((rp->p_map[S].mem_phys + rp->p_map[S].mem_len)
  684. 14673                                                 - rp->p_map[D].mem_phys);
  685. 14674         printf("%3d %-6.6s  %4x %4x %4x  %4x %4x %4x  %4x %4x %4x  %5uKn",
  686. 14675                proc_number(rp),
  687. 14676                rp->p_name,
  688. 14677                rp->p_map[T].mem_vir, rp->p_map[T].mem_phys, rp->p_map[T].mem_len,
  689. 14678                rp->p_map[D].mem_vir, rp->p_map[D].mem_phys, rp->p_map[D].mem_len,
  690. 14679                rp->p_map[S].mem_vir, rp->p_map[S].mem_phys, rp->p_map[S].mem_len,
  691. 14680                click_to_round_k(size));
  692. 14681   }
  693. 14682   if (rp == END_PROC_ADDR) rp = cproc_addr(HARDWARE); else printf("--more--r");
  694. 14683   oldrp = rp;
  695. 14684 }
  696. 14687 /*===========================================================================*
  697. 14688  *                              proc_name                                    *
  698. 14689  *===========================================================================*/
  699. 14690 PRIVATE char *proc_name(proc_nr)
  700. 14691 int proc_nr;
  701. 14692 {
  702. 14693   if (proc_nr == ANY) return "ANY";
  703. 14694   return proc_addr(proc_nr)->p_name;
  704. 14695 }
  705. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  706. src/kernel/system.c    
  707. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  708. 14700 /* This task handles the interface between file system and kernel as well as
  709. 14701  * between memory manager and kernel.  System services are obtained by sending
  710. 14702  * sys_task() a message specifying what is needed.  To make life easier for
  711. 14703  * MM and FS, a library is provided with routines whose names are of the
  712. 14704  * form sys_xxx, e.g. sys_xit sends the SYS_XIT message to sys_task.  The
  713. 14705  * message types and parameters are:
  714. 14706  *
  715. 14707  *   SYS_FORK    informs kernel that a process has forked
  716. 14708  *   SYS_NEWMAP  allows MM to set up a process memory map
  717. 14709  *   SYS_GETMAP  allows MM to get a process' memory map
  718. 14710  *   SYS_EXEC    sets program counter and stack pointer after EXEC
  719. 14711  *   SYS_XIT     informs kernel that a process has exited
  720. 14712  *   SYS_GETSP   caller wants to read out some process' stack pointer
  721. 14713  *   SYS_TIMES   caller wants to get accounting times for a process
  722. 14714  *   SYS_ABORT   MM or FS cannot go on; abort MINIX
  723. 14715  *   SYS_FRESH   start with a fresh process image during EXEC (68000 only)
  724. 14716  *   SYS_SENDSIG send a signal to a process (POSIX style)
  725. 14717  *   SYS_SIGRETURN complete POSIX-style signalling
  726. 14718  *   SYS_KILL    cause a signal to be sent via MM
  727. 14719  *   SYS_ENDSIG  finish up after SYS_KILL-type signal
  728. 14720  *   SYS_COPY    request a block of data to be copied between processes
  729. 14721  *   SYS_VCOPY   request a series of data blocks to be copied between procs
  730. 14722  *   SYS_GBOOT   copies the boot parameters to a process
  731. 14723  *   SYS_MEM     returns the next free chunk of physical memory
  732. 14724  *   SYS_UMAP    compute the physical address for a given virtual address
  733. 14725  *   SYS_TRACE   request a trace operation
  734. 14726  *
  735. 14727  * Message types and parameters:
  736. 14728  *
  737. 14729  *    m_type       PROC1     PROC2      PID     MEM_PTR
  738. 14730  * ------------------------------------------------------
  739. 14731  * | SYS_FORK   | parent  |  child  |   pid   |         |
  740. 14732  * |------------+---------+---------+---------+---------|
  741. 14733  * | SYS_NEWMAP | proc nr |         |         | map ptr |
  742. 14734  * |------------+---------+---------+---------+---------|
  743. 14735  * | SYS_EXEC   | proc nr | traced  | new sp  |         |
  744. 14736  * |------------+---------+---------+---------+---------|
  745. 14737  * | SYS_XIT    | parent  | exitee  |         |         |
  746. 14738  * |------------+---------+---------+---------+---------|
  747. 14739  * | SYS_GETSP  | proc nr |         |         |         |
  748. 14740  * |------------+---------+---------+---------+---------|
  749. 14741  * | SYS_TIMES  | proc nr |         | buf ptr |         |
  750. 14742  * |------------+---------+---------+---------+---------|
  751. 14743  * | SYS_ABORT  |         |         |         |         |
  752. 14744  * |------------+---------+---------+---------+---------|
  753. 14745  * | SYS_FRESH  | proc nr | data_cl |         |         |
  754. 14746  * |------------+---------+---------+---------+---------|
  755. 14747  * | SYS_GBOOT  | proc nr |         |         | bootptr |
  756. 14748  * |------------+---------+---------+---------+---------|
  757. 14749  * | SYS_GETMAP | proc nr |         |         | map ptr |
  758. 14750  * ------------------------------------------------------
  759. 14751  *
  760. 14752  *    m_type          m1_i1     m1_i2     m1_i3       m1_p1
  761. 14753  * ----------------+---------+---------+---------+--------------
  762. 14754  * | SYS_VCOPY     |  src p  |  dst p  | vec siz | vc addr     |
  763. 14755  * |---------------+---------+---------+---------+-------------|
  764. 14756  * | SYS_SENDSIG   | proc nr |         |         | smp         |
  765. 14757  * |---------------+---------+---------+---------+-------------|
  766. 14758  * | SYS_SIGRETURN | proc nr |         |         | scp         |
  767. 14759  * |---------------+---------+---------+---------+-------------|
  768. 14760  * | SYS_ENDSIG    | proc nr |         |         |             |
  769. 14761  * -------------------------------------------------------------
  770. 14762  *
  771. 14763  *    m_type       m2_i1     m2_i2     m2_l1     m2_l2
  772. 14764  * ------------------------------------------------------
  773. 14765  * | SYS_TRACE  | proc_nr | request |  addr   |  data   |
  774. 14766  * ------------------------------------------------------
  775. 14767  *
  776. 14768  *
  777. 14769  *    m_type       m6_i1     m6_i2     m6_i3     m6_f1
  778. 14770  * ------------------------------------------------------
  779. 14771  * | SYS_KILL   | proc_nr  |  sig    |         |         |
  780. 14772  * ------------------------------------------------------
  781. 14773  *
  782. 14774  *
  783. 14775  *    m_type      m5_c1   m5_i1    m5_l1   m5_c2   m5_i2    m5_l2   m5_l3
  784. 14776  * --------------------------------------------------------------------------
  785. 14777  * | SYS_COPY   |src seg|src proc|src vir|dst seg|dst proc|dst vir| byte ct |
  786. 14778  * --------------------------------------------------------------------------
  787. 14779  * | SYS_UMAP   |  seg  |proc nr |vir adr|       |        |       | byte ct |
  788. 14780  * --------------------------------------------------------------------------
  789. 14781  *
  790. 14782  *
  791. 14783  *    m_type      m1_i1      m1_i2      m1_i3
  792. 14784  * |------------+----------+----------+----------
  793. 14785  * | SYS_MEM    | mem base | mem size | tot mem |
  794. 14786  * ----------------------------------------------
  795. 14787  *
  796. 14788  * In addition to the main sys_task() entry point, there are 5 other minor
  797. 14789  * entry points:
  798. 14790  *   cause_sig: take action to cause a signal to occur, sooner or later
  799. 14791  *   inform:    tell MM about pending signals
  800. 14792  *   numap:     umap D segment starting from process number instead of pointer
  801. 14793  *   umap:      compute the physical address for a given virtual address
  802. 14794  *   alloc_segments: allocate segments for 8088 or higher processor
  803. 14795  */
  804. 14796
  805. 14797 #include "kernel.h"
  806. 14798 #include <signal.h>
  807. 14799 #include <unistd.h>
  808. 14800 #include <sys/sigcontext.h>
  809. 14801 #include <sys/ptrace.h>
  810. 14802 #include <minix/boot.h>
  811. 14803 #include <minix/callnr.h>
  812. 14804 #include <minix/com.h>
  813. 14805 #include "proc.h"
  814. 14806 #include "protect.h"
  815. 14807
  816. 14808 /* PSW masks. */
  817. 14809 #define IF_MASK 0x00000200
  818. 14810 #define IOPL_MASK 0x003000
  819. 14811
  820. 14812 PRIVATE message m;
  821. 14813
  822. 14814 FORWARD _PROTOTYPE( int do_abort, (message *m_ptr) );
  823. 14815 FORWARD _PROTOTYPE( int do_copy, (message *m_ptr) );
  824. 14816 FORWARD _PROTOTYPE( int do_exec, (message *m_ptr) );
  825. 14817 FORWARD _PROTOTYPE( int do_fork, (message *m_ptr) );
  826. 14818 FORWARD _PROTOTYPE( int do_gboot, (message *m_ptr) );
  827. 14819 FORWARD _PROTOTYPE( int do_getsp, (message *m_ptr) );
  828. 14820 FORWARD _PROTOTYPE( int do_kill, (message *m_ptr) );
  829. 14821 FORWARD _PROTOTYPE( int do_mem, (message *m_ptr) );
  830. 14822 FORWARD _PROTOTYPE( int do_newmap, (message *m_ptr) );
  831. 14823 FORWARD _PROTOTYPE( int do_sendsig, (message *m_ptr) );
  832. 14824 FORWARD _PROTOTYPE( int do_sigreturn, (message *m_ptr) );
  833. 14825 FORWARD _PROTOTYPE( int do_endsig, (message *m_ptr) );
  834. 14826 FORWARD _PROTOTYPE( int do_times, (message *m_ptr) );
  835. 14827 FORWARD _PROTOTYPE( int do_trace, (message *m_ptr) );
  836. 14828 FORWARD _PROTOTYPE( int do_umap, (message *m_ptr) );
  837. 14829 FORWARD _PROTOTYPE( int do_xit, (message *m_ptr) );
  838. 14830 FORWARD _PROTOTYPE( int do_vcopy, (message *m_ptr) );
  839. 14831 FORWARD _PROTOTYPE( int do_getmap, (message *m_ptr) );
  840. 14832
  841. 14833
  842. 14834 /*===========================================================================*
  843. 14835  *                              sys_task                                     *
  844. 14836  *===========================================================================*/
  845. 14837 PUBLIC void sys_task()
  846. 14838 {
  847. 14839 /* Main entry point of sys_task.  Get the message and dispatch on type. */
  848. 14840
  849. 14841   register int r;
  850. 14842
  851. 14843   while (TRUE) {
  852. 14844         receive(ANY, &m);
  853. 14845
  854. 14846         switch (m.m_type) {     /* which system call */
  855. 14847             case SYS_FORK:      r = do_fork(&m);        break;
  856. 14848             case SYS_NEWMAP:    r = do_newmap(&m);      break;
  857. 14849             case SYS_GETMAP:    r = do_getmap(&m);      break;
  858. 14850             case SYS_EXEC:      r = do_exec(&m);        break;
  859. 14851             case SYS_XIT:       r = do_xit(&m);         break;
  860. 14852             case SYS_GETSP:     r = do_getsp(&m);       break;
  861. 14853             case SYS_TIMES:     r = do_times(&m);       break;
  862. 14854             case SYS_ABORT:     r = do_abort(&m);       break;
  863. 14855             case SYS_SENDSIG:   r = do_sendsig(&m);     break;
  864. 14856             case SYS_SIGRETURN: r = do_sigreturn(&m);   break;
  865. 14857             case SYS_KILL:      r = do_kill(&m);        break;
  866. 14858             case SYS_ENDSIG:    r = do_endsig(&m);      break;
  867. 14859             case SYS_COPY:      r = do_copy(&m);        break;
  868. 14860             case SYS_VCOPY:     r = do_vcopy(&m);       break;
  869. 14861             case SYS_GBOOT:     r = do_gboot(&m);       break;
  870. 14862             case SYS_MEM:       r = do_mem(&m);         break;
  871. 14863             case SYS_UMAP:      r = do_umap(&m);        break;
  872. 14864             case SYS_TRACE:     r = do_trace(&m);       break;
  873. 14865             default:            r = E_BAD_FCN;
  874. 14866         }
  875. 14867
  876. 14868         m.m_type = r;           /* 'r' reports status of call */
  877. 14869         send(m.m_source, &m);   /* send reply to caller */
  878. 14870   }
  879. 14871 }
  880. 14874 /*===========================================================================*
  881. 14875  *                              do_fork                                      *
  882. 14876  *===========================================================================*/
  883. 14877 PRIVATE int do_fork(m_ptr)
  884. 14878 register message *m_ptr;        /* pointer to request message */
  885. 14879 {
  886. 14880 /* Handle sys_fork().  m_ptr->PROC1 has forked.  The child is m_ptr->PROC2. */
  887. 14881
  888. 14882   reg_t old_ldt_sel;
  889. 14883   register struct proc *rpc;
  890. 14884   struct proc *rpp;
  891. 14885
  892. 14886   if (!isoksusern(m_ptr->PROC1) || !isoksusern(m_ptr->PROC2))
  893. 14887         return(E_BAD_PROC);
  894. 14888   rpp = proc_addr(m_ptr->PROC1);
  895. 14889   rpc = proc_addr(m_ptr->PROC2);
  896. 14890
  897. 14891   /* Copy parent 'proc' struct to child. */
  898. 14892   old_ldt_sel = rpc->p_ldt_sel; /* stop this being obliterated by copy */
  899. 14893
  900. 14894   *rpc = *rpp;                  /* copy 'proc' struct */
  901. 14895
  902. 14896   rpc->p_ldt_sel = old_ldt_sel;
  903. 14897   rpc->p_nr = m_ptr->PROC2;     /* this was obliterated by copy */
  904. 14898
  905. 14899   rpc->p_flags |= NO_MAP;       /* inhibit the process from running */
  906. 14900
  907. 14901   rpc->p_flags &= ~(PENDING | SIG_PENDING | P_STOP);
  908. 14902
  909. 14903   /* Only 1 in group should have PENDING, child does not inherit trace status*/
  910. 14904   sigemptyset(&rpc->p_pending);
  911. 14905   rpc->p_pendcount = 0;
  912. 14906   rpc->p_pid = m_ptr->PID;      /* install child's pid */
  913. 14907   rpc->p_reg.retreg = 0;        /* child sees pid = 0 to know it is child */
  914. 14908
  915. 14909   rpc->user_time = 0;           /* set all the accounting times to 0 */
  916. 14910   rpc->sys_time = 0;
  917. 14911   rpc->child_utime = 0;
  918. 14912   rpc->child_stime = 0;
  919. 14913
  920. 14914   return(OK);
  921. 14915 }
  922. 14918 /*===========================================================================*
  923. 14919  *                              do_newmap                                    *
  924. 14920  *===========================================================================*/
  925. 14921 PRIVATE int do_newmap(m_ptr)
  926. 14922 message *m_ptr;                 /* pointer to request message */
  927. 14923 {
  928. 14924 /* Handle sys_newmap().  Fetch the memory map from MM. */
  929. 14925
  930. 14926   register struct proc *rp;
  931. 14927   phys_bytes src_phys;
  932. 14928   int caller;                   /* whose space has the new map (usually MM) */
  933. 14929   int k;                        /* process whose map is to be loaded */
  934. 14930   int old_flags;                /* value of flags before modification */
  935. 14931   struct mem_map *map_ptr;      /* virtual address of map inside caller (MM) */
  936. 14932
  937. 14933   /* Extract message parameters and copy new memory map from MM. */
  938. 14934   caller = m_ptr->m_source;
  939. 14935   k = m_ptr->PROC1;
  940. 14936   map_ptr = (struct mem_map *) m_ptr->MEM_PTR;
  941. 14937   if (!isokprocn(k)) return(E_BAD_PROC);
  942. 14938   rp = proc_addr(k);            /* ptr to entry of user getting new map */
  943. 14939
  944. 14940   /* Copy the map from MM. */
  945. 14941   src_phys = umap(proc_addr(caller), D, (vir_bytes) map_ptr, sizeof(rp->p_map));
  946. 14942   if (src_phys == 0) panic("bad call to sys_newmap", NO_NUM);
  947. 14943   phys_copy(src_phys, vir2phys(rp->p_map), (phys_bytes) sizeof(rp->p_map));
  948. 14944
  949. 14945   alloc_segments(rp);
  950. 14946   old_flags = rp->p_flags;      /* save the previous value of the flags */
  951. 14947   rp->p_flags &= ~NO_MAP;
  952. 14948   if (old_flags != 0 && rp->p_flags == 0) lock_ready(rp);
  953. 14949
  954. 14950   return(OK);
  955. 14951 }
  956. 14954 /*===========================================================================*
  957. 14955  *                              do_getmap                                    *
  958. 14956  *===========================================================================*/
  959. 14957 PRIVATE int do_getmap(m_ptr)
  960. 14958 message *m_ptr;                 /* pointer to request message */
  961. 14959 {
  962. 14960 /* Handle sys_getmap().  Report the memory map to MM. */
  963. 14961
  964. 14962   register struct proc *rp;
  965. 14963   phys_bytes dst_phys;
  966. 14964   int caller;                   /* where the map has to be stored */
  967. 14965   int k;                        /* process whose map is to be loaded */
  968. 14966   struct mem_map *map_ptr;      /* virtual address of map inside caller (MM) */
  969. 14967
  970. 14968   /* Extract message parameters and copy new memory map to MM. */
  971. 14969   caller = m_ptr->m_source;
  972. 14970   k = m_ptr->PROC1;
  973. 14971   map_ptr = (struct mem_map *) m_ptr->MEM_PTR;
  974. 14972
  975. 14973   if (!isokprocn(k))
  976. 14974         panic("do_getmap got bad proc: ", m_ptr->PROC1);
  977. 14975
  978. 14976   rp = proc_addr(k);            /* ptr to entry of the map */
  979. 14977
  980. 14978   /* Copy the map to MM. */
  981. 14979   dst_phys = umap(proc_addr(caller), D, (vir_bytes) map_ptr, sizeof(rp->p_map));
  982. 14980   if (dst_phys == 0) panic("bad call to sys_getmap", NO_NUM);
  983. 14981   phys_copy(vir2phys(rp->p_map), dst_phys, sizeof(rp->p_map));
  984. 14982
  985. 14983   return(OK);
  986. 14984 }
  987. 14987 /*===========================================================================*
  988. 14988  *                              do_exec                                      *
  989. 14989  *===========================================================================*/
  990. 14990 PRIVATE int do_exec(m_ptr)
  991. 14991 register message *m_ptr;        /* pointer to request message */
  992. 14992 {
  993. 14993 /* Handle sys_exec().  A process has done a successful EXEC. Patch it up. */
  994. 14994
  995. 14995   register struct proc *rp;
  996. 14996   reg_t sp;                     /* new sp */
  997. 14997   phys_bytes phys_name;
  998. 14998   char *np;
  999. 14999 #define NLEN (sizeof(rp->p_name)-1)
  1000. 15000
  1001. 15001   if (!isoksusern(m_ptr->PROC1)) return E_BAD_PROC;
  1002. 15002   /* PROC2 field is used as flag to indicate process is being traced */
  1003. 15003   if (m_ptr->PROC2) cause_sig(m_ptr->PROC1, SIGTRAP);
  1004. 15004   sp = (reg_t) m_ptr->STACK_PTR;
  1005. 15005   rp = proc_addr(m_ptr->PROC1);
  1006. 15006   rp->p_reg.sp = sp;            /* set the stack pointer */
  1007. 15007   rp->p_reg.pc = (reg_t) m_ptr->IP_PTR; /* set pc */
  1008. 15008   rp->p_alarm = 0;              /* reset alarm timer */
  1009. 15009   rp->p_flags &= ~RECEIVING;    /* MM does not reply to EXEC call */
  1010. 15010   if (rp->p_flags == 0) lock_ready(rp);
  1011. 15011
  1012. 15012   /* Save command name for debugging, ps(1) output, etc. */
  1013. 15013   phys_name = numap(m_ptr->m_source, (vir_bytes) m_ptr->NAME_PTR,
  1014. 15014                                                         (vir_bytes) NLEN);
  1015. 15015   if (phys_name != 0) {
  1016. 15016         phys_copy(phys_name, vir2phys(rp->p_name), (phys_bytes) NLEN);
  1017. 15017         for (np = rp->p_name; (*np & BYTE) >= ' '; np++) {}
  1018. 15018         *np = 0;
  1019. 15019   }
  1020. 15020   return(OK);
  1021. 15021 }
  1022. 15024 /*===========================================================================*
  1023. 15025  *                              do_xit                                       *
  1024. 15026  *===========================================================================*/
  1025. 15027 PRIVATE int do_xit(m_ptr)
  1026. 15028 message *m_ptr;                 /* pointer to request message */
  1027. 15029 {
  1028. 15030 /* Handle sys_xit().  A process has exited. */
  1029. 15031
  1030. 15032   register struct proc *rp, *rc;
  1031. 15033   struct proc *np, *xp;
  1032. 15034   int parent;                   /* number of exiting proc's parent */
  1033. 15035   int proc_nr;                  /* number of process doing the exit */
  1034. 15036   phys_clicks base, size;
  1035. 15037
  1036. 15038   parent = m_ptr->PROC1;        /* slot number of parent process */
  1037. 15039   proc_nr = m_ptr->PROC2;       /* slot number of exiting process */
  1038. 15040   if (!isoksusern(parent) || !isoksusern(proc_nr)) return(E_BAD_PROC);
  1039. 15041   rp = proc_addr(parent);
  1040. 15042   rc = proc_addr(proc_nr);
  1041. 15043   lock();
  1042. 15044   rp->child_utime += rc->user_time + rc->child_utime;   /* accum child times */
  1043. 15045   rp->child_stime += rc->sys_time + rc->child_stime;
  1044. 15046   unlock();
  1045. 15047   rc->p_alarm = 0;              /* turn off alarm timer */
  1046. 15048   if (rc->p_flags == 0) lock_unready(rc);
  1047. 15049
  1048. 15050   strcpy(rc->p_name, "<noname>");       /* process no longer has a name */
  1049. 15051
  1050. 15052   /* If the process being terminated happens to be queued trying to send a
  1051. 15053    * message (i.e., the process was killed by a signal, rather than it doing an
  1052. 15054    * EXIT), then it must be removed from the message queues.
  1053. 15055    */
  1054. 15056   if (rc->p_flags & SENDING) {
  1055. 15057         /* Check all proc slots to see if the exiting process is queued. */
  1056. 15058         for (rp = BEG_PROC_ADDR; rp < END_PROC_ADDR; rp++) {
  1057. 15059                 if (rp->p_callerq == NIL_PROC) continue;
  1058. 15060                 if (rp->p_callerq == rc) {
  1059. 15061                         /* Exiting process is on front of this queue. */
  1060. 15062                         rp->p_callerq = rc->p_sendlink;
  1061. 15063                         break;
  1062. 15064                 } else {
  1063. 15065                         /* See if exiting process is in middle of queue. */
  1064. 15066                         np = rp->p_callerq;
  1065. 15067                         while ( ( xp = np->p_sendlink) != NIL_PROC)
  1066. 15068                                 if (xp == rc) {
  1067. 15069                                         np->p_sendlink = xp->p_sendlink;
  1068. 15070                                         break;
  1069. 15071                                 } else {
  1070. 15072                                         np = xp;
  1071. 15073                                 }
  1072. 15074                 }
  1073. 15075         }
  1074. 15076   }
  1075. 15077
  1076. 15078   if (rc->p_flags & PENDING) --sig_procs;
  1077. 15079   sigemptyset(&rc->p_pending);
  1078. 15080   rc->p_pendcount = 0;
  1079. 15081   rc->p_flags = P_SLOT_FREE;
  1080. 15082   return(OK);
  1081. 15083 }
  1082. 15086 /*===========================================================================*
  1083. 15087  *                              do_getsp                                     *
  1084. 15088  *===========================================================================*/
  1085. 15089 PRIVATE int do_getsp(m_ptr)
  1086. 15090 register message *m_ptr;        /* pointer to request message */
  1087. 15091 {
  1088. 15092 /* Handle sys_getsp().  MM wants to know what sp is. */
  1089. 15093
  1090. 15094   register struct proc *rp;
  1091. 15095
  1092. 15096   if (!isoksusern(m_ptr->PROC1)) return(E_BAD_PROC);
  1093. 15097   rp = proc_addr(m_ptr->PROC1);
  1094. 15098   m_ptr->STACK_PTR = (char *) rp->p_reg.sp;     /* return sp here (bad type) */
  1095. 15099   return(OK);
  1096. 15100 }
  1097. 15103 /*===========================================================================*
  1098. 15104  *                              do_times                                     *
  1099. 15105  *===========================================================================*/
  1100. 15106 PRIVATE int do_times(m_ptr)
  1101. 15107 register message *m_ptr;        /* pointer to request message */
  1102. 15108 {
  1103. 15109 /* Handle sys_times().  Retrieve the accounting information. */
  1104. 15110
  1105. 15111   register struct proc *rp;
  1106. 15112
  1107. 15113   if (!isoksusern(m_ptr->PROC1)) return E_BAD_PROC;
  1108. 15114   rp = proc_addr(m_ptr->PROC1);
  1109. 15115
  1110. 15116   /* Insert the times needed by the TIMES system call in the message. */
  1111. 15117   lock();                       /* halt the volatile time counters in rp */
  1112. 15118   m_ptr->USER_TIME   = rp->user_time;
  1113. 15119   m_ptr->SYSTEM_TIME = rp->sys_time;
  1114. 15120   unlock();
  1115. 15121   m_ptr->CHILD_UTIME = rp->child_utime;
  1116. 15122   m_ptr->CHILD_STIME = rp->child_stime;
  1117. 15123   m_ptr->BOOT_TICKS  = get_uptime();
  1118. 15124   return(OK);
  1119. 15125 }
  1120. 15128 /*===========================================================================*
  1121. 15129  *                              do_abort                                     *
  1122. 15130  *===========================================================================*/
  1123. 15131 PRIVATE int do_abort(m_ptr)
  1124. 15132 message *m_ptr;                 /* pointer to request message */
  1125. 15133 {
  1126. 15134 /* Handle sys_abort.  MINIX is unable to continue.  Terminate operation. */
  1127. 15135   char monitor_code[64];
  1128. 15136   phys_bytes src_phys;
  1129. 15137
  1130. 15138   if (m_ptr->m1_i1 == RBT_MONITOR) {
  1131. 15139         /* The monitor is to run user specified instructions. */
  1132. 15140         src_phys = numap(m_ptr->m_source, (vir_bytes) m_ptr->m1_p1,
  1133. 15141                                         (vir_bytes) sizeof(monitor_code));
  1134. 15142         if (src_phys == 0) panic("bad monitor code from", m_ptr->m_source);
  1135. 15143         phys_copy(src_phys, vir2phys(monitor_code),
  1136. 15144                                         (phys_bytes) sizeof(monitor_code));
  1137. 15145         reboot_code = vir2phys(monitor_code);
  1138. 15146   }
  1139. 15147   wreboot(m_ptr->m1_i1);
  1140. 15148   return(OK);                   /* pro-forma (really EDISASTER) */
  1141. 15149 }
  1142. 15154 /*===========================================================================*
  1143. 15155  *                            do_sendsig                                     *
  1144. 15156  *===========================================================================*/
  1145. 15157 PRIVATE int do_sendsig(m_ptr)
  1146. 15158 message *m_ptr;                 /* pointer to request message */
  1147. 15159 {
  1148. 15160 /* Handle sys_sendsig, POSIX-style signal */
  1149. 15161
  1150. 15162   struct sigmsg smsg;
  1151. 15163   register struct proc *rp;
  1152. 15164   phys_bytes src_phys, dst_phys;
  1153. 15165   struct sigcontext sc, *scp;
  1154. 15166   struct sigframe fr, *frp;
  1155. 15167
  1156. 15168   if (!isokusern(m_ptr->PROC1)) return(E_BAD_PROC);
  1157. 15169   rp = proc_addr(m_ptr->PROC1);
  1158. 15170
  1159. 15171   /* Get the sigmsg structure into our address space.  */
  1160. 15172   src_phys = umap(proc_addr(MM_PROC_NR), D, (vir_bytes) m_ptr->SIG_CTXT_PTR,
  1161. 15173                   (vir_bytes) sizeof(struct sigmsg));
  1162. 15174   if (src_phys == 0)
  1163. 15175         panic("do_sendsig can't signal: bad sigmsg address from MM", NO_NUM);
  1164. 15176   phys_copy(src_phys, vir2phys(&smsg), (phys_bytes) sizeof(struct sigmsg));
  1165. 15177
  1166. 15178   /* Compute the usr stack pointer value where sigcontext will be stored. */
  1167. 15179   scp = (struct sigcontext *) smsg.sm_stkptr - 1;
  1168. 15180
  1169. 15181   /* Copy the registers to the sigcontext structure. */
  1170. 15182   memcpy(&sc.sc_regs, &rp->p_reg, sizeof(struct sigregs));
  1171. 15183
  1172. 15184   /* Finish the sigcontext initialization. */
  1173. 15185   sc.sc_flags = SC_SIGCONTEXT;
  1174. 15186
  1175. 15187   sc.sc_mask = smsg.sm_mask;
  1176. 15188
  1177. 15189   /* Copy the sigcontext structure to the user's stack. */
  1178. 15190   dst_phys = umap(rp, D, (vir_bytes) scp,
  1179. 15191                   (vir_bytes) sizeof(struct sigcontext));
  1180. 15192   if (dst_phys == 0) return(EFAULT);
  1181. 15193   phys_copy(vir2phys(&sc), dst_phys, (phys_bytes) sizeof(struct sigcontext));
  1182. 15194
  1183. 15195   /* Initialize the sigframe structure. */
  1184. 15196   frp = (struct sigframe *) scp - 1;
  1185. 15197   fr.sf_scpcopy = scp;
  1186. 15198   fr.sf_retadr2= (void (*)()) rp->p_reg.pc;
  1187. 15199   fr.sf_fp = rp->p_reg.fp;
  1188. 15200   rp->p_reg.fp = (reg_t) &frp->sf_fp;
  1189. 15201   fr.sf_scp = scp;
  1190. 15202   fr.sf_code = 0;       /* XXX - should be used for type of FP exception */
  1191. 15203   fr.sf_signo = smsg.sm_signo;
  1192. 15204   fr.sf_retadr = (void (*)()) smsg.sm_sigreturn;
  1193. 15205
  1194. 15206   /* Copy the sigframe structure to the user's stack. */
  1195. 15207   dst_phys = umap(rp, D, (vir_bytes) frp, (vir_bytes) sizeof(struct sigframe));
  1196. 15208   if (dst_phys == 0) return(EFAULT);
  1197. 15209   phys_copy(vir2phys(&fr), dst_phys, (phys_bytes) sizeof(struct sigframe));
  1198. 15210
  1199. 15211   /* Reset user registers to execute the signal handler. */
  1200. 15212   rp->p_reg.sp = (reg_t) frp;
  1201. 15213   rp->p_reg.pc = (reg_t) smsg.sm_sighandler;
  1202. 15214
  1203. 15215   return(OK);
  1204. 15216 }
  1205. 15218 /*===========================================================================*
  1206. 15219  *                            do_sigreturn                                   *
  1207. 15220  *===========================================================================*/
  1208. 15221 PRIVATE int do_sigreturn(m_ptr)
  1209. 15222 register message *m_ptr;
  1210. 15223 {
  1211. 15224 /* POSIX style signals require sys_sigreturn to put things in order before the
  1212. 15225  * signalled process can resume execution
  1213. 15226  */
  1214. 15227
  1215. 15228   struct sigcontext sc;
  1216. 15229   register struct proc *rp;
  1217. 15230   phys_bytes src_phys;
  1218. 15231
  1219. 15232   if (!isokusern(m_ptr->PROC1)) return(E_BAD_PROC);
  1220. 15233   rp = proc_addr(m_ptr->PROC1);
  1221. 15234
  1222. 15235   /* Copy in the sigcontext structure. */
  1223. 15236   src_phys = umap(rp, D, (vir_bytes) m_ptr->SIG_CTXT_PTR,
  1224. 15237                   (vir_bytes) sizeof(struct sigcontext));
  1225. 15238   if (src_phys == 0) return(EFAULT);
  1226. 15239   phys_copy(src_phys, vir2phys(&sc), (phys_bytes) sizeof(struct sigcontext));
  1227. 15240
  1228. 15241   /* Make sure that this is not just a jmp_buf. */
  1229. 15242   if ((sc.sc_flags & SC_SIGCONTEXT) == 0) return(EINVAL);
  1230. 15243
  1231. 15244   /* Fix up only certain key registers if the compiler doesn't use
  1232. 15245    * register variables within functions containing setjmp.
  1233. 15246    */
  1234. 15247   if (sc.sc_flags & SC_NOREGLOCALS) {
  1235. 15248         rp->p_reg.retreg = sc.sc_retreg;
  1236. 15249         rp->p_reg.fp = sc.sc_fp;
  1237. 15250         rp->p_reg.pc = sc.sc_pc;
  1238. 15251         rp->p_reg.sp = sc.sc_sp;
  1239. 15252         return (OK);
  1240. 15253   }
  1241. 15254   sc.sc_psw  = rp->p_reg.psw;
  1242. 15255
  1243. 15256 #if (CHIP == INTEL)
  1244. 15257   /* Don't panic kernel if user gave bad selectors. */
  1245. 15258   sc.sc_cs = rp->p_reg.cs;
  1246. 15259   sc.sc_ds = rp->p_reg.ds;
  1247. 15260   sc.sc_es = rp->p_reg.es;
  1248. 15261 #if _WORD_SIZE == 4
  1249. 15262   sc.sc_fs = rp->p_reg.fs;
  1250. 15263   sc.sc_gs = rp->p_reg.gs;
  1251. 15264 #endif
  1252. 15265 #endif
  1253. 15266
  1254. 15267   /* Restore the registers. */
  1255. 15268   memcpy(&rp->p_reg, (char *)&sc.sc_regs, sizeof(struct sigregs));
  1256. 15269
  1257. 15270   return(OK);
  1258. 15271 }
  1259. 15273 /*===========================================================================*
  1260. 15274  *                              do_kill                                      *
  1261. 15275  *===========================================================================*/
  1262. 15276 PRIVATE int do_kill(m_ptr)
  1263. 15277 register message *m_ptr;        /* pointer to request message */
  1264. 15278 {
  1265. 15279 /* Handle sys_kill(). Cause a signal to be sent to a process via MM.
  1266. 15280  * Note that this has nothing to do with the kill (2) system call, this
  1267. 15281  * is how the FS (and possibly other servers) get access to cause_sig to
  1268. 15282  * send a KSIG message to MM
  1269. 15283  */
  1270. 15284
  1271. 15285   if (!isokusern(m_ptr->PR)) return(E_BAD_PROC);
  1272. 15286   cause_sig(m_ptr->PR, m_ptr->SIGNUM);
  1273. 15287   return(OK);
  1274. 15288 }
  1275. 15291 /*===========================================================================*
  1276. 15292  *                            do_endsig                                      *
  1277. 15293  *===========================================================================*/
  1278. 15294 PRIVATE int do_endsig(m_ptr)
  1279. 15295 register message *m_ptr;        /* pointer to request message */
  1280. 15296 {
  1281. 15297 /* Finish up after a KSIG-type signal, caused by a SYS_KILL message or a call
  1282. 15298  * to cause_sig by a task
  1283. 15299  */
  1284. 15300
  1285. 15301   register struct proc *rp;
  1286. 15302
  1287. 15303   if (!isokusern(m_ptr->PROC1)) return(E_BAD_PROC);
  1288. 15304   rp = proc_addr(m_ptr->PROC1);
  1289. 15305
  1290. 15306   /* MM has finished one KSIG. */
  1291. 15307   if (rp->p_pendcount != 0 && --rp->p_pendcount == 0
  1292. 15308       && (rp->p_flags &= ~SIG_PENDING) == 0)
  1293. 15309         lock_ready(rp);
  1294. 15310   return(OK);
  1295. 15311 }
  1296. 15313 /*===========================================================================*
  1297. 15314  *                              do_copy                                      *
  1298. 15315  *===========================================================================*/
  1299. 15316 PRIVATE int do_copy(m_ptr)
  1300. 15317 register message *m_ptr;        /* pointer to request message */
  1301. 15318 {
  1302. 15319 /* Handle sys_copy().  Copy data for MM or FS. */
  1303. 15320
  1304. 15321   int src_proc, dst_proc, src_space, dst_space;
  1305. 15322   vir_bytes src_vir, dst_vir;
  1306. 15323   phys_bytes src_phys, dst_phys, bytes;
  1307. 15324
  1308. 15325   /* Dismember the command message. */
  1309. 15326   src_proc = m_ptr->SRC_PROC_NR;
  1310. 15327   dst_proc = m_ptr->DST_PROC_NR;
  1311. 15328   src_space = m_ptr->SRC_SPACE;
  1312. 15329   dst_space = m_ptr->DST_SPACE;
  1313. 15330   src_vir = (vir_bytes) m_ptr->SRC_BUFFER;
  1314. 15331   dst_vir = (vir_bytes) m_ptr->DST_BUFFER;
  1315. 15332   bytes = (phys_bytes) m_ptr->COPY_BYTES;
  1316. 15333
  1317. 15334   /* Compute the source and destination addresses and do the copy. */
  1318. 15335   if (src_proc == ABS)
  1319. 15336         src_phys = (phys_bytes) m_ptr->SRC_BUFFER;
  1320. 15337   else {
  1321. 15338         if (bytes != (vir_bytes) bytes)
  1322. 15339                 /* This would happen for 64K segments and 16-bit vir_bytes.
  1323. 15340                  * It would happen a lot for do_fork except MM uses ABS
  1324. 15341                  * copies for that case.
  1325. 15342                  */
  1326. 15343                 panic("overflow in count in do_copy", NO_NUM);
  1327. 15344
  1328. 15345         src_phys = umap(proc_addr(src_proc), src_space, src_vir,
  1329. 15346                         (vir_bytes) bytes);
  1330. 15347         }
  1331. 15348
  1332. 15349   if (dst_proc == ABS)
  1333. 15350         dst_phys = (phys_bytes) m_ptr->DST_BUFFER;
  1334. 15351   else
  1335. 15352         dst_phys = umap(proc_addr(dst_proc), dst_space, dst_vir,
  1336. 15353                         (vir_bytes) bytes);
  1337. 15354
  1338. 15355   if (src_phys == 0 || dst_phys == 0) return(EFAULT);
  1339. 15356   phys_copy(src_phys, dst_phys, bytes);
  1340. 15357   return(OK);
  1341. 15358 }
  1342. 15361 /*===========================================================================*
  1343. 15362  *                              do_vcopy                                     *
  1344. 15363  *===========================================================================*/
  1345. 15364 PRIVATE int do_vcopy(m_ptr)
  1346. 15365 register message *m_ptr;        /* pointer to request message */
  1347. 15366 {
  1348. 15367 /* Handle sys_vcopy(). Copy multiple blocks of memory */
  1349. 15368
  1350. 15369   int src_proc, dst_proc, vect_s, i;
  1351. 15370   vir_bytes src_vir, dst_vir, vect_addr;
  1352. 15371   phys_bytes src_phys, dst_phys, bytes;
  1353. 15372   cpvec_t cpvec_table[CPVEC_NR];
  1354. 15373
  1355. 15374   /* Dismember the command message. */
  1356. 15375   src_proc = m_ptr->m1_i1;
  1357. 15376   dst_proc = m_ptr->m1_i2;
  1358. 15377   vect_s = m_ptr->m1_i3;
  1359. 15378   vect_addr = (vir_bytes)m_ptr->m1_p1;
  1360. 15379
  1361. 15380   if (vect_s > CPVEC_NR) return EDOM;
  1362. 15381
  1363. 15382   src_phys= numap (m_ptr->m_source, vect_addr, vect_s * sizeof(cpvec_t));
  1364. 15383   if (!src_phys) return EFAULT;
  1365. 15384   phys_copy(src_phys, vir2phys(cpvec_table),
  1366. 15385                                 (phys_bytes) (vect_s * sizeof(cpvec_t)));
  1367. 15386
  1368. 15387   for (i = 0; i < vect_s; i++) {
  1369. 15388         src_vir= cpvec_table[i].cpv_src;
  1370. 15389         dst_vir= cpvec_table[i].cpv_dst;
  1371. 15390         bytes= cpvec_table[i].cpv_size;
  1372. 15391         src_phys = numap(src_proc,src_vir,(vir_bytes)bytes);
  1373. 15392         dst_phys = numap(dst_proc,dst_vir,(vir_bytes)bytes);
  1374. 15393         if (src_phys == 0 || dst_phys == 0) return(EFAULT);
  1375. 15394         phys_copy(src_phys, dst_phys, bytes);
  1376. 15395   }
  1377. 15396   return(OK);
  1378. 15397 }
  1379. 15400 /*==========================================================================*
  1380. 15401  *                              do_gboot                                    *
  1381. 15402  *==========================================================================*/
  1382. 15403 PUBLIC struct bparam_s boot_parameters;
  1383. 15404
  1384. 15405 PRIVATE int do_gboot(m_ptr)
  1385. 15406 message *m_ptr;                 /* pointer to request message */
  1386. 15407 {
  1387. 15408 /* Copy the boot parameters.  Normally only called during fs init. */
  1388. 15409
  1389. 15410   phys_bytes dst_phys;
  1390. 15411
  1391. 15412   dst_phys = umap(proc_addr(m_ptr->PROC1), D, (vir_bytes) m_ptr->MEM_PTR,
  1392. 15413                                 (vir_bytes) sizeof(boot_parameters));
  1393. 15414   if (dst_phys == 0) panic("bad call to SYS_GBOOT", NO_NUM);
  1394. 15415   phys_copy(vir2phys(&boot_parameters), dst_phys,
  1395. 15416                                 (phys_bytes) sizeof(boot_parameters));
  1396. 15417   return(OK);
  1397. 15418 }
  1398. 15421 /*===========================================================================*
  1399. 15422  *                              do_mem                                       *
  1400. 15423  *===========================================================================*/
  1401. 15424 PRIVATE int do_mem(m_ptr)
  1402. 15425 register message *m_ptr;        /* pointer to request message */
  1403. 15426 {
  1404. 15427 /* Return the base and size of the next chunk of memory. */
  1405. 15428
  1406. 15429   struct memory *memp;
  1407. 15430
  1408. 15431   for (memp = mem; memp < &mem[NR_MEMS]; ++memp) {
  1409. 15432         m_ptr->m1_i1 = memp->base;
  1410. 15433         m_ptr->m1_i2 = memp->size;
  1411. 15434         m_ptr->m1_i3 = tot_mem_size;
  1412. 15435         memp->size = 0;
  1413. 15436         if (m_ptr->m1_i2 != 0) break;           /* found a chunk */
  1414. 15437   }
  1415. 15438   return(OK);
  1416. 15439 }
  1417. 15442 /*==========================================================================*
  1418. 15443  *                              do_umap                                     *
  1419. 15444  *==========================================================================*/
  1420. 15445 PRIVATE int do_umap(m_ptr)
  1421. 15446 register message *m_ptr;        /* pointer to request message */
  1422. 15447 {
  1423. 15448 /* Same as umap(), for non-kernel processes. */
  1424. 15449
  1425. 15450   m_ptr->SRC_BUFFER = umap(proc_addr((int) m_ptr->SRC_PROC_NR),
  1426. 15451                            (int) m_ptr->SRC_SPACE,
  1427. 15452                            (vir_bytes) m_ptr->SRC_BUFFER,
  1428. 15453                            (vir_bytes) m_ptr->COPY_BYTES);
  1429. 15454   return(OK);
  1430. 15455 }
  1431. 15458 /*==========================================================================*
  1432. 15459  *                              do_trace                                    *
  1433. 15460  *==========================================================================*/
  1434. 15461 #define TR_PROCNR       (m_ptr->m2_i1)
  1435. 15462 #define TR_REQUEST      (m_ptr->m2_i2)
  1436. 15463 #define TR_ADDR         ((vir_bytes) m_ptr->m2_l1)
  1437. 15464 #define TR_DATA         (m_ptr->m2_l2)
  1438. 15465 #define TR_VLSIZE       ((vir_bytes) sizeof(long))
  1439. 15466
  1440. 15467 PRIVATE int do_trace(m_ptr)
  1441. 15468 register message *m_ptr;
  1442. 15469 {
  1443. 15470 /* Handle the debugging commands supported by the ptrace system call
  1444. 15471  * The commands are:
  1445. 15472  * T_STOP       stop the process
  1446. 15473  * T_OK         enable tracing by parent for this process
  1447. 15474  * T_GETINS     return value from instruction space
  1448. 15475  * T_GETDATA    return value from data space
  1449. 15476  * T_GETUSER    return value from user process table
  1450. 15477  * T_SETINS     set value from instruction space
  1451. 15478  * T_SETDATA    set value from data space
  1452. 15479  * T_SETUSER    set value in user process table
  1453. 15480  * T_RESUME     resume execution
  1454. 15481  * T_EXIT       exit
  1455. 15482  * T_STEP       set trace bit
  1456. 15483  *
  1457. 15484  * The T_OK and T_EXIT commands are handled completely by the memory manager,
  1458. 15485  * all others come here.
  1459. 15486  */
  1460. 15487
  1461. 15488   register struct proc *rp;
  1462. 15489   phys_bytes src, dst;
  1463. 15490   int i;
  1464. 15491
  1465. 15492   rp = proc_addr(TR_PROCNR);
  1466. 15493   if (rp->p_flags & P_SLOT_FREE) return(EIO);
  1467. 15494   switch (TR_REQUEST) {
  1468. 15495   case T_STOP:                  /* stop process */
  1469. 15496         if (rp->p_flags == 0) lock_unready(rp);
  1470. 15497         rp->p_flags |= P_STOP;
  1471. 15498         rp->p_reg.psw &= ~TRACEBIT;     /* clear trace bit */
  1472. 15499         return(OK);
  1473. 15500
  1474. 15501   case T_GETINS:                /* return value from instruction space */
  1475. 15502         if (rp->p_map[T].mem_len != 0) {
  1476. 15503                 if ((src = umap(rp, T, TR_ADDR, TR_VLSIZE)) == 0) return(EIO);
  1477. 15504                 phys_copy(src, vir2phys(&TR_DATA), (phys_bytes) sizeof(long));
  1478. 15505                 break;
  1479. 15506         }
  1480. 15507         /* Text space is actually data space - fall through. */
  1481. 15508
  1482. 15509   case T_GETDATA:               /* return value from data space */
  1483. 15510         if ((src = umap(rp, D, TR_ADDR, TR_VLSIZE)) == 0) return(EIO);
  1484. 15511         phys_copy(src, vir2phys(&TR_DATA), (phys_bytes) sizeof(long));
  1485. 15512         break;
  1486. 15513
  1487. 15514   case T_GETUSER:               /* return value from process table */
  1488. 15515         if ((TR_ADDR & (sizeof(long) - 1)) != 0 ||
  1489. 15516             TR_ADDR > sizeof(struct proc) - sizeof(long))
  1490. 15517                 return(EIO);
  1491. 15518         TR_DATA = *(long *) ((char *) rp + (int) TR_ADDR);
  1492. 15519         break;
  1493. 15520
  1494. 15521   case T_SETINS:                /* set value in instruction space */
  1495. 15522         if (rp->p_map[T].mem_len != 0) {
  1496. 15523                 if ((dst = umap(rp, T, TR_ADDR, TR_VLSIZE)) == 0) return(EIO);
  1497. 15524                 phys_copy(vir2phys(&TR_DATA), dst, (phys_bytes) sizeof(long));
  1498. 15525                 TR_DATA = 0;
  1499. 15526                 break;
  1500. 15527         }
  1501. 15528         /* Text space is actually data space - fall through. */
  1502. 15529
  1503. 15530   case T_SETDATA:                       /* set value in data space */
  1504. 15531         if ((dst = umap(rp, D, TR_ADDR, TR_VLSIZE)) == 0) return(EIO);
  1505. 15532         phys_copy(vir2phys(&TR_DATA), dst, (phys_bytes) sizeof(long));
  1506. 15533         TR_DATA = 0;
  1507. 15534         break;
  1508. 15535
  1509. 15536   case T_SETUSER:                       /* set value in process table */
  1510. 15537         if ((TR_ADDR & (sizeof(reg_t) - 1)) != 0 ||
  1511. 15538              TR_ADDR > sizeof(struct stackframe_s) - sizeof(reg_t))
  1512. 15539                 return(EIO);
  1513. 15540         i = (int) TR_ADDR;
  1514. 15541 #if (CHIP == INTEL)
  1515. 15542         /* Altering segment registers might crash the kernel when it
  1516. 15543          * tries to load them prior to restarting a process, so do
  1517. 15544          * not allow it.
  1518. 15545          */
  1519. 15546         if (i == (int) &((struct proc *) 0)->p_reg.cs ||
  1520. 15547             i == (int) &((struct proc *) 0)->p_reg.ds ||
  1521. 15548             i == (int) &((struct proc *) 0)->p_reg.es ||
  1522. 15549 #if _WORD_SIZE == 4
  1523. 15550             i == (int) &((struct proc *) 0)->p_reg.gs ||
  1524. 15551             i == (int) &((struct proc *) 0)->p_reg.fs ||
  1525. 15552 #endif
  1526. 15553             i == (int) &((struct proc *) 0)->p_reg.ss)
  1527. 15554                 return(EIO);
  1528. 15555 #endif
  1529. 15556         if (i == (int) &((struct proc *) 0)->p_reg.psw)
  1530. 15557                 /* only selected bits are changeable */
  1531. 15558                 SETPSW(rp, TR_DATA);
  1532. 15559         else
  1533. 15560                 *(reg_t *) ((char *) &rp->p_reg + i) = (reg_t) TR_DATA;
  1534. 15561         TR_DATA = 0;
  1535. 15562         break;
  1536. 15563
  1537. 15564   case T_RESUME:                /* resume execution */
  1538. 15565         rp->p_flags &= ~P_STOP;
  1539. 15566         if (rp->p_flags == 0) lock_ready(rp);
  1540. 15567         TR_DATA = 0;
  1541. 15568         break;
  1542. 15569
  1543. 15570   case T_STEP:                  /* set trace bit */
  1544. 15571         rp->p_reg.psw |= TRACEBIT;
  1545. 15572         rp->p_flags &= ~P_STOP;
  1546. 15573         if (rp->p_flags == 0) lock_ready(rp);
  1547. 15574         TR_DATA = 0;
  1548. 15575         break;
  1549. 15576
  1550. 15577   default:
  1551. 15578         return(EIO);
  1552. 15579   }
  1553. 15580   return(OK);
  1554. 15581 }
  1555. 15583 /*===========================================================================*
  1556. 15584  *                              cause_sig                                    *
  1557. 15585  *===========================================================================*/
  1558. 15586 PUBLIC void cause_sig(proc_nr, sig_nr)
  1559. 15587 int proc_nr;                    /* process to be signalled */
  1560. 15588 int sig_nr;                     /* signal to be sent, 1 to _NSIG */
  1561. 15589 {
  1562. 15590 /* A task wants to send a signal to a process.   Examples of such tasks are:
  1563. 15591  *   TTY wanting to cause SIGINT upon getting a DEL
  1564. 15592  *   CLOCK wanting to cause SIGALRM when timer expires
  1565. 15593  * FS also uses this to send a signal, via the SYS_KILL message.
  1566. 15594  * Signals are handled by sending a message to MM.  The tasks don't dare do
  1567. 15595  * that directly, for fear of what would happen if MM were busy.  Instead they
  1568. 15596  * call cause_sig, which sets bits in p_pending, and then carefully checks to
  1569. 15597  * see if MM is free.  If so, a message is sent to it.  If not, when it becomes
  1570. 15598  * free, a message is sent.  The process being signaled is blocked while MM
  1571. 15599  * has not seen or finished with all signals for it.  These signals are
  1572. 15600  * counted in p_pendcount, and the SIG_PENDING flag is kept nonzero while
  1573. 15601  * there are some.  It is not sufficient to ready the process when MM is
  1574. 15602  * informed, because MM can block waiting for FS to do a core dump.
  1575. 15603  */
  1576. 15604
  1577. 15605   register struct proc *rp, *mmp;
  1578. 15606
  1579. 15607   rp = proc_addr(proc_nr);
  1580. 15608   if (sigismember(&rp->p_pending, sig_nr))
  1581. 15609         return;                 /* this signal already pending */
  1582. 15610   sigaddset(&rp->p_pending, sig_nr);
  1583. 15611   ++rp->p_pendcount;            /* count new signal pending */
  1584. 15612   if (rp->p_flags & PENDING)
  1585. 15613         return;                 /* another signal already pending */
  1586. 15614   if (rp->p_flags == 0) lock_unready(rp);
  1587. 15615   rp->p_flags |= PENDING | SIG_PENDING;
  1588. 15616   ++sig_procs;                  /* count new process pending */
  1589. 15617
  1590. 15618   mmp = proc_addr(MM_PROC_NR);
  1591. 15619   if ( ((mmp->p_flags & RECEIVING) == 0) || mmp->p_getfrom != ANY) return;
  1592. 15620   inform();
  1593. 15621 }
  1594. 15624 /*===========================================================================*
  1595. 15625  *                              inform                                       *
  1596. 15626  *===========================================================================*/
  1597. 15627 PUBLIC void inform()
  1598. 15628 {
  1599. 15629 /* When a signal is detected by the kernel (e.g., DEL), or generated by a task
  1600. 15630  * (e.g. clock task for SIGALRM), cause_sig() is called to set a bit in the
  1601. 15631  * p_pending field of the process to signal.  Then inform() is called to see
  1602. 15632  * if MM is idle and can be told about it.  Whenever MM blocks, a check is
  1603. 15633  * made to see if 'sig_procs' is nonzero; if so, inform() is called.
  1604. 15634  */
  1605. 15635
  1606. 15636   register struct proc *rp;
  1607. 15637
  1608. 15638   /* MM is waiting for new input.  Find a process with pending signals. */
  1609. 15639   for (rp = BEG_SERV_ADDR; rp < END_PROC_ADDR; rp++)
  1610. 15640         if (rp->p_flags & PENDING) {
  1611. 15641                 m.m_type = KSIG;
  1612. 15642                 m.SIG_PROC = proc_number(rp);
  1613. 15643                 m.SIG_MAP = rp->p_pending;
  1614. 15644                 sig_procs--;
  1615. 15645                 if (lock_mini_send(proc_addr(HARDWARE), MM_PROC_NR, &m) != OK)
  1616. 15646                         panic("can't inform MM", NO_NUM);
  1617. 15647                 sigemptyset(&rp->p_pending); /* the ball is now in MM's court */
  1618. 15648                 rp->p_flags &= ~PENDING;/* remains inhibited by SIG_PENDING */
  1619. 15649                 lock_pick_proc();       /* avoid delay in scheduling MM */
  1620. 15650                 return;
  1621. 15651         }
  1622. 15652 }
  1623. 15655 /*===========================================================================*
  1624. 15656  *                              umap                                         *
  1625. 15657  *===========================================================================*/
  1626. 15658 PUBLIC phys_bytes umap(rp, seg, vir_addr, bytes)
  1627. 15659 register struct proc *rp;       /* pointer to proc table entry for process */
  1628. 15660 int seg;                        /* T, D, or S segment */
  1629. 15661 vir_bytes vir_addr;             /* virtual address in bytes within the seg */
  1630. 15662 vir_bytes bytes;                /* # of bytes to be copied */
  1631. 15663 {
  1632. 15664 /* Calculate the physical memory address for a given virtual address. */
  1633. 15665
  1634. 15666   vir_clicks vc;                /* the virtual address in clicks */
  1635. 15667   phys_bytes pa;                /* intermediate variables as phys_bytes */
  1636. 15668   phys_bytes seg_base;
  1637. 15669
  1638. 15670   /* If 'seg' is D it could really be S and vice versa.  T really means T.
  1639. 15671    * If the virtual address falls in the gap,  it causes a problem. On the
  1640. 15672    * 8088 it is probably a legal stack reference, since "stackfaults" are
  1641. 15673    * not detected by the hardware.  On 8088s, the gap is called S and
  1642. 15674    * accepted, but on other machines it is called D and rejected.
  1643. 15675    * The Atari ST behaves like the 8088 in this respect.
  1644. 15676    */
  1645. 15677
  1646. 15678   if (bytes <= 0) return( (phys_bytes) 0);
  1647. 15679   vc = (vir_addr + bytes - 1) >> CLICK_SHIFT;   /* last click of data */
  1648. 15680
  1649. 15681   if (seg != T)
  1650. 15682         seg = (vc < rp->p_map[D].mem_vir + rp->p_map[D].mem_len ? D : S);
  1651. 15683
  1652. 15684   if((vir_addr>>CLICK_SHIFT) >= rp->p_map[seg].mem_vir+ rp->p_map[seg].mem_len)
  1653. 15685         return( (phys_bytes) 0 );
  1654. 15686   seg_base = (phys_bytes) rp->p_map[seg].mem_phys;
  1655. 15687   seg_base = seg_base << CLICK_SHIFT;   /* segment origin in bytes */
  1656. 15688   pa = (phys_bytes) vir_addr;
  1657. 15689   pa -= rp->p_map[seg].mem_vir << CLICK_SHIFT;
  1658. 15690   return(seg_base + pa);
  1659. 15691 }
  1660. 15694 /*==========================================================================*
  1661. 15695  *                              numap                                       *
  1662. 15696  *==========================================================================*/
  1663. 15697 PUBLIC phys_bytes numap(proc_nr, vir_addr, bytes)
  1664. 15698 int proc_nr;                    /* process number to be mapped */
  1665. 15699 vir_bytes vir_addr;             /* virtual address in bytes within D seg */
  1666. 15700 vir_bytes bytes;                /* # of bytes required in segment  */
  1667. 15701 {
  1668. 15702 /* Do umap() starting from a process number instead of a pointer.  This
  1669. 15703  * function is used by device drivers, so they need not know about the
  1670. 15704  * process table.  To save time, there is no 'seg' parameter. The segment
  1671. 15705  * is always D.
  1672. 15706  */
  1673. 15707
  1674. 15708   return(umap(proc_addr(proc_nr), D, vir_addr, bytes));
  1675. 15709 }
  1676. 15711 #if (CHIP == INTEL)
  1677. 15712 /*==========================================================================*
  1678. 15713  *                              alloc_segments                              *
  1679. 15714  *==========================================================================*/
  1680. 15715 PUBLIC void alloc_segments(rp)
  1681. 15716 register struct proc *rp;
  1682. 15717 {
  1683. 15718 /* This is called only by do_newmap, but is broken out as a separate function
  1684. 15719  * because so much is hardware-dependent.
  1685. 15720  */
  1686. 15721
  1687. 15722   phys_bytes code_bytes;
  1688. 15723   phys_bytes data_bytes;
  1689. 15724   int privilege;
  1690. 15725
  1691. 15726   if (protected_mode) {
  1692. 15727         data_bytes = (phys_bytes) (rp->p_map[S].mem_vir + rp->p_map[S].mem_len)
  1693. 15728                      << CLICK_SHIFT;
  1694. 15729         if (rp->p_map[T].mem_len == 0)
  1695. 15730                 code_bytes = data_bytes;        /* common I&D, poor protect */
  1696. 15731         else
  1697. 15732                 code_bytes = (phys_bytes) rp->p_map[T].mem_len << CLICK_SHIFT;
  1698. 15733         privilege = istaskp(rp) ? TASK_PRIVILEGE : USER_PRIVILEGE;
  1699. 15734         init_codeseg(&rp->p_ldt[CS_LDT_INDEX],
  1700. 15735                      (phys_bytes) rp->p_map[T].mem_phys << CLICK_SHIFT,
  1701. 15736                      code_bytes, privilege);
  1702. 15737         init_dataseg(&rp->p_ldt[DS_LDT_INDEX],
  1703. 15738                      (phys_bytes) rp->p_map[D].mem_phys << CLICK_SHIFT,
  1704. 15739                      data_bytes, privilege);
  1705. 15740         rp->p_reg.cs = (CS_LDT_INDEX * DESC_SIZE) | TI | privilege;
  1706. 15741 #if _WORD_SIZE == 4
  1707. 15742         rp->p_reg.gs =
  1708. 15743         rp->p_reg.fs =
  1709. 15744 #endif
  1710. 15745         rp->p_reg.ss =
  1711. 15746         rp->p_reg.es =
  1712. 15747         rp->p_reg.ds = (DS_LDT_INDEX*DESC_SIZE) | TI | privilege;
  1713. 15748   } else {
  1714. 15749         rp->p_reg.cs = click_to_hclick(rp->p_map[T].mem_phys);
  1715. 15750         rp->p_reg.ss =
  1716. 15751         rp->p_reg.es =
  1717. 15752         rp->p_reg.ds = click_to_hclick(rp->p_map[D].mem_phys);
  1718. 15753   }
  1719. 15754 }
  1720. 15755 #endif /* (CHIP == INTEL) */
  1721. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1722. src/mm/mm.h    
  1723. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1724. 15800 /* This is the master header for mm.  It includes some other files
  1725. 15801  * and defines the principal constants.
  1726. 15802  */
  1727. 15803 #define _POSIX_SOURCE      1    /* tell headers to include POSIX stuff */
  1728. 15804 #define _MINIX             1    /* tell headers to include MINIX stuff */
  1729. 15805 #define _SYSTEM            1    /* tell headers that this is the kernel */
  1730. 15806
  1731. 15807 /* The following are so basic, all the *.c files get them automatically. */
  1732. 15808 #include <minix/config.h>       /* MUST be first */
  1733. 15809 #include <ansi.h>               /* MUST be second */
  1734. 15810 #include <sys/types.h>
  1735. 15811 #include <minix/const.h>
  1736. 15812 #include <minix/type.h>
  1737. 15813
  1738. 15814 #include <fcntl.h>
  1739. 15815 #include <unistd.h>
  1740. 15816 #include <minix/syslib.h>
  1741. 15817
  1742. 15818 #include <limits.h>
  1743. 15819 #include <errno.h>
  1744. 15820
  1745. 15821 #include "const.h"
  1746. 15822 #include "type.h"
  1747. 15823 #include "proto.h"
  1748. 15824 #include "glo.h"
  1749. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1750. src/mm/const.h    
  1751. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1752. 15900 /* Constants used by the Memory Manager. */
  1753. 15901
  1754. 15902 #define NO_MEM ((phys_clicks) 0)  /* returned by alloc_mem() with mem is up */
  1755. 15903
  1756. 15904 #if (CHIP == INTEL && _WORD_SIZE == 2)
  1757. 15905 /* These definitions are used in size_ok and are not needed for 386.
  1758. 15906  * The 386 segment granularity is 1 for segments smaller than 1M and 4096
  1759. 15907  * above that.  
  1760. 15908  */
  1761. 15909 #define PAGE_SIZE         16    /* how many bytes in a page (s.b.HCLICK_SIZE)*/
  1762. 15910 #define MAX_PAGES       4096    /* how many pages in the virtual addr space */
  1763. 15911 #endif
  1764. 15912
  1765. 15913 #define printf        printk
  1766. 15914
  1767. 15915 #define INIT_PID           1    /* init's process id number */
  1768. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1769. src/mm/type.h    
  1770. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1771. 16000 /* If there were any type definitions local to the Memory Manager, they would
  1772. 16001  * be here.  This file is included only for symmetry with the kernel and File
  1773. 16002  * System, which do have some local type definitions.
  1774. 16003  */
  1775. 16004
  1776. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1777. src/mm/proto.h    
  1778. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1779. 16100 /* Function prototypes. */
  1780. 16101
  1781. 16102 struct mproc;           /* need types outside of parameter list --kub */
  1782. 16103 struct stat;
  1783. 16104
  1784. 16105 /* alloc.c */
  1785. 16106 _PROTOTYPE( phys_clicks alloc_mem, (phys_clicks clicks)                 );
  1786. 16107 _PROTOTYPE( void free_mem, (phys_clicks base, phys_clicks clicks)       );
  1787. 16108 _PROTOTYPE( phys_clicks max_hole, (void)                                );
  1788. 16109 _PROTOTYPE( void mem_init, (phys_clicks *total, phys_clicks *free)      );
  1789. 16110 _PROTOTYPE( phys_clicks mem_left, (void)                                );
  1790. 16111 _PROTOTYPE( int do_brk3, (void)                                         );
  1791. 16112
  1792. 16113 /* break.c */
  1793. 16114 _PROTOTYPE( int adjust, (struct mproc *rmp,
  1794. 16115                         vir_clicks data_clicks, vir_bytes sp)           );
  1795. 16116 _PROTOTYPE( int do_brk, (void)                                          );
  1796. 16117 _PROTOTYPE( int size_ok, (int file_type, vir_clicks tc, vir_clicks dc,
  1797. 16118                         vir_clicks sc, vir_clicks dvir, vir_clicks s_vir) );
  1798. 16119
  1799. 16120 /* exec.c */
  1800. 16121 _PROTOTYPE( int do_exec, (void)                                         );
  1801. 16122 _PROTOTYPE( struct mproc *find_share, (struct mproc *mp_ign, Ino_t ino,
  1802. 16123                         Dev_t dev, time_t ctime)                        );
  1803. 16124
  1804. 16125 /* forkexit.c */
  1805. 16126 _PROTOTYPE( int do_fork, (void)                                         );
  1806. 16127 _PROTOTYPE( int do_mm_exit, (void)                                      );
  1807. 16128 _PROTOTYPE( int do_waitpid, (void)                                      );
  1808. 16129 _PROTOTYPE( void mm_exit, (struct mproc *rmp, int exit_status)          );
  1809. 16130
  1810. 16131 /* getset.c */
  1811. 16132 _PROTOTYPE( int do_getset, (void)                                       );
  1812. 16133
  1813. 16134 /* main.c */
  1814. 16135 _PROTOTYPE( void main, (void)                                           );
  1815. 16136
  1816. 16137 #if (MACHINE == MACINTOSH)
  1817. 16138 _PROTOTYPE( phys_clicks start_click, (void)                             );
  1818. 16139 #endif
  1819. 16140
  1820. 16141 _PROTOTYPE( void reply, (int proc_nr, int result, int res2, char *respt));
  1821. 16142
  1822. 16143 /* putk.c */
  1823. 16144 _PROTOTYPE( void putk, (int c)                                          );
  1824. 16145
  1825. 16146 /* signal.c */
  1826. 16147 _PROTOTYPE( int do_alarm, (void)                                        );
  1827. 16148 _PROTOTYPE( int do_kill, (void)                                         );
  1828. 16149 _PROTOTYPE( int do_ksig, (void)                                         );
  1829. 16150 _PROTOTYPE( int do_pause, (void)                                        );
  1830. 16151 _PROTOTYPE( int set_alarm, (int proc_nr, int sec)                       );
  1831. 16152 _PROTOTYPE( int check_sig, (pid_t proc_id, int signo)                   );
  1832. 16153 _PROTOTYPE( void sig_proc, (struct mproc *rmp, int sig_nr)              );
  1833. 16154 _PROTOTYPE( int do_sigaction, (void)                                    );
  1834. 16155 _PROTOTYPE( int do_sigpending, (void)                                   );
  1835. 16156 _PROTOTYPE( int do_sigprocmask, (void)                                  );
  1836. 16157 _PROTOTYPE( int do_sigreturn, (void)                                    );
  1837. 16158 _PROTOTYPE( int do_sigsuspend, (void)                                   );
  1838. 16159 _PROTOTYPE( int do_reboot, (void)                                       );
  1839. 16160
  1840. 16161 /* trace.c */
  1841. 16162 _PROTOTYPE( int do_trace, (void)                                        );
  1842. 16163 _PROTOTYPE( void stop_proc, (struct mproc *rmp, int sig_nr)             );
  1843. 16164
  1844. 16165 /* utility.c */
  1845. 16166 _PROTOTYPE( int allowed, (char *name_buf, struct stat *s_buf, int mask) );
  1846. 16167 _PROTOTYPE( int no_sys, (void)                                          );
  1847. 16168 _PROTOTYPE( void panic, (char *format, int num)                         );
  1848. 16169 _PROTOTYPE( void tell_fs, (int what, int p1, int p2, int p3)            );
  1849. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1850. src/mm/glo.h    
  1851. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1852. 16200 /* EXTERN should be extern except in table.c */
  1853. 16201 #ifdef _TABLE
  1854. 16202 #undef EXTERN
  1855. 16203 #define EXTERN
  1856. 16204 #endif
  1857. 16205
  1858. 16206 /* Global variables. */
  1859. 16207 EXTERN struct mproc *mp;        /* ptr to 'mproc' slot of current process */
  1860. 16208 EXTERN int dont_reply;          /* normally 0; set to 1 to inhibit reply */
  1861. 16209 EXTERN int procs_in_use;        /* how many processes are marked as IN_USE */
  1862. 16210
  1863. 16211 /* The parameters of the call are kept here. */
  1864. 16212 EXTERN message mm_in;           /* the incoming message itself is kept here. */
  1865. 16213 EXTERN message mm_out;          /* the reply message is built up here. */
  1866. 16214 EXTERN int who;                 /* caller's proc number */
  1867. 16215 EXTERN int mm_call;             /* system call number */
  1868. 16216
  1869. 16217 /* The following variables are used for returning results to the caller. */
  1870. 16218 EXTERN int err_code;            /* temporary storage for error number */
  1871. 16219 EXTERN int result2;             /* secondary result */
  1872. 16220 EXTERN char *res_ptr;           /* result, if pointer */
  1873. 16221
  1874. 16222 extern _PROTOTYPE (int (*call_vec[]), (void) ); /* system call handlers */
  1875. 16223 extern char core_name[];        /* file name where core images are produced */
  1876. 16224 EXTERN sigset_t core_sset;      /* which signals cause core images */
  1877. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1878. src/mm/mproc.h    
  1879. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1880. 16300 /* This table has one slot per process.  It contains all the memory management
  1881. 16301  * information for each process.  Among other things, it defines the text, data
  1882. 16302  * and stack segments, uids and gids, and various flags.  The kernel and file
  1883. 16303  * systems have tables that are also indexed by process, with the contents
  1884. 16304  * of corresponding slots referring to the same process in all three.
  1885. 16305  */
  1886. 16306
  1887. 16307 EXTERN struct mproc {
  1888. 16308   struct mem_map mp_seg[NR_SEGS];/* points to text, data, stack */
  1889. 16309   char mp_exitstatus;           /* storage for status when process exits */
  1890. 16310   char mp_sigstatus;            /* storage for signal # for killed procs */
  1891. 16311   pid_t mp_pid;                 /* process id */
  1892. 16312   pid_t mp_procgrp;             /* pid of process group (used for signals) */
  1893. 16313   pid_t mp_wpid;                /* pid this process is waiting for */
  1894. 16314   int mp_parent;                /* index of parent process */
  1895. 16315
  1896. 16316   /* Real and effective uids and gids. */
  1897. 16317   uid_t mp_realuid;             /* process' real uid */
  1898. 16318   uid_t mp_effuid;              /* process' effective uid */
  1899. 16319   gid_t mp_realgid;             /* process' real gid */
  1900. 16320   gid_t mp_effgid;              /* process' effective gid */
  1901. 16321
  1902. 16322   /* File identification for sharing. */
  1903. 16323   ino_t mp_ino;                 /* inode number of file */
  1904. 16324   dev_t mp_dev;                 /* device number of file system */
  1905. 16325   time_t mp_ctime;              /* inode changed time */
  1906. 16326
  1907. 16327   /* Signal handling information. */
  1908. 16328   sigset_t mp_ignore;           /* 1 means ignore the signal, 0 means don't */
  1909. 16329   sigset_t mp_catch;            /* 1 means catch the signal, 0 means don't */
  1910. 16330   sigset_t mp_sigmask;          /* signals to be blocked */
  1911. 16331   sigset_t mp_sigmask2;         /* saved copy of mp_sigmask */
  1912. 16332   sigset_t mp_sigpending;       /* signals being blocked */
  1913. 16333   struct sigaction mp_sigact[_NSIG + 1]; /* as in sigaction(2) */
  1914. 16334   vir_bytes mp_sigreturn;       /* address of C library __sigreturn function */
  1915. 16335
  1916. 16336   /* Backwards compatibility for signals. */
  1917. 16337   sighandler_t mp_func;         /* all sigs vectored to a single user fcn */
  1918. 16338
  1919. 16339   unsigned mp_flags;            /* flag bits */
  1920. 16340   vir_bytes mp_procargs;        /* ptr to proc's initial stack arguments */
  1921. 16341 } mproc[NR_PROCS];
  1922. 16342
  1923. 16343 /* Flag values */
  1924. 16344 #define IN_USE           001    /* set when 'mproc' slot in use */
  1925. 16345 #define WAITING          002    /* set by WAIT system call */
  1926. 16346 #define HANGING          004    /* set by EXIT system call */
  1927. 16347 #define PAUSED           010    /* set by PAUSE system call */
  1928. 16348 #define ALARM_ON         020    /* set when SIGALRM timer started */
  1929. 16349 #define SEPARATE         040    /* set if file is separate I & D space */
  1930. 16350 #define TRACED          0100    /* set if process is to be traced */
  1931. 16351 #define STOPPED         0200    /* set if process stopped for tracing */
  1932. 16352 #define SIGSUSPENDED    0400    /* set by SIGSUSPEND system call */
  1933. 16353
  1934. 16354 #define NIL_MPROC ((struct mproc *) 0)
  1935. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1936. src/mm/param.h    
  1937. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1938. 16400 /* The following names are synonyms for the variables in the input message. */
  1939. 16401 #define addr            mm_in.m1_p1
  1940. 16402 #define exec_name       mm_in.m1_p1
  1941. 16403 #define exec_len        mm_in.m1_i1
  1942. 16404 #define func            mm_in.m6_f1
  1943. 16405 #define grpid           (gid_t) mm_in.m1_i1
  1944. 16406 #define namelen         mm_in.m1_i1
  1945. 16407 #define pid             mm_in.m1_i1
  1946. 16408 #define seconds         mm_in.m1_i1
  1947. 16409 #define sig             mm_in.m6_i1
  1948. 16410 #define stack_bytes     mm_in.m1_i2
  1949. 16411 #define stack_ptr       mm_in.m1_p2
  1950. 16412 #define status          mm_in.m1_i1
  1951. 16413 #define usr_id          (uid_t) mm_in.m1_i1
  1952. 16414 #define request         mm_in.m2_i2
  1953. 16415 #define taddr           mm_in.m2_l1
  1954. 16416 #define data            mm_in.m2_l2
  1955. 16417 #define sig_nr          mm_in.m1_i2
  1956. 16418 #define sig_nsa         mm_in.m1_p1
  1957. 16419 #define sig_osa         mm_in.m1_p2
  1958. 16420 #define sig_ret         mm_in.m1_p3
  1959. 16421 #define sig_set         mm_in.m2_l1
  1960. 16422 #define sig_how         mm_in.m2_i1
  1961. 16423 #define sig_flags       mm_in.m2_i2
  1962. 16424 #define sig_context     mm_in.m2_p1
  1963. 16425 #ifdef _SIGMESSAGE
  1964. 16426 #define sig_msg         mm_in.m1_i1
  1965. 16427 #endif
  1966. 16428 #define reboot_flag     mm_in.m1_i1
  1967. 16429 #define reboot_code     mm_in.m1_p1
  1968. 16430 #define reboot_size     mm_in.m1_i2
  1969. 16431
  1970. 16432 /* The following names are synonyms for the variables in the output message. */
  1971. 16433 #define reply_type      mm_out.m_type
  1972. 16434 #define reply_i1        mm_out.m2_i1
  1973. 16435 #define reply_p1        mm_out.m2_p1
  1974. 16436 #define ret_mask        mm_out.m2_l1    
  1975. 16437
  1976. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1977. src/mm/table.c    
  1978. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1979. 16500 /* This file contains the table used to map system call numbers onto the
  1980. 16501  * routines that perform them.
  1981. 16502  */
  1982. 16503
  1983. 16504 #define _TABLE
  1984. 16505
  1985. 16506 #include "mm.h"
  1986. 16507 #include <minix/callnr.h>
  1987. 16508 #include <signal.h>
  1988. 16509 #include "mproc.h"
  1989. 16510 #include "param.h"
  1990. 16511
  1991. 16512 /* Miscellaneous */
  1992. 16513 char core_name[] = "core";      /* file name where core images are produced */
  1993. 16514
  1994. 16515 _PROTOTYPE (int (*call_vec[NCALLS]), (void) ) = {
  1995. 16516         no_sys,         /*  0 = unused  */
  1996. 16517         do_mm_exit,     /*  1 = exit    */
  1997. 16518         do_fork,        /*  2 = fork    */
  1998. 16519         no_sys,         /*  3 = read    */
  1999. 16520         no_sys,         /*  4 = write   */
  2000. 16521         no_sys,         /*  5 = open    */
  2001. 16522         no_sys,         /*  6 = close   */
  2002. 16523         do_waitpid,     /*  7 = wait    */
  2003. 16524         no_sys,         /*  8 = creat   */
  2004. 16525         no_sys,         /*  9 = link    */
  2005. 16526         no_sys,         /* 10 = unlink  */
  2006. 16527         do_waitpid,     /* 11 = waitpid */
  2007. 16528         no_sys,         /* 12 = chdir   */
  2008. 16529         no_sys,         /* 13 = time    */
  2009. 16530         no_sys,         /* 14 = mknod   */
  2010. 16531         no_sys,         /* 15 = chmod   */
  2011. 16532         no_sys,         /* 16 = chown   */
  2012. 16533         do_brk,         /* 17 = break   */
  2013. 16534         no_sys,         /* 18 = stat    */
  2014. 16535         no_sys,         /* 19 = lseek   */
  2015. 16536         do_getset,      /* 20 = getpid  */
  2016. 16537         no_sys,         /* 21 = mount   */
  2017. 16538         no_sys,         /* 22 = umount  */
  2018. 16539         do_getset,      /* 23 = setuid  */
  2019. 16540         do_getset,      /* 24 = getuid  */
  2020. 16541         no_sys,         /* 25 = stime   */
  2021. 16542         do_trace,       /* 26 = ptrace  */
  2022. 16543         do_alarm,       /* 27 = alarm   */
  2023. 16544         no_sys,         /* 28 = fstat   */
  2024. 16545         do_pause,       /* 29 = pause   */
  2025. 16546         no_sys,         /* 30 = utime   */
  2026. 16547         no_sys,         /* 31 = (stty)  */
  2027. 16548         no_sys,         /* 32 = (gtty)  */
  2028. 16549         no_sys,         /* 33 = access  */
  2029. 16550         no_sys,         /* 34 = (nice)  */
  2030. 16551         no_sys,         /* 35 = (ftime) */
  2031. 16552         no_sys,         /* 36 = sync    */
  2032. 16553         do_kill,        /* 37 = kill    */
  2033. 16554         no_sys,         /* 38 = rename  */
  2034. 16555         no_sys,         /* 39 = mkdir   */
  2035. 16556         no_sys,         /* 40 = rmdir   */
  2036. 16557         no_sys,         /* 41 = dup     */
  2037. 16558         no_sys,         /* 42 = pipe    */
  2038. 16559         no_sys,         /* 43 = times   */
  2039. 16560         no_sys,         /* 44 = (prof)  */
  2040. 16561         no_sys,         /* 45 = unused  */
  2041. 16562         do_getset,      /* 46 = setgid  */
  2042. 16563         do_getset,      /* 47 = getgid  */
  2043. 16564         no_sys,         /* 48 = (signal)*/
  2044. 16565         no_sys,         /* 49 = unused  */
  2045. 16566         no_sys,         /* 50 = unused  */
  2046. 16567         no_sys,         /* 51 = (acct)  */
  2047. 16568         no_sys,         /* 52 = (phys)  */
  2048. 16569         no_sys,         /* 53 = (lock)  */
  2049. 16570         no_sys,         /* 54 = ioctl   */
  2050. 16571         no_sys,         /* 55 = fcntl   */
  2051. 16572         no_sys,         /* 56 = (mpx)   */
  2052. 16573         no_sys,         /* 57 = unused  */
  2053. 16574         no_sys,         /* 58 = unused  */
  2054. 16575         do_exec,        /* 59 = execve  */
  2055. 16576         no_sys,         /* 60 = umask   */
  2056. 16577         no_sys,         /* 61 = chroot  */
  2057. 16578         do_getset,      /* 62 = setsid  */
  2058. 16579         do_getset,      /* 63 = getpgrp */
  2059. 16580
  2060. 16581         do_ksig,        /* 64 = KSIG: signals originating in the kernel */
  2061. 16582         no_sys,         /* 65 = UNPAUSE */
  2062. 16583         no_sys,         /* 66 = unused  */
  2063. 16584         no_sys,         /* 67 = REVIVE  */
  2064. 16585         no_sys,         /* 68 = TASK_REPLY  */
  2065. 16586         no_sys,         /* 69 = unused  */
  2066. 16587         no_sys,         /* 70 = unused  */
  2067. 16588         do_sigaction,   /* 71 = sigaction   */
  2068. 16589         do_sigsuspend,  /* 72 = sigsuspend  */
  2069. 16590         do_sigpending,  /* 73 = sigpending  */
  2070. 16591         do_sigprocmask, /* 74 = sigprocmask */
  2071. 16592         do_sigreturn,   /* 75 = sigreturn   */
  2072. 16593         do_reboot,      /* 76 = reboot  */
  2073. 16594 };
  2074. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  2075. src/mm/main.c    
  2076. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  2077. 16600 /* This file contains the main program of the memory manager and some related
  2078. 16601  * procedures.  When MINIX starts up, the kernel runs for a little while,
  2079. 16602  * initializing itself and its tasks, and then it runs MM and FS.  Both MM
  2080. 16603  * and FS initialize themselves as far as they can.  FS then makes a call to
  2081. 16604  * MM, because MM has to wait for FS to acquire a RAM disk.  MM asks the
  2082. 16605  * kernel for all free memory and starts serving requests.
  2083. 16606  *
  2084. 16607  * The entry points into this file are:
  2085. 16608  *   main:      starts MM running
  2086. 16609  *   reply:     reply to a process making an MM system call
  2087. 16610  */
  2088. 16611
  2089. 16612 #include "mm.h"
  2090. 16613 #include <minix/callnr.h>
  2091. 16614 #include <minix/com.h>
  2092. 16615 #include <signal.h>
  2093. 16616 #include <fcntl.h>
  2094. 16617 #include <sys/ioctl.h>
  2095. 16618 #include "mproc.h"
  2096. 16619 #include "param.h"
  2097. 16620
  2098. 16621 FORWARD _PROTOTYPE( void get_work, (void)                               );
  2099. 16622 FORWARD _PROTOTYPE( void mm_init, (void)                                );
  2100. 16623
  2101. 16624 /*===========================================================================*
  2102. 16625  *                              main                                         *
  2103. 16626  *===========================================================================*/
  2104. 16627 PUBLIC void main()
  2105. 16628 {
  2106. 16629 /* Main routine of the memory manager. */
  2107. 16630
  2108. 16631   int error;
  2109. 16632
  2110. 16633   mm_init();                    /* initialize memory manager tables */
  2111. 16634
  2112. 16635   /* This is MM's main loop-  get work and do it, forever and forever. */
  2113. 16636   while (TRUE) {
  2114. 16637         /* Wait for message. */
  2115. 16638         get_work();             /* wait for an MM system call */
  2116. 16639         mp = &mproc[who];
  2117. 16640
  2118. 16641         /* Set some flags. */
  2119. 16642         error = OK;
  2120. 16643         dont_reply = FALSE;
  2121. 16644         err_code = -999;
  2122. 16645
  2123. 16646         /* If the call number is valid, perform the call. */
  2124. 16647         if (mm_call < 0 || mm_call >= NCALLS)
  2125. 16648                 error = EBADCALL;
  2126. 16649         else
  2127. 16650                 error = (*call_vec[mm_call])();
  2128. 16651
  2129. 16652         /* Send the results back to the user to indicate completion. */
  2130. 16653         if (dont_reply) continue;       /* no reply for EXIT and WAIT */
  2131. 16654         if (mm_call == EXEC && error == OK) continue;
  2132. 16655         reply(who, error, result2, res_ptr);
  2133. 16656   }
  2134. 16657 }
  2135. 16660 /*===========================================================================*
  2136. 16661  *                              get_work                                     *
  2137. 16662  *===========================================================================*/
  2138. 16663 PRIVATE void get_work()
  2139. 16664 {
  2140. 16665 /* Wait for the next message and extract useful information from it. */
  2141. 16666
  2142. 16667   if (receive(ANY, &mm_in) != OK) panic("MM receive error", NO_NUM);
  2143. 16668   who = mm_in.m_source;         /* who sent the message */
  2144. 16669   mm_call = mm_in.m_type;       /* system call number */
  2145. 16670 }
  2146. 16673 /*===========================================================================*
  2147. 16674  *                              reply                                        *
  2148. 16675  *===========================================================================*/
  2149. 16676 PUBLIC void reply(proc_nr, result, res2, respt)
  2150. 16677 int proc_nr;                    /* process to reply to */
  2151. 16678 int result;                     /* result of the call (usually OK or error #)*/
  2152. 16679 int res2;                       /* secondary result */
  2153. 16680 char *respt;                    /* result if pointer */
  2154. 16681 {
  2155. 16682 /* Send a reply to a user process. */
  2156. 16683
  2157. 16684   register struct mproc *proc_ptr;
  2158. 16685
  2159. 16686   proc_ptr = &mproc[proc_nr];
  2160. 16687   /* 
  2161. 16688    * To make MM robust, check to see if destination is still alive.  This
  2162. 16689    * validy check must be skipped if the caller is a task.
  2163. 16690    */
  2164. 16691   if ((who >=0) && ((proc_ptr->mp_flags&IN_USE) == 0 || 
  2165. 16692         (proc_ptr->mp_flags&HANGING))) return;
  2166. 16693
  2167. 16694   reply_type = result;
  2168. 16695   reply_i1 = res2;
  2169. 16696   reply_p1 = respt;
  2170. 16697   if (send(proc_nr, &mm_out) != OK) panic("MM can't reply", NO_NUM);
  2171. 16698 }
  2172. 16701 /*===========================================================================*
  2173. 16702  *                              mm_init                                      *
  2174. 16703  *===========================================================================*/
  2175. 16704 PRIVATE void mm_init()
  2176. 16705 {
  2177. 16706 /* Initialize the memory manager. */
  2178. 16707
  2179. 16708   static char core_sigs[] = {
  2180. 16709         SIGQUIT, SIGILL, SIGTRAP, SIGABRT,
  2181. 16710         SIGEMT, SIGFPE, SIGUSR1, SIGSEGV,
  2182. 16711         SIGUSR2, 0 };
  2183. 16712   register int proc_nr;
  2184. 16713   register struct mproc *rmp;
  2185. 16714   register char *sig_ptr;
  2186. 16715   phys_clicks ram_clicks, total_clicks, minix_clicks, free_clicks, dummy;
  2187. 16716   message mess;
  2188. 16717   struct mem_map kernel_map[NR_SEGS];
  2189. 16718   int mem;
  2190. 16719
  2191. 16720   /* Build the set of signals which cause core dumps. Do it the Posix
  2192. 16721    * way, so no knowledge of bit positions is needed.
  2193. 16722    */
  2194. 16723   sigemptyset(&core_sset);
  2195. 16724   for (sig_ptr = core_sigs; *sig_ptr != 0; sig_ptr++)
  2196. 16725         sigaddset(&core_sset, *sig_ptr);
  2197. 16726
  2198. 16727   /* Get the memory map of the kernel to see how much memory it uses,
  2199. 16728    * including the gap between address 0 and the start of the kernel.
  2200. 16729    */
  2201. 16730   sys_getmap(SYSTASK, kernel_map);
  2202. 16731   minix_clicks = kernel_map[S].mem_phys + kernel_map[S].mem_len;
  2203. 16732
  2204. 16733   /* Initialize MM's tables. */
  2205. 16734   for (proc_nr = 0; proc_nr <= INIT_PROC_NR; proc_nr++) {
  2206. 16735         rmp = &mproc[proc_nr];
  2207. 16736         rmp->mp_flags |= IN_USE;
  2208. 16737         sys_getmap(proc_nr, rmp->mp_seg);
  2209. 16738         if (rmp->mp_seg[T].mem_len != 0) rmp->mp_flags |= SEPARATE;
  2210. 16739         minix_clicks += (rmp->mp_seg[S].mem_phys + rmp->mp_seg[S].mem_len)
  2211. 16740                                 - rmp->mp_seg[T].mem_phys;
  2212. 16741   }
  2213. 16742   mproc[INIT_PROC_NR].mp_pid = INIT_PID;
  2214. 16743   sigemptyset(&mproc[INIT_PROC_NR].mp_ignore);
  2215. 16744   sigemptyset(&mproc[INIT_PROC_NR].mp_catch);
  2216. 16745   procs_in_use = LOW_USER + 1;
  2217. 16746
  2218. 16747   /* Wait for FS to send a message telling the RAM disk size then go "on-line".
  2219. 16748    */
  2220. 16749   if (receive(FS_PROC_NR, &mess) != OK)
  2221. 16750         panic("MM can't obtain RAM disk size from FS", NO_NUM);
  2222. 16751
  2223. 16752   ram_clicks = mess.m1_i1;
  2224. 16753
  2225. 16754   /* Initialize tables to all physical mem. */
  2226. 16755   mem_init(&total_clicks, &free_clicks);
  2227. 16756
  2228. 16757   /* Print memory information. */
  2229. 16758   printf("nMemory size =%5dK   ", click_to_round_k(total_clicks));
  2230. 16759   printf("MINIX =%4dK   ", click_to_round_k(minix_clicks));
  2231. 16760   printf("RAM disk =%5dK   ", click_to_round_k(ram_clicks));
  2232. 16761   printf("Available =%5dKnn", click_to_round_k(free_clicks));
  2233. 16762
  2234. 16763   /* Tell FS to continue. */
  2235. 16764   if (send(FS_PROC_NR, &mess) != OK)
  2236. 16765         panic("MM can't sync up with FS", NO_NUM);
  2237. 16766
  2238. 16767   /* Tell the memory task where my process table is for the sake of ps(1). */
  2239. 16768   if ((mem = open("/dev/mem", O_RDWR)) != -1) {
  2240. 16769         ioctl(mem, MIOCSPSINFO, (void *) mproc);
  2241. 16770         close(mem);
  2242. 16771   }
  2243. 16772 }
  2244. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  2245. src/mm/forkexit.c    
  2246. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  2247. 16800 /* This file deals with creating processes (via FORK) and deleting them (via
  2248. 16801  * EXIT/WAIT).  When a process forks, a new slot in the 'mproc' table is
  2249. 16802  * allocated for it, and a copy of the parent's core image is made for the
  2250. 16803  * child.  Then the kernel and file system are informed.  A process is removed
  2251. 16804  * from the 'mproc' table when two events have occurred: (1) it has exited or
  2252. 16805  * been killed by a signal, and (2) the parent has done a WAIT.  If the process
  2253. 16806  * exits first, it continues to occupy a slot until the parent does a WAIT.
  2254. 16807  *
  2255. 16808  * The entry points into this file are:
  2256. 16809  *   do_fork:    perform the FORK system call
  2257. 16810  *   do_mm_exit: perform the EXIT system call (by calling mm_exit())
  2258. 16811  *   mm_exit:    actually do the exiting
  2259. 16812  *   do_wait:    perform the WAITPID or WAIT system call
  2260. 16813  */
  2261. 16814
  2262. 16815
  2263. 16816 #include "mm.h"
  2264. 16817 #include <sys/wait.h>
  2265. 16818 #include <minix/callnr.h>
  2266. 16819 #include <signal.h>
  2267. 16820 #include "mproc.h"
  2268. 16821 #include "param.h"
  2269. 16822
  2270. 16823 #define LAST_FEW            2   /* last few slots reserved for superuser */
  2271. 16824
  2272. 16825 PRIVATE pid_t next_pid = INIT_PID+1;    /* next pid to be assigned */
  2273. 16826
  2274. 16827 FORWARD _PROTOTYPE (void cleanup, (register struct mproc *child) );
  2275. 16828
  2276. 16829 /*===========================================================================*
  2277. 16830  *                              do_fork                                      *
  2278. 16831  *===========================================================================*/
  2279. 16832 PUBLIC int do_fork()
  2280. 16833 {
  2281. 16834 /* The process pointed to by 'mp' has forked.  Create a child process. */
  2282. 16835
  2283. 16836   register struct mproc *rmp;   /* pointer to parent */
  2284. 16837   register struct mproc *rmc;   /* pointer to child */
  2285. 16838   int i, child_nr, t;
  2286. 16839   phys_clicks prog_clicks, child_base = 0;
  2287. 16840   phys_bytes prog_bytes, parent_abs, child_abs; /* Intel only */
  2288. 16841
  2289. 16842  /* If tables might fill up during FORK, don't even start since recovery half
  2290. 16843   * way through is such a nuisance.
  2291. 16844   */
  2292. 16845   rmp = mp;
  2293. 16846   if (procs_in_use == NR_PROCS) return(EAGAIN);
  2294. 16847   if (procs_in_use >= NR_PROCS-LAST_FEW && rmp->mp_effuid != 0)return(EAGAIN);
  2295. 16848
  2296. 16849   /* Determine how much memory to allocate.  Only the data and stack need to
  2297. 16850    * be copied, because the text segment is either shared or of zero length.
  2298. 16851    */
  2299. 16852   prog_clicks = (phys_clicks) rmp->mp_seg[S].mem_len;
  2300. 16853   prog_clicks += (rmp->mp_seg[S].mem_vir - rmp->mp_seg[D].mem_vir);
  2301. 16854   prog_bytes = (phys_bytes) prog_clicks << CLICK_SHIFT;
  2302. 16855   if ( (child_base = alloc_mem(prog_clicks)) == NO_MEM) return(EAGAIN);
  2303. 16856
  2304. 16857   /* Create a copy of the parent's core image for the child. */
  2305. 16858   child_abs = (phys_bytes) child_base << CLICK_SHIFT;
  2306. 16859   parent_abs = (phys_bytes) rmp->mp_seg[D].mem_phys << CLICK_SHIFT;
  2307. 16860   i = sys_copy(ABS, 0, parent_abs, ABS, 0, child_abs, prog_bytes);
  2308. 16861   if (i < 0) panic("do_fork can't copy", i);
  2309. 16862
  2310. 16863   /* Find a slot in 'mproc' for the child process.  A slot must exist. */
  2311. 16864   for (rmc = &mproc[0]; rmc < &mproc[NR_PROCS]; rmc++)
  2312. 16865         if ( (rmc->mp_flags & IN_USE) == 0) break;
  2313. 16866
  2314. 16867   /* Set up the child and its memory map; copy its 'mproc' slot from parent. */
  2315. 16868   child_nr = (int)(rmc - mproc);        /* slot number of the child */
  2316. 16869   procs_in_use++;
  2317. 16870   *rmc = *rmp;                  /* copy parent's process slot to child's */
  2318. 16871
  2319. 16872   rmc->mp_parent = who;         /* record child's parent */
  2320. 16873   rmc->mp_flags &= ~TRACED;     /* child does not inherit trace status */
  2321. 16874   /* A separate I&D child keeps the parents text segment.  The data and stack
  2322. 16875    * segments must refer to the new copy.
  2323. 16876    */
  2324. 16877   if (!(rmc->mp_flags & SEPARATE)) rmc->mp_seg[T].mem_phys = child_base;
  2325. 16878   rmc->mp_seg[D].mem_phys = child_base;
  2326. 16879   rmc->mp_seg[S].mem_phys = rmc->mp_seg[D].mem_phys + 
  2327. 16880                         (rmp->mp_seg[S].mem_vir - rmp->mp_seg[D].mem_vir);
  2328. 16881   rmc->mp_exitstatus = 0;
  2329. 16882   rmc->mp_sigstatus = 0;
  2330. 16883
  2331. 16884   /* Find a free pid for the child and put it in the table. */
  2332. 16885   do {
  2333. 16886         t = 0;                  /* 't' = 0 means pid still free */
  2334. 16887         next_pid = (next_pid < 30000 ? next_pid + 1 : INIT_PID + 1);
  2335. 16888         for (rmp = &mproc[0]; rmp < &mproc[NR_PROCS]; rmp++)
  2336. 16889                 if (rmp->mp_pid == next_pid || rmp->mp_procgrp == next_pid) {
  2337. 16890                         t = 1;
  2338. 16891                         break;
  2339. 16892                 }
  2340. 16893         rmc->mp_pid = next_pid; /* assign pid to child */
  2341. 16894   } while (t);
  2342. 16895
  2343. 16896   /* Tell kernel and file system about the (now successful) FORK. */
  2344. 16897   sys_fork(who, child_nr, rmc->mp_pid, child_base); /* child_base is 68K only*/
  2345. 16898   tell_fs(FORK, who, child_nr, rmc->mp_pid);
  2346. 16899
  2347. 16900   /* Report child's memory map to kernel. */
  2348. 16901   sys_newmap(child_nr, rmc->mp_seg);
  2349. 16902
  2350. 16903   /* Reply to child to wake it up. */
  2351. 16904   reply(child_nr, 0, 0, NIL_PTR);
  2352. 16905   return(next_pid);              /* child's pid */
  2353. 16906 }
  2354. 16909 /*===========================================================================*
  2355. 16910  *                              do_mm_exit                                   *
  2356. 16911  *===========================================================================*/
  2357. 16912 PUBLIC int do_mm_exit()
  2358. 16913 {
  2359. 16914 /* Perform the exit(status) system call. The real work is done by mm_exit(),
  2360. 16915  * which is also called when a process is killed by a signal.
  2361. 16916  */
  2362. 16917
  2363. 16918   mm_exit(mp, status);
  2364. 16919   dont_reply = TRUE;            /* don't reply to newly terminated process */
  2365. 16920   return(OK);                   /* pro forma return code */
  2366. 16921 }
  2367. 16924 /*===========================================================================*
  2368. 16925  *                              mm_exit                                      *
  2369. 16926  *===========================================================================*/
  2370. 16927 PUBLIC void mm_exit(rmp, exit_status)
  2371. 16928 register struct mproc *rmp;     /* pointer to the process to be terminated */
  2372. 16929 int exit_status;                /* the process' exit status (for parent) */
  2373. 16930 {
  2374. 16931 /* A process is done.  Release most of the process' possessions.  If its
  2375. 16932  * parent is waiting, release the rest, else hang.
  2376. 16933  */
  2377. 16934
  2378. 16935   register int proc_nr;
  2379. 16936   int parent_waiting, right_child;
  2380. 16937   pid_t pidarg, procgrp;
  2381. 16938   phys_clicks base, size, s;            /* base and size used on 68000 only */
  2382. 16939
  2383. 16940   proc_nr = (int) (rmp - mproc);        /* get process slot number */
  2384. 16941
  2385. 16942   /* Remember a session leader's process group. */
  2386. 16943   procgrp = (rmp->mp_pid == mp->mp_procgrp) ? mp->mp_procgrp : 0;
  2387. 16944
  2388. 16945   /* If the exited process has a timer pending, kill it. */
  2389. 16946   if (rmp->mp_flags & ALARM_ON) set_alarm(proc_nr, (unsigned) 0);
  2390. 16947
  2391. 16948   /* Tell the kernel and FS that the process is no longer runnable. */
  2392. 16949   tell_fs(EXIT, proc_nr, 0, 0);  /* file system can free the proc slot */
  2393. 16950   sys_xit(rmp->mp_parent, proc_nr, &base, &size);
  2394. 16951
  2395. 16952   /* Release the memory occupied by the child. */
  2396. 16953   if (find_share(rmp, rmp->mp_ino, rmp->mp_dev, rmp->mp_ctime) == NULL) {
  2397. 16954         /* No other process shares the text segment, so free it. */
  2398. 16955         free_mem(rmp->mp_seg[T].mem_phys, rmp->mp_seg[T].mem_len);
  2399. 16956   }
  2400. 16957   /* Free the data and stack segments. */
  2401. 16958   free_mem(rmp->mp_seg[D].mem_phys,
  2402. 16959       rmp->mp_seg[S].mem_vir + rmp->mp_seg[S].mem_len - rmp->mp_seg[D].mem_vir);
  2403. 16960
  2404. 16961   /* The process slot can only be freed if the parent has done a WAIT. */
  2405. 16962   rmp->mp_exitstatus = (char) exit_status;
  2406. 16963   pidarg = mproc[rmp->mp_parent].mp_wpid;       /* who's being waited for? */
  2407. 16964   parent_waiting = mproc[rmp->mp_parent].mp_flags & WAITING;
  2408. 16965   if (pidarg == -1 || pidarg == rmp->mp_pid || -pidarg == rmp->mp_procgrp)
  2409. 16966         right_child = TRUE;             /* child meets one of the 3 tests */
  2410. 16967   else
  2411. 16968         right_child = FALSE;            /* child fails all 3 tests */
  2412. 16969   if (parent_waiting && right_child)
  2413. 16970         cleanup(rmp);                   /* tell parent and release child slot */
  2414. 16971   else
  2415. 16972         rmp->mp_flags |= HANGING;       /* parent not waiting, suspend child */
  2416. 16973
  2417. 16974   /* If the process has children, disinherit them.  INIT is the new parent. */
  2418. 16975   for (rmp = &mproc[0]; rmp < &mproc[NR_PROCS]; rmp++) {
  2419. 16976         if (rmp->mp_flags & IN_USE && rmp->mp_parent == proc_nr) {
  2420. 16977                 /* 'rmp' now points to a child to be disinherited. */
  2421. 16978                 rmp->mp_parent = INIT_PROC_NR;
  2422. 16979                 parent_waiting = mproc[INIT_PROC_NR].mp_flags & WAITING;
  2423. 16980                 if (parent_waiting && (rmp->mp_flags & HANGING)) cleanup(rmp);
  2424. 16981         }
  2425. 16982   }
  2426. 16983
  2427. 16984   /* Send a hangup to the process' process group if it was a session leader. */
  2428. 16985   if (procgrp != 0) check_sig(-procgrp, SIGHUP);
  2429. 16986 }
  2430. 16989 /*===========================================================================*
  2431. 16990  *                              do_waitpid                                   *
  2432. 16991  *===========================================================================*/
  2433. 16992 PUBLIC int do_waitpid()
  2434. 16993 {
  2435. 16994 /* A process wants to wait for a child to terminate. If one is already waiting,
  2436. 16995  * go clean it up and let this WAIT call terminate.  Otherwise, really wait.
  2437. 16996  * Both WAIT and WAITPID are handled by this code.
  2438. 16997  */
  2439. 16998
  2440. 16999   register struct mproc *rp;
  2441. 17000   int pidarg, options, children, res2;
  2442. 17001
  2443. 17002   /* A process calling WAIT never gets a reply in the usual way via the
  2444. 17003    * reply() in the main loop (unless WNOHANG is set or no qualifying child
  2445. 17004    * exists).  If a child has already exited, the routine cleanup() sends 
  2446. 17005    * the reply to awaken the caller.
  2447. 17006    */
  2448. 17007
  2449. 17008   /* Set internal variables, depending on whether this is WAIT or WAITPID. */
  2450. 17009   pidarg  = (mm_call == WAIT ? -1 : pid);       /* first param of waitpid */
  2451. 17010   options = (mm_call == WAIT ?  0 : sig_nr);    /* third param of waitpid */
  2452. 17011   if (pidarg == 0) pidarg = -mp->mp_procgrp;    /* pidarg < 0 ==> proc grp */
  2453. 17012
  2454. 17013   /* Is there a child waiting to be collected? At this point, pidarg != 0:
  2455. 17014    *    pidarg  >  0 means pidarg is pid of a specific process to wait for
  2456. 17015    *    pidarg == -1 means wait for any child
  2457. 17016    *    pidarg  < -1 means wait for any child whose process group = -pidarg
  2458. 17017    */
  2459. 17018   children = 0;
  2460. 17019   for (rp = &mproc[0]; rp < &mproc[NR_PROCS]; rp++) {
  2461. 17020         if ( (rp->mp_flags & IN_USE) && rp->mp_parent == who) {
  2462. 17021                 /* The value of pidarg determines which children qualify. */
  2463. 17022                 if (pidarg  > 0 && pidarg != rp->mp_pid) continue;
  2464. 17023                 if (pidarg < -1 && -pidarg != rp->mp_procgrp) continue;
  2465. 17024
  2466. 17025                 children++;             /* this child is acceptable */
  2467. 17026                 if (rp->mp_flags & HANGING) {
  2468. 17027                         /* This child meets the pid test and has exited. */
  2469. 17028                         cleanup(rp);    /* this child has already exited */
  2470. 17029                         dont_reply = TRUE;
  2471. 17030                         return(OK);
  2472. 17031                 }
  2473. 17032                 if ((rp->mp_flags & STOPPED) && rp->mp_sigstatus) {
  2474. 17033                         /* This child meets the pid test and is being traced.*/
  2475. 17034                         res2 =  0177 | (rp->mp_sigstatus << 8);
  2476. 17035                         reply(who, rp->mp_pid, res2, NIL_PTR);
  2477. 17036                         dont_reply = TRUE;
  2478. 17037                         rp->mp_sigstatus = 0;
  2479. 17038                         return(OK);
  2480. 17039                 }
  2481. 17040         }
  2482. 17041   }
  2483. 17042
  2484. 17043   /* No qualifying child has exited.  Wait for one, unless none exists. */
  2485. 17044   if (children > 0) {
  2486. 17045         /* At least 1 child meets the pid test exists, but has not exited. */
  2487. 17046         if (options & WNOHANG) return(0);    /* parent does not want to wait */
  2488. 17047         mp->mp_flags |= WAITING;             /* parent wants to wait */
  2489. 17048         mp->mp_wpid = (pid_t) pidarg;        /* save pid for later */
  2490. 17049         dont_reply = TRUE;                   /* do not reply now though */
  2491. 17050         return(OK);                          /* yes - wait for one to exit */
  2492. 17051   } else {
  2493. 17052         /* No child even meets the pid test.  Return error immediately. */
  2494. 17053         return(ECHILD);                      /* no - parent has no children */
  2495. 17054   }
  2496. 17055 }
  2497. 17058 /*===========================================================================*
  2498. 17059  *                              cleanup                                      *
  2499. 17060  *===========================================================================*/
  2500. 17061 PRIVATE void cleanup(child)
  2501. 17062 register struct mproc *child;   /* tells which process is exiting */
  2502. 17063 {
  2503. 17064 /* Finish off the exit of a process.  The process has exited or been killed
  2504. 17065  * by a signal, and its parent is waiting.
  2505. 17066  */
  2506. 17067
  2507. 17068   int exitstatus;
  2508. 17069
  2509. 17070   /* Wake up the parent. */
  2510. 17071   exitstatus = (child->mp_exitstatus << 8) | (child->mp_sigstatus & 0377);
  2511. 17072   reply(child->mp_parent, child->mp_pid, exitstatus, NIL_PTR);
  2512. 17073   mproc[child->mp_parent].mp_flags &= ~WAITING; /* parent no longer waiting */
  2513. 17074
  2514. 17075   /* Release the process table entry. */
  2515. 17076   child->mp_flags = 0;
  2516. 17077   procs_in_use--;
  2517. 17078 }
  2518. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  2519. src/mm/exec.c    
  2520. ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  2521. 17100 /* This file handles the EXEC system call.  It performs the work as follows:
  2522. 17101  *    - see if the permissions allow the file to be executed
  2523. 17102  *    - read the header and extract the sizes
  2524. 17103  *    - fetch the initial args and environment from the user space
  2525. 17104  *    - allocate the memory for the new process
  2526. 17105  *    - copy the initial stack from MM to the process
  2527. 17106  *    - read in the text and data segments and copy to the process
  2528. 17107  *    - take care of setuid and setgid bits
  2529. 17108  *    - fix up 'mproc' table
  2530. 17109  *    - tell kernel about EXEC
  2531. 17110  *    - save offset to initial argc (for ps)
  2532. 17111  *
  2533. 17112  * The entry points into this file are:
  2534. 17113  *   do_exec:    perform the EXEC system call
  2535. 17114  *   find_share: find a process whose text segment can be shared
  2536. 17115  */
  2537. 17116
  2538. 17117 #include "mm.h"
  2539. 17118 #include <sys/stat.h>
  2540. 17119 #include <minix/callnr.h>
  2541. 17120 #include <a.out.h>
  2542. 17121 #include <signal.h>
  2543. 17122 #include <string.h>
  2544. 17123 #include "mproc.h"
  2545. 17124 #include "param.h"
  2546. 17125
  2547. 17126 FORWARD _PROTOTYPE( void load_seg, (int fd, int seg, vir_bytes seg_bytes) );
  2548. 17127 FORWARD _PROTOTYPE( int new_mem, (struct mproc *sh_mp, vir_bytes text_bytes,
  2549. 17128                 vir_bytes data_bytes, vir_bytes bss_bytes,
  2550. 17129                 vir_bytes stk_bytes, phys_bytes tot_bytes)              );
  2551. 17130 FORWARD _PROTOTYPE( void patch_ptr, (char stack [ARG_MAX ], vir_bytes base) );
  2552. 17131 FORWARD _PROTOTYPE( int read_header, (int fd, int *ft, vir_bytes *text_bytes,
  2553. 17132                 vir_bytes *data_bytes, vir_bytes *bss_bytes,
  2554. 17133                 phys_bytes *tot_bytes, long *sym_bytes, vir_clicks sc,
  2555. 17134                 vir_bytes *pc)                                          );
  2556. 17135
  2557. 17136
  2558. 17137 /*===========================================================================*
  2559. 17138  *                              do_exec                                      *
  2560. 17139  *===========================================================================*/
  2561. 17140 PUBLIC int do_exec()
  2562. 17141 {
  2563. 17142 /* Perform the execve(name, argv, envp) call.  The user library builds a
  2564. 17143  * complete stack image, including pointers, args, environ, etc.  The stack
  2565. 17144  * is copied to a buffer inside MM, and then to the new core image.
  2566. 17145  */
  2567. 17146
  2568. 17147   register struct mproc *rmp;
  2569. 17148   struct mproc *sh_mp;
  2570. 17149   int m, r, fd, ft, sn;
  2571. 17150   static char mbuf[ARG_MAX];    /* buffer for stack and zeroes */
  2572. 17151   static char name_buf[PATH_MAX]; /* the name of the file to exec */
  2573. 17152   char *new_sp, *basename;
  2574. 17153   vir_bytes src, dst, text_bytes, data_bytes, bss_bytes, stk_bytes, vsp;
  2575. 17154   phys_bytes tot_bytes;         /* total space for program, including gap */
  2576. 17155   long sym_bytes;
  2577. 17156   vir_clicks sc;
  2578. 17157   struct stat s_buf;
  2579. 17158   vir_bytes pc;
  2580. 17159
  2581. 17160   /* Do some validity checks. */
  2582. 17161   rmp = mp;
  2583. 17162   stk_bytes = (vir_bytes) stack_bytes;
  2584. 17163   if (stk_bytes > ARG_MAX) return(ENOMEM);      /* stack too big */
  2585. 17164   if (exec_len <= 0 || exec_len > PATH_MAX) return(EINVAL);
  2586. 17165
  2587. 17166   /* Get the exec file name and see if the file is executable. */
  2588. 17167   src = (vir_bytes) exec_name;
  2589. 17168   dst = (vir_bytes) name_buf;
  2590. 17169   r = sys_copy(who, D, (phys_bytes) src,
  2591. 17170                 MM_PROC_NR, D, (phys_bytes) dst, (phys_bytes) exec_len);
  2592. 17171   if (r != OK) return(r);       /* file name not in user data segment */
  2593. 17172   tell_fs(CHDIR, who, FALSE, 0);        /* switch to the user's FS environ. */
  2594. 17173   fd = allowed(name_buf, &s_buf, X_BIT);        /* is file executable? */
  2595. 17174   if (fd < 0) return(fd);       /* file was not executable */
  2596. 17175
  2597. 17176   /* Read the file header and extract the segment sizes. */
  2598. 17177   sc = (stk_bytes + CLICK_SIZE - 1) >> CLICK_SHIFT;
  2599. 17178   m = read_header(fd, &ft, &text_bytes, &data_bytes, &bss_bytes, 
  2600. 17179                                         &tot_bytes, &sym_bytes, sc, &pc);
  2601. 17180   if (m < 0) {
  2602. 17181         close(fd);              /* something wrong with header */
  2603. 17182         return(ENOEXEC);
  2604. 17183   }
  2605. 17184
  2606. 17185   /* Fetch the stack from the user before destroying the old core image. */
  2607. 17186   src = (vir_bytes) stack_ptr;
  2608. 17187   dst = (vir_bytes) mbuf;
  2609. 17188   r = sys_copy(who, D, (phys_bytes) src,
  2610. 17189                         MM_PROC_NR, D, (phys_bytes) dst, (phys_bytes)stk_bytes);
  2611. 17190   if (r != OK) {
  2612. 17191         close(fd);              /* can't fetch stack (e.g. bad virtual addr) */
  2613. 17192         return(EACCES);
  2614. 17193   }
  2615. 17194
  2616. 17195   /* Can the process' text be shared with that of one already running? */
  2617. 17196   sh_mp = find_share(rmp, s_buf.st_ino, s_buf.st_dev, s_buf.st_ctime);
  2618. 17197
  2619. 17198   /* Allocate new memory and release old memory.  Fix map and tell kernel. */
  2620. 17199   r = new_mem(sh_mp, text_bytes, data_bytes, bss_bytes, stk_bytes, tot_bytes);
  2621. 17200   if (r != OK) {
  2622. 17201         close(fd);              /* insufficient core or program too big */
  2623. 17202         return(r);
  2624. 17203   }
  2625. 17204
  2626. 17205   /* Save file identification to allow it to be shared. */
  2627. 17206   rmp->mp_ino = s_buf.st_ino;
  2628. 17207   rmp->mp_dev = s_buf.st_dev;
  2629. 17208   rmp->mp_ctime = s_buf.st_ctime;
  2630. 17209
  2631. 17210   /* Patch up stack and copy it from MM to new core image. */
  2632. 17211   vsp = (vir_bytes) rmp->mp_seg[S].mem_vir << CLICK_SHIFT;
  2633. 17212   vsp += (vir_bytes) rmp->mp_seg[S].mem_len << CLICK_SHIFT;
  2634. 17213   vsp -= stk_bytes;
  2635. 17214   patch_ptr(mbuf, vsp);
  2636. 17215   src = (vir_bytes) mbuf;
  2637. 17216   r = sys_copy(MM_PROC_NR, D, (phys_bytes) src,
  2638. 17217                         who, D, (phys_bytes) vsp, (phys_bytes)stk_bytes);
  2639. 17218   if (r != OK) panic("do_exec stack copy err", NO_NUM);
  2640. 17219
  2641. 17220   /* Read in text and data segments. */
  2642. 17221   if (sh_mp != NULL) {
  2643. 17222         lseek(fd, (off_t) text_bytes, SEEK_CUR);  /* shared: skip text */
  2644. 17223   } else {
  2645. 17224         load_seg(fd, T, text_bytes);
  2646. 17225   }
  2647. 17226   load_seg(fd, D, data_bytes);
  2648. 17227
  2649. 17228
  2650. 17229   close(fd);                    /* don't need exec file any more */
  2651. 17230
  2652. 17231   /* Take care of setuid/setgid bits. */
  2653. 17232   if ((rmp->mp_flags & TRACED) == 0) { /* suppress if tracing */
  2654. 17233         if (s_buf.st_mode & I_SET_UID_BIT) {
  2655. 17234                 rmp->mp_effuid = s_buf.st_uid;
  2656. 17235                 tell_fs(SETUID,who, (int)rmp->mp_realuid, (int)rmp->mp_effuid);
  2657. 17236         }
  2658. 17237         if (s_buf.st_mode & I_SET_GID_BIT) {
  2659. 17238                 rmp->mp_effgid = s_buf.st_gid;
  2660. 17239                 tell_fs(SETGID,who, (int)rmp->mp_realgid, (int)rmp->mp_effgid);
  2661. 17240         }
  2662. 17241   }
  2663. 17242
  2664. 17243   /* Save offset to initial argc (for ps) */
  2665. 17244   rmp->mp_procargs = vsp;
  2666. 17245
  2667. 17246   /* Fix 'mproc' fields, tell kernel that exec is done,  reset caught sigs. */
  2668. 17247   for (sn = 1; sn <= _NSIG; sn++) {
  2669. 17248         if (sigismember(&rmp->mp_catch, sn)) {
  2670. 17249                 sigdelset(&rmp->mp_catch, sn);
  2671. 17250                 rmp->mp_sigact[sn].sa_handler = SIG_DFL;
  2672. 17251                 sigemptyset(&rmp->mp_sigact[sn].sa_mask);
  2673. 17252         }
  2674. 17253   }
  2675. 17254
  2676. 17255   rmp->mp_flags &= ~SEPARATE;   /* turn off SEPARATE bit */
  2677. 17256   rmp->mp_flags |= ft;          /* turn it on for separate I & D files */
  2678. 17257   new_sp = (char *) vsp;
  2679. 17258
  2680. 17259   tell_fs(EXEC, who, 0, 0);     /* allow FS to handle FD_CLOEXEC files */
  2681. 17260
  2682. 17261   /* System will save command line for debugging, ps(1) output, etc. */
  2683. 17262   basename = strrchr(name_buf, '/');
  2684. 17263   if (basename == NULL) basename = name_buf; else basename++;