123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695 |
- /*
- * Installation code for REDUCE in the Codemist CSL version. This
- * code is intended for use with MSDOS, and is compiled with
- * Zortech C release 3.0
- */
- /*
- * Copyright (C) 1992, Codemist Ltd.
- * This code has been modelled after suggestions from Jed Marti
- */
- /* Signature: 18bb67af 11-Oct-1993 */
- /*
- * Here, right at the top of this program, I give a list of the
- * files that are to be unpacked.
- */
-
- typedef struct arch
- {
- char *name;
- int loaded;
- } arch;
- static arch archives[8] =
- {
- {"EXE286.CAR", 0},
- {"EXE386.CAR", 0},
- {"IMG.CAR", 0},
- {"DOCS.CAR", 0},
- {"SRC1.CAR", 0},
- {"SRC2.CAR", 0},
- {"SRC3.CAR", 0},
- {"README.CAR", 0}
- };
- #include <stdio.h>
- #include <stdlib.h>
- #include <ctype.h>
- #include <signal.h>
- #include <setjmp.h>
- #include <errno.h>
- #include <string.h>
- #include <math.h>
- #include <time.h>
- #include <disp.h> /* The Zortech C fast screen-output package */
- #include <conio.h> /* unbuffered keyboard input etc */
- #include <direct.h> /* for mkdir() to create directories */
- #include <dos.h> /* so that I can select which drive is current */
- #define EXTRACT 1
- #define INSTALL 1
- static jmp_buf escape_buffer;
- static void ctrl_c_handler(int n)
- {
- longjmp(escape_buffer, 1);
- }
- static void show_txt(int row, int col, char *msg);
- #include "car.c" /* File decompression technology */
- #define white_on_blue 0x17
- #define white_on_black 0x07
- #define black_on_white 0x70
- /*
- * filep(dr, fn) - returns nonzero if drive dr and file name exists, 0 if not.
- */
- static FILE *filep(char *dr, char *fn)
- {
- char fname[256];
- FILE *fh;
- sprintf(fname, "%s%s\0", dr, fn);
- return fopen(fname, "rb");
- }
- /*
- * There is a smallish area in the middle of the screen that I
- * actually use here - clear it to the background colour (blue) that
- * I am using.
- */
- static void clear_work_area()
- {
- disp_fillbox(256*white_on_blue + ' ', 7, 11, 18, 68);
- }
- static void show_txt(int row, int col, char *msg)
- {
- int c, l = col;
- disp_move(row, col);
- while ((c = *msg++) != 0)
- { if (c == '\n')
- { while (l <= 68)
- { disp_putc(' ');
- l++;
- }
- disp_move(++row, col);
- l = col;
- }
- else
- { disp_putc(c);
- l++;
- }
- }
- }
- /*
- * This routine asks the use a question. The question is passed as
- * MSG and it will ask the user to type in a value. The default
- * will be provided in VAL, and that will be what is used if the user
- * just hits ENTER. maxlen indicates the size of the VAL buffer, so
- * that (with luck) I can avoid overwriting things. The message HELP
- * is displayed so that the user understands what the question is
- * supposed to be about. Returns non-zero if user wants to give up.
- */
- static int request_string(int maxlen, char *msg, char *help,
- char *val, char *dflt)
- {
- int i, j, hwm;
- clear_work_area();
- disp_setattr(white_on_black);
- show_txt(8, 11, msg);
- show_txt(12, 11, help);
- disp_setattr(black_on_white);
- strcpy(val, dflt);
- show_txt(10, 16, val);
- hwm = i = strlen(val);
- disp_move(10, 16+i);
- disp_flush();
- for (;;)
- { int c = getch();
- if (c == 0) c = 0x100 + getch(); /* Extended character */
- switch (c)
- {
- case 0x03:
- case 0x1b: /* ^C and ESC */
- return 1;
- case 0x08: /* backspace deletes char before the cursor */
- if (i == 0) continue;
- i--;
- case 0xff:
- case 0x100+'S': /* DELete deletes char at cursor */
- if (i == hwm) continue;
- disp_move(10, 16+i);
- hwm--;
- for (j=i; j<hwm; j++)
- { int c = val[j+1];
- disp_putc(c);
- val[j] = c;
- }
- disp_setattr(white_on_blue);
- disp_putc(' ');
- disp_setattr(black_on_white);
- disp_move(10, 16+i);
- disp_flush();
- continue;
- case '\r':
- case '\n': /* ENTER */
- hwm--; /* I truncate off any trailing blanks here */
- while (hwm!=0 && val[hwm]==' ') hwm--;
- val[hwm+1] = 0;
- return 0;
- case 0x100+'R': /* Insert */
- if (hwm >= maxlen) continue;
- hwm++;
- for (j=hwm; j>i; j--) val[j] = val[j-1];
- val[i] = ' ';
- disp_move(10, 16+i);
- for (j=i; j<hwm; j++) disp_putc(val[j]);
- disp_move(10, 16+i);
- continue;
- case 0x100+'K': /* Left arrow */
- if (i == 0) continue;
- i--;
- disp_move(10, 16+i);
- disp_flush();
- continue;
- case 0x100+'M': /* Right arrow */
- if (i == hwm) continue;
- i++;
- disp_move(10, 16+i);
- disp_flush();
- continue;
- case 0x100+'G': /* Home - reinstate default text */
- disp_move(10, 16);
- disp_setattr(white_on_blue);
- for (j=0; j<hwm; j++) disp_putc(' ');
- disp_setattr(black_on_white);
- strcpy(val, dflt);
- show_txt(10, 16, val);
- hwm = i = strlen(val);
- disp_move(10, 16+i);
- disp_flush();
- continue;
- default:
- if (c >= 0x100) continue; /* Ignore extended chars */
- if (i < maxlen)
- { val[i++] = c;
- disp_putc(c);
- disp_flush();
- if (i > hwm) hwm = i;
- }
- continue;
- }
- }
- }
- /*
- * request_char is like request_string, except that the input needed from
- * the user will be just one character long. The result is dumped into the
- * first character of VAL.
- */
- static int request_char(char *msg, char *help, char *val)
- {
- clear_work_area();
- disp_setattr(white_on_black);
- show_txt(8, 11, msg);
- show_txt(12, 11, help);
- disp_setattr(black_on_white);
- show_txt(10, 16, val);
- disp_move(10, 16);
- disp_flush();
- for (;;)
- { int c = getch();
- switch (c & 0x7f)
- {
- case 0x03:
- case 0x1b: /* ^C and ESC */
- return 1;
- case '\r':
- case '\n': /* ENTER */
- return 0;
- case 0x08:
- case 0x7f: /* backspace and DELete - mapped to blank */
- c = ' ';
- default:
- c = toupper(c);
- val[0] = c;
- disp_move(10, 16);
- disp_putc(c);
- disp_move(10, 16);
- disp_flush();
- continue;
- }
- break;
- }
- return 0;
- }
- /*
- * request_ok() displays the given message and just waits for the user
- * to press any (printing) key. It is used for diagnostic messages and
- * similar reports.
- */
- static void request_ok(char *help)
- {
- clear_work_area();
- disp_setattr(white_on_black);
- show_txt(8, 11, "Press a key (e.g. ENTER) to continue\n");
- show_txt(12, 11, help);
- disp_move(10, 16);
- disp_flush();
- (void)getch(); /* All characters behave the same here */
- }
- /*
- * This is an interface to code found in the #included file "car.c"
- * that contains the Codemist file compression code and archive
- * utility. It sets up various workspace and invokes file decompression.
- */
-
- static void decompress_files()
- {
- int32 i;
- for (i=0; i<PREDICTION_SIZE; i++)
- { prediction1[i] = ' ';
- prediction2[i] = '\n';
- }
- memcpy(current_frequencies, default_frequencies,
- sizeof(default_frequencies));
- CRC = 1;
- while (extract_files()) continue;
- { unsigned32 calculated_CRC = CRC;
- unsigned32 stored_CRC = get4(); /* WARNING: get4() changes CRC! */
- if (calculated_CRC != stored_CRC)
- fprintf(stderr, "Warning: CRC failure - archive may be corrupt\n");
- }
- fclose(archive_file);
- }
- static int install(int default_source)
- {
- int cpu = cputype(); /* Used to select version to install */
- char *xdir, sdrive[4], dest[48], yes_no[4];
- char msg[1000], fname[256];
- int i, l, c, something_found;
- int disable_exe = 0, want_exe = 1, want_doc = 1, want_src = 1;
- if (cpu < 2)
- { request_ok("Unsuitable computer - can not install REDUCE\n"
- "REDUCE needs at least an 80286 computer (and if you\n"
- "have an 80386 or 80486 a faster version will be\n"
- "installed for you). This machine seems to be an 8086\n"
- "or 80186. You will not be allowed to install\n"
- "executable binaries, but can load sources or docs\n");
- disable_exe = 1;
- }
- if (cpu < 3) archives[1].loaded = 1, xdir = "csl286";
- else archives[0].loaded = 1, xdir = "csl386";
-
- for (;;)
- { if (disable_exe) want_exe = 0;
- else
- { for (;;)
- { sprintf(yes_no, "Y");
- if (request_char(
- "Do you want to install executable binaries?\n",
- "If you select Y and type ENTER then executable binaries\n"
- "for REDUCE will be installed onto your hard disc. If you\n"
- "type N and then ENTER binaries will not be installed.\n"
- "The binary files involved will be called r35.exe and\n"
- "r35.img\n", yes_no)) goto abandon;
- c = toupper(yes_no[0]);
- if (c == 'Y' || c == 'N') break;
- }
- want_exe = (c == 'Y');
- }
- for (;;)
- { sprintf(yes_no, "Y");
- if (request_char("Do you want to install documentation?\n",
- "If you select Y and type ENTER then machine-readable\n"
- "documentation for REDUCE will be installed onto your hard\n"
- "disc. If you type N and then ENTER it will not be\n"
- "installed. The files involved will be placed in a\n"
- "subdirectory called \"DOC\" and most of them are formatted\n"
- "using the TeX text layout system\n", yes_no)) goto abandon;
- c = toupper(yes_no[0]);
- if (c == 'Y' || c == 'N') break;
- }
- want_doc = (c == 'Y');
- #ifdef PERSONAL
- want_src = 0;
- #else
- for (;;)
- { sprintf(yes_no, "Y");
- if (request_char("Do you want to install source files?\n",
- "If you select Y and type ENTER then all source files for\n"
- "REDUCE will be installed onto your hard disc. If you type\n"
- "N and then ENTER binaries will not be installed. The\n"
- "REDUCE source files will be put in a directory \"SRC\", and\n"
- "many further files relating to the underlying Lisp (CSL)\n"
- "will also appear\n", yes_no)) goto abandon;
- c = toupper(yes_no[0]);
- if (c == 'Y' || c == 'N') break;
- }
- want_src = (c == 'Y');
- #endif
- for (;;)
- { sprintf(yes_no, "Y");
- if (want_exe || want_doc || want_src)
- sprintf(msg, "You have chosen to install%s%s%s\n",
- (want_exe ? "\n Executable binaries" : ""),
- (want_doc ? "\n Documentation" : ""),
- (want_src ? "\n Source code" : ""));
- else sprintf(msg, "You seem not to want to install anything!\n");
- if (request_char("Proceed with installation?\n",
- msg, yes_no)) goto abandon;
- c = toupper(yes_no[0]);
- if (c == 'Y' || c == 'N') break;
- else if (c == 'Q') goto abandon;
- }
- if (c == 'N') continue; /* Give user a chance to try again */
- break;
- }
- if (!want_exe)
- archives[0].loaded = archives[1].loaded = archives[2].loaded = 1;
- if (!want_doc)
- archives[3].loaded = 1;
- if (!want_src)
- archives[4].loaded = archives[5].loaded = archives[6].loaded = 1;
- for (;;)
- { sdrive[0] = default_source;
- sdrive[1] = ':';
- sdrive[2] = 0;
- if (request_char("Floppy Drive Name\n",
- "This is the drive identifier where the Codemist Floppy\n"
- "disks will be mounted. Common values are A or B. If you\n"
- "want the default, just type the ENTER key. To change the\n"
- "value, enter a new letter, any of A-Z will be accepted.\n"
- "Any other key will cause the installation procedure to\n"
- "be aborted.", sdrive)) goto abandon;
- sdrive[0] = toupper(sdrive[0]);
- if (!('A' <= sdrive[0] && sdrive[0] <= 'Z')) goto abandon;
- select_destination:
- if (request_string(44, "Hard Disk Drive and Directory\n",
- "This is the drive and directory you wish to install\n"
- "Codemist REDUCE 3.4 on. If you want the default, just\n"
- "type the ENTER key. To change the value, enter a new\n"
- "letter. If the directory does not exist, it will be\n"
- "created for you.", dest, "C:\\REDUCE")) goto abandon;
- for (;;)
- { sprintf(yes_no, "Y");
- sprintf(msg, "Source Disk Drive %s\n"
- "Where to install %s\n"
- "Are these values correct (Y or N)\n"
- "Type Q or ^C to abandon installation\n", sdrive, dest);
- if (request_char("Proceed with installation?\n",
- msg, yes_no)) goto abandon;
- c = toupper(yes_no[0]);
- if (c == 'Y' || c == 'N') break;
- else if (c == 'Q') goto abandon;
- }
- if (c == 'N') continue; /* Give user a chance to try again */
- /*
- * Now I try to ensure that the directory indicated exists... First
- * I select the relevant drive as current, that is always supposing that
- * the user's response started off as "X:" for some X.
- */
- if (dest[0] != 0 && dest[1] == ':')
- { unsigned int want, ndrives, found;
- want = toupper(dest[0]) - 'A' + 1;
- dos_setdrive(want, &ndrives);
- /*
- * I check that the desired drive was indeed selected by checking which
- * drive is current after I (attempt to) make the selection.
- */
- dos_getdrive(&found);
- if (found != want) goto select_failed;
- }
- l = strlen(dest);
- for (i=1; i<=l; i++)
- { c = dest[i];
- if ((c == '\\' || c == 0) &&
- dest[i-1] != ':' && dest[i-1] != '\\')
- { int w;
- dest[i] = 0;
- w = mkdir(dest);
- if (w == 0 || /* created OK */
- errno == EACCES) /* already present */
- { dest[i] = c;
- continue;
- }
- yes_no[0] = 'Y'; yes_no[1] = 0;
- sprintf(msg, "The directory\n"
- "%s\n"
- "could not be created for you. Type Y/ENTER\n"
- "to select another destination, anything else to\n"
- "quit.\n", dest);
- if (request_char("Try again?\n", msg, yes_no)) goto abandon;
- if (toupper(yes_no[0]) != 'Y') goto abandon;
- else goto select_destination;
- }
- }
- /*
- * Now the destination directory ought to exist - select it as
- * current.
- * I will leave this drive and directory selected when I exit from
- * the installation process (even if installation fails)
- */
- if (chdir(dest) == 0) break;
- /*
- * If I fail to select the desired drive I give the user a chance to
- * select another.
- */
- select_failed:
- yes_no[0] = 'Y'; yes_no[1] = 0;
- sprintf(msg, "The directory\n"
- "%s\n"
- "could not be selected. Type Y/ENTER\n"
- "to select another destination, anything else to\n"
- "quit.\n", dest);
- if (request_char("Try again?\n", msg, yes_no)) goto abandon;
- if (toupper(yes_no[0]) != 'Y') goto abandon;
- else goto select_destination;
- }
- /*
- * I want two sub-directories created here - SLOW is used to hold
- * log files from the slow version of the system, while COUNTS holds
- * statistics that can indicate which (bytecoded) functions are being used
- * most heavily.
- */
- if (mkdir("slow") != 0 && errno != EACCES) goto installation_failed;
- if (mkdir("counts") != 0 && errno != EACCES) goto installation_failed;
- if (mkdir(xdir) != 0 && errno != EACCES) goto installation_failed;
- if (chdir(xdir) != 0) goto installation_failed;
- /*
- * Now I have the guts of the installation process - I have a list of
- * archives that I would like to find and will accept discs in any order
- * until I have found them all.
- */
- for (;;)
- {
- something_found = 0;
- look_for_another:
- for (i=0; i<8; i++)
- { if (archives[i].loaded) continue;
- goto another_to_load;
- }
- break; /* Installation complete */
- another_to_load:
- for (i=0; i<8; i++)
- { if (!archives[i].loaded)
- { archive_file = filep(sdrive, archives[i].name);
- if (archive_file == NULL) continue;
- something_found = 1;
- clear_work_area();
- sprintf(msg, "File %s to be uncompressed\n", archives[i].name);
- show_txt(9, 16, msg);
- decompress_files();
- archives[i].loaded = 1;
- goto look_for_another;
- }
- }
- if (something_found)
- request_ok(
- "Please load the next disc of the REDUCE distribution\n"
- "set in your floppy disc drive and type a character\n"
- "(e.g. ENTER). Type ^C to abort installation.\n");
- else request_ok(
- "*** No relevant files were found on that disc ***\n"
- "This may be because you are not loading the whole of\n"
- "REDUCE. Please load the next disc of the distribution\n"
- "set in your floppy disc drive and type a character\n"
- "(e.g. ENTER). Type ^C to abort installation.\n");
- }
- request_ok(
- "Installation complete. The directory with the REDUCE\n"
- "files will now be selected as current. Please check\n"
- "the file \"READ.ME\" for last-minute notes.\n");
- return EXIT_SUCCESS;
- installation_failed:
- request_ok(
- "The installation script was unable to create\n"
- "one of the files or directories needed for REDUCE.\n");
- return EXIT_FAILURE;
- abandon:
- request_ok("Installation of REDUCE incomplete.\n");
- return EXIT_FAILURE;
- }
- /*
- * The following procedure displays one line of text, centred in the
- * box I have drawn on the screen.
- */
- static void disp(int row, char *s)
- {
- int len = strlen(s);
- disp_move(row, 40-len/2);
- disp_puts(s);
- }
- int main(int argc, char *argv[])
- {
- int rc, vm = disp_getmode();
- int nrows, ncols, crow, ccol;
- unsigned short *ssave;
- /*
- * Firstly I grab the initial contents of the screen so that I can
- * restore it at the end.
- */
- disp_open();
- nrows = disp_numrows; ncols = disp_numcols;
- crow = disp_cursorrow; ccol = disp_cursorcol;
- ssave = (unsigned short *)malloc(nrows*ncols*sizeof(short));
- if (ssave == NULL)
- { disp_close();
- fprintf(stderr, "Insufficient memory for installation of REDUCE\n");
- exit(EXIT_FAILURE);
- }
- disp_peekbox(ssave, 0, 0, nrows-1, ncols-1);
- disp_close();
- /*
- * Now I grab the memory that will be needed for file decompression, so
- * that use of malloc() is all done early and thus out-of-memory
- * messages can be dealt with before anything else is messed up.
- */
- prediction1 = (unsigned char *)malloc((size_t)PREDICTION_SIZE);
- prediction2 = (unsigned char *)malloc((size_t)PREDICTION_SIZE);
- fast_decode_table =
- (int32 *)malloc((size_t)(FAST_DECODE_TABLESIZE*sizeof(int32)));
- char_being_coded = (int *)malloc((size_t)(2*CODE_SIZE*sizeof(int)));
- number_of_occurences = (int32 *)malloc((size_t)(2*CODE_SIZE*sizeof(int32)));
- character_index = (int *)malloc((size_t)(2*CODE_SIZE*sizeof(int)));
- corresponding_character = (int *)malloc((size_t)(CODE_SIZE*sizeof(int)));
- huffman_coded_bits =
- (unsigned32 *)malloc((size_t)(CODE_SIZE*sizeof(unsigned32)));
- length_of_code = (char *)malloc((size_t)CODE_SIZE);
- /*
- * I re-use the double-length array character_index[] as one of the
- * arrays needed for decoding...
- */
- sorted_huffman_codes = (unsigned32 *)&character_index[0];
- if (prediction1 == NULL ||
- prediction2 == NULL ||
- fast_decode_table == NULL ||
- char_being_coded == NULL ||
- number_of_occurences == NULL ||
- character_index == NULL ||
- corresponding_character == NULL ||
- huffman_coded_bits == NULL ||
- length_of_code == NULL)
- { fprintf(stderr, "Insufficient memory for installation of REDUCE\n");
- exit(EXIT_FAILURE);
- }
- /*
- * Now force myself into an 80*25 alphanumeric screen mode
- */
- if (vm != 3) disp_setmode(3);
- disp_open();
- /*
- * Clear the screen
- */
- disp_move(0, 0); disp_eeop();
- /*
- * Draw a box, with a border, for installation messages to live within
- */
- disp_fillbox(256*white_on_blue + ' ', 4, 9, 21, 70);
- disp_box(0, white_on_blue, 4, 9, 21, 70);
- disp_setattr(white_on_blue);
- #ifdef PERSONAL
- disp(6, "Personal R E D U C E 3.4.1");
- #else
- disp(6, "R E D U C E 3.4.1");
- #endif
- disp(19, "Codemist Ltd, Alta, Horsecombe Vale, Combe Down,");
- disp(20, "Bath BA2 5QR, England. Telephone/Fax: +44 225 837430");
- switch (setjmp(escape_buffer))
- {
- case 0:
- signal(SIGTERM, ctrl_c_handler);
- rc = install(argc > 0 ? toupper(argv[0][0]) : 'A');
- break;
- default:
- signal(SIGTERM, SIG_IGN);
- rc = EXIT_FAILURE;
- break;
- }
- /*
- * Re-instate the screen-mode that was active when I started
- */
- disp_close();
- if (vm != 3) disp_setmode(vm);
- /*
- * Restore previous screen contents and cursor position
- */
- disp_open();
- disp_pokebox(ssave, 0, 0, nrows-1, ncols-1);
- disp_move(crow, ccol);
- disp_flush();
- disp_close();
- exit(rc);
- return 0;
- }
- /* end of install.c */
|