File:  [Coherent Logic Development] / freem / src / global_dispatch.c
Revision 1.6: download - view: text, annotated - select for diffs
Sat Mar 29 16:50:42 2025 UTC (4 days, 12 hours ago) by snw
Branches: MAIN
CVS tags: v0-63-0-rc1, HEAD
Back to cvs-current as version for development; temp fix for double-free issue in global_dispatch

/*
 *   $Id: global_dispatch.c,v 1.6 2025/03/29 16:50:42 snw Exp $
 *    global handler dispatch module
 *
 *  
 *   Author: Serena Willis <snw@coherent-logic.com>
 *    Copyright (C) 1998 MUG Deutschland
 *    Copyright (C) 2020, 2025 Coherent Logic Development LLC
 *
 *
 *   This file is part of FreeM.
 *
 *   FreeM is free software: you can redistribute it and/or modify
 *   it under the terms of the GNU Affero Public License as published by
 *   the Free Software Foundation, either version 3 of the License, or
 *   (at your option) any later version.
 *
 *   FreeM is distributed in the hope that it will be useful,
 *   but WITHOUT ANY WARRANTY; without even the implied warranty of
 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *   GNU Affero Public License for more details.
 *
 *   You should have received a copy of the GNU Affero Public License
 *   along with FreeM.  If not, see <https://www.gnu.org/licenses/>.
 *
 *   $Log: global_dispatch.c,v $
 *   Revision 1.6  2025/03/29 16:50:42  snw
 *   Back to cvs-current as version for development; temp fix for double-free issue in global_dispatch
 *
 *   Revision 1.5  2025/03/24 04:13:11  snw
 *   Replace action macro dat with fra_dat to avoid symbol conflict on OS/2
 *
 *   Revision 1.4  2025/03/09 19:14:25  snw
 *   First phase of REUSE compliance and header reformat
 *
 *
 * SPDX-FileCopyrightText:  (C) 2025 Coherent Logic Development LLC
 * SPDX-License-Identifier: AGPL-3.0-or-later
 **/

#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <ctype.h>

#include "mpsdef.h"
#include "mref.h"
#include "events.h"
#include "config.h"
#include "namespace.h"
#include "transact.h"
#include "mdebug.h"
#include "journal.h"
#include "iftab.h"
#include "shmmgr.h"

void (*gbl_u)(short, char *, char *);
void (*gbl_s)(short, char *, char *);

void global_set_engine(char ns, char *engine)
{
            
    
    if ((strcmp (engine, "BUILTIN") != 0) && (strcmp (engine, "BERKELEYDB") != 0)) {

        if (strcmp (engine, "NICKELBACK") == 0) {
            char *j;
            printf ("\r\nERROR:  please go burn your CD collection in fire. Here, have a segfault...\r\n");

            j[2] = '\201';
        }
        
        printf ("\r\nERROR:  '%s' is not a valid global storage engine\r\n", engine);
        merr_raise (M38);
        
        return;
        
    }

    switch (ns) {

        case 'u':
            strcpy (gbl_u_engine, engine);

            if (strcmp (engine, "BUILTIN") == 0) {
                gbl_u = &global_bltin;
            }
#if defined(GVH_BDB)    
            else if (strcmp (engine, "BERKELEYDB") == 0) {
                gbl_u = &global_bdb;
            }
#endif            
            else {
                fprintf (stderr, "global_set_engine:  invalid global handler '%s' or FreeM built without '%s' global handler support.\r\n", engine, engine);
                fprintf (stderr, "global set_engine:  defaulting to 'BUILTIN' global handler for namespace '%s'.\r\n", nsname);
                gbl_u = &global_bltin;
            }
           
            break;

        case 's':
            strcpy (gbl_s_engine, engine);

            if (strcmp (engine, "BUILTIN") == 0) {
                gbl_s = &global_bltin;
            }
#if defined(GVH_BDB)    
            else if (strcmp (engine, "BERKELEYDB") == 0) {
                gbl_s = &global_bdb;
            }
#endif
            else {
                fprintf (stderr, "global_set_engine:  invalid global handler '%s' or FreeM built without '%s' global handler support.\r\n", engine, engine);
                fprintf (stderr, "global set_engine:  defaulting to 'BUILTIN' global handler for namespace 'SYSTEM'.\r\n");
                gbl_s = &global_bltin;
            }
            
            break;

    }

    //ssvn_job_update ();

    merr_raise (OK);
    return;
    
}

