Annotation of freem/src/init.c, revision 1.1
1.1 ! snw 1: /*
! 2: * *
! 3: * * *
! 4: * * *
! 5: * ***************
! 6: * * * * *
! 7: * * MUMPS *
! 8: * * * * *
! 9: * ***************
! 10: * * *
! 11: * * *
! 12: * *
! 13: *
! 14: * init.c
! 15: * FreeM initialization
! 16: *
! 17: *
! 18: * Author: Serena Willis <jpw@coherent-logic.com>
! 19: * Copyright (C) 1998 MUG Deutschland
! 20: * Copyright (C) 2020 Coherent Logic Development LLC
! 21: *
! 22: *
! 23: * This file is part of FreeM.
! 24: *
! 25: * FreeM is free software: you can redistribute it and/or modify
! 26: * it under the terms of the GNU Affero Public License as published by
! 27: * the Free Software Foundation, either version 3 of the License, or
! 28: * (at your option) any later version.
! 29: *
! 30: * FreeM is distributed in the hope that it will be useful,
! 31: * but WITHOUT ANY WARRANTY; without even the implied warranty of
! 32: * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
! 33: * GNU Affero Public License for more details.
! 34: *
! 35: * You should have received a copy of the GNU Affero Public License
! 36: * along with FreeM. If not, see <https://www.gnu.org/licenses/>.
! 37: *
! 38: **/
! 39:
! 40: #include <stdio.h>
! 41: #include <stdlib.h>
! 42: #include <string.h>
! 43: #include <unistd.h>
! 44: #include <limits.h>
! 45: #include <sys/types.h>
! 46: #include <sys/stat.h>
! 47: #include <pwd.h>
! 48: #include <time.h>
! 49: #include <errno.h>
! 50: #include <sys/ioctl.h>
! 51: #include <termios.h>
! 52:
! 53: #include "config.h"
! 54:
! 55: #if defined(HAVE_MWAPI_MOTIF)
! 56: # include <Xm/Xm.h>
! 57: #endif
! 58:
! 59: #include "mpsdef.h"
! 60: #include "transact.h"
! 61: #include "namespace.h"
! 62: #include "events.h"
! 63: #include "mdebug.h"
! 64: #include "shmmgr.h"
! 65: #include "locktab.h"
! 66: #include "jobtab.h"
! 67: #include "datatypes.h"
! 68: #include "objects.h"
! 69:
! 70: #ifdef HAVE_LIBREADLINE
! 71: # if defined(HAVE_READLINE_READLINE_H)
! 72: # include <readline/readline.h>
! 73: # elif defined(HAVE_READLINE_H)
! 74: # include <readline.h>
! 75: # else /* !defined(HAVE_READLINE_H) */
! 76: extern char *readline ();
! 77: # endif /* !defined(HAVE_READLINE_H) */
! 78: char *cmdline = NULL;
! 79: #else /* !defined(HAVE_READLINE_READLINE_H) */
! 80: /* no readline */
! 81: #endif /* HAVE_LIBREADLINE */
! 82:
! 83: #ifdef HAVE_READLINE_HISTORY
! 84: # if defined(HAVE_READLINE_HISTORY_H)
! 85: # include <readline/history.h>
! 86: # elif defined(HAVE_HISTORY_H)
! 87: # include <history.h>
! 88: # else /* !defined(HAVE_HISTORY_H) */
! 89: extern void add_history ();
! 90: extern int write_history ();
! 91: extern int read_history ();
! 92: # endif /* defined(HAVE_READLINE_HISTORY_H) */
! 93: /* no history */
! 94: #endif /* HAVE_READLINE_HISTORY */
! 95:
! 96: #if defined(HAVE_WIRINGPI_H)
! 97: # include <wiringPi.h>
! 98: #endif
! 99:
! 100: #if !defined(PATH_MAX) && defined(_SCO_DS)
! 101: # define PATH_MAX 4096
! 102: #endif
! 103:
! 104: #if !defined(PATH_MAX) && defined(__gnu_hurd__)
! 105: # define PATH_MAX 1024
! 106: #endif
! 107:
! 108: #if !defined(PATH_MAX) && defined(__sun__)
! 109: # include <limits.h>
! 110: #endif
! 111:
! 112: #if defined(__NetBSD__) || defined(__FreeBSD__) || defined(__OpenBSD__)
! 113: # include <sys/syslimits.h>
! 114: #endif
! 115:
! 116: #define SHMKEY 0x990120
! 117: #define SHMSIZ 1048576
! 118:
! 119: void init_process(void);
! 120: void init_devtable(void);
! 121: void init_signals(void);
! 122: void init_timezone(void);
! 123: void init_freem_path(void);
! 124:
! 125: #if defined(HAVE_LIBREADLINE)
! 126: void init_readline(void);
! 127: #endif
! 128:
! 129: void init_execution_context(void);
! 130: void init_io(void);
! 131: void init_random_number(void);
! 132: void init_ztrap(void);
! 133: void init_ssvn(void);
! 134: void init_terminal(void);
! 135: void init_estack(void);
! 136:
! 137: void init_mwapi(void);
! 138:
! 139: short init (char *namespace_name)
! 140: {
! 141: short retval;
! 142:
! 143: init_process ();
! 144: init_devtable ();
! 145: init_signals ();
! 146: init_freem_path ();
! 147: init_timezone ();
! 148:
! 149: #if defined(HAVE_LIBREADLINE)
! 150: init_readline ();
! 151: #endif
! 152:
! 153: init_execution_context ();
! 154:
! 155: if (run_daemon == FALSE) {
! 156: init_io ();
! 157: }
! 158:
! 159: init_random_number ();
! 160: init_ztrap ();
! 161:
! 162: retval = shm_init (shm_init_size);
! 163:
! 164: if (retval == SHMS_GET_ERR) {
! 165: fprintf (stderr, "init: error initializing shared memory [errno %d]\r\n", errno);
! 166: exit (1);
! 167: }
! 168:
! 169: symtab_init ();
! 170: tp_init ();
! 171:
! 172: set_namespace (namespace_name, FALSE);
! 173:
! 174: if (first_process) {
! 175: fprintf (stderr, "init: we are the first process in the environment (pid %d)\r\n", pid);
! 176: }
! 177:
! 178: if (first_process) fprintf (stderr, "init: initializing job table\r\n");
! 179: jobtab_init ();
! 180:
! 181: if (first_process) fprintf (stderr, "init: adding job to job table\r\n");
! 182: job_init (FALSE);
! 183:
! 184: if (first_process) fprintf (stderr, "init: initializing structured system variables\r\n");
! 185: init_ssvn ();
! 186:
! 187: if (first_process) fprintf (stderr, "init: initializing terminal\r\n");
! 188: init_terminal ();
! 189:
! 190: if (first_process) fprintf (stderr, "init: initializing asynchronous events\r\n");
! 191: evt_init ();
! 192:
! 193: if (first_process) fprintf (stderr, "init: initializing debugger\r\n");
! 194: dbg_init ();
! 195:
! 196: if (first_process) fprintf (stderr, "init: initializing error stack\r\n");
! 197: init_estack();
! 198:
! 199: etrap[0] = EOL;
! 200: ecode[0] = EOL;
! 201: estack = 0;
! 202:
! 203: init_mwapi();
! 204:
! 205: if (merr () == OK) {
! 206: return TRUE;
! 207: }
! 208:
! 209: return FALSE;
! 210: }
! 211:
! 212: void init_process (void)
! 213: {
! 214: pid = getpid (); /* get $J = process ID */
! 215: umask (0); /* protection bits mask to full rights */
! 216: snprintf (fp_conversion, 9, "%%.%df\201", DBL_DIG);
! 217:
! 218: if (fp_mode) {
! 219: zprecise = DBL_DIG;
! 220: }
! 221: else {
! 222: zprecise = 100;
! 223: }
! 224: }
! 225:
! 226: void init_devtable (void)
! 227: {
! 228: register int i;
! 229: register int j;
! 230:
! 231: for (j = 0; j <= MAXDEV; j++) { /* init. translation tables */
! 232:
! 233: for (i = 0; i < 256; i++) {
! 234: G0I[j][i] = (char) i;
! 235: G0O[j][i] = (char) i;
! 236: G1I[j][i] = (char) i;
! 237: G1O[j][i] = (char) i;
! 238: }
! 239:
! 240: G0I[j][UNSIGN (EOL)] = NUL;
! 241: G0O[j][UNSIGN (EOL)] = NUL;
! 242: G1I[j][UNSIGN (EOL)] = NUL;
! 243: G1O[j][UNSIGN (EOL)] = NUL;
! 244: G0I[j][UNSIGN (DELIM)] = NUL;
! 245: G0O[j][UNSIGN (DELIM)] = NUL;
! 246: G1I[j][UNSIGN (DELIM)] = NUL;
! 247: G1O[j][UNSIGN (DELIM)] = NUL;
! 248: G0I[j][256] = EOL;
! 249: G0O[j][256] = EOL;
! 250: G1I[j][256] = EOL;
! 251: G1O[j][256] = EOL;
! 252:
! 253: }
! 254:
! 255: #ifdef SCO
! 256: #ifndef HACK_NOXLATE
! 257: G0I[HOME][245] = 64;
! 258: G0O[HOME][64] = 245; /* Paragraph */
! 259: G0I[HOME][142] = 91;
! 260: G0O[HOME][91] = 142; /* A umlaut */
! 261: G0I[HOME][153] = 92;
! 262: G0O[HOME][92] = 153; /* O umlaut */
! 263: G0I[HOME][154] = 93;
! 264: G0O[HOME][93] = 154; /* U umlaut */
! 265: G0I[HOME][132] = 123;
! 266: G0O[HOME][123] = 132; /* a umlaut */
! 267: G0I[HOME][148] = 124;
! 268: G0O[HOME][124] = 148; /* o umlaut */
! 269: G0I[HOME][129] = 125;
! 270: G0O[HOME][125] = 129; /* u umlaut */
! 271: G0I[HOME][225] = 126;
! 272: G0O[HOME][126] = 225; /* sharp s */
! 273: #endif/*HACK_NOXLATE*/
! 274:
! 275: /* DEC Special graphics */
! 276: G1I[HOME][254] = 96;
! 277: G1O[HOME][96] = 254; /* diamond */
! 278: G1I[HOME][176] = 97;
! 279: G1O[HOME][97] = 176; /* checker board */
! 280: G1I[HOME][241] = 99;
! 281: G1O[HOME][99] = 241; /* FF */
! 282: G1I[HOME][242] = 100;
! 283: G1O[HOME][100] = 242; /* CR */
! 284: G1I[HOME][243] = 101;
! 285: G1O[HOME][101] = 243; /* LF */
! 286: G1I[HOME][248] = 102;
! 287: G1O[HOME][102] = 248; /* degree sign */
! 288: G1I[HOME][241] = 103;
! 289: G1O[HOME][103] = 241; /* plus minus */
! 290: G1I[HOME][244] = 104;
! 291: G1O[HOME][104] = 244; /* NL */
! 292: G1I[HOME][251] = 105;
! 293: G1O[HOME][105] = 251; /* VT */
! 294: G1I[HOME][217] = 106;
! 295: G1O[HOME][106] = 217; /* lower right corner */
! 296: G1I[HOME][191] = 107;
! 297: G1O[HOME][107] = 191; /* upper right corner */
! 298: G1I[HOME][218] = 108;
! 299: G1O[HOME][108] = 218; /* upper left corner */
! 300: G1I[HOME][192] = 109;
! 301: G1O[HOME][109] = 192; /* lower left corner */
! 302: G1I[HOME][197] = 110;
! 303: G1O[HOME][110] = 197; /* cross */
! 304: G1I[HOME][200] = 111;
! 305: G1O[HOME][111] = 200; /* linescan 5 */
! 306: G1I[HOME][201] = 112;
! 307: G1O[HOME][112] = 201; /* linescan 4 */
! 308: G1I[HOME][196] = 113;
! 309: G1O[HOME][113] = 196; /* linescan 3 */
! 310: G1I[HOME][202] = 114;
! 311: G1O[HOME][114] = 202; /* linescan 2 */
! 312: G1I[HOME][203] = 115;
! 313: G1O[HOME][115] = 203; /* linescan 1 */
! 314: G1I[HOME][195] = 116;
! 315: G1O[HOME][116] = 195; /* left junction */
! 316: G1I[HOME][180] = 117;
! 317: G1O[HOME][117] = 180; /* right junction */
! 318: G1I[HOME][193] = 118;
! 319: G1O[HOME][118] = 193; /* lower junction */
! 320: G1I[HOME][194] = 119;
! 321: G1O[HOME][119] = 194; /* upper junction */
! 322: G1I[HOME][179] = 120;
! 323: G1O[HOME][120] = 179; /* vertival bar */
! 324: G1I[HOME][243] = 121;
! 325: G1O[HOME][121] = 243; /* lower equals */
! 326: G1I[HOME][242] = 122;
! 327: G1O[HOME][122] = 242; /* greater equals */
! 328: G1I[HOME][227] = 123;
! 329: G1O[HOME][123] = 227; /* pi */
! 330: G1I[HOME][246] = 124;
! 331: G1O[HOME][124] = 246; /* not equals */
! 332: G1I[HOME][128] = 125;
! 333: G1O[HOME][125] = 128; /* euro sign */
! 334: G1I[HOME][250] = 126;
! 335: G1O[HOME][126] = 250; /* centered dot */
! 336: #endif /* SCO */
! 337: }
! 338:
! 339: void init_signals (void)
! 340: {
! 341: sig_init ();
! 342: }
! 343:
! 344: void init_timezone (void)
! 345: {
! 346:
! 347: struct tm lt;
! 348: struct tm gt;
! 349:
! 350: unsigned long gmt;
! 351: unsigned long lmt;
! 352:
! 353: long clock;
! 354:
! 355: #ifdef __CYGWIN__
! 356:
! 357: tzset (); /* may be required in order */
! 358: /* to guarantee _timezone set */
! 359: #else
! 360:
! 361: clock = time (0L);
! 362: lt = *localtime (&clock);
! 363: gt = *gmtime (&clock);
! 364:
! 365: /* This is awkward but I think it is portable: steve_morris */
! 366: gmt = gt.tm_year * 365;
! 367: gmt = (gmt + gt.tm_yday) * 24;
! 368: gmt = (gmt + gt.tm_hour) * 60;
! 369: gmt = (gmt + gt.tm_min);
! 370:
! 371: lmt = lt.tm_year * 365;
! 372: lmt = (lmt + lt.tm_yday) * 24;
! 373: lmt = (lmt + lt.tm_hour) * 60;
! 374: lmt = (lmt + lt.tm_min);
! 375:
! 376: FreeM_timezone = (gmt - lmt) * 60;
! 377: tzoffset = -FreeM_timezone;
! 378:
! 379: #endif /* __CYGWIN__ */
! 380:
! 381:
! 382: }
! 383:
! 384: void init_freem_path (void)
! 385: {
! 386:
! 387: if((freem_path = malloc(PATH_MAX + 1)) == NULL) {
! 388: fprintf(stderr, "Can't allocate freem_path. Exiting.");
! 389:
! 390: exit(1);
! 391: }
! 392:
! 393: freem_path[0] = NUL;
! 394:
! 395: /* check where I'm being executed from */
! 396: #ifdef __linux__
! 397: readlink ("/proc/self/exe", freem_path, PATH_MAX);
! 398: #endif
! 399: #ifdef __FreeBSD__
! 400: readlink ("/proc/curproc/file", freem_path, PATH_MAX);
! 401: #endif
! 402: #ifdef __sun
! 403: readlink ("/proc/self/path/a.out", freem_path, PATH_MAX);
! 404: #endif
! 405:
! 406: if(freem_path[0] == NUL) {
! 407: /* we don't know where we came from */
! 408: }
! 409:
! 410: getcwd (curdir, PATHLEN);
! 411: stcnv_c2m (curdir);
! 412:
! 413: }
! 414:
! 415: #if defined(HAVE_LIBREADLINE)
! 416: void init_readline (void)
! 417: {
! 418: uid_t uid = geteuid ();
! 419: struct passwd *pw = getpwuid (uid);
! 420: char *pw_buf;
! 421:
! 422: pw_buf = (char *) calloc (strlen(pw->pw_dir) + 1, sizeof(char));
! 423: strcpy (pw_buf, pw->pw_dir);
! 424:
! 425: snprintf (history_file, 256, "%s/.freem_history", pw_buf);
! 426:
! 427: free (pw_buf);
! 428:
! 429: using_history ();
! 430: read_history (history_file);
! 431: }
! 432: #endif
! 433:
! 434: void init_execution_context (void)
! 435: {
! 436: register int i;
! 437:
! 438: obj_init ();
! 439:
! 440: merr_clear ();
! 441:
! 442: codptr = code;
! 443: code[0] = EOL; /* init code_pointer */
! 444: partition = calloc ((unsigned) (PSIZE + 2), 1);
! 445:
! 446: if (partition == NULL) exit (2); /* could not allocate stuff... */
! 447:
! 448: for (i = 0; i < MAXNO_OF_RBUF; i++) {
! 449: rbuf_flags[i].standard = standard;
! 450: }
! 451:
! 452: for (i = 0; i < NESTLEVLS; i++) {
! 453: extr_types[i] = DT_STRING;
! 454: }
! 455:
! 456: symlen = PSIZE;
! 457: s = &partition[PSIZE] - 256; /* pointer to symlen_offset */
! 458: argptr = partition; /* pointer to beg of tmp-storage */
! 459:
! 460: svntable = calloc ((unsigned) (UDFSVSIZ + 1), 1);
! 461: if (svntable == NULL) exit (2); /* could not allocate stuff... */
! 462:
! 463: svnlen = UDFSVSIZ; /* begin of udf_svn_table */
! 464: buff = calloc ((unsigned) NO_OF_RBUF * (unsigned) PSIZE0, 1); /* routine buffer pool */
! 465: if (buff == NULL) exit (2); /* could not allocate stuff... */
! 466:
! 467:
! 468: newstack = calloc ((unsigned) NSIZE, 1);
! 469: if (newstack == NULL) exit (2); /* could not allocate stuff... */
! 470:
! 471: #ifdef DEBUG_NEWPTR
! 472: printf("Allocating newptr stack...\r\n");
! 473: #endif
! 474:
! 475: newptr = newstack;
! 476: newlimit = newstack + NSIZE - 1024;
! 477:
! 478:
! 479: namstck = calloc ((unsigned) NESTLEVLS * 13, 1);
! 480: if (namstck == NULL) exit (2); /* could not allocate stuff... */
! 481:
! 482: *namstck = EOL;
! 483: *(namstck + 1) = EOL;
! 484: namptr = namstck; /* routine name stack pointer */
! 485: framstck = calloc ((unsigned) NESTLEVLS * 256, 1);
! 486: if (framstck == NULL) exit (2); /* could not allocate stuff... */
! 487:
! 488: *framstck = EOL;
! 489: *(framstck + 1) = EOL;
! 490: dofrmptr = framstck; /* DO_frame stack pointer */
! 491: cmdstack = calloc ((unsigned) NESTLEVLS * 256, 1);
! 492: if (cmdstack == NULL) exit (2); /* could not allocate stuff... */
! 493:
! 494: cmdptr = cmdstack; /* command stack */
! 495:
! 496: rouend = rouins = rouptr = buff;
! 497: roucur = buff + (NO_OF_RBUF * PSIZE0 + 1);
! 498: *rouptr = EOL;
! 499: *(rouptr + 1) = EOL;
! 500: *(rouptr + 2) = EOL;
! 501:
! 502: err_suppl[0] = EOL; /* empty out supplemental error info */
! 503: }
! 504:
! 505: void init_estack (void)
! 506: {
! 507: stcpy (merr_stack[0].PLACE, "xecline()\201");
! 508: }
! 509:
! 510: #if defined(HAVE_MWAPI_MOTIF)
! 511: void init_mwapi (void)
! 512: {
! 513: /*
! 514: if (getenv("DISPLAY") != NULL) {
! 515: gtk_init (0, NULL);
! 516: }
! 517: */
! 518: //TODO: init Motif/libXt
! 519: }
! 520: #else
! 521: void init_mwapi (void)
! 522: {
! 523: return;
! 524: }
! 525: #endif
! 526:
! 527: void init_io (void)
! 528: {
! 529: register int i;
! 530:
! 531: /* initialize screen */
! 532: setbuf (stdin, NULL); /* no input buffering */
! 533: glvnflag.all = 0L;
! 534: stcpy (buff, "\201");
! 535: writeHOME (buff);
! 536: sq_modes[0] = '+';
! 537: for (i = 0; i <= MAXDEV; ug_buf[i++][0] = EOL); /* init read-buffers */
! 538:
! 539: crlf[HOME] = frm_filter;
! 540:
! 541: if (hardcopy) zbreakon = ENABLE; /* enable CTRL/B */
! 542:
! 543: set_io (MUMPS); /* set i/o parameters */
! 544:
! 545: #if !defined(__AMIGA)
! 546: if (ttyname (HOME)) { /* for $IO of HOME */
! 547: strcpy (dev[HOME], ttyname (HOME));
! 548: dev[HOME][strlen (dev[HOME])] = EOL;
! 549: }
! 550: else {
! 551: dev[HOME][0] = EOL; /* ...we are in a pipe */
! 552: }
! 553: #else
! 554: strcpy (dev[HOME], "CONSOLE:");
! 555: #endif
! 556:
! 557: /* init function keys */
! 558: for (i = 0; i < 44; zfunkey[i++][0] = EOL);
! 559: }
! 560:
! 561: void init_random_number (void)
! 562: {
! 563:
! 564: srand (time (NULL));
! 565:
! 566: if ((nrandom = time (0L) * getpid ()) < 0) {
! 567: nrandom = (-nrandom);
! 568: }
! 569:
! 570: }
! 571:
! 572: void init_ztrap (void)
! 573: {
! 574:
! 575: if (frm_filter) {
! 576: ztrap[0][0] = EOL; /* no default ztrap for filters */
! 577: }
! 578: else if (startuprou[0] == '^') {
! 579: stcpy (ztrap[0], startuprou);
! 580: }
! 581: else {
! 582: stcpy (ztrap[0], "^%SYSINIT\201");
! 583: }
! 584:
! 585: /* $ZT to be xecuted on startup */
! 586:
! 587: stcpy (ztrap[NESTLEVLS + 1], ztrap[0]); /* DSM V.2 error trapping */
! 588:
! 589: }
! 590:
! 591: void init_ssvn(void)
! 592: {
! 593: ssvn_job_update ();
! 594: ssvn_display_update ();
! 595: ssvn_routine_update ();
! 596: ssvn_library_update ();
! 597: if (first_process) ssvn_system_update ();
! 598: }
! 599:
! 600: void init_terminal(void)
! 601: {
! 602: xpos[HOME] = 80;
! 603: ypos[HOME] = 24;
! 604: }
! 605:
! 606: void reset_terminal(void)
! 607: {
! 608: struct termio tpara;
! 609:
! 610: ioctl (0, TCGETA, &tpara);
! 611:
! 612: tpara.c_lflag |= (ECHO | ICANON); /* enable echo/no cbreak mode */
! 613: tpara.c_iflag |= ICRNL; /* cr-lf mapping */
! 614: tpara.c_oflag |= ONLCR; /* cr-lf mapping */
! 615: tpara.c_cc[VMIN] = EOT;
! 616: tpara.c_cc[VTIME] = -1;
! 617:
! 618: ioctl (0, TCSETA, &tpara);
! 619: }
! 620:
! 621: void cleanup (void)
! 622: {
! 623: char k_buf[256];
! 624: int ch;
! 625:
! 626: /* remove this job's entry from ^$JOB SSVN */
! 627: snprintf (k_buf, 255, "^$JOB\202%d\201", pid);
! 628: symtab_shm (kill_sym, k_buf, " \201");
! 629:
! 630: reset_terminal ();
! 631:
! 632: if (tp_level > 0) {
! 633:
! 634: if (direct_mode == TRUE) {
! 635: fprintf (stderr, "UNCOMMITTED TRANSACTIONS EXIST:\n\n");
! 636: tp_tdump ();
! 637: set_io (UNIX);
! 638: fprintf (stderr, "\nWould you like to c)ommit or r)ollback the above transactions and their operations? ($TLEVEL = %d) ", tp_level);
! 639:
! 640: for (;;) {
! 641: ch = fgetc (stdin);
! 642:
! 643: if (ch == 'c' || ch == 'C') {
! 644: while (tp_level > 0) tp_tcommit ();
! 645:
! 646: fprintf (stderr, "\n\nTransactions have been committed.\n");
! 647:
! 648: break;
! 649: }
! 650: else if (ch == 'r' || ch == 'R') {
! 651: tp_trollback (tp_level);
! 652:
! 653: fprintf (stderr, "\n\nTransactions have been rolled back.\n");
! 654:
! 655: break;
! 656: }
! 657: else {
! 658: fprintf (stderr, "\n\nInvalid input '%c'. Must choose c)ommit or r)ollback.\n", ch);
! 659: }
! 660: }
! 661: }
! 662: else {
! 663: fprintf (stderr, "Uncommitted transactions exist. Rolling back.\n");
! 664: tp_trollback (tp_level);
! 665: }
! 666: }
! 667:
! 668: #if defined(HAVE_LIBREADLINE)
! 669: write_history (history_file);
! 670: #endif
! 671:
! 672: locktab_unlock_all ();
! 673: job_remove (pid);
! 674:
! 675: shm_exit ();
! 676:
! 677: if (run_daemon == TRUE) {
! 678:
! 679: if (pid_fd != -1) {
! 680: lockf (pid_fd, F_ULOCK, 0);
! 681: close (pid_fd);
! 682: }
! 683:
! 684: if (pid_file_path != NULL) {
! 685: unlink (pid_file_path);
! 686: }
! 687:
! 688: }
! 689:
! 690:
! 691:
! 692: free (buff); /* free previously allocated space */
! 693: free (svntable);
! 694: if (partition) free (partition);
! 695: if (apartition) free (apartition);
! 696:
! 697:
! 698: free (newstack);
! 699:
! 700:
! 701: if (v22size) free (v22ali);
! 702:
! 703: return;
! 704: } /* end of cleanup */
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>