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

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

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