Annotation of freem/src/transact.c, revision 1.6

1.1       snw         1: /*
1.6     ! snw         2:  *   $Id: transact.c,v 1.5 2025/03/24 02:54:47 snw Exp $
1.1       snw         3:  *    FreeM transaction processing support
                      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: transact.c,v $
1.6     ! snw        27:  *   Revision 1.5  2025/03/24 02:54:47  snw
        !            28:  *   Transaction compat fixes for OS/2
        !            29:  *
1.5       snw        30:  *   Revision 1.4  2025/03/09 19:50:47  snw
                     31:  *   Second phase of REUSE compliance and header reformat
                     32:  *
1.4       snw        33:  *
                     34:  * SPDX-FileCopyrightText:  (C) 2025 Coherent Logic Development LLC
                     35:  * SPDX-License-Identifier: AGPL-3.0-or-later
1.1       snw        36:  **/
                     37: 
                     38: #include <string.h>
                     39: #include <stddef.h>
                     40: #include <stdio.h>
                     41: #include <stdlib.h>
                     42: #include <unistd.h>
                     43: #include <errno.h>
                     44: 
                     45: #include "mpsdef.h"
                     46: #include "transact.h"
                     47: #include "iftab.h"
                     48: #include "journal.h"
                     49: #include "shmmgr.h"
                     50: #include "mref.h"
                     51: #include "tp_check.h"
                     52: 
                     53: #define FALSE   0
                     54: #define TRUE    1
                     55: 
