dosinst.c 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695
  1. /*
  2. * Installation code for REDUCE in the Codemist CSL version. This
  3. * code is intended for use with MSDOS, and is compiled with
  4. * Zortech C release 3.0
  5. */
  6. /*
  7. * Copyright (C) 1992, Codemist Ltd.
  8. * This code has been modelled after suggestions from Jed Marti
  9. */
  10. /* Signature: 18bb67af 11-Oct-1993 */
  11. /*
  12. * Here, right at the top of this program, I give a list of the
  13. * files that are to be unpacked.
  14. */
  15. typedef struct arch
  16. {
  17. char *name;
  18. int loaded;
  19. } arch;
  20. static arch archives[8] =
  21. {
  22. {"EXE286.CAR", 0},
  23. {"EXE386.CAR", 0},
  24. {"IMG.CAR", 0},
  25. {"DOCS.CAR", 0},
  26. {"SRC1.CAR", 0},
  27. {"SRC2.CAR", 0},
  28. {"SRC3.CAR", 0},
  29. {"README.CAR", 0}
  30. };
  31. #include <stdio.h>
  32. #include <stdlib.h>
  33. #include <ctype.h>
  34. #include <signal.h>
  35. #include <setjmp.h>
  36. #include <errno.h>
  37. #include <string.h>
  38. #include <math.h>
  39. #include <time.h>
  40. #include <disp.h> /* The Zortech C fast screen-output package */
  41. #include <conio.h> /* unbuffered keyboard input etc */
  42. #include <direct.h> /* for mkdir() to create directories */
  43. #include <dos.h> /* so that I can select which drive is current */
  44. #define EXTRACT 1
  45. #define INSTALL 1
  46. static jmp_buf escape_buffer;
  47. static void ctrl_c_handler(int n)
  48. {
  49. longjmp(escape_buffer, 1);
  50. }
  51. static void show_txt(int row, int col, char *msg);
  52. #include "car.c" /* File decompression technology */
  53. #define white_on_blue 0x17
  54. #define white_on_black 0x07
  55. #define black_on_white 0x70
  56. /*
  57. * filep(dr, fn) - returns nonzero if drive dr and file name exists, 0 if not.
  58. */
  59. static FILE *filep(char *dr, char *fn)
  60. {
  61. char fname[256];
  62. FILE *fh;
  63. sprintf(fname, "%s%s\0", dr, fn);
  64. return fopen(fname, "rb");
  65. }
  66. /*
  67. * There is a smallish area in the middle of the screen that I
  68. * actually use here - clear it to the background colour (blue) that
  69. * I am using.
  70. */
  71. static void clear_work_area()
  72. {
  73. disp_fillbox(256*white_on_blue + ' ', 7, 11, 18, 68);
  74. }
  75. static void show_txt(int row, int col, char *msg)
  76. {
  77. int c, l = col;
  78. disp_move(row, col);
  79. while ((c = *msg++) != 0)
  80. { if (c == '\n')
  81. { while (l <= 68)
  82. { disp_putc(' ');
  83. l++;
  84. }
  85. disp_move(++row, col);
  86. l = col;
  87. }
  88. else
  89. { disp_putc(c);
  90. l++;
  91. }
  92. }
  93. }
  94. /*
  95. * This routine asks the use a question. The question is passed as
  96. * MSG and it will ask the user to type in a value. The default
  97. * will be provided in VAL, and that will be what is used if the user
  98. * just hits ENTER. maxlen indicates the size of the VAL buffer, so
  99. * that (with luck) I can avoid overwriting things. The message HELP
  100. * is displayed so that the user understands what the question is
  101. * supposed to be about. Returns non-zero if user wants to give up.
  102. */
  103. static int request_string(int maxlen, char *msg, char *help,
  104. char *val, char *dflt)
  105. {
  106. int i, j, hwm;
  107. clear_work_area();
  108. disp_setattr(white_on_black);
  109. show_txt(8, 11, msg);
  110. show_txt(12, 11, help);
  111. disp_setattr(black_on_white);
  112. strcpy(val, dflt);
  113. show_txt(10, 16, val);
  114. hwm = i = strlen(val);
  115. disp_move(10, 16+i);
  116. disp_flush();
  117. for (;;)
  118. { int c = getch();
  119. if (c == 0) c = 0x100 + getch(); /* Extended character */
  120. switch (c)
  121. {
  122. case 0x03:
  123. case 0x1b: /* ^C and ESC */
  124. return 1;
  125. case 0x08: /* backspace deletes char before the cursor */
  126. if (i == 0) continue;
  127. i--;
  128. case 0xff:
  129. case 0x100+'S': /* DELete deletes char at cursor */
  130. if (i == hwm) continue;
  131. disp_move(10, 16+i);
  132. hwm--;
  133. for (j=i; j<hwm; j++)
  134. { int c = val[j+1];
  135. disp_putc(c);
  136. val[j] = c;
  137. }
  138. disp_setattr(white_on_blue);
  139. disp_putc(' ');
  140. disp_setattr(black_on_white);
  141. disp_move(10, 16+i);
  142. disp_flush();
  143. continue;
  144. case '\r':
  145. case '\n': /* ENTER */
  146. hwm--; /* I truncate off any trailing blanks here */
  147. while (hwm!=0 && val[hwm]==' ') hwm--;
  148. val[hwm+1] = 0;
  149. return 0;
  150. case 0x100+'R': /* Insert */
  151. if (hwm >= maxlen) continue;
  152. hwm++;
  153. for (j=hwm; j>i; j--) val[j] = val[j-1];
  154. val[i] = ' ';
  155. disp_move(10, 16+i);
  156. for (j=i; j<hwm; j++) disp_putc(val[j]);
  157. disp_move(10, 16+i);
  158. continue;
  159. case 0x100+'K': /* Left arrow */
  160. if (i == 0) continue;
  161. i--;
  162. disp_move(10, 16+i);
  163. disp_flush();
  164. continue;
  165. case 0x100+'M': /* Right arrow */
  166. if (i == hwm) continue;
  167. i++;
  168. disp_move(10, 16+i);
  169. disp_flush();
  170. continue;
  171. case 0x100+'G': /* Home - reinstate default text */
  172. disp_move(10, 16);
  173. disp_setattr(white_on_blue);
  174. for (j=0; j<hwm; j++) disp_putc(' ');
  175. disp_setattr(black_on_white);
  176. strcpy(val, dflt);
  177. show_txt(10, 16, val);
  178. hwm = i = strlen(val);
  179. disp_move(10, 16+i);
  180. disp_flush();
  181. continue;
  182. default:
  183. if (c >= 0x100) continue; /* Ignore extended chars */
  184. if (i < maxlen)
  185. { val[i++] = c;
  186. disp_putc(c);
  187. disp_flush();
  188. if (i > hwm) hwm = i;
  189. }
  190. continue;
  191. }
  192. }
  193. }
  194. /*
  195. * request_char is like request_string, except that the input needed from
  196. * the user will be just one character long. The result is dumped into the
  197. * first character of VAL.
  198. */
  199. static int request_char(char *msg, char *help, char *val)
  200. {
  201. clear_work_area();
  202. disp_setattr(white_on_black);
  203. show_txt(8, 11, msg);
  204. show_txt(12, 11, help);
  205. disp_setattr(black_on_white);
  206. show_txt(10, 16, val);
  207. disp_move(10, 16);
  208. disp_flush();
  209. for (;;)
  210. { int c = getch();
  211. switch (c & 0x7f)
  212. {
  213. case 0x03:
  214. case 0x1b: /* ^C and ESC */
  215. return 1;
  216. case '\r':
  217. case '\n': /* ENTER */
  218. return 0;
  219. case 0x08:
  220. case 0x7f: /* backspace and DELete - mapped to blank */
  221. c = ' ';
  222. default:
  223. c = toupper(c);
  224. val[0] = c;
  225. disp_move(10, 16);
  226. disp_putc(c);
  227. disp_move(10, 16);
  228. disp_flush();
  229. continue;
  230. }
  231. break;
  232. }
  233. return 0;
  234. }
  235. /*
  236. * request_ok() displays the given message and just waits for the user
  237. * to press any (printing) key. It is used for diagnostic messages and
  238. * similar reports.
  239. */
  240. static void request_ok(char *help)
  241. {
  242. clear_work_area();
  243. disp_setattr(white_on_black);
  244. show_txt(8, 11, "Press a key (e.g. ENTER) to continue\n");
  245. show_txt(12, 11, help);
  246. disp_move(10, 16);
  247. disp_flush();
  248. (void)getch(); /* All characters behave the same here */
  249. }
  250. /*
  251. * This is an interface to code found in the #included file "car.c"
  252. * that contains the Codemist file compression code and archive
  253. * utility. It sets up various workspace and invokes file decompression.
  254. */
  255. static void decompress_files()
  256. {
  257. int32 i;
  258. for (i=0; i<PREDICTION_SIZE; i++)
  259. { prediction1[i] = ' ';
  260. prediction2[i] = '\n';
  261. }
  262. memcpy(current_frequencies, default_frequencies,
  263. sizeof(default_frequencies));
  264. CRC = 1;
  265. while (extract_files()) continue;
  266. { unsigned32 calculated_CRC = CRC;
  267. unsigned32 stored_CRC = get4(); /* WARNING: get4() changes CRC! */
  268. if (calculated_CRC != stored_CRC)
  269. fprintf(stderr, "Warning: CRC failure - archive may be corrupt\n");
  270. }
  271. fclose(archive_file);
  272. }
  273. static int install(int default_source)
  274. {
  275. int cpu = cputype(); /* Used to select version to install */
  276. char *xdir, sdrive[4], dest[48], yes_no[4];
  277. char msg[1000], fname[256];
  278. int i, l, c, something_found;
  279. int disable_exe = 0, want_exe = 1, want_doc = 1, want_src = 1;
  280. if (cpu < 2)
  281. { request_ok("Unsuitable computer - can not install REDUCE\n"
  282. "REDUCE needs at least an 80286 computer (and if you\n"
  283. "have an 80386 or 80486 a faster version will be\n"
  284. "installed for you). This machine seems to be an 8086\n"
  285. "or 80186. You will not be allowed to install\n"
  286. "executable binaries, but can load sources or docs\n");
  287. disable_exe = 1;
  288. }
  289. if (cpu < 3) archives[1].loaded = 1, xdir = "csl286";
  290. else archives[0].loaded = 1, xdir = "csl386";
  291. for (;;)
  292. { if (disable_exe) want_exe = 0;
  293. else
  294. { for (;;)
  295. { sprintf(yes_no, "Y");
  296. if (request_char(
  297. "Do you want to install executable binaries?\n",
  298. "If you select Y and type ENTER then executable binaries\n"
  299. "for REDUCE will be installed onto your hard disc. If you\n"
  300. "type N and then ENTER binaries will not be installed.\n"
  301. "The binary files involved will be called r35.exe and\n"
  302. "r35.img\n", yes_no)) goto abandon;
  303. c = toupper(yes_no[0]);
  304. if (c == 'Y' || c == 'N') break;
  305. }
  306. want_exe = (c == 'Y');
  307. }
  308. for (;;)
  309. { sprintf(yes_no, "Y");
  310. if (request_char("Do you want to install documentation?\n",
  311. "If you select Y and type ENTER then machine-readable\n"
  312. "documentation for REDUCE will be installed onto your hard\n"
  313. "disc. If you type N and then ENTER it will not be\n"
  314. "installed. The files involved will be placed in a\n"
  315. "subdirectory called \"DOC\" and most of them are formatted\n"
  316. "using the TeX text layout system\n", yes_no)) goto abandon;
  317. c = toupper(yes_no[0]);
  318. if (c == 'Y' || c == 'N') break;
  319. }
  320. want_doc = (c == 'Y');
  321. #ifdef PERSONAL
  322. want_src = 0;
  323. #else
  324. for (;;)
  325. { sprintf(yes_no, "Y");
  326. if (request_char("Do you want to install source files?\n",
  327. "If you select Y and type ENTER then all source files for\n"
  328. "REDUCE will be installed onto your hard disc. If you type\n"
  329. "N and then ENTER binaries will not be installed. The\n"
  330. "REDUCE source files will be put in a directory \"SRC\", and\n"
  331. "many further files relating to the underlying Lisp (CSL)\n"
  332. "will also appear\n", yes_no)) goto abandon;
  333. c = toupper(yes_no[0]);
  334. if (c == 'Y' || c == 'N') break;
  335. }
  336. want_src = (c == 'Y');
  337. #endif
  338. for (;;)
  339. { sprintf(yes_no, "Y");
  340. if (want_exe || want_doc || want_src)
  341. sprintf(msg, "You have chosen to install%s%s%s\n",
  342. (want_exe ? "\n Executable binaries" : ""),
  343. (want_doc ? "\n Documentation" : ""),
  344. (want_src ? "\n Source code" : ""));
  345. else sprintf(msg, "You seem not to want to install anything!\n");
  346. if (request_char("Proceed with installation?\n",
  347. msg, yes_no)) goto abandon;
  348. c = toupper(yes_no[0]);
  349. if (c == 'Y' || c == 'N') break;
  350. else if (c == 'Q') goto abandon;
  351. }
  352. if (c == 'N') continue; /* Give user a chance to try again */
  353. break;
  354. }
  355. if (!want_exe)
  356. archives[0].loaded = archives[1].loaded = archives[2].loaded = 1;
  357. if (!want_doc)
  358. archives[3].loaded = 1;
  359. if (!want_src)
  360. archives[4].loaded = archives[5].loaded = archives[6].loaded = 1;
  361. for (;;)
  362. { sdrive[0] = default_source;
  363. sdrive[1] = ':';
  364. sdrive[2] = 0;
  365. if (request_char("Floppy Drive Name\n",
  366. "This is the drive identifier where the Codemist Floppy\n"
  367. "disks will be mounted. Common values are A or B. If you\n"
  368. "want the default, just type the ENTER key. To change the\n"
  369. "value, enter a new letter, any of A-Z will be accepted.\n"
  370. "Any other key will cause the installation procedure to\n"
  371. "be aborted.", sdrive)) goto abandon;
  372. sdrive[0] = toupper(sdrive[0]);
  373. if (!('A' <= sdrive[0] && sdrive[0] <= 'Z')) goto abandon;
  374. select_destination:
  375. if (request_string(44, "Hard Disk Drive and Directory\n",
  376. "This is the drive and directory you wish to install\n"
  377. "Codemist REDUCE 3.4 on. If you want the default, just\n"
  378. "type the ENTER key. To change the value, enter a new\n"
  379. "letter. If the directory does not exist, it will be\n"
  380. "created for you.", dest, "C:\\REDUCE")) goto abandon;
  381. for (;;)
  382. { sprintf(yes_no, "Y");
  383. sprintf(msg, "Source Disk Drive %s\n"
  384. "Where to install %s\n"
  385. "Are these values correct (Y or N)\n"
  386. "Type Q or ^C to abandon installation\n", sdrive, dest);
  387. if (request_char("Proceed with installation?\n",
  388. msg, yes_no)) goto abandon;
  389. c = toupper(yes_no[0]);
  390. if (c == 'Y' || c == 'N') break;
  391. else if (c == 'Q') goto abandon;
  392. }
  393. if (c == 'N') continue; /* Give user a chance to try again */
  394. /*
  395. * Now I try to ensure that the directory indicated exists... First
  396. * I select the relevant drive as current, that is always supposing that
  397. * the user's response started off as "X:" for some X.
  398. */
  399. if (dest[0] != 0 && dest[1] == ':')
  400. { unsigned int want, ndrives, found;
  401. want = toupper(dest[0]) - 'A' + 1;
  402. dos_setdrive(want, &ndrives);
  403. /*
  404. * I check that the desired drive was indeed selected by checking which
  405. * drive is current after I (attempt to) make the selection.
  406. */
  407. dos_getdrive(&found);
  408. if (found != want) goto select_failed;
  409. }
  410. l = strlen(dest);
  411. for (i=1; i<=l; i++)
  412. { c = dest[i];
  413. if ((c == '\\' || c == 0) &&
  414. dest[i-1] != ':' && dest[i-1] != '\\')
  415. { int w;
  416. dest[i] = 0;
  417. w = mkdir(dest);
  418. if (w == 0 || /* created OK */
  419. errno == EACCES) /* already present */
  420. { dest[i] = c;
  421. continue;
  422. }
  423. yes_no[0] = 'Y'; yes_no[1] = 0;
  424. sprintf(msg, "The directory\n"
  425. "%s\n"
  426. "could not be created for you. Type Y/ENTER\n"
  427. "to select another destination, anything else to\n"
  428. "quit.\n", dest);
  429. if (request_char("Try again?\n", msg, yes_no)) goto abandon;
  430. if (toupper(yes_no[0]) != 'Y') goto abandon;
  431. else goto select_destination;
  432. }
  433. }
  434. /*
  435. * Now the destination directory ought to exist - select it as
  436. * current.
  437. * I will leave this drive and directory selected when I exit from
  438. * the installation process (even if installation fails)
  439. */
  440. if (chdir(dest) == 0) break;
  441. /*
  442. * If I fail to select the desired drive I give the user a chance to
  443. * select another.
  444. */
  445. select_failed:
  446. yes_no[0] = 'Y'; yes_no[1] = 0;
  447. sprintf(msg, "The directory\n"
  448. "%s\n"
  449. "could not be selected. Type Y/ENTER\n"
  450. "to select another destination, anything else to\n"
  451. "quit.\n", dest);
  452. if (request_char("Try again?\n", msg, yes_no)) goto abandon;
  453. if (toupper(yes_no[0]) != 'Y') goto abandon;
  454. else goto select_destination;
  455. }
  456. /*
  457. * I want two sub-directories created here - SLOW is used to hold
  458. * log files from the slow version of the system, while COUNTS holds
  459. * statistics that can indicate which (bytecoded) functions are being used
  460. * most heavily.
  461. */
  462. if (mkdir("slow") != 0 && errno != EACCES) goto installation_failed;
  463. if (mkdir("counts") != 0 && errno != EACCES) goto installation_failed;
  464. if (mkdir(xdir) != 0 && errno != EACCES) goto installation_failed;
  465. if (chdir(xdir) != 0) goto installation_failed;
  466. /*
  467. * Now I have the guts of the installation process - I have a list of
  468. * archives that I would like to find and will accept discs in any order
  469. * until I have found them all.
  470. */
  471. for (;;)
  472. {
  473. something_found = 0;
  474. look_for_another:
  475. for (i=0; i<8; i++)
  476. { if (archives[i].loaded) continue;
  477. goto another_to_load;
  478. }
  479. break; /* Installation complete */
  480. another_to_load:
  481. for (i=0; i<8; i++)
  482. { if (!archives[i].loaded)
  483. { archive_file = filep(sdrive, archives[i].name);
  484. if (archive_file == NULL) continue;
  485. something_found = 1;
  486. clear_work_area();
  487. sprintf(msg, "File %s to be uncompressed\n", archives[i].name);
  488. show_txt(9, 16, msg);
  489. decompress_files();
  490. archives[i].loaded = 1;
  491. goto look_for_another;
  492. }
  493. }
  494. if (something_found)
  495. request_ok(
  496. "Please load the next disc of the REDUCE distribution\n"
  497. "set in your floppy disc drive and type a character\n"
  498. "(e.g. ENTER). Type ^C to abort installation.\n");
  499. else request_ok(
  500. "*** No relevant files were found on that disc ***\n"
  501. "This may be because you are not loading the whole of\n"
  502. "REDUCE. Please load the next disc of the distribution\n"
  503. "set in your floppy disc drive and type a character\n"
  504. "(e.g. ENTER). Type ^C to abort installation.\n");
  505. }
  506. request_ok(
  507. "Installation complete. The directory with the REDUCE\n"
  508. "files will now be selected as current. Please check\n"
  509. "the file \"READ.ME\" for last-minute notes.\n");
  510. return EXIT_SUCCESS;
  511. installation_failed:
  512. request_ok(
  513. "The installation script was unable to create\n"
  514. "one of the files or directories needed for REDUCE.\n");
  515. return EXIT_FAILURE;
  516. abandon:
  517. request_ok("Installation of REDUCE incomplete.\n");
  518. return EXIT_FAILURE;
  519. }
  520. /*
  521. * The following procedure displays one line of text, centred in the
  522. * box I have drawn on the screen.
  523. */
  524. static void disp(int row, char *s)
  525. {
  526. int len = strlen(s);
  527. disp_move(row, 40-len/2);
  528. disp_puts(s);
  529. }
  530. int main(int argc, char *argv[])
  531. {
  532. int rc, vm = disp_getmode();
  533. int nrows, ncols, crow, ccol;
  534. unsigned short *ssave;
  535. /*
  536. * Firstly I grab the initial contents of the screen so that I can
  537. * restore it at the end.
  538. */
  539. disp_open();
  540. nrows = disp_numrows; ncols = disp_numcols;
  541. crow = disp_cursorrow; ccol = disp_cursorcol;
  542. ssave = (unsigned short *)malloc(nrows*ncols*sizeof(short));
  543. if (ssave == NULL)
  544. { disp_close();
  545. fprintf(stderr, "Insufficient memory for installation of REDUCE\n");
  546. exit(EXIT_FAILURE);
  547. }
  548. disp_peekbox(ssave, 0, 0, nrows-1, ncols-1);
  549. disp_close();
  550. /*
  551. * Now I grab the memory that will be needed for file decompression, so
  552. * that use of malloc() is all done early and thus out-of-memory
  553. * messages can be dealt with before anything else is messed up.
  554. */
  555. prediction1 = (unsigned char *)malloc((size_t)PREDICTION_SIZE);
  556. prediction2 = (unsigned char *)malloc((size_t)PREDICTION_SIZE);
  557. fast_decode_table =
  558. (int32 *)malloc((size_t)(FAST_DECODE_TABLESIZE*sizeof(int32)));
  559. char_being_coded = (int *)malloc((size_t)(2*CODE_SIZE*sizeof(int)));
  560. number_of_occurences = (int32 *)malloc((size_t)(2*CODE_SIZE*sizeof(int32)));
  561. character_index = (int *)malloc((size_t)(2*CODE_SIZE*sizeof(int)));
  562. corresponding_character = (int *)malloc((size_t)(CODE_SIZE*sizeof(int)));
  563. huffman_coded_bits =
  564. (unsigned32 *)malloc((size_t)(CODE_SIZE*sizeof(unsigned32)));
  565. length_of_code = (char *)malloc((size_t)CODE_SIZE);
  566. /*
  567. * I re-use the double-length array character_index[] as one of the
  568. * arrays needed for decoding...
  569. */
  570. sorted_huffman_codes = (unsigned32 *)&character_index[0];
  571. if (prediction1 == NULL ||
  572. prediction2 == NULL ||
  573. fast_decode_table == NULL ||
  574. char_being_coded == NULL ||
  575. number_of_occurences == NULL ||
  576. character_index == NULL ||
  577. corresponding_character == NULL ||
  578. huffman_coded_bits == NULL ||
  579. length_of_code == NULL)
  580. { fprintf(stderr, "Insufficient memory for installation of REDUCE\n");
  581. exit(EXIT_FAILURE);
  582. }
  583. /*
  584. * Now force myself into an 80*25 alphanumeric screen mode
  585. */
  586. if (vm != 3) disp_setmode(3);
  587. disp_open();
  588. /*
  589. * Clear the screen
  590. */
  591. disp_move(0, 0); disp_eeop();
  592. /*
  593. * Draw a box, with a border, for installation messages to live within
  594. */
  595. disp_fillbox(256*white_on_blue + ' ', 4, 9, 21, 70);
  596. disp_box(0, white_on_blue, 4, 9, 21, 70);
  597. disp_setattr(white_on_blue);
  598. #ifdef PERSONAL
  599. disp(6, "Personal R E D U C E 3.4.1");
  600. #else
  601. disp(6, "R E D U C E 3.4.1");
  602. #endif
  603. disp(19, "Codemist Ltd, Alta, Horsecombe Vale, Combe Down,");
  604. disp(20, "Bath BA2 5QR, England. Telephone/Fax: +44 225 837430");
  605. switch (setjmp(escape_buffer))
  606. {
  607. case 0:
  608. signal(SIGTERM, ctrl_c_handler);
  609. rc = install(argc > 0 ? toupper(argv[0][0]) : 'A');
  610. break;
  611. default:
  612. signal(SIGTERM, SIG_IGN);
  613. rc = EXIT_FAILURE;
  614. break;
  615. }
  616. /*
  617. * Re-instate the screen-mode that was active when I started
  618. */
  619. disp_close();
  620. if (vm != 3) disp_setmode(vm);
  621. /*
  622. * Restore previous screen contents and cursor position
  623. */
  624. disp_open();
  625. disp_pokebox(ssave, 0, 0, nrows-1, ncols-1);
  626. disp_move(crow, ccol);
  627. disp_flush();
  628. disp_close();
  629. exit(rc);
  630. return 0;
  631. }
  632. /* end of install.c */