Annotation of freem/src/symtab_bltin.c, revision 1.1
1.1 ! snw 1: /*
! 2: * *
! 3: * * *
! 4: * * *
! 5: * ***************
! 6: * * * * *
! 7: * * MUMPS *
! 8: * * * * *
! 9: * ***************
! 10: * * *
! 11: * * *
! 12: * *
! 13: *
! 14: * symtab.c
! 15: * FreeM local system table and user-defined special variable table
! 16: *
! 17: *
! 18: * Author: Serena Willis <jpw@coherent-logic.com>
! 19: * Copyright (C) 1998 MUG Deutschland
! 20: * Copyright (C) 2020 Coherent Logic Development LLC
! 21: *
! 22: *
! 23: * This file is part of FreeM.
! 24: *
! 25: * FreeM is free software: you can redistribute it and/or modify
! 26: * it under the terms of the GNU Affero Public License as published by
! 27: * the Free Software Foundation, either version 3 of the License, or
! 28: * (at your option) any later version.
! 29: *
! 30: * FreeM is distributed in the hope that it will be useful,
! 31: * but WITHOUT ANY WARRANTY; without even the implied warranty of
! 32: * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
! 33: * GNU Affero Public License for more details.
! 34: *
! 35: * You should have received a copy of the GNU Affero Public License
! 36: * along with FreeM. If not, see <https://www.gnu.org/licenses/>.
! 37: *
! 38: **/
! 39:
! 40: #include <stdlib.h>
! 41: #define ZNEW 'N'
! 42: #include "mpsdef.h"
! 43: #include <string.h>
! 44: #include <sys/ipc.h>
! 45: #include <sys/shm.h>
! 46: #include <sys/sem.h>
! 47: #include <unistd.h>
! 48: #include "mdebug.h"
! 49: #include "merr.h"
! 50: #include "consttbl.h"
! 51: #include "shmmgr.h"
! 52:
! 53: /* Turn this on to get tons of lovely debugging messages about
! 54: symbol-table calls */
! 55: /* #define DEBUG_SYM */
! 56:
! 57: short restoring_consts = FALSE;
! 58: int semid_symtab;
! 59:
! 60: #if !defined(__OpenBSD__) && !defined(__APPLE__)
! 61: union semun {
! 62: int val; /* Value for SETVAL */
! 63: struct semid_ds *buf; /* Buffer for IPC_STAT, IPC_SET */
! 64: unsigned short *array; /* Array for GETALL, SETALL */
! 65: struct seminfo *__buf; /* Buffer for IPC_INFO
! 66: (Linux-specific) */
! 67: };
! 68: #endif
! 69:
! 70:
! 71: long str2long(char *string)
! 72: {
! 73: int loop = 0;
! 74: int mult = 1;
! 75: int exp = 1;
! 76: long value = 0;
! 77:
! 78: if (string[0] == '-') {
! 79: mult = -1;
! 80: string++;
! 81: }
! 82:
! 83: while(string[loop] != EOL && string[loop] >= '0' && string[loop] <= '9') loop++;
! 84:
! 85: loop--;
! 86:
! 87: while(loop > -1) {
! 88: value += (string[loop] - '0') * exp;
! 89: exp *= 10; loop--;
! 90: }
! 91:
! 92: value *= mult;
! 93:
! 94: return value;
! 95: }
! 96:
! 97: void symtab_init ()
! 98: {
! 99: register int i;
! 100: union semun arg;
! 101: key_t symtab_sk;
! 102: symtab_sk = ftok (config_file, 6);
! 103:
! 104:
! 105: if (first_process) {
! 106:
! 107: for (i = 0; i < 128; i++) {
! 108: shm_config->hdr->alphptr[i] = 0L;
! 109: }
! 110:
! 111: shm_config->hdr->symlen = PSIZE;
! 112: shm_config->hdr->s = &mbpartition[PSIZE] - 256;
! 113: shm_config->hdr->PSIZE = DEFPSIZE;
! 114: shm_config->hdr->argptr = mbpartition;
! 115:
! 116: fprintf (stderr, "symtab_init: initializing memory-backed globals\r\n");
! 117:
! 118: semid_symtab = semget (symtab_sk, 1, 0666 | IPC_CREAT);
! 119: if (semid_symtab == -1) {
! 120: fprintf (stderr, "symtab_init: failed to create symbol table semaphore\r\n");
! 121: exit (1);
! 122: }
! 123: else {
! 124: fprintf (stderr, "symtab_init: symbol table semaphore created with semid %d\r\n", semid_symtab);
! 125: }
! 126:
! 127: arg.val = 1;
! 128: if (semctl (semid_symtab, 0, SETVAL, arg) == -1) {
! 129: fprintf (stderr, "symtab_init: failed to initialize symbol table semaphore\r\n");
! 130: exit (1);
! 131: }
! 132: else {
! 133: fprintf (stderr, "symtab_init: symbol table semaphore initialized\r\n");
! 134: }
! 135:
! 136: fprintf (stderr, "symtab_init: allocating partition for memory-backed globals\r\n");
! 137:
! 138: mbpartition = (char *) shm_alloc ((size_t) PSIZE + 2);
! 139: NULLPTRCHK(mbpartition,"symtab_init");
! 140:
! 141: shm_config->hdr->partition = mbpartition;
! 142:
! 143: if (symtab_get_sem ()) {
! 144: for (i = 0; i < 128; i++) shm_config->hdr->alphptr[i] = 0L;
! 145: symtab_release_sem ();
! 146: }
! 147:
! 148: }
! 149: else {
! 150:
! 151: semid_symtab = semget (symtab_sk, 1, 0);
! 152: if (semid_symtab == -1) {
! 153: fprintf (stderr, "symtab_init: could not attach to symbol table semaphore\r\n");
! 154: exit (1);
! 155: }
! 156:
! 157: mbpartition = shm_config->hdr->partition;
! 158:
! 159: }
! 160:
! 161: }
! 162:
! 163: short have_symtab_sem = FALSE;
! 164:
! 165: short symtab_get_sem(void)
! 166: {
! 167: int tries;
! 168: struct sembuf s = {0, -1, IPC_NOWAIT};
! 169:
! 170: if (have_symtab_sem) {
! 171: return TRUE;
! 172: }
! 173:
! 174: for (tries = 0; tries < 5; tries++) {
! 175:
! 176: if (semop (semid_symtab, &s, 1) != -1) {
! 177: have_symtab_sem = TRUE;
! 178: return TRUE;
! 179: }
! 180:
! 181: sleep (1);
! 182:
! 183: }
! 184: fprintf (stderr, "symtab_get_sem: fail\r\n");
! 185:
! 186: have_symtab_sem = FALSE;
! 187: return FALSE;
! 188: }
! 189:
! 190: void symtab_release_sem(void)
! 191: {
! 192: struct sembuf s = {0, 1, 0};
! 193:
! 194: semop (semid_symtab, &s, 1);
! 195:
! 196: have_symtab_sem = FALSE;
! 197: }
! 198:
! 199:
! 200: void symtab_shm (short action, char *key, char *data) /* symbol table functions */
! 201: {
! 202: char *old_s;
! 203: char *old_argptr;
! 204: long old_psize;
! 205: long old_symlen;
! 206: unsigned long stptrs[128];
! 207: register int i;
! 208: char *old_partition = partition;
! 209: partition = mbpartition;
! 210:
! 211: writing_mb = TRUE;
! 212:
! 213: if (symtab_get_sem ()) {
! 214:
! 215: /* save off current non-shared symtab state */
! 216: old_s = s;
! 217: old_argptr = argptr;
! 218: old_psize = PSIZE;
! 219: old_symlen = symlen;
! 220: for (i = 0; i < 128; i++) {
! 221: stptrs[i] = alphptr[i];
! 222: }
! 223:
! 224: /* replace symtab state with the values from the shared symtab */
! 225: s = shm_config->hdr->s;
! 226: argptr = shm_config->hdr->argptr;
! 227: PSIZE = shm_config->hdr->PSIZE;
! 228: symlen = shm_config->hdr->symlen;
! 229: for (i = 0; i < 128; i++) {
! 230: alphptr[i] = shm_config->hdr->alphptr[i];
! 231: }
! 232:
! 233: /* execute the action (symtab_bltin will now be working with the shared symbol table) */
! 234: symtab_bltin (action, key, data);
! 235:
! 236: /* copy new alphptr state back to shared memory */
! 237: for (i = 0; i < 128; i++) {
! 238: shm_config->hdr->alphptr[i] = alphptr[i];
! 239: }
! 240:
! 241: /* restore non-shared symtab alphptr state */
! 242: for (i = 0; i < 128; i++) {
! 243: alphptr[i] = stptrs[i];
! 244: }
! 245:
! 246: /* write the new shared symtab state back to shared memory */
! 247: shm_config->hdr->s = s;
! 248: shm_config->hdr->argptr = argptr;
! 249: shm_config->hdr->PSIZE = PSIZE;
! 250: shm_config->hdr->symlen = symlen;
! 251:
! 252: /* restore the non-shared symtab state */
! 253: s = old_s;
! 254: argptr = old_argptr;
! 255: PSIZE = old_psize;
! 256: symlen = old_symlen;
! 257:
! 258: symtab_release_sem ();
! 259:
! 260: }
! 261: else {
! 262: fprintf (stderr, "symtab_shm: failed to acquire symbol table sempahore\r\n");
! 263: }
! 264:
! 265: writing_mb = FALSE;
! 266: partition = old_partition;
! 267:
! 268: }
! 269:
! 270: /* local symbol table management */
! 271: /* (+)functions are now re-implemented */
! 272: /* (!)functions are new */
! 273: /* (?)re-implemented, with issues */
! 274: /* +set_sym +get_sym */
! 275:
! 276: /* +kill_sym +$data */
! 277: /* +kill_all +$fra_order */
! 278: /* +killexcl +fra_query */
! 279: /* +new_sym +bigquery */
! 280: /* +new_all +getinc */
! 281: /* +newexcl */
! 282: /* +killone +m_alias */
! 283: /* !merge_sym +zdata */
! 284: /* !pop_sym */
! 285:
! 286:
! 287:
! 288: /* The symbol table is placed at the high end of 'partition'. It begins at
! 289: * 'symlen' and ends at 'PSIZE'. The layout is
! 290: * (keylength)(key...)(<EOL>)(datalength)(data...[<EOL>])
! 291: * The keys are sorted in $order sequence.
! 292: *
! 293: * ****possible future layout with less space requirements****
! 294: * (keylength)(statusbyte)(key...)[(datalength)(data...[<EOL>])]
! 295: * 'keylength' is the length of 'key' overhead bytes not included.
! 296: * 'statusbyte' is an indicator with the following bits:
! 297: * 0 (LSB) 1=data information is missing 0=there is a data field
! 298: * 1 1=key is numeric 0=key is alphabetic
! 299: * 2..7 0..number of previous key_pieces
! 300: * note, that the status byte of a defined unsubscripted variable
! 301: * is zero.
! 302: * If a subscripted variable is stored, the variablename and each
! 303: * subscript are separate entries in the symbol table.
! 304: * E.g. S USA("CA",6789)="California" ; with $D(ABC)=0 before the set
! 305: * then the following format is used:
! 306: * (3)( 1)ABC
! 307: * (2)(1*4+1)CA
! 308: * (4)(2*4+2)6789(10)California
! 309: * ****end of "possible future layout"****
! 310: * To have the same fast access regardless of the position in the
! 311: * alphabet for each character a pointer to the first variable beginning
! 312: * with that letter is maintained. (0 indicates there's no such var.)
! 313: */
! 314:
! 315: void symtab_bltin (short action, char *key, char *data) /* symbol table functions */
! 316: {
! 317: /* must be static: */
! 318: static unsigned long tryfast = 0L; /* last $order reference */
! 319:
! 320: /* the following variables may */
! 321: /* be static or not */
! 322: static unsigned short nocompact = TRUE; /* flag: do not compact symtab if */
! 323:
! 324: /* value becomes shorter */
! 325: /* be static or dynamic: */
! 326:
! 327: static long keyl, datal; /* length of key and data */
! 328: static long kill_from;
! 329: static char tmp1[256], tmp2[256], tmp3[256];
! 330:
! 331: register long i, j, k, k1;
! 332: char tt_with[STRLEN];
! 333: char tt_key[STRLEN];
! 334:
! 335: #ifdef DEBUG_SYM
! 336:
! 337: int i0, i1;
! 338: char *start;
! 339:
! 340: #endif
! 341:
! 342: if (restoring_consts == FALSE) {
! 343: if (((action % 2) == 0) && const_is_defined (key)) {
! 344: merr_raise (CMODIFY);
! 345: return;
! 346: }
! 347: }
! 348:
! 349: if (action == kill_all) goto no_with;
! 350: if ((stlen (key) >= 5) && (strncmp (key, "%INT.", 5) == 0)) goto no_with;
! 351: if (strncmp (key, "^$", 2) == 0) goto no_with;
! 352: if (strncmp (key, "$", 1) == 0) goto no_with;
! 353:
! 354: stcpy (tt_with, i_with);
! 355: stcpy (tt_key, key);
! 356:
! 357: stcnv_m2c (tt_with);
! 358: stcnv_m2c (tt_key);
! 359:
! 360: snprintf (key, 100, "%s%s\201\201", tt_with, tt_key);
! 361:
! 362:
! 363: no_with:
! 364:
! 365:
! 366: if (dbg_enable_watch && ((action % 2) == 0)) dbg_fire_watch (key);
! 367:
! 368: if (key && key[1] != '$') stcpy (zloc, key);
! 369:
! 370: if (v22ptr) {
! 371:
! 372: procv22 (key);
! 373:
! 374: if (key[0] == '^') {
! 375:
! 376: char zrsav[256];
! 377: int naksav;
! 378: char gosav[256];
! 379:
! 380: stcpy (zrsav, zref);
! 381:
! 382: naksav = nakoffs;
! 383:
! 384: stcpy (gosav, g_o_val);
! 385: global (action, key, data);
! 386:
! 387: stcpy (zref, zrsav);
! 388:
! 389: nakoffs = naksav;
! 390:
! 391: stcpy (l_o_val, g_o_val);
! 392: stcpy (g_o_val, gosav);
! 393:
! 394: return;
! 395:
! 396: }
! 397:
! 398: }
! 399:
! 400: /* process optional limitations */
! 401: if (glvnflag.all && key[0] >= '%' && key[0] <= 'z') {
! 402:
! 403: if ((i = glvnflag.one[0])) { /* number of significant chars */
! 404:
! 405: j = 0;
! 406:
! 407: while ((k1 = key[j]) != DELIM && k1 != EOL) {
! 408:
! 409: if (j >= i) {
! 410:
! 411: while ((k1 = key[++j]) != DELIM && k1 != EOL) ;
! 412:
! 413: stcpy (&key[i], &key[j]);
! 414:
! 415: break;
! 416:
! 417: }
! 418:
! 419: j++;
! 420: }
! 421:
! 422: }
! 423:
! 424: if (glvnflag.one[1]) { /* upper/lower sensitivity */
! 425:
! 426: j = 0;
! 427:
! 428: while ((k1 = key[j]) != DELIM && k1 != EOL) {
! 429:
! 430: if (k1 >= 'a' && k1 <= 'z') key[j] = k1 - 32;
! 431:
! 432: j++;
! 433:
! 434: }
! 435:
! 436: }
! 437:
! 438:
! 439: if ((i = glvnflag.one[2])) {
! 440:
! 441: /* IMPACT: x11-94-28 */
! 442: if (stlen (key) > i) {
! 443: merr_raise (M75);
! 444: return;
! 445: } /* key length limit */
! 446:
! 447: }
! 448:
! 449: if ((i = glvnflag.one[3])) { /* subscript length limit */
! 450:
! 451: j = 0;
! 452:
! 453: while ((k1 = key[j++]) != DELIM && k1 != EOL) ;
! 454:
! 455: if (k1 == DELIM) {
! 456:
! 457: k = 0;
! 458:
! 459: for (;;) {
! 460:
! 461: k1 = key[j++];
! 462:
! 463: if (k1 == DELIM || k1 == EOL) {
! 464:
! 465: if (k > i) {
! 466: merr_raise (M75);
! 467: return;
! 468: }
! 469:
! 470: k = 0;
! 471:
! 472: }
! 473:
! 474: if (k1 == EOL) break;
! 475:
! 476: k++;
! 477:
! 478: }
! 479: }
! 480: }
! 481: }
! 482:
! 483:
! 484:
! 485: if (aliases && (action != m_alias)) { /* there are aliases */
! 486:
! 487: i = 0;
! 488: j = 0;
! 489:
! 490: while (i < aliases) {
! 491:
! 492: k1 = i + UNSIGN (ali[i]) + 1;
! 493:
! 494: /* is current reference an alias ??? */
! 495: j = 0;
! 496:
! 497: while (ali[++i] == key[j]) {
! 498:
! 499: if (ali[i] == EOL) break;
! 500:
! 501: j++;
! 502:
! 503: }
! 504:
! 505: /* yes, it is, so resolve it now! */
! 506: if (ali[i] == EOL && (key[j] == EOL || key[j] == DELIM)) {
! 507:
! 508: stcpy (tmp1, key);
! 509: stcpy (key, &ali[i + 1]);
! 510: stcat (key, &tmp1[j]);
! 511:
! 512: i = 0;
! 513:
! 514: continue; /* try again, it might be a double alias! */
! 515:
! 516: }
! 517:
! 518: i = k1;
! 519:
! 520: }
! 521:
! 522: }
! 523:
! 524: #ifdef DEBUG_SYM
! 525:
! 526: printf("DEBUG (%d): ",action);
! 527:
! 528: if(key) {
! 529:
! 530: printf("[key] is [");
! 531:
! 532: for(loop=0; key[loop] != EOL; loop++) printf("%c",(key[loop] == DELIM) ? '!' : key[loop]);
! 533:
! 534: printf("]\r\n");
! 535:
! 536: }
! 537: else {
! 538: printf("No key passed in.\r\n");
! 539: }
! 540:
! 541: if(data) {
! 542:
! 543: printf("[data] (datalen) is [");
! 544:
! 545: for(loop=0; data[loop] != EOL; loop++) printf("%c", data[loop]);
! 546:
! 547: printf("] (%d)\r\n",stlen(data));
! 548: printf("[Numeric?] is [%d]\r\n",is_numeric(data));
! 549:
! 550: }
! 551: else {
! 552: printf("No data passed in.\r\n");
! 553: }
! 554:
! 555: #endif
! 556:
! 557: switch (action) {
! 558:
! 559:
! 560: case get_sym: /* retrieve */
! 561:
! 562:
! 563: /* OLD get_sym routine */
! 564: if ((i = alphptr[(int) key[0]])) {
! 565:
! 566: // printf ("alphptr match; writing_mb = %d\r\n", writing_mb);
! 567:
! 568: k = 1;
! 569: j = i + 1; /* first char always matches! */
! 570:
! 571: do {
! 572:
! 573: while (key[k] == partition[++j]) { /* compare keys */
! 574:
! 575: if (key[k] == EOL) {
! 576:
! 577: /* IMPACT: x11-94-28 */
! 578: i = UNSIGN (partition[++j]);
! 579:
! 580: if (i < 4) {
! 581:
! 582: k = 0;
! 583:
! 584: while (k < i) data[k++] = partition[++j];
! 585:
! 586: }
! 587: else {
! 588: stcpy0 (data, &partition[j + 1], i);
! 589: }
! 590:
! 591: data[i] = EOL;
! 592:
! 593: return;
! 594:
! 595: }
! 596:
! 597: k++;
! 598:
! 599: }
! 600:
! 601: i += UNSIGN (partition[i]); /* skip key */
! 602: i += UNSIGN (partition[i]) + 1; /* skip data */
! 603:
! 604: k = 0;
! 605: j = i;
! 606:
! 607: } while (i < PSIZE);
! 608: }
! 609:
! 610: merr_raise (M6);
! 611: data[0] = EOL;
! 612:
! 613: return;
! 614:
! 615:
! 616:
! 617:
! 618:
! 619: case set_sym: /* store/create variable */
! 620:
! 621:
! 622: /* HANDLE ISVs FROM unnew() */
! 623:
! 624: if (key[0] == '$') {
! 625:
! 626: switch (key[1]) {
! 627:
! 628: case 't': /* $TEST */
! 629:
! 630: test = data[0];
! 631: break;
! 632:
! 633: case 'z': /* $Z ISVs */
! 634:
! 635: if (key[2] == 'r') { /* $ZREFERENCE / $REFERENCE */
! 636: stcpy (zref, data);
! 637: }
! 638:
! 639: break;
! 640:
! 641: }
! 642:
! 643: }
! 644:
! 645: datal = stlen (data); /* data length */
! 646:
! 647:
! 648:
! 649:
! 650:
! 651: /* Old set_sym routine */
! 652: /* check whether the key has subscripts or not */
! 653: if ((keyl = stlen (key) + 2) > STRLEN) {
! 654: merr_raise (M75);
! 655: return;
! 656: }
! 657:
! 658: k1 = 0;
! 659: i = 1;
! 660:
! 661: while (key[i] != EOL) {
! 662:
! 663: if (key[i++] == DELIM) {
! 664: k1 = i;
! 665: break;
! 666: }
! 667:
! 668: }
! 669:
! 670: if ((i = alphptr[(int) key[0]])) { /* previous entry */
! 671:
! 672: j = i + 1;
! 673: k = 1;
! 674:
! 675: }
! 676: else {
! 677:
! 678: i = symlen;
! 679: j = i;
! 680: k = 0;
! 681:
! 682: }
! 683:
! 684: if (k1 == 0) /* key was unsubscripted */
! 685:
! 686: /* compare keys */
! 687: while (i < PSIZE) {
! 688:
! 689: while (key[k] == partition[++j]) {
! 690:
! 691: if (key[k] == EOL) goto old;
! 692:
! 693: k++;
! 694:
! 695: }
! 696:
! 697: if (key[k] < partition[j]) break;
! 698:
! 699: i += UNSIGN (partition[i]); /* skip key */
! 700: i += UNSIGN (partition[i]) + 1; /* skip data */
! 701:
! 702: j = i;
! 703: k = 0;
! 704:
! 705: }
! 706: else { /* key was subscripted */
! 707:
! 708: /* compare keys */
! 709: while (i < PSIZE) {
! 710:
! 711: while (key[k] == partition[++j]) {
! 712:
! 713: if (key[k] == EOL) goto old;
! 714:
! 715: k++;
! 716:
! 717: }
! 718:
! 719:
! 720: if (k < k1) {
! 721: if (key[k] < partition[j]) break;
! 722: }
! 723: else {
! 724:
! 725: long m, n, o, ch;
! 726:
! 727: /* get complete subscripts */
! 728: n = k;
! 729:
! 730: while (key[--n] != DELIM) ;
! 731:
! 732: n++;
! 733: m = j + n - k;
! 734: o = 0;
! 735:
! 736: while ((ch = tmp3[o++] = partition[m++]) != EOL && ch != DELIM) ;
! 737:
! 738: if (ch == DELIM) tmp3[--o] = EOL;
! 739:
! 740: o = 0;
! 741:
! 742: while ((ch = tmp2[o++] = key[n++]) != EOL && ch != DELIM) ;
! 743:
! 744: if (ch == DELIM) tmp2[--o] = EOL;
! 745:
! 746: if (collate (tmp3, tmp2) == FALSE) {
! 747: if (stcmp (tmp2, tmp3) || ch == EOL) break;
! 748: }
! 749:
! 750: }
! 751:
! 752: i += UNSIGN (partition[i]); /* skip key */
! 753: i += UNSIGN (partition[i]) + 1; /* skip data */
! 754:
! 755: j = i;
! 756: k = 0;
! 757:
! 758: }
! 759: }
! 760:
! 761: /* if entry found, i pointer to searched entry
! 762: * else entry not found, i pointer to alphabetically next entry */
! 763:
! 764: /* new entry */
! 765: if (setop) {
! 766:
! 767: tmp1[0] = EOL;
! 768:
! 769: m_op (tmp1, data, setop);
! 770:
! 771: setop = 0;
! 772:
! 773: if (merr () > OK) return;
! 774:
! 775: datal = stcpy (data, tmp1);
! 776:
! 777: }
! 778:
! 779: k = i;
! 780: j = key[0];
! 781: i = keyl + datal + 1;
! 782:
! 783: if (alphptr['%']) alphptr['%'] -= i;
! 784:
! 785: for (k1 = 'A'; k1 <= j; k1++) {
! 786: if (alphptr[k1]) alphptr[k1] -= i;
! 787: }
! 788:
! 789: i = k - i;
! 790:
! 791: if (alphptr[j] == 0 || alphptr[j] > i) alphptr[j] = i;
! 792:
! 793: j = keyl + datal + 1;
! 794: i = symlen - j;
! 795:
! 796: if (i <= 256) { /* more space needed. try to get it */
! 797:
! 798: long dif = 0L;
! 799:
! 800: dif = getpmore ();
! 801:
! 802: if (dif == 0) {
! 803: merr_raise (STORE);
! 804: return;
! 805: }
! 806:
! 807: data = argptr;
! 808: i += dif;
! 809: k += dif;
! 810:
! 811: }
! 812:
! 813: symlen = i;
! 814: s = &partition[i] - 256;
! 815:
! 816: stcpy0 (&partition[i], &partition[j + i], k - i);
! 817:
! 818: i = k - (keyl + datal + 1);
! 819: partition[i++] = (char) (keyl);
! 820:
! 821: stcpy (&partition[i], key); /* store new key */
! 822:
! 823: i += keyl - 1;
! 824:
! 825: /* IMPACT: x11-94-28 */
! 826: partition[i++] = (char) (datal);
! 827:
! 828: stcpy0 (&partition[i], data, datal); /* store new data */
! 829:
! 830: return;
! 831:
! 832: /* there is a previous value */
! 833: old:
! 834: i += UNSIGN (partition[i]);
! 835:
! 836: if (setop) {
! 837:
! 838: j = UNSIGN (partition[i]);
! 839: stcpy0 (tmp1, &partition[i + 1], j);
! 840:
! 841: tmp1[j] = EOL;
! 842:
! 843: m_op (tmp1, data, setop);
! 844:
! 845: setop = 0;
! 846:
! 847: if (merr () > OK) return;
! 848:
! 849: datal = stcpy (data, tmp1);
! 850:
! 851: }
! 852:
! 853: old0: /* entry from getinc */
! 854:
! 855: /* IMPACT: x11-94-28 */
! 856: j = UNSIGN (partition[i]) - datal;
! 857:
! 858: if (j < 0) { /* more space needed */
! 859:
! 860: if ((symlen + j) <= 256) {
! 861:
! 862: long dif = 0L;
! 863:
! 864: dif = getpmore ();
! 865:
! 866: if (dif == 0L) {
! 867: merr_raise (STORE);
! 868: return;
! 869: }
! 870:
! 871: data = argptr;
! 872: i += dif;
! 873:
! 874: }
! 875:
! 876: for (k = 36; k < key[0]; k++) {
! 877: if (alphptr[k])
! 878: alphptr[k] += j;
! 879: }
! 880:
! 881: if (alphptr[k] && alphptr[k] < i) alphptr[k] += j;
! 882:
! 883: stcpy0 (&partition[symlen + j], &partition[symlen], i - symlen);
! 884:
! 885: i += j;
! 886: symlen += j;
! 887: s = &partition[symlen] - 256;
! 888: tryfast = 0;
! 889:
! 890: }
! 891: else if (j > 0) { /* surplus space */
! 892:
! 893: if (nocompact) {
! 894:
! 895: /* in a dynamic environment it is sufficient to */
! 896: /* set newdatalength=olddatalength */
! 897:
! 898: datal += j;
! 899:
! 900: }
! 901: else {
! 902:
! 903: /* instead of compression of the local symbol table, */
! 904: /* which the following piece of code does */
! 905:
! 906: symlen += j;
! 907: s = &partition[symlen] - 256;
! 908:
! 909: for (k = 36; k < key[0]; k++) {
! 910: if (alphptr[k]) alphptr[k] += j;
! 911: }
! 912:
! 913: if (alphptr[k] && alphptr[k] < i) alphptr[k] += j;
! 914:
! 915: i += j;
! 916: k = i;
! 917: j = i - j;
! 918:
! 919: while (i >= symlen) {
! 920: partition[i--] = partition[j--];
! 921: }
! 922:
! 923: i = k;
! 924: tryfast = 0;
! 925: nocompact = TRUE;
! 926:
! 927: }
! 928: }
! 929:
! 930: /* IMPACT: x11-94-28 */
! 931: partition[i++] = (char) (datal);
! 932: j = datal;
! 933:
! 934: if (j < 4) {
! 935:
! 936: k = 0;
! 937:
! 938: while (k < j) partition[i++] = data[k++];
! 939:
! 940: return;
! 941:
! 942: }
! 943:
! 944: stcpy0 (&partition[i], data, j); /* store new data */
! 945:
! 946: return;
! 947:
! 948: /* end of set_sym section */
! 949:
! 950:
! 951: case dat:
! 952:
! 953:
! 954: /* note: we assume EOL<DELIM<ASCII */
! 955: data[0] = '0';
! 956: data[1] = EOL;
! 957:
! 958: if ((i = alphptr[(int) key[0]])) {
! 959:
! 960: data[2] = EOL;
! 961: j = i + 1;
! 962: k = 1;
! 963:
! 964: do {
! 965:
! 966: while ((k1 = key[k] - partition[++j]) == 0) { /* compare keys */
! 967:
! 968: if (key[k] == EOL) break;
! 969:
! 970: k++;
! 971:
! 972: }
! 973:
! 974: if (k1 == 0) {
! 975: data[0] = '1';
! 976: }
! 977: else {
! 978:
! 979: if (partition[j] == DELIM && key[k] == EOL) {
! 980:
! 981: data[1] = data[0];
! 982: data[0] = '1';
! 983:
! 984: return;
! 985:
! 986: }
! 987:
! 988: if (k1 < 0 && k < 2) return;
! 989:
! 990: }
! 991:
! 992: i += UNSIGN (partition[i]); /* skip key */
! 993: i += UNSIGN (partition[i]) + 1; /* skip data */
! 994:
! 995: j = i;
! 996: k = 0;
! 997:
! 998: } while (i < PSIZE);
! 999:
! 1000: }
! 1001:
! 1002: return;
! 1003:
! 1004:
! 1005:
! 1006: /* end of $data section */
! 1007:
! 1008:
! 1009:
! 1010:
! 1011: case getinc: /* increment by one and retrieve */
! 1012:
! 1013: #ifdef DEBUG_SYM
! 1014:
! 1015: printf ("DEBUG GETINC: ");
! 1016: printf ("[key] is [");
! 1017:
! 1018: for (loop = 0; key[loop] != EOL; loop++) printf ("%c", (key[loop] == DELIM) ? '!' : key[loop]);
! 1019:
! 1020: printf("]\r\n");
! 1021: #endif
! 1022:
! 1023:
! 1024: if ((i = alphptr[(int) key[0]])) {
! 1025:
! 1026: j = i + 1;
! 1027: k = 1;
! 1028:
! 1029: do {
! 1030:
! 1031: while (key[k] == partition[++j]) { /* compare keys */
! 1032:
! 1033: if (key[k] == EOL) {
! 1034:
! 1035: i = UNSIGN (partition[++j]);
! 1036: stcpy0 (data, &partition[j + 1], i);
! 1037:
! 1038: data[i] = EOL; /* data retrieved ... now increment */
! 1039:
! 1040: /****************increment by one*******************/
! 1041: if (i == 0) i++; /* if data was empty use EOL as dummy value */
! 1042: if (i > 1 && data[0] == '0') i++; /* leading zero use EOL as dummy value */
! 1043:
! 1044: k = 0;
! 1045:
! 1046: while (k < i) {
! 1047:
! 1048: if ((k1 = data[k++]) < '0' || k1 > '9') { /* no positive integer */
! 1049:
! 1050: numlit (data);
! 1051:
! 1052: tmp1[0] = '1';
! 1053: tmp1[1] = EOL;
! 1054:
! 1055: add (data, tmp1);
! 1056:
! 1057: datal = stlen (data);
! 1058:
! 1059: i = j;
! 1060:
! 1061: nocompact = FALSE; /* getinc needs compacted symtab */
! 1062:
! 1063: goto old0;
! 1064:
! 1065: }
! 1066:
! 1067: }
! 1068:
! 1069: k1 = k--; /* length of string */
! 1070:
! 1071: while ((partition[j + 1 + k] = ++data[k]) > '9') {
! 1072:
! 1073: partition[j + 1 + k] = '0';
! 1074: data[k--] = '0';
! 1075:
! 1076: if (k < 0) {
! 1077:
! 1078: k = k1;
! 1079:
! 1080: while (k >= 0) {
! 1081: data[k + 1] = data[k];
! 1082: k--;
! 1083: }
! 1084:
! 1085: data[0] = '1';
! 1086: s = &partition[--symlen] - 256;
! 1087:
! 1088: if (alphptr['%']) alphptr['%']--;
! 1089:
! 1090: for (k = 'A'; k <= key[0]; k++) {
! 1091: if (alphptr[k]) alphptr[k]--;
! 1092: }
! 1093:
! 1094: k = j - 1;
! 1095: j = symlen;
! 1096:
! 1097: stcpy0 (&partition[j], &partition[j + 1], k - j);
! 1098:
! 1099: partition[k] = (char) ++i;
! 1100: partition[++k] = '1';
! 1101:
! 1102: return;
! 1103:
! 1104: }
! 1105:
! 1106: }
! 1107:
! 1108: return;
! 1109:
! 1110: /************end increment by one*******************/
! 1111:
! 1112: }
! 1113:
! 1114: k++;
! 1115:
! 1116: }
! 1117:
! 1118: /** if (key[k]<partition[j]) break; **/
! 1119: i += UNSIGN (partition[i]); /* skip key */
! 1120: i += UNSIGN (partition[i]) + 1; /* skip data */
! 1121:
! 1122: j = i;
! 1123: k = 0;
! 1124:
! 1125: } while (i < PSIZE);
! 1126:
! 1127: }
! 1128:
! 1129: data[0] = EOL;
! 1130: merr_raise (UNDEF);
! 1131:
! 1132: return;
! 1133:
! 1134:
! 1135:
! 1136: case fra_order: /* next one please */
! 1137:
! 1138:
! 1139: if (ordercnt < 0) goto zinv;
! 1140:
! 1141: k1 = (j = stcpy (tmp1, key) - 1);
! 1142:
! 1143: while (tmp1[k1] != DELIM) {
! 1144: if ((--k1) <= 0) goto unsubscr;
! 1145: }
! 1146:
! 1147: tmp1[++k1] = EOL;
! 1148:
! 1149: stcpy (tmp2, &key[k1]);
! 1150:
! 1151: if (ordercnt == 0) {
! 1152:
! 1153: stcpy (data, tmp2);
! 1154:
! 1155: l_o_val[0] = EOL;
! 1156:
! 1157: return;
! 1158:
! 1159: }
! 1160:
! 1161: data[0] = EOL;
! 1162:
! 1163: if ((i = alphptr[(int) key[0]]) == 0) {
! 1164:
! 1165: l_o_val[0] = EOL;
! 1166:
! 1167:
! 1168: return;
! 1169:
! 1170: }
! 1171:
! 1172: /***************************/
! 1173: /* frequent special case: the key of which we search the next
! 1174: * entry is defined ! */
! 1175: if (tmp2[0] != EOL) {
! 1176:
! 1177: if (tryfast && stcmp (key, &partition[tryfast + 1]) == 0) {
! 1178:
! 1179: j = tryfast;
! 1180: j += UNSIGN (partition[j]); /* skip key */
! 1181: j += UNSIGN (partition[j]) + 1; /* skip data */
! 1182:
! 1183: goto begorder;
! 1184:
! 1185: }
! 1186:
! 1187: k = 1;
! 1188: j = i + 1; /* first char always matches! */
! 1189:
! 1190: do {
! 1191:
! 1192: while (key[k] == partition[++j]) { /* compare keys */
! 1193:
! 1194: if (key[k++] == EOL) {
! 1195: j = i;
! 1196: goto begorder;
! 1197: }
! 1198:
! 1199: }
! 1200:
! 1201: i += UNSIGN (partition[i]); /* skip key */
! 1202: i += UNSIGN (partition[i]) + 1; /* skip data */
! 1203:
! 1204: k = 0;
! 1205: j = i;
! 1206:
! 1207: } while (i < PSIZE);
! 1208:
! 1209: }
! 1210:
! 1211: /* the key was not defined */
! 1212: /***************************/
! 1213: j = alphptr[(int) key[0]];
! 1214:
! 1215: begorder:
! 1216:
! 1217: do {
! 1218:
! 1219: if (key[0] != partition[j + 1]) {
! 1220:
! 1221: l_o_val[0] = EOL;
! 1222: data[0] = EOL;
! 1223:
! 1224: return;
! 1225:
! 1226: }
! 1227:
! 1228: stcpy0 (data, &partition[j + 1], k1);
! 1229:
! 1230: data[k1] = EOL;
! 1231:
! 1232: if (stcmp (tmp1, data) == 0) {
! 1233:
! 1234: stcpy (data, &partition[j + 1 + k1]); /* index on same level */
! 1235:
! 1236: k = 0;
! 1237:
! 1238: while (data[k] != EOL && data[k] != DELIM) k++;
! 1239:
! 1240: data[k] = EOL;
! 1241:
! 1242: if (collate (tmp2, data)) {
! 1243:
! 1244: if (--ordercnt <= 0) {
! 1245:
! 1246: tryfast = j;
! 1247:
! 1248: /* save data value for inspection with $V(110) */
! 1249: j += UNSIGN (partition[j]); /* skip key */
! 1250: k = UNSIGN (partition[j++]);
! 1251: stcpy0 (l_o_val, &partition[j], k);
! 1252:
! 1253: l_o_val[k] = EOL;
! 1254:
! 1255: return;
! 1256:
! 1257: }
! 1258:
! 1259: ordercounter++;
! 1260:
! 1261: }
! 1262:
! 1263: }
! 1264:
! 1265: j += UNSIGN (partition[j]); /* skip key */
! 1266: j += UNSIGN (partition[j]) + 1; /* skip data */
! 1267:
! 1268: } while (j < PSIZE);
! 1269:
! 1270: data[0] = EOL;
! 1271: tryfast = 0;
! 1272: l_o_val[0] = EOL;
! 1273:
! 1274: return;
! 1275:
! 1276: /* end of $order section */
! 1277:
! 1278:
! 1279: case kill_all:
! 1280:
! 1281: genocid:
! 1282:
! 1283:
! 1284:
! 1285:
! 1286: /* Old genocide routine */
! 1287: alphptr['%'] = 0;
! 1288:
! 1289: for (i = 'A'; i <= 'z'; alphptr[i++] = 0) ;
! 1290:
! 1291: symlen = PSIZE;
! 1292: s = &partition[symlen] - 256;
! 1293: tryfast = 0;
! 1294:
! 1295: ssvn_system_update ();
! 1296: ssvn_job_update ();
! 1297: ssvn_routine_update ();
! 1298:
! 1299: #if defined(HAVE_MWAPI_MOTIF)
! 1300: ssvn_display_update ();
! 1301: #endif
! 1302:
! 1303: const_restore ();
! 1304:
! 1305: return;
! 1306:
! 1307:
! 1308:
! 1309:
! 1310: case kill_sym: /* kill them dirty bloody variables */
! 1311:
! 1312:
! 1313: /* Old Kill Routine */
! 1314:
! 1315: if ((i = alphptr[(int) key[0]]) == 0) return; /* damn - nothing to kill */
! 1316:
! 1317: kill_from = 0;
! 1318:
! 1319: while (i < PSIZE) {
! 1320:
! 1321: j = i;
! 1322: k = 0;
! 1323:
! 1324: while ((k1 = key[k]) == partition[++j]) { /* compare keys */
! 1325:
! 1326: if (k1 == EOL) break;
! 1327:
! 1328: k++;
! 1329:
! 1330: }
! 1331:
! 1332: if (k1 == EOL && (partition[j] == DELIM || partition[j] == EOL)) {
! 1333:
! 1334: if (kill_from == 0) kill_from = i;
! 1335:
! 1336: }
! 1337: else {
! 1338: if (kill_from) break;
! 1339: }
! 1340:
! 1341: i += UNSIGN (partition[i]); /* skip key */
! 1342: i += UNSIGN (partition[i]) + 1; /* skip data */
! 1343:
! 1344: }
! 1345:
! 1346: k_entry: /* entry from killone section */
! 1347:
! 1348:
! 1349: if (kill_from) {
! 1350:
! 1351: j = i - kill_from;
! 1352: symlen += j;
! 1353: s = &partition[symlen] - 256;
! 1354:
! 1355: for (k = 36; k < key[0]; k++) {
! 1356: if (alphptr[k]) alphptr[k] += j;
! 1357: }
! 1358:
! 1359: if (alphptr[k] == kill_from) {
! 1360:
! 1361: alphptr[k] = i;
! 1362:
! 1363: if (partition[i + 1] != key[0]) alphptr[k] = 0;
! 1364:
! 1365: }
! 1366: else {
! 1367: alphptr[k] += j;
! 1368: }
! 1369:
! 1370: /* j=i-j; while(i>symlen) partition[--i]=partition[--j]; */
! 1371: stcpy1 (&partition[i - 1], &partition[i - j - 1], i - symlen);
! 1372:
! 1373: }
! 1374:
! 1375: tryfast = 0;
! 1376:
! 1377: return;
! 1378:
! 1379:
! 1380: /* end of kill_sym section */
! 1381:
! 1382: case killone: /* kill one variable, not descendants */
! 1383:
! 1384:
! 1385: if ((i = alphptr[(int) key[0]]) == 0) return; /* nothing to kill */
! 1386:
! 1387: kill_from = 0;
! 1388:
! 1389: while (i < PSIZE) {
! 1390:
! 1391: j = i;
! 1392: k = 0;
! 1393:
! 1394: while ((k1 = key[k]) == partition[++j]) { /* compare keys */
! 1395:
! 1396: if (k1 == EOL) break;
! 1397:
! 1398: k++;
! 1399:
! 1400: }
! 1401:
! 1402: k = i;
! 1403: i += UNSIGN (partition[i]); /* skip key */
! 1404: i += UNSIGN (partition[i]) + 1; /* skip data */
! 1405:
! 1406: if (k1 == EOL) {
! 1407:
! 1408: if (partition[j] == DELIM) return; /* descendant */
! 1409:
! 1410: kill_from = k;
! 1411:
! 1412: goto k_entry;
! 1413:
! 1414: }
! 1415:
! 1416: }
! 1417:
! 1418: tryfast = 0;
! 1419:
! 1420: return;
! 1421:
! 1422:
! 1423: /* end of killone section */
! 1424:
! 1425: case killexcl: /* exclusive kill */
! 1426:
! 1427:
! 1428: i = symlen;
! 1429:
! 1430: while (i < PSIZE) {
! 1431:
! 1432: tmp2[0] = SP;
! 1433: kill_from = i;
! 1434:
! 1435: stcpy (tmp3, &partition[i + 1]);
! 1436: stcpy (&tmp2[1], tmp3);
! 1437: stcat (tmp2, " \201");
! 1438:
! 1439: i += UNSIGN (partition[i]);
! 1440: i += UNSIGN (partition[i]) + 1;
! 1441:
! 1442: if (kill_ok (key, tmp2) == 0) continue; /* don't kill */
! 1443:
! 1444: while (i < PSIZE) {
! 1445:
! 1446: j = i;
! 1447: k = 0;
! 1448:
! 1449: while ((k1 = tmp3[k]) == partition[++j]) { /* compare keys */
! 1450:
! 1451: if (k1 == EOL) break;
! 1452:
! 1453: k++;
! 1454:
! 1455: }
! 1456:
! 1457: if (k1 != EOL || (partition[j] != DELIM && partition[j] != EOL)) break;
! 1458:
! 1459: i += UNSIGN (partition[i]); /* skip key */
! 1460: i += UNSIGN (partition[i]) + 1; /* skip data */
! 1461:
! 1462: }
! 1463:
! 1464: j = i - kill_from;
! 1465: symlen += j;
! 1466: s = &partition[symlen] - 256;
! 1467:
! 1468: for (k = 36; k < tmp3[0]; k++) {
! 1469: if (alphptr[k]) alphptr[k] += j;
! 1470: }
! 1471:
! 1472: if (alphptr[k] == kill_from) {
! 1473:
! 1474: alphptr[k] = i;
! 1475:
! 1476: if (partition[i + 1] != tmp3[0]) alphptr[k] = 0;
! 1477:
! 1478: }
! 1479: else {
! 1480: alphptr[k] += j;
! 1481: }
! 1482:
! 1483: stcpy1 (&partition[i - 1], &partition[i - j - 1], i - symlen);
! 1484:
! 1485: i = kill_from + j;
! 1486:
! 1487: }
! 1488:
! 1489: tryfast = 0;
! 1490:
! 1491: return;
! 1492:
! 1493: /* end of killexcl section */
! 1494:
! 1495: case fra_query: /* next entry */
! 1496: case bigquery:
! 1497:
! 1498:
! 1499: if (ordercnt == 0) {
! 1500:
! 1501: l_o_val[0] = EOL;
! 1502:
! 1503: zname (data, key + stlen (i_with));
! 1504:
! 1505: return;
! 1506:
! 1507: }
! 1508:
! 1509: /***************************/
! 1510: /* frequent special case: the key which we search for is the next
! 1511: * entry */
! 1512:
! 1513: if ((i = alphptr[(int) key[0]])) {
! 1514:
! 1515: if (stcmp (key, &partition[tryfast + 1]) == 0) {
! 1516: i = tryfast;
! 1517: }
! 1518: else {
! 1519:
! 1520: j = i;
! 1521:
! 1522: do {
! 1523:
! 1524: if (stcmp (key, &partition[j + 1]) == 0) {
! 1525: i = j;
! 1526: break;
! 1527: }
! 1528:
! 1529: j += UNSIGN (partition[j]); /* skip key */
! 1530: j += UNSIGN (partition[j]) + 1; /* skip data */
! 1531:
! 1532: } while (j < PSIZE);
! 1533:
! 1534: }
! 1535: }
! 1536: else {
! 1537: i = symlen; /* no previous entry */
! 1538: }
! 1539: /***************************/
! 1540:
! 1541:
! 1542: /* check whether the key has subscripts or not */
! 1543: k1 = 0;
! 1544: k = 1;
! 1545:
! 1546: while (key[k] != EOL) {
! 1547:
! 1548: if (key[k++] == DELIM) {
! 1549: k1 = k;
! 1550: break;
! 1551: }
! 1552:
! 1553: }
! 1554:
! 1555: while (i < PSIZE) {
! 1556:
! 1557: j = i;
! 1558: k = 0;
! 1559:
! 1560: while (key[k] == partition[++j]) { /* compare keys */
! 1561:
! 1562: if (key[k] == EOL) break;
! 1563:
! 1564: k++;
! 1565:
! 1566: }
! 1567:
! 1568: if (key[k] == EOL) {
! 1569:
! 1570: if (partition[j] == EOL) {
! 1571: i += UNSIGN (partition[i]);
! 1572: i += UNSIGN (partition[i]) + 1;
! 1573: }
! 1574:
! 1575: break;
! 1576:
! 1577: }
! 1578:
! 1579: if (k < k1 || k1 == 0) {
! 1580:
! 1581: if (key[k] < partition[j]) break;
! 1582:
! 1583: }
! 1584: else {
! 1585: long m, n, o, ch;
! 1586:
! 1587: /* get complete subscripts */
! 1588: n = k;
! 1589:
! 1590: while (key[--n] != DELIM) ;
! 1591:
! 1592: n++;
! 1593: m = j + n - k;
! 1594: o = 0;
! 1595:
! 1596: while ((ch = tmp2[o++] = key[n++]) != EOL && ch != DELIM) ;
! 1597:
! 1598: if (ch == DELIM) tmp2[--o] = EOL;
! 1599:
! 1600: o = 0;
! 1601:
! 1602: while ((ch = tmp3[o++] = partition[m++]) != EOL && ch != DELIM) ;
! 1603:
! 1604: if (ch == DELIM) tmp3[--o] = EOL;
! 1605:
! 1606: if (collate (tmp2, tmp3)) break;
! 1607:
! 1608: }
! 1609:
! 1610: i += UNSIGN (partition[i]); /* skip key */
! 1611: i += UNSIGN (partition[i]) + 1; /* skip data */
! 1612:
! 1613: }
! 1614:
! 1615: /* multiple backward query */
! 1616: if (ordercnt < 0) {
! 1617:
! 1618: j = symlen;
! 1619: k = ordercnt - 1;
! 1620:
! 1621: while (j < i) { /* count entries */
! 1622:
! 1623: j += UNSIGN (partition[j]); /* skip key */
! 1624: j += UNSIGN (partition[j]) + 1; /* skip data */
! 1625:
! 1626: k++;
! 1627:
! 1628: }
! 1629:
! 1630: if (k < 0) {
! 1631:
! 1632: data[0] = EOL;
! 1633: l_o_val[0] = EOL;
! 1634:
! 1635: return;
! 1636:
! 1637: }
! 1638:
! 1639: i = symlen;
! 1640:
! 1641: while (--k >= 0) {
! 1642:
! 1643: i += UNSIGN (partition[i]); /* skip key */
! 1644: i += UNSIGN (partition[i]) + 1; /* skip data */
! 1645:
! 1646: }
! 1647:
! 1648: }
! 1649: /* end: multiple backward query */
! 1650:
! 1651: while (--ordercnt > 0) { /* multiple forward $query */
! 1652:
! 1653: if (i >= PSIZE) break;
! 1654:
! 1655: i += UNSIGN (partition[i]); /* skip key */
! 1656: i += UNSIGN (partition[i]) + 1; /* skip data */
! 1657:
! 1658: }
! 1659:
! 1660: /* now 'i' is pointer to 'next' entry */
! 1661: tryfast = i;
! 1662:
! 1663: /* save data value for inspection with $V(110) */
! 1664: j = i;
! 1665:
! 1666: j += UNSIGN (partition[j]);
! 1667: k = UNSIGN (partition[j]);
! 1668:
! 1669: stcpy0 (l_o_val, &partition[j + 1], k);
! 1670: l_o_val[k] = EOL;
! 1671:
! 1672: keyl = i;
! 1673: keyl += UNSIGN (partition[i++]) - 2;
! 1674:
! 1675: /* action==bigquery may return a result in a different lvn */
! 1676: /* which is illegal with $query() */
! 1677: if (action == fra_query) {
! 1678:
! 1679: k = 0; /* is result same lvn? */
! 1680:
! 1681: while (partition[i+k] == key[k]) {
! 1682:
! 1683: if (key[k] == DELIM) break;
! 1684:
! 1685: k++;
! 1686:
! 1687: }
! 1688:
! 1689: if (partition[i+k] != DELIM) i = keyl + 1; /* discard result! */
! 1690:
! 1691: }
! 1692:
! 1693: if (i <= keyl) {
! 1694: zname (data, &partition[i + stlen (i_with)]);
! 1695: }
! 1696: else {
! 1697: data[0] = EOL;
! 1698: }
! 1699:
! 1700: return;
! 1701: /* end of $query section */
! 1702:
! 1703: zinv: /* previous one please */
! 1704:
! 1705: data[0] = EOL;
! 1706: l_o_val[0] = EOL;
! 1707:
! 1708: k1 = (j = stcpy (tmp1, key) - 1);
! 1709:
! 1710: while (tmp1[k1] != DELIM) {
! 1711:
! 1712: if ((--k1) <= 0) {
! 1713: merr_raise (NEXTER);
! 1714: return;
! 1715: }
! 1716:
! 1717: }
! 1718:
! 1719: tmp1[++k1] = EOL;
! 1720:
! 1721: stcpy (tmp2, &key[k1]);
! 1722:
! 1723: if (tmp2[0] == EOL) {
! 1724:
! 1725: tmp2[0] = DEL;
! 1726: tmp2[1] = DEL;
! 1727: tmp2[2] = EOL;
! 1728:
! 1729: }
! 1730:
! 1731: k = (int) (key[0]);
! 1732:
! 1733: if (alphptr[k] == 0) return;
! 1734:
! 1735: j = alphptr[k];
! 1736:
! 1737: do {
! 1738:
! 1739: if (key[0] != partition[j + 1]) goto zinvend;
! 1740:
! 1741: stcpy0 (tmp3, &partition[j + 1], k1);
! 1742:
! 1743: tmp3[k1] = EOL;
! 1744:
! 1745: if (stcmp (tmp1, tmp3) == 0) {
! 1746:
! 1747: stcpy (tmp3, &partition[j + 1 + k1]); /* index on same level */
! 1748:
! 1749: k = 0;
! 1750:
! 1751: while (tmp3[k] != EOL && tmp3[k] != DELIM) k++;
! 1752:
! 1753: tmp3[k] = EOL;
! 1754:
! 1755: if (collate (tmp3, tmp2) == FALSE) goto zinvend;
! 1756:
! 1757: stcpy (data, tmp3);
! 1758:
! 1759: /* save data value for inspection with $V(110) */
! 1760: i = j;
! 1761:
! 1762: i += UNSIGN (partition[i]);
! 1763: k = UNSIGN (partition[i]);
! 1764:
! 1765: stcpy0 (l_o_val, &partition[i + 1], k);
! 1766:
! 1767: l_o_val[k] = EOL;
! 1768:
! 1769: }
! 1770:
! 1771: j += UNSIGN (partition[j]); /* skip key */
! 1772: j += UNSIGN (partition[j]) + 1; /* skip data */
! 1773:
! 1774: } while (j < PSIZE);
! 1775:
! 1776: zinvend:
! 1777:
! 1778: if (data[0] == EOL) return;
! 1779:
! 1780: ordercounter++;
! 1781:
! 1782: if (++ordercnt >= 0) return;
! 1783:
! 1784: stcpy (&key[k1], data);
! 1785:
! 1786: goto zinv;
! 1787:
! 1788:
! 1789:
! 1790:
! 1791:
! 1792: /* end of $zinverse section */
! 1793:
! 1794:
! 1795: case new_sym: /* new one symbol */
! 1796:
! 1797: if (key[0] == '$') { /* $svn: save current value on new stack */
! 1798:
! 1799:
! 1800:
! 1801: if (newptr > newlimit && getnewmore ()) return;
! 1802:
! 1803: if ((key[1] | 0140) == 't') { /* NEW $TEST */
! 1804:
! 1805: *newptr++ = test;
! 1806: *newptr++ = EOL;
! 1807: *newptr++ = 1;
! 1808:
! 1809: k1 = stcpy (newptr, "$t\201");
! 1810:
! 1811: newptr += k1;
! 1812: *newptr++ = EOL;
! 1813: *newptr++ = k1;
! 1814: *newptr++ = set_sym;
! 1815:
! 1816: //if (mcmnd != ZNEW) test = FALSE;
! 1817:
! 1818: return;
! 1819:
! 1820: }
! 1821:
! 1822: if ((key[1] | 0140) == 'j') { /* NEW $JOB */
! 1823:
! 1824: *newptr++ = pid / 256;
! 1825: *newptr++ = pid % 256;
! 1826: *newptr++ = EOL;
! 1827: *newptr++ = 2;
! 1828:
! 1829: k1 = stcpy (newptr, "$j\201");
! 1830:
! 1831: newptr += k1;
! 1832: *newptr++ = EOL;
! 1833: *newptr++ = k1;
! 1834: *newptr++ = set_sym;
! 1835:
! 1836: return;
! 1837:
! 1838: }
! 1839:
! 1840: if (((key[1] | 0140) == 'z') && ((key[2] | 0140) == 'i')) { /* NEW $ZINRPT */
! 1841:
! 1842: *newptr++ = breakon;
! 1843: *newptr++ = EOL;
! 1844: *newptr++ = 1;
! 1845:
! 1846: k1 = stcpy (newptr, "$zi\201");
! 1847:
! 1848: newptr += k1;
! 1849: *newptr++ = EOL;
! 1850: *newptr++ = k1;
! 1851: *newptr++ = set_sym;
! 1852:
! 1853: return;
! 1854:
! 1855: }
! 1856:
! 1857:
! 1858: /* NEW $ETRAP added 10 Oct 2020, JPW */
! 1859: if (((key[1] | 0140) == 'e') && ((key[2] | 0140) == 't')) { /* NEW $ETRAP */
! 1860:
! 1861: j = stcpy (newptr, etrap);
! 1862:
! 1863: newptr += j;
! 1864: *newptr++ = EOL;
! 1865: *newptr++ = j;
! 1866:
! 1867: k1 = stcpy (newptr, "$et\201");
! 1868:
! 1869: newptr += k1;
! 1870: *newptr++ = EOL;
! 1871: *newptr++ = k1;
! 1872: *newptr++ = set_sym;
! 1873:
! 1874: return;
! 1875:
! 1876: }
! 1877:
! 1878: /* NEW $ESTACK added 12 Oct 2020, JPW */
! 1879: if (((key[1] | 0140) == 'e') && ((key[2] | 0140) == 's')) { /* NEW $ESTACK */
! 1880:
! 1881: char esbuf[256];
! 1882:
! 1883: snprintf (esbuf, 255, "%d\201", estack);
! 1884:
! 1885: j = stcpy (newptr, esbuf);
! 1886:
! 1887: newptr += j;
! 1888: *newptr++ = EOL;
! 1889: *newptr++ = j;
! 1890:
! 1891: k1 = stcpy (newptr, "$es\201");
! 1892:
! 1893: newptr += k1;
! 1894: *newptr++ = EOL;
! 1895: *newptr++ = k1;
! 1896: *newptr++ = set_sym;
! 1897:
! 1898: estack = 0;
! 1899:
! 1900: return;
! 1901:
! 1902: }
! 1903:
! 1904: j = stcpy (newptr, zref); /* NEW $ZREFERENCE */
! 1905:
! 1906: newptr += j;
! 1907: *newptr++ = EOL;
! 1908: *newptr++ = j;
! 1909:
! 1910: k1 = stcpy (newptr, "$zr\201");
! 1911:
! 1912: newptr += k1;
! 1913: *newptr++ = EOL;
! 1914: *newptr++ = nakoffs;
! 1915:
! 1916: k1++;
! 1917:
! 1918: *newptr++ = k1;
! 1919: *newptr++ = set_sym;
! 1920:
! 1921: if (mcmnd != ZNEW) zref[0] = EOL;
! 1922:
! 1923: return;
! 1924:
! 1925:
! 1926: }
! 1927:
! 1928:
! 1929:
! 1930: if ((i = alphptr[(int) key[0]])) { /* is there something to be saved?/killed */
! 1931:
! 1932: /* always FALSE with special variables */
! 1933: kill_from = 0;
! 1934:
! 1935: while (i < PSIZE) {
! 1936:
! 1937: j = i;
! 1938: k = 0;
! 1939:
! 1940: while ((k1 = key[k]) == partition[++j]) { /* compare keys */
! 1941:
! 1942: if (k1 == EOL) break;
! 1943:
! 1944: k++;
! 1945:
! 1946: }
! 1947:
! 1948: if (k1 == EOL && (partition[j] == DELIM || partition[j] == EOL)) {
! 1949:
! 1950: if (kill_from == 0) kill_from = i;
! 1951:
! 1952: }
! 1953: else {
! 1954: if (kill_from) break;
! 1955: }
! 1956:
! 1957: if (kill_from) { /* save current values on new stack */
! 1958:
! 1959: j = UNSIGN (partition[i]);
! 1960: k = i + 1;
! 1961: k1 = j;
! 1962: i += j;
! 1963: j = UNSIGN (partition[i]);
! 1964:
! 1965: if (newptr > newlimit && getnewmore ()) return;
! 1966:
! 1967: #ifdef DEBUG_SYM
! 1968:
! 1969: start = newptr;
! 1970:
! 1971: #endif
! 1972:
! 1973: stcpy0 (newptr, &partition[i + 1], j);
! 1974:
! 1975: newptr += j;
! 1976: *newptr++ = EOL;
! 1977: *newptr++ = j;
! 1978:
! 1979: i += (j + 1);
! 1980:
! 1981: stcpy0 (newptr, &partition[k], k1);
! 1982:
! 1983: newptr += k1;
! 1984: *newptr++ = EOL;
! 1985: *newptr++ = k1;
! 1986: *newptr++ = set_sym;
! 1987:
! 1988: #ifdef DEBUG_SYM
! 1989:
! 1990: printf ("SAVING [newptr] newptr became [");
! 1991:
! 1992: while (start < newptr) {
! 1993:
! 1994: printf ("%c(%d)", (*start==EOL) ? ('!') : *start, *start);
! 1995:
! 1996: start++;
! 1997:
! 1998: }
! 1999:
! 2000: printf("{%d}]\r\n", *(newptr - 1));
! 2001:
! 2002: #endif
! 2003:
! 2004: }
! 2005: else {
! 2006:
! 2007: i += UNSIGN (partition[i]); /* skip key */
! 2008: i += UNSIGN (partition[i]) + 1; /* skip data */
! 2009:
! 2010: }
! 2011:
! 2012: }
! 2013:
! 2014: if (kill_from && mcmnd != ZNEW) {
! 2015:
! 2016: j = i - kill_from;
! 2017: symlen += j;
! 2018: s = &partition[symlen] - 256;
! 2019:
! 2020: for (k = 36; k < key[0]; k++) {
! 2021: if (alphptr[k]) alphptr[k] += j;
! 2022: }
! 2023:
! 2024: if (alphptr[k] == kill_from) {
! 2025:
! 2026: alphptr[k] = i;
! 2027:
! 2028: if (partition[i + 1] != key[0]) alphptr[k] = 0;
! 2029:
! 2030: }
! 2031: else {
! 2032: alphptr[k] += j;
! 2033: }
! 2034:
! 2035: stcpy1 (&partition[i - 1], &partition[i - j - 1], i - symlen);
! 2036:
! 2037: }
! 2038:
! 2039: tryfast = 0;
! 2040:
! 2041: }
! 2042:
! 2043: if (newptr > newlimit && getnewmore ()) return;
! 2044:
! 2045: #ifdef DEBUG_SYM
! 2046: start = newptr;
! 2047: #endif
! 2048:
! 2049: j = stcpy (newptr, key);
! 2050:
! 2051: newptr += j;
! 2052: *newptr++ = EOL;
! 2053: *newptr++ = j;
! 2054: *newptr++ = kill_sym;
! 2055:
! 2056: #ifdef DEBUG_SYM
! 2057:
! 2058: printf ("KILLING [newptr] newptr became [");
! 2059:
! 2060: while (start < newptr) {
! 2061: printf ("%c(%d)", (*start == EOL) ? ('!') : *start,*start );
! 2062:
! 2063: start++;
! 2064:
! 2065: }
! 2066:
! 2067: printf ("{%d}]\r\n", *(newptr - 1));
! 2068:
! 2069: #endif
! 2070:
! 2071: return;
! 2072:
! 2073: /* end of new_sym section */
! 2074:
! 2075:
! 2076: case new_all: /* new all symbols */
! 2077:
! 2078:
! 2079:
! 2080: i = symlen;
! 2081:
! 2082: while (i < PSIZE) {
! 2083:
! 2084: j = UNSIGN (partition[i]);
! 2085: k = i + 1;
! 2086: k1 = j;
! 2087: i += j;
! 2088: j = UNSIGN (partition[i]);
! 2089:
! 2090: if (newptr > newlimit && getnewmore ()) return;
! 2091:
! 2092: stcpy0 (newptr, &partition[i + 1], j);
! 2093:
! 2094: newptr += j;
! 2095: *newptr++ = EOL;
! 2096: *newptr++ = j;
! 2097: i += (j + 1);
! 2098:
! 2099: stcpy0 (newptr, &partition[k], k1);
! 2100:
! 2101: newptr += k1;
! 2102: *newptr++ = EOL;
! 2103: *newptr++ = k1;
! 2104: *newptr++ = set_sym;
! 2105:
! 2106: }
! 2107:
! 2108: *newptr++ = kill_all;
! 2109:
! 2110: if (mcmnd == ZNEW) return;
! 2111:
! 2112: goto genocid; /* ... and now kill them all */
! 2113:
! 2114: /* end of new_all section */
! 2115:
! 2116:
! 2117: case newexcl: /* new all except specified */
! 2118:
! 2119:
! 2120:
! 2121: i = symlen;
! 2122:
! 2123: while (i < PSIZE) {
! 2124:
! 2125: tmp2[0] = SP;
! 2126: kill_from = i;
! 2127:
! 2128: stcpy (tmp3, &partition[i + 1]);
! 2129: stcpy (&tmp2[1], tmp3);
! 2130: stcat (tmp2, " \201");
! 2131:
! 2132: if (kill_ok (key, tmp2) == 0) { /* don't new */
! 2133:
! 2134: i += UNSIGN (partition[i]);
! 2135: i += UNSIGN (partition[i]) + 1;
! 2136:
! 2137: continue;
! 2138:
! 2139: }
! 2140:
! 2141: j = UNSIGN (partition[i]);
! 2142: k = i + 1;
! 2143: k1 = j;
! 2144: i += j;
! 2145: j = UNSIGN (partition[i]);
! 2146:
! 2147: if (newptr > newlimit && getnewmore ()) return;
! 2148:
! 2149: stcpy0 (newptr, &partition[i + 1], j);
! 2150:
! 2151: newptr += j;
! 2152: *newptr++ = EOL;
! 2153: *newptr++ = j;
! 2154: i += (j + 1);
! 2155:
! 2156: stcpy0 (newptr, &partition[k], k1);
! 2157:
! 2158: newptr += k1;
! 2159: *newptr++ = EOL;
! 2160: *newptr++ = k1;
! 2161: *newptr++ = set_sym;
! 2162:
! 2163: while (i < PSIZE) {
! 2164:
! 2165: j = i;
! 2166: k = 0;
! 2167:
! 2168: while ((k1 = tmp3[k]) == partition[++j]) { /* compare keys */
! 2169:
! 2170: if (k1 == EOL) break;
! 2171:
! 2172: k++;
! 2173:
! 2174: }
! 2175:
! 2176: if (k1 != EOL || (partition[j] != DELIM && partition[j] != EOL)) break;
! 2177:
! 2178: j = UNSIGN (partition[i]);
! 2179: k = i + 1;
! 2180: k1 = j;
! 2181: i += j;
! 2182: j = UNSIGN (partition[i]);
! 2183:
! 2184: if (newptr > newlimit && getnewmore ()) return;
! 2185:
! 2186: stcpy0 (newptr, &partition[i + 1], j);
! 2187:
! 2188: newptr += j;
! 2189: *newptr++ = EOL;
! 2190: *newptr++ = j;
! 2191: i += (j + 1);
! 2192:
! 2193: stcpy0 (newptr, &partition[k], k1);
! 2194:
! 2195: newptr += k1;
! 2196: *newptr++ = EOL;
! 2197: *newptr++ = k1;
! 2198: *newptr++ = set_sym;
! 2199:
! 2200: }
! 2201:
! 2202: if (mcmnd == ZNEW) continue;
! 2203:
! 2204: j = i - kill_from;
! 2205: symlen += j;
! 2206: s = &partition[symlen] - 256;
! 2207:
! 2208: for (k = 36; k < tmp3[0]; k++) {
! 2209:
! 2210: if (alphptr[k]) alphptr[k] += j;
! 2211:
! 2212: }
! 2213:
! 2214: if (alphptr[k] == kill_from) {
! 2215:
! 2216: alphptr[k] = i;
! 2217:
! 2218: if (partition[i + 1] != tmp3[0]) alphptr[k] = 0;
! 2219:
! 2220: }
! 2221: else {
! 2222: alphptr[k] += j;
! 2223: }
! 2224:
! 2225: stcpy1 (&partition[i - 1], &partition[i - j - 1], i - symlen);
! 2226:
! 2227: i = kill_from + j;
! 2228:
! 2229: }
! 2230:
! 2231: tryfast = 0;
! 2232:
! 2233: if (newptr > newlimit && getnewmore ()) return;
! 2234:
! 2235: j = stcpy (newptr, key);
! 2236:
! 2237: newptr += (j + 1);
! 2238: *newptr++ = j;
! 2239: *newptr++ = killexcl;
! 2240:
! 2241: return;
! 2242:
! 2243:
! 2244: /* end of newexcl section */
! 2245:
! 2246:
! 2247: case m_alias: /* define an alias of a variable */
! 2248:
! 2249:
! 2250: /* process stuff */
! 2251: if (stcmp (key, data) == 0) return; /* sorry, that's no alias */
! 2252:
! 2253: if (data[0] == EOL) { /* delete an alias from the table */
! 2254:
! 2255: if (aliases) { /* there are aliases */
! 2256:
! 2257: i = 0;
! 2258:
! 2259: while (i < aliases) {
! 2260:
! 2261: k = i;
! 2262: k1 = i + UNSIGN (ali[i]) + 1;
! 2263: j = 0; /* is current reference an alias ??? */
! 2264:
! 2265: while (ali[++i] == key[j]) {
! 2266:
! 2267: if (ali[i] == EOL) break;
! 2268:
! 2269: j++;
! 2270:
! 2271: }
! 2272:
! 2273: /* yes, it is, so resolve it now! */
! 2274: if (ali[i] == EOL && key[j] == EOL) {
! 2275:
! 2276: if (aliases > k1) stcpy0 (&ali[k], &ali[k1], aliases - k1);
! 2277:
! 2278: aliases -= (k1 - k);
! 2279:
! 2280: return;
! 2281:
! 2282: }
! 2283:
! 2284: i = k1;
! 2285:
! 2286: }
! 2287:
! 2288: }
! 2289:
! 2290: return;
! 2291:
! 2292: }
! 2293:
! 2294: /* new entry to alias table. there is no check agains duplicate entries */
! 2295: i = stlen (key);
! 2296: j = stlen (data);
! 2297:
! 2298: ali[aliases++] = (char) (i + j + 2); /* byte for fast skipping */
! 2299:
! 2300: stcpy (&ali[aliases], key);
! 2301: aliases += (i + 1);
! 2302:
! 2303: stcpy (&ali[aliases], data);
! 2304: aliases += (j + 1);
! 2305:
! 2306: /* write note to unmake the alias */
! 2307: j = stcpy (newptr, key);
! 2308: newptr += (j + 1);
! 2309: *newptr++ = j;
! 2310: *newptr++ = m_alias;
! 2311:
! 2312: return;
! 2313:
! 2314: case zdata: /* nonstandard data function */
! 2315:
! 2316:
! 2317:
! 2318: {
! 2319: long counties[128];
! 2320: int icnt, icnt0;
! 2321:
! 2322: i = 0;
! 2323:
! 2324: while (i < 128) counties[i++] = 0L; /* init count; */
! 2325:
! 2326: /* note: we assume EOL<DELIM<ASCII */
! 2327:
! 2328: icnt = 0;
! 2329: i = 0;
! 2330:
! 2331: while ((j = key[i++]) != EOL) {
! 2332: if (j == DELIM) {
! 2333: icnt++;
! 2334: }
! 2335: }
! 2336:
! 2337: if ((i = alphptr[(int) key[0]])) {
! 2338:
! 2339: data[2] = EOL;
! 2340: j = i + 1;
! 2341: k = 1;
! 2342:
! 2343: do {
! 2344:
! 2345: icnt0 = j + 1;
! 2346:
! 2347: while ((k1 = key[k] - partition[++j]) == 0) { /* compare keys */
! 2348:
! 2349: if (key[k] == EOL) break;
! 2350:
! 2351: k++;
! 2352:
! 2353: }
! 2354:
! 2355: if (k1 == 0) {
! 2356: counties[0] = 1;
! 2357: }
! 2358: else {
! 2359:
! 2360: if (partition[j] == DELIM && key[k] == EOL) {
! 2361:
! 2362: int ch;
! 2363:
! 2364: j = icnt0;
! 2365: icnt0 = 0;
! 2366:
! 2367: while ((ch = partition[j++]) != EOL) {
! 2368:
! 2369: if (ch == DELIM) {
! 2370: icnt0++;
! 2371: }
! 2372:
! 2373: }
! 2374:
! 2375: if (icnt0 <= icnt) break;
! 2376:
! 2377: counties[icnt0 - icnt]++;
! 2378:
! 2379: }
! 2380:
! 2381: /* if (k1<0 && k<2) break; */
! 2382:
! 2383: }
! 2384:
! 2385: i += UNSIGN (partition[i]); /* skip key */
! 2386: i += UNSIGN (partition[i]) + 1; /* skip data */
! 2387:
! 2388: j = i;
! 2389: k = 0;
! 2390:
! 2391: } while (i < PSIZE);
! 2392:
! 2393: }
! 2394:
! 2395: i = 128;
! 2396:
! 2397: while (counties[--i] == 0L) ;
! 2398:
! 2399: lintstr (data, counties[0]);
! 2400:
! 2401: j = 1;
! 2402: tmp1[0] = ',';
! 2403:
! 2404: while (j <= i) {
! 2405:
! 2406: lintstr (&tmp1[1], counties[j++]);
! 2407: stcat (data, tmp1);
! 2408:
! 2409: }
! 2410:
! 2411: return;
! 2412: } /* end of $zdata section */
! 2413:
! 2414: } /* end of action switch */
! 2415:
! 2416:
! 2417: /* return next variable or array name - non standard */
! 2418: unsubscr:
! 2419:
! 2420: if (standard) {
! 2421: merr_raise (NEXTER);
! 2422: return;
! 2423: }
! 2424:
! 2425: j = key[0];
! 2426: data[0] = EOL;
! 2427:
! 2428: while (alphptr[j] == 0) {
! 2429: if (++j >= DEL) return;
! 2430: }
! 2431:
! 2432: i = alphptr[j];
! 2433:
! 2434: while (i < PSIZE) {
! 2435:
! 2436: j = i;
! 2437: k = 0;
! 2438:
! 2439: while ((k1 = key[k] - partition[++j]) == 0) { /* compare keys */
! 2440:
! 2441: if (key[k] == EOL) break;
! 2442:
! 2443: k++;
! 2444:
! 2445: }
! 2446:
! 2447: if (k1 < 0 && (partition[j] != DELIM || key[k] != EOL)) {
! 2448:
! 2449: j = i;
! 2450: i = 0;
! 2451:
! 2452: while ((data[i] = partition[++j]) != EOL) {
! 2453:
! 2454: if (data[i] == DELIM) {
! 2455: data[i] = EOL;
! 2456: break;
! 2457: }
! 2458:
! 2459: i++;
! 2460:
! 2461: }
! 2462:
! 2463: return;
! 2464:
! 2465: }
! 2466:
! 2467: i += UNSIGN (partition[i]); /* skip key */
! 2468: i += UNSIGN (partition[i]) + 1; /* skip data */
! 2469:
! 2470: }
! 2471:
! 2472: return;
! 2473:
! 2474: } /* end of symtab() */
! 2475:
! 2476:
! 2477: /******************************************************************************/
! 2478: short int collate (s, t)
! 2479: char *s;
! 2480: char *t;
! 2481: /* if 't' follows 's' in MUMPS collating sequence a 1 is returned
! 2482: * otherwise 0
! 2483: */
! 2484: {
! 2485: short dif;
! 2486:
! 2487: if (s[0] == EOL) return (t[0] != EOL); /* the empty one is the leader! */
! 2488: if (t[0] == EOL) return FALSE;
! 2489: if ((dif = stcmp (t, s)) == 0) return FALSE;
! 2490:
! 2491: if (numeric (s)) { /* then come numerics */
! 2492:
! 2493: if (numeric (t) == FALSE) return TRUE;
! 2494:
! 2495: return comp (s, t);
! 2496:
! 2497: }
! 2498:
! 2499: if (numeric (t)) return FALSE;
! 2500:
! 2501: return dif > 0;
! 2502:
! 2503: } /* end of collate() */
! 2504:
! 2505: /******************************************************************************/
! 2506: short int numeric (char *str)
! 2507: /**
! 2508: * boolean function that tests
! 2509: * whether str is a canonical
! 2510: * numeric
! 2511: */
! 2512: {
! 2513: register int ptr = 0, ch;
! 2514: register int point;
! 2515:
! 2516:
! 2517:
! 2518: if (str[0] == '-') {
! 2519: ptr = 1;
! 2520: }
! 2521: if (str[ptr] == EOL) {
! 2522: return FALSE;
! 2523: }
! 2524: if (str[ptr] == '0') return str[1] == EOL; /* leading zero */
! 2525:
! 2526: point = FALSE;
! 2527:
! 2528: while ((ch = str[ptr++]) != EOL) {
! 2529:
! 2530:
! 2531: if (ch > '9') {
! 2532: return FALSE;
! 2533: }
! 2534:
! 2535: if (ch < '0') {
! 2536:
! 2537: if (ch != '.') return FALSE;
! 2538: if (point) return FALSE; /* multiple points */
! 2539:
! 2540: point = TRUE;
! 2541:
! 2542: }
! 2543:
! 2544: }
! 2545:
! 2546: if (point) {
! 2547:
! 2548: if ((ch = str[ptr - 2]) == '0') return FALSE; /* trailing zero */
! 2549: if (ch == '.') return FALSE; /* trailing point */
! 2550: }
! 2551: return TRUE;
! 2552: } /* end of numeric() */
! 2553:
! 2554: /******************************************************************************/
! 2555: short int comp (s, t)
! 2556: char *s;
! 2557: char *t;
! 2558: /* s and t are strings representing */
! 2559: /* MUMPS numbers. comp returns t>s */
! 2560: {
! 2561:
! 2562: register int s1 = s[0], t1 = t[0], point = '.';
! 2563:
! 2564: #if !defined(_AIX)
! 2565: if (fp_mode) {
! 2566: double fp_s;
! 2567: double fp_t;
! 2568:
! 2569: stcnv_m2c (s);
! 2570: stcnv_m2c (t);
! 2571:
! 2572: fp_s = atof (s);
! 2573: fp_t = atof (t);
! 2574:
! 2575: return fp_t > fp_s;
! 2576: }
! 2577: #endif
! 2578:
! 2579: if (s1 != t1) {
! 2580:
! 2581: if (s1 == '-') return TRUE; /* s<0<t */
! 2582: if (t1 == '-') return FALSE; /* t<0<s */
! 2583: if (s1 == point && t1 == '0') return FALSE; /* s>0; t==0 */
! 2584: if (t1 == point && s1 == '0') return TRUE; /* t>0; s==0 */
! 2585:
! 2586: }
! 2587:
! 2588: if (t1 == '-') {
! 2589:
! 2590: char *a;
! 2591:
! 2592: a = &t[1];
! 2593: t = &s[1];
! 2594: s = a;
! 2595:
! 2596: }
! 2597:
! 2598: s1 = 0;
! 2599:
! 2600: while (s[s1] > point) s1++; /* Note: EOL<'.' */
! 2601:
! 2602: t1 = 0;
! 2603:
! 2604: while (t[t1] > point) t1++;
! 2605:
! 2606: if (t1 > s1) return TRUE;
! 2607: if (t1 < s1) return FALSE;
! 2608:
! 2609: while (*t == *s) {
! 2610:
! 2611: if (*t == EOL) return FALSE;
! 2612:
! 2613: t++;
! 2614: s++;
! 2615:
! 2616: }
! 2617:
! 2618: if (*t > *s) return TRUE;
! 2619:
! 2620: return FALSE;
! 2621:
! 2622: } /* end of comp() */
! 2623: /******************************************************************************/
! 2624: void intstr (str, integ) /* converts integer to string */
! 2625: char *str;
! 2626: short integ;
! 2627: {
! 2628:
! 2629: if (integ < 0) {
! 2630: integ = (-integ);
! 2631: *str++ = '-';
! 2632: }
! 2633:
! 2634: if (integ < 10) {
! 2635:
! 2636: *str++ = integ + '0';
! 2637: *str = EOL;
! 2638:
! 2639: return;
! 2640:
! 2641: }
! 2642: else if (integ < 100) {
! 2643: str += 2;
! 2644: }
! 2645: else if (integ < 1000) {
! 2646: str += 3;
! 2647: }
! 2648: else if (integ < 10000) {
! 2649: str += 4;
! 2650: }
! 2651: else {
! 2652: str += 5;
! 2653: }
! 2654:
! 2655: *str = EOL;
! 2656:
! 2657: do {
! 2658: *(--str) = integ % 10 + '0';
! 2659: } while (integ /= 10);
! 2660:
! 2661: return;
! 2662: } /* end of intstr() */
! 2663:
! 2664: /******************************************************************************/
! 2665: void lintstr (str, integ) /* converts long integer to string */
! 2666: char *str;
! 2667: long integ;
! 2668: {
! 2669: char result[11]; /* 32 bit = 10 digits+sign */
! 2670: register int i = 0;
! 2671:
! 2672: if (integ < 0) {
! 2673: integ = (-integ);
! 2674: *str++ = '-';
! 2675: }
! 2676:
! 2677: do {
! 2678: result[i++] = integ % 10 + '0';
! 2679: } while (integ /= 10);
! 2680:
! 2681: do {
! 2682: *str++ = result[--i];
! 2683: } while (i > 0);
! 2684:
! 2685: *str = EOL;
! 2686:
! 2687: return;
! 2688:
! 2689: } /* end of lintstr() */
! 2690:
! 2691: /****************************************************************/
! 2692:
! 2693: /* user defined special variable table management */
! 2694: /* The symbol table is placed at the high end of 'svntable'. It begins at
! 2695: * 'svnlen' and ends at 'UDFSVSIZ'. The layout is
! 2696: * (keylength)(key...)(<EOL>)(datalength)(data...[<EOL>])
! 2697: * The keys are sorted in alphabetic sequence.
! 2698: *
! 2699: * To have the same fast access regardless of the position in the
! 2700: * alphabet for each character a pointer to the first variable beginning
! 2701: * with that letter is maintained. (0 indicates there's no such var.)
! 2702: */
! 2703: void udfsvn (action, key, data) /* symbol table functions */
! 2704: short action; /* set_sym get_sym */
! 2705: char *key; /* lvn as ASCII-string */
! 2706: char *data;
! 2707: {
! 2708:
! 2709: long keyl; /* length of key */
! 2710: long datal; /* length of data */
! 2711: register long int i, j, k, k1;
! 2712:
! 2713:
! 2714:
! 2715: #ifdef DEBUG_SYM
! 2716:
! 2717: char *start;
! 2718:
! 2719: #endif
! 2720:
! 2721: switch (action) {
! 2722:
! 2723:
! 2724: case get_sym: /* retrieve */
! 2725:
! 2726:
! 2727: if ((i = svnaptr[(int) key[0]])) {
! 2728:
! 2729: k = 1;
! 2730: j = i + 1; /* first char always matches! */
! 2731:
! 2732: do {
! 2733:
! 2734: while (key[k] == svntable[++j]) { /* compare keys */
! 2735:
! 2736: if (key[k++] == EOL) {
! 2737:
! 2738: i = UNSIGN (svntable[++j]);
! 2739: stcpy0 (data, &svntable[j + 1], i);
! 2740: data[i] = EOL;
! 2741:
! 2742: return;
! 2743: }
! 2744:
! 2745: }
! 2746:
! 2747: i += UNSIGN (svntable[i]); /* skip key */
! 2748: i += UNSIGN (svntable[i]) + 1; /* skip data */
! 2749:
! 2750: k = 0;
! 2751: j = i;
! 2752:
! 2753: } while (i < UDFSVSIZ);
! 2754:
! 2755: }
! 2756:
! 2757: merr_raise (ILLFUN);
! 2758: return;
! 2759:
! 2760:
! 2761: case set_sym: /* store/create variable; */
! 2762:
! 2763:
! 2764: if ((keyl = stlen (key) + 2) > STRLEN) {
! 2765: merr_raise (M75);
! 2766: return;
! 2767: } /* key length +2 */
! 2768:
! 2769: datal = stlen (data); /* data length */
! 2770:
! 2771: if ((i = svnaptr[(int) key[0]])) { /* previous entry */
! 2772:
! 2773: j = i + 1;
! 2774: k = 1;
! 2775:
! 2776: }
! 2777: else {
! 2778:
! 2779: i = svnlen;
! 2780: j = i;
! 2781: k = 0;
! 2782:
! 2783: }
! 2784:
! 2785: while (i < UDFSVSIZ) { /* compare keys */
! 2786:
! 2787: while (key[k] == svntable[++j]) {
! 2788:
! 2789: if (key[k] == EOL) goto old;
! 2790:
! 2791: k++;
! 2792:
! 2793: }
! 2794:
! 2795: if (key[k] < svntable[j]) break;
! 2796:
! 2797: i += UNSIGN (svntable[i]); /* skip key */
! 2798: i += UNSIGN (svntable[i]) + 1; /* skip data */
! 2799: j = i;
! 2800: k = 0;
! 2801:
! 2802: }
! 2803:
! 2804: /* if entry found, i pointer to searched entry
! 2805: * else entry not found, i pointer to alphabetically next entry */
! 2806: /* new entry */
! 2807:
! 2808: k = i;
! 2809: j = key[0];
! 2810: i = keyl + datal + 1;
! 2811:
! 2812: if (svnlen <= i) {
! 2813:
! 2814: long dif;
! 2815:
! 2816: dif = getumore ();
! 2817:
! 2818: if (dif == 0L) return;
! 2819:
! 2820: k += dif;
! 2821:
! 2822: }
! 2823:
! 2824: for (k1 = 'a'; k1 <= j; k1++) {
! 2825: if (svnaptr[k1]) svnaptr[k1] -= i;
! 2826: }
! 2827:
! 2828: i = k - i;
! 2829:
! 2830: if (svnaptr[j] == 0 || svnaptr[j] > i) svnaptr[j] = i;
! 2831:
! 2832: i = (svnlen -= (j = keyl + datal + 1));
! 2833: stcpy0 (&svntable[i], &svntable[j + i], k - i);
! 2834:
! 2835: i = k - (keyl + datal + 1);
! 2836: svntable[i++] = (char) (keyl);
! 2837: stcpy (&svntable[i], key); /* store new key */
! 2838:
! 2839: i += keyl - 1;
! 2840: svntable[i++] = (char) (datal);
! 2841: stcpy0 (&svntable[i], data, datal); /* store new data */
! 2842:
! 2843: return;
! 2844:
! 2845: /* there is a previous value */
! 2846: old:
! 2847:
! 2848: i += UNSIGN (svntable[i]);
! 2849: j = UNSIGN (svntable[i]) - datal;
! 2850:
! 2851: if (j < 0) { /* more space needed */
! 2852:
! 2853: if (svnlen <= (-j)) {
! 2854:
! 2855: long dif;
! 2856:
! 2857: dif = getumore ();
! 2858:
! 2859: if (dif == 0L) return;
! 2860:
! 2861: i += dif;
! 2862:
! 2863: }
! 2864:
! 2865: svnlen += j;
! 2866:
! 2867: for (k = 'a'; k < key[0]; k++) {
! 2868: if (svnaptr[k]) svnaptr[k] += j;
! 2869: }
! 2870:
! 2871: if (svnaptr[k] && svnaptr[k] < i) svnaptr[k] += j;
! 2872:
! 2873: k = i + j;
! 2874: i = svnlen;
! 2875: stcpy0 (&svntable[i], &svntable[i - j], k - i);
! 2876:
! 2877: i = k;
! 2878:
! 2879: }
! 2880: else if (j > 0) { /* surplus space */
! 2881:
! 2882: svnlen += j;
! 2883:
! 2884: for (k = 'a'; k < key[0]; k++) {
! 2885: if (svnaptr[k]) svnaptr[k] += j;
! 2886: }
! 2887:
! 2888: if (svnaptr[k] && svnaptr[k] < i) svnaptr[k] += j;
! 2889:
! 2890: i += j;
! 2891: k = i;
! 2892: j = i - j;
! 2893:
! 2894: while (i >= svnlen) {
! 2895: svntable[i--] = svntable[j--];
! 2896: }
! 2897:
! 2898: i = k;
! 2899:
! 2900: }
! 2901:
! 2902: svntable[i++] = (char) (datal);
! 2903:
! 2904: stcpy0 (&svntable[i], data, datal); /* store new data */
! 2905:
! 2906: return;
! 2907: /* end of set_sym section */
! 2908: }
! 2909: } /* end user defined special variable table */
! 2910:
! 2911:
! 2912: /******************************************************************************/
! 2913: long getpmore ()
! 2914: { /* try to get more 'partition' space. returns size increment */
! 2915:
! 2916: long siz;
! 2917: long dif;
! 2918:
! 2919: if (autopsize == FALSE) return 0L;
! 2920:
! 2921: siz = PSIZE;
! 2922:
! 2923: if (siz % 1024) siz = (siz & ~01777) + 02000; /* round for full kB; */
! 2924:
! 2925: siz += 01777;
! 2926: dif = siz - PSIZE;
! 2927:
! 2928: if (newpsize (siz)) return 0L;
! 2929:
! 2930: return dif;
! 2931:
! 2932: } /* end getpmore */
! 2933:
! 2934: /******************************************************************************/
! 2935: long getumore ()
! 2936: { /* try to get more udfsvntab space. returns size increment */
! 2937: long siz, dif;
! 2938:
! 2939: if (autousize == FALSE) {
! 2940: merr_raise (STORE);
! 2941: return 0L;
! 2942: }
! 2943:
! 2944: siz = UDFSVSIZ;
! 2945:
! 2946: if (siz % 1024) siz = (siz & ~01777) + 02000; /* round for full kB; */
! 2947:
! 2948: siz += 01777;
! 2949: dif = siz - UDFSVSIZ;
! 2950:
! 2951: if (newusize (siz)) {
! 2952: merr_raise (STORE);
! 2953: return 0L;
! 2954: }
! 2955:
! 2956: return dif;
! 2957:
! 2958: } /* end getumore */
! 2959:
! 2960: /******************************************************************************/
! 2961: long getrmore ()
! 2962: { /* try to get more routine space. returns size increment */
! 2963: long siz, dif;
! 2964: short i;
! 2965:
! 2966: if (autorsize == FALSE) {
! 2967: merr_raise (PGMOV);
! 2968: return 0L;
! 2969: }
! 2970:
! 2971: siz = PSIZE0;
! 2972:
! 2973: if (siz % 1024) siz = (siz & ~01777) + 02000; /* round for full kB; */
! 2974:
! 2975: siz += 01777;
! 2976: dif = siz - PSIZE0;
! 2977:
! 2978: for (i = 0; i < NO_OF_RBUF; i++) { /* empty routine buffer */
! 2979: pgms[i][0] = EOL;
! 2980: ages[i] = 0L;
! 2981: }
! 2982:
! 2983: if (newrsize (siz, NO_OF_RBUF)) {
! 2984: merr_raise (PGMOV);
! 2985: return 0L;
! 2986: }
! 2987:
! 2988: return dif;
! 2989:
! 2990: } /* end getrmore */
! 2991:
! 2992: /******************************************************************************/
! 2993: short int getnewmore ()
! 2994: { /* enlarge new_buffers */
! 2995: char *newbuf;
! 2996: int i;
! 2997: long dif;
! 2998:
! 2999: newbuf = calloc ((unsigned) (NSIZE + 4096), 1); /* new_buffer */
! 3000:
! 3001: if (newbuf == NULL) { /* could not allocate stuff... */
! 3002: merr_raise (STKOV);
! 3003: return 1;
! 3004: }
! 3005:
! 3006: stcpy0 (newbuf, newstack, (long) NSIZE);
! 3007:
! 3008: dif = newbuf - newstack;
! 3009:
! 3010: free (newstack); /* free previously allocated space */
! 3011:
! 3012: newstack = newbuf;
! 3013: NSIZE += 4096;
! 3014: newptr += dif;
! 3015: newlimit = newstack + NSIZE - 1024;
! 3016: i = 0;
! 3017:
! 3018: while (i <= nstx) {
! 3019:
! 3020: if (nestnew[i]) nestnew[i] += dif;
! 3021:
! 3022: i++;
! 3023:
! 3024: }
! 3025:
! 3026: return 0;
! 3027:
! 3028: } /* end getnewmore() */
! 3029: /******************************************************************************/
! 3030:
! 3031:
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>