1.5       snw        56: #if !defined(__OpenBSD__) && !defined(__APPLE__) && !defined(__OS2__)
1.1       snw        57: union semun {
                     58:     int              val;    /* Value for SETVAL */
                     59:     struct semid_ds *buf;    /* Buffer for IPC_STAT, IPC_SET */
                     60:     unsigned short  *array;  /* Array for GETALL, SETALL */
                     61:     struct seminfo  *__buf;  /* Buffer for IPC_INFO
                     62:                                 (Linux-specific) */
                     63: };
                     64: #endif
                     65: 
                     66: void m_log (int, const char *);
                     67: 
                     68: int semid_tp;
                     69: int tp_committing = FALSE;
                     70: int tp_level = 0;
                     71: tp_transaction transactions[TP_MAX_NEST];
                     72: 
                     73: 
                     74: void tp_init(void)
                     75: {
                     76:     union semun arg;
                     77:     char err[255];
                     78:     key_t tp_sk;
                     79: 
                     80:     tp_sk = ftok (config_file, 4);
                     81:     
                     82:     if (first_process) {
                     83: 
                     84:         semid_tp = semget (tp_sk, 1, 0666 | IPC_CREAT);
                     85:         if (semid_tp == -1) {
                     86:             fprintf (stderr, "tp_init:  failed to create transaction processing semaphore [errno %d]\r\n", errno);
                     87:             exit (1);
                     88:         }
                     89: 
                     90:         arg.val = 1;
                     91:         if (semctl (semid_tp, 0, SETVAL, arg) == -1) {
                     92:             fprintf (stderr, "tp_init:  failed to initialize transaction processing semaphore\r\n");
                     93:             exit (1);
                     94:         }
                     95: 
                     96:     }
                     97:     else {
                     98:         
                     99:         semid_tp = semget (tp_sk, 1, 0);
                    100:         if (semid_tp == -1) {
                    101:             fprintf (stderr, "tp_init:  could not attach to transaction processing semaphore [errno %d]\r\n", errno);
                    102:             exit (1);
                    103:         }
                    104: 
                    105:     }
                    106: 
                    107:     return;
                    108: 
                    109: }
                    110: 
                    111: short tp_get_sem(void)
                    112: {
                    113:     int tries;
                    114:     struct sembuf s = {0, -1, 0};
                    115: 
                    116:     char msgbuf[100];
                    117: 
                    118:     snprintf (msgbuf, 99, "tp_get_sem:  process %d attempting to acquire transaction processing semaphore", pid);
                    119:     m_log (1, msgbuf);
                    120:     
                    121:     
                    122:     /* our process already owns the semaphore */
                    123:     if (shm_config->hdr->tp_owner == pid) {
                    124: 
                    125:         snprintf (msgbuf, 99, "tp_get_sem:  process %d increments transaction processing semaphore counter", pid);
                    126:         m_log (1, msgbuf);
                    127:     
                    128:         
                    129:         if (first_process == TRUE) {
                    130:             fprintf (stderr, "tp_get_sem:  daemon process increments critical section counter\r\n");
                    131:         }
                    132: 
                    133: 
                    134:         shm_config->hdr->tp_semctr++;
                    135: 
                    136:         return TRUE;
                    137:     }
                    138: 
                    139:     if (first_process == TRUE) {
                    140:         fprintf (stderr, "tp_get_sem:  daemon process enters critical section\r\n");
                    141:     }
                    142: 
                    143:     
                    144:     for (tries = 0; tries < 10; tries++) {
                    145: 
                    146:         if (semop (semid_tp, &s, 1) != -1) {
                    147:             shm_config->hdr->tp_owner = pid;
                    148:             shm_config->hdr->tp_semctr = 1;
                    149: 
                    150:             snprintf (msgbuf, 99, "tp_get_sem:  process %d takes transaction processing semaphore", pid);
                    151:             m_log (1, msgbuf);
                    152:     
                    153:             
                    154:             if (first_process == TRUE) {
                    155:                 fprintf (stderr, "tp_get_sem:  daemon process takes transaction processing semaphore\r\n");
                    156:             }
                    157: 
                    158:             return TRUE;
                    159:         }
                    160: 
                    161:         snprintf (msgbuf, 99, "tp_get_sem:  process %d attempting to acquire transaction processing semaphore (tries = %d)", pid, tries);
                    162:         m_log (1, msgbuf);
                    163:     
                    164: 
                    165:         sleep (1);
                    166: 
                    167:     }
                    168: 
                    169:     return FALSE;
                    170:     
                    171: }
                    172: 
                    173: void tp_release_sem(void)
                    174: {
                    175: 
                    176:     char msgbuf[100];
                    177: 
                    178:     if (shm_config->hdr->tp_semctr == 1) {
                    179: 
                    180:         struct sembuf s = {0, 1, 0};
                    181: 
                    182:         if (first_process == TRUE) {
                    183:             fprintf (stderr, "tp_release_sem:  daemon process leaves critical section\r\n");
                    184:         }
                    185: 
                    186:         
                    187:         shm_config->hdr->tp_semctr = 0;
                    188:         shm_config->hdr->tp_owner = 0;
                    189: 
                    190:         if (first_process == TRUE) {
                    191:             fprintf (stderr, "tp_release_sem:  daemon process relinquishes transaction processing semaphore\r\n");
                    192:         }
                    193: 
                    194: 
                    195:         snprintf (msgbuf, 99, "tp_get_sem:  process %d releases transaction processing semaphore", pid);
                    196:         m_log (1, msgbuf);
                    197: 
                    198:         
                    199:         semop (semid_tp, &s, 1);
                    200:         
                    201:     }
                    202:     else {
                    203: 
                    204:         if (first_process == TRUE) {
                    205:             fprintf (stderr, "tp_release_sem:  daemon process decrements critical section counter\r\n");
                    206:         }
                    207:         
                    208:         snprintf (msgbuf, 99, "tp_get_sem:  process %d decrements transaction processing semaphore counter", pid);
                    209:         m_log (1, msgbuf);
                    210:         
                    211:         shm_config->hdr->tp_semctr--;
                    212:     }
                    213: 
                    214:     
                    215: }
                    216: 
                    217: int tp_tstart(char *tp_id, short serial, short restartable, char **sym_save)
                    218: {
                    219:     if (tp_level == TP_MAX_NEST) {
                    220:         char m[256];
                    221: 
                    222:         snprintf (m, 256, "Attempt to exceed TP_MAX_NEST. Transaction aborted.\r\n\201");
                    223:         write_m (m);
                    224: 
                    225:         return FALSE;
                    226:     }
                    227: 
                    228:     if (((serial == TRUE) && (tp_get_sem () == TRUE)) ||
                    229:         (serial == FALSE)) {
                    230: 
                    231:         tp_level++;
                    232:         
                    233:         jnl_ent_write (JNLA_TSTART, "", "");
                    234:     
                    235:         strcpy (transactions[tp_level].tp_id, tp_id);
                    236:         
                    237:         transactions[tp_level].serial = serial;
                    238:         transactions[tp_level].restartable = restartable;
                    239:         
                    240:         transactions[tp_level].opcount = 0;
                    241:       
                    242:         return TRUE;
                    243:         
                    244:     }
                    245:     else {
                    246:         fprintf (stderr, "tp_tstart:  could not get transaction processing semaphore\r\n");
                    247:         exit (1);
                    248:     }
                    249:     
                    250: 
                    251: }
                    252: 
                    253: int tp_add_op(short islock, short action, char *key, char *data)
                    254: {
                    255:     int oc;
                    256:     freem_ref_t *gr;
                    257: 
                    258:     gr = (freem_ref_t *) malloc (sizeof (freem_ref_t));
                    259:     NULLPTRCHK(gr,"tp_add_op");
                    260:     
                    261:     mref_init (gr, MREF_RT_GLOBAL, "");
                    262:     internal_to_mref (gr, key);
                    263:     
                    264:     if (transactions[tp_level].opcount == TP_MAX_OPS) {
                    265:         char m[256];
                    266: 
                    267:         snprintf (m, 256, "Attempt to exceed TP_MAX_OPS at transaction level %d. Rolling back.\r\n\201", tp_level);
                    268:         write_m (m);
                    269: 
                    270:         free (gr);
                    271: 
                    272:         tp_trollback (1);
                    273:         tp_cleanup (1);
                    274: 
                    275:         if (transactions[tp_level].serial == TRUE) {
                    276:             tp_release_sem();
                    277:         }
                    278:         
                    279:         return FALSE;
                    280:     }
                    281:     
                    282:     /* update transaction-in-flight symbol table */
                    283:     switch (action) {
                    284: 
                    285:         case lock_inc:
                    286:         case lock_dec:
                    287:         case lock_old:
                    288:         case set_sym:
                    289:             iftab_insert (action, key, data, tp_level);
                    290:             break;
                    291: 
                    292:         case kill_sym:
                    293:             iftab_kill (key);
                    294:             break;
                    295:             
                    296:     }
                    297: 
                    298:     if (transactions[tp_level].serial == TRUE) {
                    299:         /* mark the global for checkpointing */
                    300:         cptab_insert (tp_level, gr->name);
                    301:     }
                    302: 
                    303:     free (gr);
                    304: 
                    305:     transactions[tp_level].opcount = transactions[tp_level].opcount + 1;
                    306: 
                    307:     oc = transactions[tp_level].opcount;
                    308: 
                    309:     transactions[tp_level].ops[oc].is_lock = islock;
                    310:     transactions[tp_level].ops[oc].action = action;
                    311: 
1.2       snw       312:     stcpy ((char *) &transactions[tp_level].ops[oc].key, key);
                    313:     stcpy ((char *) &transactions[tp_level].ops[oc].data, data);
1.1       snw       314:        
                    315:  
                    316:     return TRUE;
                    317: }
                    318: 
                    319: int tp_tcommit(void)
                    320: {
                    321:     register int i;
                    322:     short is_serial = transactions[tp_level].serial;
                    323:     
                    324:     tp_committing = TRUE;
                    325: 
                    326:     if (is_serial) {
                    327:         /* checkpoint all globals involved in transaction */
                    328:         cptab_precommit (tp_level);
                    329:     }
                    330:     
                    331:     for(i = 1; i <= transactions[tp_level].opcount; i++) {
                    332:         
                    333:         if (transactions[tp_level].ops[i].is_lock == FALSE) {
                    334:             global (transactions[tp_level].ops[i].action, transactions[tp_level].ops[i].key, transactions[tp_level].ops[i].data);
                    335: 
                    336:             if (merr () > OK) goto commit_error;
                    337:                 
                    338:         }
                    339:         
                    340:     }
                    341: 
                    342:     jnl_ent_write (JNLA_TCOMMIT, "\201", "\201");
                    343: 
                    344:     if (is_serial) {
                    345:         cptab_postcommit (tp_level);
                    346:     }
                    347: 
                    348:     goto commit_done;
                    349:     
                    350: commit_error:
                    351:     
                    352:     tp_trollback (1);
                    353: 
                    354: commit_done:
                    355: 
                    356:     tp_cleanup (1);
                    357:     
                    358:     tp_committing = FALSE;
                    359: 
                    360:     if (is_serial) {
                    361:         tp_release_sem ();
                    362:     }
                    363:     
                    364:     return TRUE;
                    365: }
                    366: 
                    367: int tp_cleanup(int levels)
                    368: {
                    369:     register int i;
                    370: 
                    371:     for (i = tp_level; i >= (((tp_level - levels) >= 0) ? tp_level - levels : 0); i--) {
                    372:         iftab_pop_tlevel (i);
                    373:     }
                    374: 
                    375:     tp_level = ((tp_level - levels) >= 0) ? tp_level - levels : 0;
                    376: 
                    377:     return TRUE;
                    378: }
                    379: 
                    380: int tp_trollback(int levels)
                    381: {
                    382:     register int i;
                    383:     register int j;
                    384: 
                    385: //    for (i = 0; i < levels; i++) {
                    386:     for (i = tp_level; i >= (((tp_level - levels) >= 0) ? tp_level - levels : 0); i--) {
                    387:         
                    388:         for (j = 1; j <= transactions[i].opcount; j++) {
                    389:             
                    390:             if (transactions[i].ops[j].is_lock == TRUE) {
                    391:                 locktab_decrement (transactions[i].ops[j].key, -1);
                    392:             }
                    393:             
                    394:         }
                    395: 
                    396:         if (transactions[i].serial == TRUE) {
                    397:             cptab_rollback (i);
                    398:         }
                    399:         
                    400:     }
                    401: 
                    402: 
                    403:     return TRUE;
                    404: }
                    405: 
                    406: int tp_trestart(void)
                    407: {
                    408:     return 0;
                    409: }
                    410: 
                    411: void tp_tdump(void)
                    412: {
                    413: 
                    414:     int i, j;
                    415: 
                    416:     char tkey[256];
                    417:     char tdata[256];
                    418:     char tact[256];
                    419:     
                    420:     set_io (UNIX);
                    421: 
                    422:     if (tp_level == 0) {
                    423:         printf("No transaction is active.\n");
                    424: 
                    425:         return;
                    426:     }
                    427: 
                    428:     for(i = 1; i <= tp_level; i++) {
                    429: 
                    430:         if(i == tp_level) {
                    431:             printf(" $TLEVEL %d*\n", i);
                    432:         }
                    433:         else {
                    434:             printf(" $TLEVEL %d\n", i);
                    435:         }
                    436: 
                    437:         printf("  Operations for Transaction ID: %s [%s%s]\n",
                    438:                transactions[i].tp_id,
                    439:                ((transactions[i].restartable == TRUE) ? "RESTARTABLE" : "NON-RESTARTABLE"),
                    440:                ((transactions[i].serial == TRUE) ? " SERIAL" : " BATCH"));
                    441: 
                    442:         printf ("\n   %-10s%-15s%-15s\n", "OP. NO.", "ACTION", "KEY/DATA");
                    443:         printf ("   %-10s%-15s%-15s\n", "-------", "------", "--------");
                    444:         
                    445:         
                    446:         for(j = 1; j <= transactions[i].opcount; j++) {          
                    447:             stcpy (tkey, transactions[i].ops[j].key);
                    448:             stcnv_m2c (tkey);
                    449:             stcpy (tdata, transactions[i].ops[j].data);
                    450:             stcnv_m2c (tdata);
                    451: 
                    452:             tp_get_op_name (tact, transactions[i].ops[j].action);
                    453: 
                    454:             if (transactions[i].ops[j].action == set_sym) {
                    455:                 printf ("   %-10d%-15s%s=%s\n", j, tact, tkey, tdata);
                    456:             }
                    457:             else {
                    458:                 printf ("   %-10d%-15s%s\n", j, tact, tkey); 
                    459:             }
                    460:             
                    461:         }
                    462: 
                    463:         cptab_dump (i);
                    464:         
                    465:     }
                    466: 
                    467: 
                    468:     set_io (MUMPS);
                    469: }
                    470: 
                    471: void tp_get_op_name(char *buf, const short action)
                    472: {
                    473:     switch (action) {
                    474:         
                    475:         case set_sym:
                    476:             strcpy (buf, "SET");
                    477:             break;
                    478: 
                    479:         case killone:
                    480:         case kill_sym:
                    481:         case kill_all:
                    482:         case killexcl:
                    483:             strcpy (buf, "KILL");
                    484:             break;
                    485: 
                    486:         case new_sym:
                    487:         case new_all:
                    488:         case newexcl:
                    489:             strcpy (buf, "NEW");
                    490:             break;
                    491: 
                    492:         case get_sym:
                    493:             strcpy (buf, "GET");
                    494:             break;
                    495: 
1.6     ! snw       496:         case fra_dat:
1.1       snw       497:             strcpy (buf, "$DATA");
                    498:             break;
                    499: 
                    500:         case fra_order:
                    501:             strcpy (buf, "$ORDER");
                    502:             break;
                    503: 
                    504:         case fra_query:
                    505:         case bigquery:
                    506:             strcpy (buf, "$QUERY");
                    507:             break;
                    508: 
                    509:         case getinc:
                    510:             strcpy (buf, "$INCREMENT");
                    511:             break;
                    512: 
                    513:         case getnext:
                    514:             strcpy (buf, "$NEXT");
                    515:             break;
                    516: 
                    517:         case lock_inc:
                    518:             strcpy (buf, "LOCK (INCR)");
                    519:             break;
                    520: 
                    521:         case lock_old:
                    522:             strcpy (buf, "LOCK (TRAD)");
                    523:             break;
                    524:             
                    525:     }
                    526:             
                    527: }

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