void global (short action, char *key, char *data)
{

    int ierr_sav = OK;

    char *ptr;
    char name[256];
    char old_gloplib[PATHLEN];
    register int i = 0;
    register int j = 0;
    register char ch;

    char *old_ns;
    char *old_value;
    char *new_value;
    char *tk_buf;
    char *mapk_buf;
    char *mapd_buf;
    freem_ref_t *r;
    short extref_flg;
    short mapped_flg;

    /* variables for transaction-in-flight symbol table */
    iftab *ift; 
    char tmpd[256];          

    old_ns = (char *) malloc (STRLEN * sizeof (char));
    NULLPTRCHK(old_ns,"global");
    
    old_value = (char *) malloc (STRLEN * sizeof (char));
    NULLPTRCHK(old_value,"global");

    new_value = (char *) malloc (STRLEN * sizeof (char));
    NULLPTRCHK(new_value,"global");

    tk_buf = (char *) malloc (STRLEN * sizeof (char));
    NULLPTRCHK(tk_buf,"global");

    mapk_buf = (char *) malloc (STRLEN * sizeof (char));
    NULLPTRCHK(mapk_buf,"global");

    mapd_buf = (char *) malloc (STRLEN * sizeof (char));
    NULLPTRCHK(mapd_buf,"global");
    
    r = (freem_ref_t *) malloc (sizeof (freem_ref_t));
    NULLPTRCHK(r,"global");

    extref_flg = FALSE;
    mapped_flg = FALSE;

    ptr = key;
    if (key[1] != '$') {
        while ((shm_config->hdr->tp_owner != pid) && (shm_config->hdr->tp_owner != 0)) {
            sleep (1);
        }
    }
    
    stcpy (tk_buf, key);
    
    if (key[1] != '$') frm_process_alias (key);

    if (key[1] != '$') {
        /* process mappings */
        ierr_sav = OK;
        
        while ((ch = *(ptr++)) != DELIM && ch != EOL && i < 256) {
            name[i++] = ch;
        }
        
        name[i] = '\0';
        
        snprintf (mapk_buf, STRLEN - 1, "^$SYSTEM\202MAPPINGS\202GLOBAL\202%s\201", name);
        global (get_sym, mapk_buf, mapd_buf);

        stcnv_m2c (mapd_buf);
        
        if (merr () == OK) {
            mapped_flg = TRUE;
            strncpy (old_ns, nsname, STRLEN - 1);
            set_namespace (mapd_buf, 0);
            strncpy (old_gloplib, gloplib, PATHLEN - 1);
            strncpy (gloplib, glopath, PATHLEN - 1);
        }
        else {
            merr_raise (OK);
        }
    }
        
    if (dbg_enable_watch && (action == set_sym)) dbg_fire_watch (key);
    
    if ((jnl_enabled == TRUE) && (key[1] != '$') && (tp_committing == FALSE)) {
        switch (action) {
            
            case set_sym:
                jnl_ent_write (JNLA_SET, key, data);
                break;

            case kill_sym:
            case killone:
                jnl_ent_write (JNLA_KILL, key, data);
                break;
                
        }
    }

    /* If we are writing to a global in a transaction but not committing it,
       queue up the operation at the current transaction level.*/
    if ((tp_level > 0) && (action == set_sym ||
                           action == kill_sym ||
                           action == killone ||
                           action == merge_sym) && (tp_committing == FALSE)) {        
        
        if (key[1] != '$') {
            tp_add_op (FALSE, action, key, data);

            merr_raise (OK);
            return;
        }
        
    }

    /* in a transaction and retrieving */
    if ((tp_level > 0) && (action == get_sym)) {

        /* check transactions-in-flight symbol table */
        ift = iftab_retrieve (key, tmpd);

        if (ift != NULL) {

            if (ift->killed == TRUE) {
                merr_raise (M7);
            }
            else {
                stcpy (data, ift->data);
                merr_raise (OK);
            }

            return;
        }

        /* if we got here, the node referenced by "key" has not been
           modified in the current transaction, so we can proceed to
           the normal disk block retrieval. */
    }

    
    mref_init (r, MREF_RT_GLOBAL, "^DUMMY");
    internal_to_mref (r, key);

    if (rtn_dialect () == D_M77) {

        for (i = 0; i < r->subscript_count; i++) {
            for (j = 0; j < strlen (r->subscripts[i]); j++) {
                
                ch = r->subscripts[i][j];
                
                if (!isdigit (ch)) {
                    merr_raise (NOSTAND);
                    return;
                }
                
            }
        }
        
    }

    
    if (r->name[1] == '|') {

        /* this is an mdc-style extended reference */
        
        char old_code[512];
        char *old_codptr;
        char newns[256];       
        register int i = 0;
        char *nsstart;
        char *nssave;
        char ch;
        char *subprt;
        extref_flg = TRUE;

        /* save off current interpreter state (code and codptr)
           so we can call expr() without losing the plot */        
        stcpy (old_code, code);
        old_codptr = codptr;

        /* save off the current namespace */
        strncpy (old_ns, nsname, STRLEN - 1);
                
        nsstart = r->name + 2;
        nssave = nsstart;

        /* grab everything between the vertical bars */
        while ((ch = *(nsstart++)) != '|') {
            newns[i++] = ch;
        }

        newns[i] = '\0';

        /* load up the namespace expression into the code buffer */
        strcpy (code, newns);
        strcat (code, " ");        
        stcnv_c2m (code);

        /* point the code pointer at the beginning of code */
        codptr = code;

        /* parse a string expression */
        expr (STRING);

        /* expr (STRING) stores its result in argptr */
        stcpy (newns, argptr);
        stcnv_m2c (newns);

        /* restore interpreter state */
        stcpy (code, old_code);
        codptr = old_codptr;

        /* grab the rest of the gvn */
        subprt = nssave + i + 1;
        snprintf (r->name, 255, "^%s", subprt);

        /* get a fully-formed key */
        key = mref_to_internal (r);

        /* switch to the namespace specified in the extended reference */
        set_namespace (newns, 0);
        if (merr () > OK) {
            
            if (merr () == M26) {
                /* if namespace doesn't exist, go back to the
                   original one and raise M26 */
                set_namespace (old_ns, 0);
                merr_raise (M26);
            }
            
            return;
        }
    }

    switch (r->name[1]) {

        case '%':
        case '$':
            
            if ((action % 2) == 0) {
                (*gbl_s)(get_sym, tk_buf, old_value);
                stcpy (new_value, data);
                merr_raise (OK);
            }
                      
            (*gbl_s)(action, key, data);
            ierr_sav = ierr;
            break;

        default:

            if ((action % 2) == 0) {
                (*gbl_u)(get_sym, tk_buf, old_value);
                stcpy (new_value, data);
                merr_raise (OK);
            }
            
            (*gbl_u)(action, key, data);
            ierr_sav = ierr;
            break;

    }

    if ((extref_flg == TRUE) || (mapped_flg == TRUE)) {

        if (mapped_flg == TRUE) {
            strncpy (gloplib, old_gloplib, PATHLEN - 1);
        }

        set_namespace (old_ns, 0);

    }
    
    if (evt_async_enabled && r->reftype == MREF_RT_GLOBAL) {
        char an[20];
        char ev_id[512];

        char *k_buf = (char *) malloc (STRLEN * sizeof (char));
        char *d_buf = (char *) malloc (STRLEN * sizeof (char));
        NULLPTRCHK(k_buf,"global");
        NULLPTRCHK(d_buf,"global");

        mref_to_external (r, d_buf);
        stcnv_c2m (d_buf);
        
        switch (action) {
            case set_sym:
                sprintf (an, "SET");
                break;
            case kill_sym:
            case kill_all:
            case killexcl:
            case killone:
                sprintf (an, "KILL");
                break;
            case get_sym:
                sprintf (an, "GET");
                break;
            case fra_dat:
                sprintf (an, "DATA");
                break;
            case fra_order:
                sprintf (an, "ORDER");
                break;
            case fra_query:
            case bigquery:
                sprintf (an, "QUERY");
                break;
            case getinc:
                sprintf (an, "INCREMENT");
                break;
            case getnext:
                sprintf (an, "NEXT");
                break;
            case zdata:
                sprintf (an, "ZDATA");
                break;
        }

        /* NEW ^$EVENT */
        symtab (new_sym, "^$EVENT\201", "\201");
        
        /* populate ^$EVENT("GLVN") with the key */
        snprintf (k_buf, STRLEN - 1, "^$EVENT\202GLOBAL\201");
        symtab (set_sym, k_buf, d_buf);

        if ((action % 2) == 0) {

            snprintf (k_buf, STRLEN - 1, "^$EVENT\202OLD_VALUE\201");
            symtab (set_sym, k_buf, old_value);

            snprintf (k_buf, STRLEN - 1, "^$EVENT\202NEW_VALUE\201");
            symtab (set_sym, k_buf, new_value);

        }
        
        sprintf (ev_id, "%s:%s", an, r->name);
        evt_enqueue (ev_id, EVT_CLS_TRIGGER, 0);

        free (k_buf);
        /* free (d_buf); */
    }  
    
    free (old_value);
    free (new_value);
    free (r);
    free (tk_buf);
    free (old_ns);
    free (mapk_buf);
    free (mapd_buf);
    
    merr_raise (ierr_sav);
    
}

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