Annotation of freem/src/mumps.c, revision 1.5

1.1       snw         1: /*
1.5     ! snw         2:  *   $Id: mumps.c,v 1.4 2025/03/09 19:50:47 snw Exp $
1.1       snw         3:  *    main module of freem
                      4:  *
                      5:  *  
1.3       snw         6:  *   Author: Serena Willis <snw@coherent-logic.com>
1.1       snw         7:  *    Copyright (C) 1998 MUG Deutschland
1.4       snw         8:  *    Copyright (C) 2020, 2025 Coherent Logic Development LLC
1.1       snw         9:  *
                     10:  *
                     11:  *   This file is part of FreeM.
                     12:  *
                     13:  *   FreeM is free software: you can redistribute it and/or modify
                     14:  *   it under the terms of the GNU Affero Public License as published by
                     15:  *   the Free Software Foundation, either version 3 of the License, or
                     16:  *   (at your option) any later version.
                     17:  *
                     18:  *   FreeM is distributed in the hope that it will be useful,
                     19:  *   but WITHOUT ANY WARRANTY; without even the implied warranty of
                     20:  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
                     21:  *   GNU Affero Public License for more details.
                     22:  *
                     23:  *   You should have received a copy of the GNU Affero Public License
                     24:  *   along with FreeM.  If not, see <https://www.gnu.org/licenses/>.
                     25:  *
1.5     ! snw        26:  *   $Log: mumps.c,v $
        !            27:  *   Revision 1.4  2025/03/09 19:50:47  snw
        !            28:  *   Second phase of REUSE compliance and header reformat
        !            29:  *
1.4       snw        30:  *
                     31:  * SPDX-FileCopyrightText:  (C) 2025 Coherent Logic Development LLC
                     32:  * SPDX-License-Identifier: AGPL-3.0-or-later
1.1       snw        33:  **/
                     34: 
                     35: #include <stdlib.h>
                     36: #include <stddef.h>
                     37: #include "mpsdef.h"
                     38: #include "errmsg.h"
                     39: #include "iniconf.h"
                     40: #include "namespace.h"
                     41: #include "transact.h"
                     42: #include "init.h"
                     43: #include "consttbl.h"
                     44: #include <setjmp.h>
                     45: #include <stdio.h>
                     46: #include <signal.h>
                     47: #include <time.h>
                     48: #include <unistd.h>
                     49: #include <sys/types.h>
                     50: #include <sys/wait.h>
                     51: #include <sys/stat.h>
                     52: #include <limits.h>
                     53: #include <sys/types.h>
                     54: #include <pwd.h>
                     55: #include <string.h>
                     56: #include <fcntl.h>
                     57: #include "version.h"
                     58: #include "shmmgr.h"
                     59: #include "jobtab.h"
                     60: #include <errno.h>
                     61: #include <pwd.h>
                     62: #include <grp.h>
                     63: 
                     64: 
                     65: #if defined(HAVE_GETOPT_H)
                     66: # include <getopt.h>
                     67: #endif
                     68: 
                     69: #if !defined(PATH_MAX) && defined(_SCO_DS)
                     70: # define PATH_MAX 1024
                     71: #endif
                     72: 
                     73: extern int xecline(int typ);
                     74: extern char *getenv(const char *name);
                     75: void freem_usage(void);
                     76: void freem_print_version(void);
                     77: void init_ztrap(void);
                     78: void m_log (int, const char *);
                     79: 
                     80: int main (int argc, char **argv, char **envp)
                     81: {
                     82:     pid_t fork_pid = 0;
                     83:     short dx_mode = 0;
                     84:     
                     85:     int c;
                     86:     int import_env = FALSE;
                     87:     short skip_init = 0;
                     88:     
                     89:     int option_index = 0;
                     90: 
                     91:     char dx_mcode[512];
                     92:     char startup_routine[256];
                     93:     short routine_mode;
                     94:     char m_dialect[50];
                     95: 
                     96:     char nsnbuf[256];
                     97: 
                     98:     char d_username[40];
                     99:     char d_groupname[40];
                    100:     struct group *d_grp;
                    101:     struct passwd *d_user;
                    102:     gid_t d_gid;
                    103:     uid_t d_uid;
                    104: 
                    105:     short custom_user = FALSE;
                    106:     short custom_group = FALSE;
                    107:     
                    108: #if defined(HAVE_GETOPT_LONG)
                    109:     struct option long_options[] = {       
                    110:         {"help", no_argument, 0, 'h'},
                    111:         {"filter", no_argument, &frm_filter, TRUE},
                    112:         {"standard", required_argument, 0, 's'},
                    113:         {"import", no_argument, &import_env, TRUE},
                    114:         {"quiet", no_argument, &quiet_mode, TRUE},
                    115:         {"restricted", no_argument, &restricted_mode, TRUE},
                    116:         {"routine", required_argument, 0, 'r'},
                    117:         {"namespace", required_argument, 0, 'n'},
                    118:         {"environment", required_argument, 0, 'e'},
                    119:        {"version", no_argument, 0, 'v'},
                    120:        {"execute", required_argument, 0, 'x'},
                    121:         {"daemon", no_argument, 0, 'd'},
                    122:         {"nofork", no_argument, 0, 'k'},
                    123:         {"pidfile", required_argument, 0, 'p'},
                    124:         {"shmsize", required_argument, 0, 'S'},
                    125:         {"user", required_argument, 0, 'u'},
                    126:         {"group", required_argument, 0, 'g'},
                    127:         {0, 0, 0, 0}
                    128:     };
                    129: #endif
                    130:     
                    131:     char **env;      
                    132:     char *varname = (char *) calloc(STRLEN, sizeof(char));
                    133:     char *varval = (char *) calloc(STRLEN, sizeof(char));
                    134: 
                    135:     char *symname = (char *) calloc(STRLEN, sizeof(char));
                    136:     char *symval = (char *) calloc(STRLEN, sizeof(char));
                    137: 
                    138:     int namelen;
                    139:     int vallen;
                    140: 
                    141:     char cli_rtn_path[PATH_MAX] = {0};
                    142:     char *cli_rtn_file;
                    143:     char cli_rtn_name[256];
                    144: 
                    145:     routine_mode = FALSE;
                    146:     strcpy (m_dialect, "FREEM");
                    147:     
                    148:     m_argc = argc;                     /* save arguments count     */
                    149:     m_argv = argv;                     /* save arguments string    */
                    150:     m_envp = envp;                     /* save environment pointer */
                    151: 
                    152:     strncpy (shm_env, "DEFAULT", 8);    /* establish default environment name */
                    153:     
                    154:     strncpy (d_username, "freem", 40);
                    155:     strncpy (d_groupname, "freem", 40);
                    156:     
                    157:     strcpy (zb, argv[0]);              /* name with which mumps has been called */
                    158:     stcnv_c2m (zb);
                    159:     strcpy (stack0, argv[0]);
                    160:     stcnv_c2m (stack0);
                    161: 
                    162: 
                    163:     /* strncpy (config_file, SYSCONFDIR"/freem.conf", 100); */
                    164: 
                    165: 
                    166: 
                    167: #if defined(HAVE_GETOPT_LONG)
                    168:     
                    169:     extern char *optarg;
                    170:     extern int optind, optopt;
                    171:     
                    172:     while (1) {
                    173:         
                    174:         c = getopt_long (argc, argv, "hsfiqRr:n:e:vx:dkpS:u:g:", long_options, &option_index);
                    175: 
                    176:         if (c == -1) break;
                    177:        if (c == '?') freem_usage ();
                    178:        
                    179:         switch (c) {
                    180:             case 'h':
                    181:                 freem_usage ();
                    182:                 break;
                    183: 
                    184:             case 'f':
                    185:                 frm_filter = TRUE;
                    186:                 break;
                    187: 
                    188:             case 'i':
                    189:                 import_env = TRUE;
                    190:                 break;
                    191: 
                    192:             case 'q':
                    193:                 quiet_mode = TRUE;
                    194:                 break;
                    195: 
                    196:             case 'e': /* specify FreeM environment */
                    197:                 strncpy (shm_env, optarg, 255);
                    198:                 break;
                    199: 
                    200:             case 'R':
                    201:                 restricted_mode = TRUE;
                    202:                 break;
                    203: 
                    204:             case 'r': /* startup routine */
                    205:                 direct_mode = FALSE;
                    206:                 usermode = 0;
                    207: 
                    208:                 strcpy (startup_routine, optarg);
                    209:                 startup_routine[strlen (startup_routine)] = '\201';
                    210: 
                    211:                 routine_mode = TRUE;
                    212:                 break;
                    213: 
                    214:             case 'n': /* namespace */
                    215:             {
                    216:                 if (validate_namespace (optarg) == TRUE) {                
                    217:                     strcpy (nsname, optarg);
                    218:                 }
                    219:                 else {
                    220:                     fprintf (stderr, "freem:  invalid namespace '%s'\n", optarg);
                    221:                     exit (1);
                    222:                 }
                    223:                 
                    224:                 break;
                    225:             }
                    226:            case 'v': /* version */
                    227:                freem_print_version ();
                    228:                break;
                    229: 
                    230:            case 'x': /* execute */
                    231:                direct_mode = FALSE;
                    232:                usermode = 0;
                    233:                dx_mode = 1;
                    234:                
                    235:                strncpy (dx_mcode, optarg, 512 - 1);
                    236: 
                    237:                stcnv_c2m (dx_mcode);
                    238: 
                    239:                break;
                    240: 
                    241:             case 's': /* --standard */
                    242: 
                    243:                 if (strcmp (optarg, "M77") == 0) {
                    244:                     standard = D_M77;
                    245:                     strcpy (m_dialect, "M 1977");
                    246:                 }
                    247:                 else if (strcmp (optarg, "M84") == 0) {
                    248:                     standard = D_M84;
                    249:                     strcpy (m_dialect, "M 1984");
                    250:                 }
                    251:                 else if (strcmp (optarg, "M90") == 0) {
                    252:                     standard = D_M90;
                    253:                     strcpy (m_dialect, "M 1990");
                    254:                 }
                    255:                 else if (strcmp (optarg, "M95") == 0) {
                    256:                     standard = D_M95;
                    257:                     strcpy (m_dialect, "M 1995");
                    258:                 }
                    259:                 else if (strcmp (optarg, "MDS") == 0) {
                    260:                     standard = D_MDS;
                    261:                     strcpy (m_dialect, "Millennium Draft Standard");
                    262:                 }
                    263:                 else if (strcmp (optarg, "M5") == 0) {
                    264:                     standard = D_M5;
                    265:                     strcpy (m_dialect, "M5");
                    266:                 }
                    267:                 else if (strcmp (optarg, "FREEM") == 0) {
                    268:                     standard = D_FREEM;
                    269:                     strcpy (m_dialect, "FREEM");
                    270:                 }
                    271:                 else {
                    272:                     freem_usage ();
                    273:                 }
                    274:                 
                    275:                 break;                
                    276: 
                    277:             case 'd': /* --daemon */
                    278:                 run_daemon = TRUE;                
                    279:                 break;
                    280: 
                    281:             case 'k': /* --nofork */
                    282:                nofork = TRUE;
                    283:                 break;
                    284: 
                    285:             case 'p': /* --pidfile */
                    286:                 pid_file_path = strdup (optarg);
                    287:                 break;
                    288: 
                    289:             case 'S': /* --shmsize */
                    290:                 shm_init_size = atol (optarg);
                    291:                 break;
                    292: 
                    293:             case 'u': /* --user */
                    294:                 strncpy (d_username, optarg, 40);
                    295:                 custom_user = TRUE;
                    296:                 break;
                    297: 
                    298:             case 'g': /* --group */
                    299:                 strncpy (d_groupname, optarg, 40);
                    300:                 custom_group = TRUE;
                    301:                 break;
                    302:                 
                    303: 
                    304:         } 
                    305: 
                    306:     }
                    307: 
                    308: #else
                    309:     
                    310:     {
                    311:         extern char *optarg;
                    312:         extern int optind, optopt;
                    313: 
                    314:         while ((c = getopt (argc, argv, "hsfiqRr:n:e:vx:dkpS:u:g:")) != -1) {
                    315: 
                    316:            if (c == '?') freem_usage ();
                    317:            
                    318:             switch (c) {
                    319: 
                    320:                case 'h':
                    321:                    freem_usage ();
                    322:                    break;
                    323:                
                    324:                 case 'f':
                    325:                     frm_filter = TRUE;
                    326:                     break;
                    327: 
                    328:                 case 'i':
                    329:                     import_env = TRUE;
                    330:                     break;
                    331: 
                    332:                 case 'q':
                    333:                     quiet_mode = TRUE;
                    334:                     break;
                    335: 
                    336:                 case 'e':  /* specify FreeM environment */
                    337:                     strncpy (shm_env, optarg, 255);
                    338:                     break;
                    339: 
                    340:                 case 'R':
                    341:                     restricted_mode = TRUE;
                    342:                     break;
                    343:                     
                    344:                 case 'r': /* startup routine */
                    345:                     direct_mode = FALSE;
                    346:                     usermode = 0;
                    347: 
                    348:                     strcpy (startup_routine, optarg);
                    349:                     startup_routine[strlen (startup_routine)] = '\201';
                    350: 
                    351:                     routine_mode = TRUE;
                    352:                     break;
                    353: 
                    354:                 case 'n': /* namespace */
                    355:                     strcpy (nsname, optarg);               
                    356:                     break;
                    357: 
                    358:                case 'v':
                    359:                    freem_print_version ();
                    360:                    break;
                    361: 
                    362:                case 'x': /* execute */
                    363:                    direct_mode = FALSE;
                    364:                    usermode = 0;
                    365:                    dx_mode = 1;
                    366:                    
                    367:                    strncpy (dx_mcode, optarg, 512 - 1);
                    368:                    
                    369:                    stcnv_c2m (dx_mcode);
                    370: 
                    371:                     break;
                    372:                     
                    373:                 case 'd': /* --daemon */
                    374:                     run_daemon = TRUE;
                    375:                     break;
                    376: 
                    377:                 case 'k': /* --nofork */
                    378:                     nofork = TRUE;
                    379:                     break;
                    380: 
                    381:                 case 'p': /* --pidfile */
                    382:                     pid_file_path = strdup (optarg);
                    383:                     break;
                    384: 
                    385:                 case 's': /* --standard */
                    386:                     
                    387:                     if (strcmp (optarg, "M77") == 0) {
                    388:                         standard = D_M77;
                    389:                         strcpy (m_dialect, "M 1977");
                    390:                     }
                    391:                     else if (strcmp (optarg, "M84") == 0) {
                    392:                         standard = D_M84;
                    393:                         strcpy (m_dialect, "M 1984");
                    394:                     }
                    395:                     else if (strcmp (optarg, "M90") == 0) {
                    396:                         standard = D_M90;
                    397:                         strcpy (m_dialect, "M 1990");
                    398:                     }
                    399:                     else if (strcmp (optarg, "M95") == 0) {
                    400:                         standard = D_M95;
                    401:                         strcpy (m_dialect, "M 1995");
                    402:                     }
                    403:                     else if (strcmp (optarg, "MDS") == 0) {
                    404:                         standard = D_MDS;
                    405:                         strcpy (m_dialect, "Millennium Draft Standard");
                    406:                     }
                    407:                     else if (strcmp (optarg, "M5") == 0) {
                    408:                         standard = D_M5;
                    409:                         strcpy (m_dialect, "M5");
                    410:                     }
                    411:                     else if (strcmp (optarg, "FREEM") == 0) {
                    412:                         standard = D_FREEM;
                    413:                         strcpy (m_dialect, "FREEM");
                    414:                     }
                    415:                     else {
                    416:                         freem_usage ();
                    417:                     }
                    418:                 
                    419:                     break;                
                    420: 
                    421:                     
                    422:                 case 'S': /* --shmsize */
                    423:                     shm_init_size = atol (optarg);
                    424:                     break;
                    425: 
                    426:                 case 'u': /* --user */
                    427:                     strncpy (d_username, optarg, 40);
                    428:                     custom_user = TRUE;
                    429:                     break;
                    430:                     
                    431:                 case 'g': /* --group */
                    432:                     strncpy (d_groupname, optarg, 40);
                    433:                     custom_group = TRUE;
                    434:                     break;
                    435: 
                    436:                     
                    437:             } 
                    438:         }
                    439:     }
                    440: #endif
                    441:     snprintf (config_file, 4096, "%s/freem/%s/freem.conf", SYSCONFDIR, shm_env);
                    442:     
                    443:     if (run_daemon == TRUE && geteuid() == 0) {
                    444: 
                    445:         if (custom_group) {
                    446:             d_grp = getgrnam (d_groupname);
                    447: 
                    448:             if (d_grp == NULL) {
                    449:                 fprintf (stderr, "freem:  invalid group '%s'\n", d_groupname);
                    450:                 exit (1);
                    451:             }
                    452:             
                    453:             d_gid = d_grp->gr_gid;
                    454:         }
                    455: 
                    456:         if (custom_user) {
                    457:             d_user = getpwnam (d_username);
                    458: 
                    459:             if (d_user == NULL) {
                    460:                 fprintf (stderr, "freem:  invalid user '%s'\n", d_username);
                    461:                 exit (1);
                    462:             }
                    463:             
                    464:             d_uid = d_user->pw_uid;
                    465:         }
                    466:         
                    467:     }
                    468:     
                    469:     if ((nofork == TRUE) && (run_daemon == FALSE)) {
                    470:         freem_usage ();
                    471:         exit (1);
                    472:     }
                    473: 
                    474:     if ((run_daemon == TRUE) && (nofork == FALSE)) {
                    475: 
                    476:         int fork_fd;
                    477:         
                    478:         /* daemonize */
                    479: 
                    480:         fork_pid = fork ();
                    481: 
                    482:         if (fork_pid < 0) {
                    483:             fprintf (stderr, "freem:  failure in fork()\r\n");
                    484:             m_log (1, "failure in initial fork()\r\n");
                    485:             exit (1);
                    486:         }
                    487: 
                    488:         if (fork_pid > 0) {
                    489:             exit (0);
                    490:         }
                    491: 
                    492:         if (setsid () < 0) {
                    493:             fprintf (stderr, "freem:  failure in setsid()\r\n");
                    494:             m_log (1, "failure in setsid()\r\n");
                    495:             exit (1);
                    496:         }
                    497: 
                    498:         signal (SIGCHLD, SIG_IGN);
                    499: 
                    500:         fork_pid = fork ();
                    501: 
                    502:         if (fork_pid < 0) {
                    503:             fprintf (stderr, "freem:  failure in fork()\r\n");
                    504:             m_log (1, "failure in second fork()\r\n");
                    505:             exit (1);
                    506:         }
                    507: 
                    508:         if (fork_pid > 0) {
                    509:             exit (0);
                    510:             m_log (1, "exiting from second fork");
                    511:         }
                    512: 
                    513:         umask (0);
                    514: 
                    515:         chdir ("/");       
                    516: 
                    517:         for (fork_fd = sysconf (_SC_OPEN_MAX); fork_fd > 0; fork_fd--) {
                    518:             close (fork_fd);
                    519:         }
                    520:         
                    521:         if (geteuid () == 0) {
                    522:             /* shed privileges */
                    523: 
                    524:             if (custom_group) {
                    525:                 fprintf (stderr, "freem:  switching to group %s\n", d_groupname);
                    526:                 m_log (1, "switching groups");
                    527:                 
                    528:                 if (setgid (d_gid) == -1) {
                    529:                     fprintf (stderr, "freem:  failure switching GID\n");
                    530:                     m_log (1, "failure switching GIDs");
                    531:                     exit (1);
                    532:                 }
                    533:             }
                    534: 
                    535: 
                    536:             if (custom_user) {                
                    537:                 fprintf (stderr, "freem:  switching to username %s\n", d_username);
                    538:                 m_log (1, "switching users");
                    539:                 
                    540:                 if (setuid (d_uid) == -1) {
                    541:                     fprintf (stderr, "freem:  failure switching UID\n");
                    542:                     m_log (1, "failure switching UIDs");
                    543:                     exit (1);
                    544:                 }
                    545:                 if (chdir (d_user->pw_dir) == -1) {
                    546:                     fprintf (stderr, "freem:  chdir failure\n");
                    547:                     m_log (1, "failure in chdir");
                    548:                     exit (1);
                    549:                 }            
                    550:             }
                    551: 
                    552:         }
                    553:         else {
                    554:             fprintf (stderr, "not euid 0");
                    555:         }
                    556: 
                    557:        freopen ("/dev/null", "r", stdin);
                    558:        freopen ("/dev/null", "w+", stdout);
                    559:        freopen ("/dev/null", "w+", stderr);
                    560: 
                    561:         run_daemon = TRUE;
                    562:         nofork = FALSE;        
                    563:         
                    564:         if (pid_file_path == NULL) {
                    565:             /* no PID file specified. choose one. */
                    566:             uid_t pid_uid;
                    567:             char *home_directory;
                    568:            
                    569:             pid_file_path = (char *) calloc (PATH_MAX, sizeof (char));
                    570:             NULLPTRCHK(pid_file_path,"main");
                    571: 
                    572:             home_directory = (char *) calloc (PATH_MAX, sizeof (char));
                    573:             NULLPTRCHK(home_directory,"main");
                    574:             
                    575:             pid_uid = geteuid ();
                    576: 
                    577:             if (pid_uid == 0) {
                    578:                 /* we're running as root */
                    579:                 strcpy (pid_file_path, "/var/run/freem.pid");
                    580:             }
                    581:             else {
                    582:                 /* our user is a normie */
                    583:                 struct passwd *pw = getpwuid (pid_uid);
                    584: 
                    585:                 if (pw == NULL) {
                    586:                     m_log (1, "main:  failure in getpwuid()");
                    587:                 }
                    588:                 
                    589:                 strcpy (home_directory, pw->pw_dir);               
                    590:                 snprintf (pid_file_path, PATH_MAX - 1, "%s/.freem.pid", home_directory);                
                    591:             }
                    592: 
                    593:             free (home_directory);
                    594:             
                    595:         }
                    596: 
                    597:         {
                    598:             char pidfile_buf[256];
                    599:             int errsav;
                    600: 
                    601:             m_log (1, pid_file_path);
                    602: 
                    603:             pid_fd = open (pid_file_path, O_RDWR | O_CREAT, 0640);
                    604:             errsav = errno;
                    605:             
                    606:             if (pid_fd < 0) {
                    607:                 m_log (1, "freem:  could not open PID file");
                    608:                 m_log (1, strerror (errsav));
                    609:                 exit (1);
                    610:             }
                    611: 
                    612:             if (lockf (pid_fd, F_TLOCK, 0) < 0) {
                    613:                 errsav = errno;
                    614:                 m_log (1, "freem: could not lock PID file - perhaps already running?");
                    615:                 m_log (1, strerror (errsav));
                    616:                 exit (1);
                    617:             }
                    618: 
                    619:             sprintf (pidfile_buf, "%d\n", getpid ());
                    620:             write (pid_fd, pidfile_buf, strlen (pidfile_buf));
                    621:          
                    622:         }
                    623:        
                    624:         
                    625:     } /* END of daemonization */
                    626:     
                    627:     /* handle passing of an arbitrary .m file on the command line */
                    628:     /* this is most often used for shebang-line scripts. */
                    629:     if (optind < argc) {
                    630: 
                    631:         /* not valid for daemon mode */
                    632:         if (run_daemon == TRUE) {
                    633:             fprintf (stderr, "freem:  cannot pass --daemon flag in shebang line\r\n");
                    634:             exit (1);
                    635:         }
                    636:         
                    637:         /* bail if file does not exist */
                    638:         if (access (argv[optind], F_OK) == -1) {
                    639: 
                    640:             set_io (UNIX);
                    641:             
                    642:             fprintf (stderr, "Routine %s does not exist.\n", argv[optind]);
                    643: 
                    644:             exit (1);
                    645: 
                    646:         }
                    647: 
                    648:        skip_init = 1;
                    649: 
                    650:        /* initialize FreeM environment */
                    651:        strncpy (nsnbuf, nsname, 255);    
                    652:        if (init (nsnbuf) == FALSE) {
                    653:            
                    654:            set_io (UNIX);
                    655:            fprintf (stderr, "\nError initializing FreeM.\n");
                    656:            
                    657:            exit (1);            
                    658:            
                    659:        }        
                    660: 
                    661:        
                    662:         direct_mode = FALSE;
                    663:         usermode = 0;
                    664: 
                    665:         /* was a path specified at all? */
                    666:         if (strchr (argv[optind], '/') == NULL) {
                    667:            
                    668:             /* the entirety of argv[optind] is the filename */
                    669:             cli_rtn_file = argv[optind];            
                    670: 
                    671:             /* use the current directory */
                    672:             sprintf (cli_rtn_path, ".");
                    673: 
                    674:         }
                    675:         else {
                    676: 
                    677:             /* isolate the filename from the path */
                    678:             cli_rtn_file = strrchr (argv[optind], '/') + 1;
                    679:             
                    680:             /* isolate the routine name from the filename */
                    681:             strncpy (cli_rtn_name, cli_rtn_file, strchr (cli_rtn_file, '.') - cli_rtn_file);
                    682:             
                    683:             /* isolate the path from the routine file */
                    684:             strncpy (cli_rtn_path, argv[optind], strrchr (argv[optind], '/') - argv[optind]);      
                    685:            
                    686: /*         set_io (UNIX);
                    687:            printf ("cli_rtn_name = '%s' cli_rtn_path = '%s'\n", cli_rtn_name, cli_rtn_path);
                    688:            set_io (MUMPS);
                    689: */         
                    690:         }
                    691: 
                    692:         /* do we have a file extension? */
                    693:         if (strchr (cli_rtn_file, '.') != NULL) {
                    694:             
                    695:             /* if so, just remove it */
                    696:             strncpy (cli_rtn_name, cli_rtn_file, strchr (cli_rtn_file, '.') - cli_rtn_file);
                    697: 
                    698:         }
                    699:         else {
                    700: 
                    701:             /* otherwise, just take a direct copy */
                    702:             strcpy (cli_rtn_name, cli_rtn_file);
                    703: 
                    704:         }
                    705: 
                    706:         /* make this the startup routine */
                    707:         snprintf (startuprou, 256, "^%s\201", cli_rtn_name);
                    708: 
                    709:         /* re-work the namespace config to search for the 
                    710:            routine in the discovered path */
                    711:         if (cli_rtn_name[0] == '%') {
                    712: 
                    713:             snprintf (rou0plib, 256, "%s\201", cli_rtn_path);
                    714:             snprintf (rou1plib, 256, "%s\201", cli_rtn_path);
                    715: 
                    716:         }
                    717:         else {
                    718: 
                    719:             snprintf (rou0path, 256, "%s\201", cli_rtn_path);
                    720:             snprintf (rou1path, 256, "%s\201", cli_rtn_path);
                    721: 
                    722:         }
                    723: 
                    724:     }
                    725: 
                    726: 
                    727:     if (!file_exists (config_file)) {
                    728: 
                    729:         set_io (UNIX);
                    730:         fprintf (stderr, "\nFreeM has not been configured. Please run 'fmadm configure'.\n\n\n\n");
                    731:         
                    732:         exit (2);
                    733: 
                    734:     }
                    735: 
                    736:     if (!skip_init) {
                    737:        /* initialize FreeM environment */
                    738:        strncpy (nsnbuf, nsname, 255);    
                    739:        if (init (nsnbuf) == FALSE) {
                    740:            
                    741:            set_io (UNIX);
                    742:            fprintf (stderr, "\nError initializing FreeM.\n");
                    743: 
                    744:            exit (1);
                    745: 
                    746:        }
                    747:     }
                    748: 
                    749:     if (first_process == TRUE) {
                    750:         
                    751:         char verstr[500];
                    752:         pid_t stop_requester;
                    753: 
                    754:         if (run_daemon == FALSE) {
                    755:             fprintf (stderr, "freem:  re-run with --daemon or -d command-line flags\r\n");
                    756:             cleanup ();
                    757:             exit (1);
                    758:         }
                    759:         
                    760:         stcpy (verstr, FREEM_VERSION_STR);
                    761:         stcnv_m2c (verstr);
                    762:         
                    763:         fprintf (stderr, "Coherent Logic Development FreeM version %s\r\n", verstr);
1.2       snw       764:         fprintf (stderr, "freem:  shared memory for environment %s initialized (%ld bytes of shared memory @ '%p')\r\nfreem:  system ready\r\n", shm_env, (long) shm_init_size, shm_config->dta);
1.1       snw       765:         
                    766:         for (;;) {
                    767: 
                    768:             job_set_status (pid, JSTAT_HOUSEKEEPING);
                    769:             
                    770: 
                    771:             if (shm_config->hdr->maintenance_mode == 1) {
                    772: 
                    773:                 job_slot_t *slot;
                    774: 
                    775:                 fprintf (stderr, "freem:  entering maintenance mode\r\n");
                    776:                 m_log (1, "freem:  entering maintenance mode");
                    777:                 
                    778:                 for (slot = shm_config->hdr->jobtab_head; slot != NULL; slot = slot->next) {
                    779: 
                    780:                     if ((slot->pid != pid) && ((slot->flags & JFLG_FMADM) != JFLG_FMADM)) {
                    781:                         kill (slot->pid, SIGINT);
                    782:                     }
                    783: 
                    784:                 }
                    785:                 
                    786:             }
                    787:             
                    788:             if ((stop_requester = job_stop_requested (pid)) != 0) {
                    789:                 int connected_jobs;
                    790:                 
                    791:                 job_set_status (pid, JSTAT_SHUTDOWN);
                    792:                 
                    793:                 connected_jobs = job_count ();
                    794:                 
                    795:                 fprintf (stderr, "freem:  STOP requested by pid %d\r\n", stop_requester);
                    796:                 fprintf (stderr, "freem:  there are %d job(s) connected to this environment\r\n", connected_jobs);
                    797:                 
                    798:                 if (connected_jobs > 1) {
                    799:                     
                    800:                     fprintf (stderr, "freem:  asking non-daemon job(s) to disconnect and halt...\r\n");
                    801:                     job_request_all_stop ();
                    802: 
                    803:                     fprintf (stderr, "freem:  waiting 5 seconds for job(s) to disconnect...\r\n");
                    804:                     sleep (5);
                    805: 
                    806:                     connected_jobs = job_count ();
                    807:                     if (connected_jobs > 1) {
                    808:                         fprintf (stderr, "freem:  sending SIGTERM to %d job(s)...\r\n", connected_jobs);
                    809:                         job_signal_all (SIGTERM);
                    810:                         fprintf (stderr, "freem:  waiting 5 seconds for job(s) to disconnect...\r\n");
                    811:                         
                    812:                         sleep (5);
                    813:                     }
                    814:                     
                    815:                     connected_jobs = job_count ();
                    816:                     if (connected_jobs > 1) {
                    817:                         fprintf (stderr, "freem:  sending SIGKILL to %d job(s)...\r\n", connected_jobs);
                    818:                         job_signal_all (SIGKILL);
                    819:                     }
                    820:                    
                    821:                     job_gc_mark ();
                    822:                     job_gc_sweep ();
                    823:                     
                    824:                 }
                    825: 
                    826:                 fprintf (stderr, "freem:  terminating\r\n");
                    827:                 cleanup ();
                    828:                 exit (0);
                    829:                 
                    830:             }
                    831:             
                    832:             job_gc_mark ();
                    833:             job_set_status (pid, JSTAT_IDLE);
                    834:             sleep (1);
                    835:             
                    836:             job_set_status (pid, JSTAT_HOUSEKEEPING);
                    837:             job_gc_sweep ();
                    838:             sleep (1);
                    839:         }
                    840:             
                    841:     }
                    842:         
                    843:     
                    844: #if !defined(_AIX)
                    845:     if(import_env == TRUE) {
                    846: 
                    847:        int i_maxlen = 255;
                    848:        
                    849:         for(env = envp; *env != 0; env++) {
                    850:            
                    851:             namelen = 0;
                    852:             vallen = 0;
                    853:             
                    854:             varname = strtok(*env, "=");
                    855:             varval = strtok(NULL, "=");
                    856: 
                    857:             if(varval != NULL) {
                    858:                 namelen = strlen (varname);
                    859:                 vallen = strlen (varval);
                    860: 
                    861:                snprintf (symname, i_maxlen, "ENV.%s\201\201", varname);
                    862:                 strncpy (symval, varval, i_maxlen);
                    863:                
                    864:                 stcnv_c2m (symval);
                    865: 
                    866:                 symtab (set_sym, symname, symval);
                    867:             }
                    868:         }
                    869:     }
                    870: #endif    
                    871: 
                    872:     
                    873:     if (direct_mode == TRUE && quiet_mode == FALSE) {
                    874: 
                    875:        char verstr[500];
                    876:         char version[256];
                    877:        
                    878:        stcpy (verstr, FREEM_VERSION_STR);
                    879:        stcnv_m2c (verstr);
                    880:                
                    881:         snprintf (version, 255, "\r\nCoherent Logic Development FreeM version %s [DIALECT: %s%s]\r\n\201", verstr, m_dialect, (restricted_mode == TRUE ? "/RESTRICTED" : ""));
                    882:        write_m (version);
                    883: 
1.4       snw       884:        snprintf (version, 255, "Copyright (C) 2014, 2020, 2021, 2023, 2025 Coherent Logic Development LLC\r\n\r\n\201");
1.1       snw       885:        write_m (version);
                    886: 
1.5     ! snw       887:         /*
1.1       snw       888:         printf ("Environment:       \t%s\r\n", shm_env);
                    889:         printf ("Environment Daemon:\tPID %d\r\n", shm_config->hdr->first_process);
                    890:         printf ("Interpreter Process:\tPID %d\r\n", pid);
1.5     ! snw       891:         */
1.1       snw       892: 
                    893:     }
                    894:     else {
                    895:         write_m ("\r\n\r\n\201");
                    896:     }
                    897: 
                    898:     if (dx_mode) {
                    899:        char k_buf[512];        
                    900: 
                    901:        snprintf (k_buf, 512 - 1, "%%TMPINITMCODE\201\201");
                    902:        symtab (set_sym, k_buf, dx_mcode);
                    903:         const_define (k_buf, dx_mcode);
                    904:     }
                    905: 
                    906:     if (routine_mode) {
                    907:         char k_buf[512];
                    908: 
                    909:         snprintf (k_buf, 512 - 1, "%%TMPINITROUTINE\201\201");
                    910:         symtab (set_sym, k_buf, startup_routine);
                    911:         const_define (k_buf, startup_routine);
                    912:     }
                    913:     
                    914:     /* run mumps */
                    915:     xecline (1);
                    916: 
                    917:     exit (0);  /* we should never reach that statement */
                    918: 
                    919: } /* end of main() */
                    920: 
                    921: void freem_usage(void)
                    922: {
                    923:     fprintf (stdout, "\nusage:  freem [OPTION...]\n\n");
                    924:     
                    925:     fprintf (stdout, "OPTIONS:\n\n");
                    926:     
                    927: #if defined(HAVE_GETOPT_LONG)
                    928:     fprintf (stdout, "\t-h, --help\n\t\tdisplays this help message\n\n");
                    929:     fprintf (stdout, "\t-i, --import\n\t\timports UNIX environment variables as M locals\n\n");
                    930:     fprintf (stdout, "\t-e <environment-name>, --environment=<environment-name>\n\t\tsets active environment to <environment-name> (DEFAULT if unspecified)\n\n");
                    931:     fprintf (stdout, "\t-f, --filter\n\t\tallows M code to be used as a filter\n\n");
                    932:     fprintf (stdout, "\t-n <NAMESPACE>, --namespace=<NAMESPACE>\n\t\tselects <NAMESPACE> as the startup namespace instead of USER\n\n");
                    933:     fprintf (stdout, "\t-q, --quiet\n\t\tdisables startup messages and prompt string\n\n");
                    934:     fprintf (stdout, "\t-r <LABEL^ROUTINE>, --routine=<LABEL^ROUTINE>\n\t\texecute <LABEL^ROUTINE> on startup instead of entering direct mode\n\n");
                    935:     fprintf (stdout, "\t-s, --standard\n\t\trestrict access to FreeM vendor extensions not present in relevant standards*\n\n");
                    936:     fprintf (stdout, "\t-v, --version\n\t\tdisplay FreeM version information\n\n");
                    937:     fprintf (stdout, "\t-x <MCODE>, --execute=<MCODE>\n\t\texecute M code <MCODE> on startup\n\n");
                    938:     fprintf (stdout, "\t-d, --daemon\n\t\trun the FreeM daemon (one and only one FreeM daemon must always be running)\n\n");
                    939:     fprintf (stdout, "\t-k, --nofork\n\t\trun the FreeM daemon in foreground (requires --daemon)\n\n");
                    940:     fprintf (stdout, "\t-p <PIDFILE>, --pidfile=<PIDFILE>\n\t\tuse <PIDFILE> to record the PID of the FreeM daemon\n\n\n");
                    941:     fprintf (stdout, "\t-S <BYTES>, --shmsize=<BYTES>\n\t\tsets the size of the shared memory segment where FreeM stores the job table, lock table, and IPC table.\n");
                    942: #else
                    943:     fprintf (stdout, "\t-h\n\t\tdisplays this help message\n\n");
                    944:     fprintf (stdout, "\t-i\n\t\timports UNIX environment variables as M locals\n\n");
                    945:     fprintf (stdout, "\t-e <environment-name>\n\t\tsets active environment to <environment-name> (DEFAULT if unspecified)\n\n");
                    946:     fprintf (stdout, "\t-f\n\t\tallows M code to be used as a filter\n\n");
                    947:     fprintf (stdout, "\t-n <NAMESPACE>\n\t\tselects <NAMESPACE> as the startup namespace instead of USER\n\n");
                    948:     fprintf (stdout, "\t-q\n\t\tdisables startup messages and prompt string\n\n");
                    949:     fprintf (stdout, "\t-r <LABEL^ROUTINE>\n\t\texecute <LABEL^ROUTINE> on startup instead of entering direct mode\n\n");
                    950:     fprintf (stdout, "\t-s\n\t\trestrict access to FreeM vendor extensions not present in relevant standards*\n\n");
                    951:     fprintf (stdout, "\t-v\n\t\tdisplay FreeM version information\n\n");
                    952:     fprintf (stdout, "\t-x <MCODE>\n\t\texecute M code <MCODE> on startup\n\n");
                    953:     fprintf (stdout, "\t-d\n\t\trun the FreeM daemon (one and only one FreeM daemon must always be running)\n\n");
                    954:     fprintf (stdout, "\t-k\n\t\trun the FreeM daemon in foreground (requires --daemon)\n\n");
                    955:     fprintf (stdout, "\t-p <PIDFILE>\n\t\tuse <PIDFILE> to record the PID of the FreeM daemon\n\n\n");
                    956:     fprintf (stdout, "\t-S <BYTES>\n\t\tsets the size of the shared memory segment where FreeM stores the job table, lock table, and IPC table.\n");
                    957: #endif    
                    958:     fprintf (stdout, "\t\t  - Each concurrent job takes %d bytes (1 page) of shared memory\n", PG_SIZE);
                    959:     fprintf (stdout, "\t\t  - Each LOCK takes %d bytes (2 pages) of shared memory\n", PG_SIZE * 2);
                    960:     fprintf (stdout, "\t\t  - Each IPC takes %d bytes (1 page) of shared memory\n\n", PG_SIZE);
                    961:     fprintf (stdout, "\t* FreeM attempts to conform (at least loosely) to the Millennium Draft Standard when this mode is selected.\n\n\n");
                    962:     fprintf (stdout, "Report bugs to: freem-bugs@coherent-logic.com\n");
                    963:     fprintf (stdout, "FreeM home page: <https://freem.coherent-logic.com>\n\n");   
                    964:     
                    965:     exit (1);
                    966: }
                    967: 
                    968: void freem_print_version(void)
                    969: {
                    970:     char verstr[500];
                    971:     stcpy (verstr, FREEM_VERSION_STR);
                    972:     stcnv_m2c (verstr);
                    973:     
                    974:     fprintf (stdout, "Coherent Logic Development FreeM %s\n", verstr);
                    975:     fprintf (stdout, "Copyright (C) 2014, 2020, 2021, 2023 Coherent Logic Development LLC\n\n");
                    976:     fprintf (stdout, "License AGPLv3+: GNU AGPL version 3 or later <https://gnu.org/license/agpl-3.0.html>\n");
                    977:     fprintf (stdout, "This is free software: you are free to change and redistribute it.\n");
                    978:     fprintf (stdout, "There is NO WARRANTY, to the extent permitted by law.\n");
                    979:     
                    980:     exit (0);
                    981: }

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>