Annotation of freem/src/fma_globals.c, revision 1.4
1.1 snw 1: /*
1.4 ! snw 2: * $Id$
1.1 snw 3: * fmadm - globals
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.4 ! snw 26: * $Log$
! 27: *
! 28: * SPDX-FileCopyrightText: (C) 2025 Coherent Logic Development LLC
! 29: * SPDX-License-Identifier: AGPL-3.0-or-later
1.1 snw 30: **/
31:
32: #include <stdlib.h>
33: #include <stdio.h>
34: #include <stdlib.h>
35: #include <string.h>
36: #include <dirent.h>
37: #include <time.h>
38: #include <unistd.h>
39: #include <sys/types.h>
40: #include <sys/stat.h>
41: #include <fcntl.h>
42:
43: #include "fmadm.h"
44:
45: void gl (char *global, short kf, short df, short nr, int chop);
46: int gverify (char *gpath);
47: static short int g_collate (); /* if 't' follows 's' in MUMPS collating */
48: int key_check (char *key);
1.2 snw 49: void check (unsigned long blknbr);
50: static short int g_collate (char s[], char t[]); /* if 't' follows 's' in MUMPS collating */
51: void show (char key[]);
1.1 snw 52:
53: /* global variables for gverify */
54: short filedes; /* file descriptor */
55: char block[MAXLEV][BLOCKLEN]; /* block to be read */
56: char *blck; /* dto. as pointer */
57: unsigned long llink[MAXLEV]; /* left link pointers */
58: unsigned long rlink[MAXLEV]; /* right link pointers */
59: short offsets[MAXLEV]; /* offsets */
60: short type[MAXLEV]; /* block type */
61: extern short level; /* current level */
62: unsigned long no_of_blks; /* number of blocks */
63: unsigned long freeblks; /* free blocks list */
64: char key[256]; /* current key in pointer block scan */
65: short exstat = 0; /* exit status */
66: void showpath (); /* display path of pointers */
67: /* end of global variables for gverify */
68:
69:
70: int fma_globals_list (int optc, char **opts)
71: {
72: DIR *dir;
73: struct dirent *ent;
74:
75: char filename[STRLEN];
76:
77: int ct = 0;
78:
79: printf ("\nFreeM Global Listing\n");
80: printf ("--------------------\n\n");
81:
82: printf ("Namespace: %s\n", fma_namespace);
83: printf ("Global Path: %s\n\n", fma_global_path);
84:
85: if ((dir = opendir (fma_global_path)) == NULL) {
86: fprintf (stderr, "fmadm: could not open global directory %s\n", fma_global_path);
87: return 1;
88: }
89:
90: while ((ent = readdir (dir)) != NULL) {
91:
92: strncpy (filename, ent->d_name, STRLEN - 1);
93:
94: if (filename[0] == '^' && filename[1] != '$') {
95: printf (" %s\n", filename);
96: ct++;
97: }
98:
99: }
100:
101: printf ("\n\n - %d globals found\n\n", ct);
102: closedir (dir);
103:
104: return 0;
105: }
106:
107: int fma_globals_examine (int optc, char **opts)
108: {
109:
110: DIR *dir;
111: struct dirent *ent;
112:
113: char gpath[STRLEN];
114: int i;
115: int ct = 0;
116:
117: if ((dir = opendir (fma_global_path)) == NULL) {
118: fprintf (stderr, "fmadm: could not open global directory %s\n", fma_global_path);
119: return 1;
120: }
121:
122: printf ("\nFreeM Global Examine\n");
123: printf ("--------------------\n\n");
124:
125: printf ("Namespace: %s\n", fma_namespace);
126: printf ("Global Path: %s\n", fma_global_path);
127: printf ("Globals Selected: ");
128:
129:
130: if (optc > 1) {
131:
132: for (i = fma_base_opt; i <= optc; i++) {
133: printf ("%s ", opts[i]);
134: }
135:
136: printf ("\n\n");
137:
138: }
139: else {
140: printf ("[ENTIRE NAMESPACE]\n\n");
141: }
142:
143:
144: if (optc > 1) {
145:
146: for (i = fma_base_opt; i < optc; i++) {
147:
148: snprintf (gpath, STRLEN - 1, "%s/%s", fma_global_path, opts[i]);
149: gl (gpath, FALSE, FALSE, FALSE, strlen (fma_global_path));
150:
151: ct++;
152:
153: }
154:
155: }
156: else {
157:
158: while ((ent = readdir (dir)) != NULL) {
159:
160: strncpy (gpath, ent->d_name, STRLEN - 1);
161:
162: if (gpath[0] == '^' && gpath[1] != '$') {
163:
164: snprintf (gpath, STRLEN - 1, "%s/%s", fma_global_path, ent->d_name);
165:
166: gl (gpath, FALSE, FALSE, FALSE, strlen (fma_global_path));
167:
168: ct++;
169:
170: }
171:
172: }
173:
174: }
175:
176: printf ("\n\n");
177: printf (" - %d globals examined\n\n", ct);
178:
179: return 0;
180:
181: }
182:
183: int fma_globals_remove (int optc, char **opts)
184: {
185:
186: char gpath[STRLEN];
187: int i;
188: int ct = 0;
189: int er = 0;
190: int tot = 0;
191:
192: printf ("\nFreeM Global Removal\n");
193: printf ("--------------------\n\n");
194:
195: printf ("Namespace: %s\n", fma_namespace);
196: printf ("Global Path: %s\n\n", fma_global_path);
197:
198: for (i = fma_base_opt; i < optc; i++) {
199: printf ("%-10s\t", opts[i]);
200:
201: snprintf (gpath, STRLEN - 1, "%s/%s", fma_global_path, opts[i]);
202:
203: if (unlink (gpath) == -1) {
204: printf ("[FAIL]\n");
205: er++;
206: }
207: else {
208: printf ("[OK]\n");
209: ct++;
210: }
211:
212: tot++;
213:
214: }
215:
216: printf ("\nRemoved %d globals [%d errors/%d attempted]\n\n", ct, er, tot);
217:
218: return 0;
219:
220: }
221:
222: int fma_globals_verify (int optc, char **opts)
223: {
224: DIR *dir;
225: struct dirent *ent;
226:
227: char gpath[STRLEN];
228: int i;
229: int ct = 0;
230:
231: if ((dir = opendir (fma_global_path)) == NULL) {
232: fprintf (stderr, "fmadm: could not open global directory %s\n", fma_global_path);
233: return 1;
234: }
235:
236: printf ("\nFreeM Global Verify\n");
237: printf ("-------------------\n\n");
238:
239: printf ("Namespace: %s\n", fma_namespace);
240: printf ("Global Path: %s\n", fma_global_path);
241: printf ("Globals Selected: ");
242:
243: if (optc > fma_base_opt) {
244:
245: for (i = fma_base_opt; i < optc; i++) {
246: printf ("%s ", opts[i]);
247: }
248:
249: printf ("\n\n");
250:
251: }
252: else {
253: printf ("[ENTIRE NAMESPACE]\n\n");
254: }
255:
256:
257: if (optc > fma_base_opt) {
258:
259: for (i = fma_base_opt; i < optc; i++) {
260:
261: snprintf (gpath, STRLEN - 1, "%s/%s", fma_global_path, opts[i]);
262:
263: exstat = 0;
264: gverify (gpath);
265:
266: printf ("\n\t%d error(s) in %s\n", exstat, gpath);
267:
268: ct++;
269:
270: }
271:
272: }
273: else {
274:
275: while ((ent = readdir (dir)) != NULL) {
276:
277: strncpy (gpath, ent->d_name, STRLEN - 1);
278:
279: if (gpath[0] == '^') {
280:
281: snprintf (gpath, STRLEN - 1, "%s/%s", fma_global_path, ent->d_name);
282:
283: exstat = 0;
284: gverify (gpath);
285:
286: printf ("\n\t%d errors in %s\n", exstat, gpath);
287:
288: ct++;
289:
290: }
291:
292: }
293:
294: }
295:
296: printf ("\n\n");
297: printf (" - %d globals verified\n\n", ct);
298:
299: return 0;
300:
301: }
302:
303:
304: /* PRIVATE FUNCTIONS */
305:
306: /***
307: * gl(): list global
308: *
309: * global: name of global
310: * kf: key flag
311: * df: data flag
312: * nr: naked reference flag
313: */
314: void gl (char *global, short kf, short df, short nr, int chop)
315: {
316: short filedes; /* file descriptor */
317: char block[BLOCKLEN]; /* block to be read */
318: char key[512];
319: char prevkey[512]; /* previous key */
320: char data[1024]; /* if data has CTRLs it may become */
321: /* so long */
322: unsigned long blknbr;
323: short offset;
324: short length;
325: short koffs;
326: short dkf = TRUE;
327:
328: short CtrlFlag;
329: short n, k1;
330: register int i, j, k, ch;
331:
332: if ((filedes = open (global, 0)) == -1) {
333: printf ("%s: cannot open\012\015", global);
334:
335: return;
336: }
337:
338: if (kf == FALSE && df == FALSE) {
339: kf = TRUE;
340: df = TRUE;
341: }
342: else {
343: dkf = FALSE;
344: }
345:
346: blknbr = ROOT;
347: prevkey[0] = 0;
348: prevkey[1] = 0;
349: prevkey[2] = 0;
350:
351: for (;;) {
352:
353: lseek (filedes, blknbr * BLOCKLEN, 0);
354:
355: if (read (filedes, block, BLOCKLEN) == 0) {
356: fprintf (stderr, "\015*** something wrong ***\033[K\n\r");
357: exit (0);
358: }
359:
360: if (block[BTYP] == DATA) goto first;
361:
362: i = UNSIGN (block[0]) + 2;
363: blknbr = UNSIGN (block[i]) * 65536 + UNSIGN (block[i + 1]) * 256 + UNSIGN (block[i + 2]);
364:
365: }
366:
367: again:
368:
369: if (blknbr == 0) {
370: close (filedes);
371: return;
372: }
373:
374: lseek (filedes, blknbr * 1024L, 0);
375: read (filedes, block, BLOCKLEN);
376:
377: first:
378:
379: offset = UNSIGN (block[OFFS]) * 256 + UNSIGN (block[OFFS + 1]);
380: blknbr = UNSIGN (block[RLPTR]) * 65536 + UNSIGN (block[RLPTR + 1]) * 256 + UNSIGN (block[RLPTR + 2]);
381:
382: i = 0;
383: while (i < offset) {
384:
385: length = UNSIGN (block[i++]);
386: k = koffs = UNSIGN (block[i++]);
387:
388: if ((i + length) > offset) break;
389:
390: for (j = 0; j < length; j++) key[k++] = block[i++];
391:
392: key[k] = g_EOL;
393: {
394:
395: short ch0, i, j, k;
396:
397: j = 0;
398: i = 0;
399: data[j++] = '(';
400: k = 1;
401:
402: while ((ch = UNSIGN (key[i++])) != g_EOL) {
403:
404: if (k) {
405: k = 0;
406: if (ch > SP) data[j++] = '"';
407: }
408:
409: ch0 = (ch >= SP ? (ch >> 1) : /* 'string' chars */
410: (ch < 20 ? (ch >> 1) + '0' : /* 0...9 */
411: (ch >> 1) + SP)); /* '.' or '-' */
412:
413: if (ch0 == DEL) {
414: if ((ch0 = (UNSIGN (key[i++]) >> 1)) == DEL) {
415: ch0 = (UNSIGN (key[i++]) >> 1) + DEL;
416: }
417:
418: ch0 += DEL;
419: data[j] = '<';
420: data[++j] = '0' + ch0 / 100;
421: data[++j] = '0' + (ch0 % 100) / 10;
422: data[++j] = '0' + ch0 % 10;
423: data[++j] = '>';
424:
425: }
426: else {
427: data[j] = ch0;
428: }
429:
430: if (data[j++] == '"') data[j++] = '"';
431:
432: if (ch & 01) {
433:
434: if (ch > SP) data[j++] = '"';
435:
436: data[j++] = ',';
437: k = 1;
438:
439: }
440:
441: }
442:
443: data[j--] = 0;
444: data[j] = ')';
445:
446: if (j == 0) data[0] = 0;
447:
448: while (j >= 0) {
449: if ((ch = data[--j]) < SP || ch >= DEL) break;
450: }
451:
452: k1 = 0;
453: if (nr) {
454:
455: if (prevkey[0]) {
456:
457: n = ch = 0;
458:
459: while (data[n] == prevkey[n]) {
460:
461: if (prevkey[n] == '"') ch = !ch;
462: if (!ch && k1 == 0 && (prevkey[n] == '(')) k1 = n + 1;
463: if (!ch && (prevkey[n] == ',')) k1 = n + 1;
464:
465: n++;
466:
467: }
468:
469: while (prevkey[n]) {
470:
471: if (prevkey[n] == '"') ch = !ch;
472:
473: if (!ch && (prevkey[n] == ',')) {
474: k1 = 0;
475: break;
476: }
477:
478: n++;
479:
480: }
481:
482: }
483:
484: strcpy (prevkey, data);
485:
486: if (k1 > 1) {
487: strcpy (&data[1], &data[k1]);
488: }
489:
490: }
491:
492: if (j < 0) {
493:
494: if (kf) {
495:
496: if (k1) {
497: printf ("%c%s", '^', data);
498: }
499: else {
500: printf ("%s%s", global + chop + 1, data);
501: }
502:
503: }
504:
505: if (dkf && !nr) {
506: printf ("=");
507: }
508: else if (kf) {
509: printf ("\n");
510: }
511:
512: }
513: else {
514: fprintf (stderr, "[%d][%d] <illegal subscript>\n", length, koffs);
515: }
516:
517: }
518:
519: length = UNSIGN (block[i++]);
520:
521: stcpy0 (data, &block[i], (long) length);
522:
523: data[length] = EOL;
524:
525: if (numeric (data)) {
526: data[length] = 0;
527: i += length;
528: }
529: else {
530:
531: CtrlFlag = 0;
532: data[0] = '"';
533: k = 1;
534:
535: while (length-- > 0) {
536:
537: ch = UNSIGN (block[i++]);
538:
539: if ((ch >= SP) && (ch < DEL)) {
540:
541: if (CtrlFlag) { /* close Bracket after CTRL */
542:
543: data[k++] = ')';
544: data[k++] = '_';
545: data[k++] = '"';
546:
547: CtrlFlag = 0;
548:
549: }
550:
551: if ((data[k++] = ch) == '"') data[k++] = ch;
552:
553: }
554: else {
555:
556: if (((ch >= NUL) && (ch < SP)) || ch == DEL) {
557:
558: if (CtrlFlag) {
559: data[k++] = ',';
560: }
561: else {
562:
563: if (k > 1) {
564: data[k++] = '"';
565: data[k++] = '_';
566: }
567: else {
568: k = 0;
569: }
570:
571: data[k++] = '$';
572: data[k++] = 'C';
573: data[k++] = '(';
574:
575: CtrlFlag = 1;
576:
577: }
578:
579: if (ch == DEL) {
580: data[k++] = '1';
581: ch -= 100;
582: }
583:
584: if (ch >= 10) {
585: data[k++] = ch / 10 + '0';
586: ch = ch % 10;
587: }
588:
589: data[k++] = ch + '0';
590:
591: }
592: else {
593:
594: if (CtrlFlag) { /* close Bracket after CTRL */
595:
596: data[k++] = ')';
597: data[k++] = '_';
598: data[k++] = '"';
599:
600: CtrlFlag = 0;
601:
602: }
603:
604: data[k++] = '<';
605:
606: if (ch > 99) {
607: data[k++] = '0' + (ch / 100);
608: ch = ch % 100;
609: }
610:
611: if (ch > 9) {
612: data[k++] = '0' + (ch / 10);
613: ch = ch % 10;
614: }
615:
616: data[k++] = '0' + ch;
617: data[k++] = '>';
618:
619: }
620: }
621: }
622:
623: if (CtrlFlag) {
624: data[k++] = ')';
625: }
626: else {
627: data[k++] = '"';
628: }
629:
630: data[k] = 0;
631:
632: }
633:
634: if (df) printf ("%s\n", data);
635:
636: }
637:
638: if (i != offset) fprintf (stderr, "\nwrong offset %d vs. %d\n", offset, i);
639:
640: goto again;
641:
642: }
643:
644: int gverify (char *gpath)
645: {
646: register int j;
647:
648: printf ("\n%s:\n\n", gpath);
649:
650: if ((filedes = open (gpath, 0)) == -1) {
651: fprintf (stderr, "Cannot open file %s\007\n\r", gpath);
652: return 1;
653: }
654:
655: j = 0;
656:
657: while (j < MAXLEV) {
658:
659: rlink[j] = 0;
660: llink[j] = 0;
661:
662: j++;
663:
664: }
665:
666: level = 0;
667:
668: check (ROOT);
669:
670: j = 1; /* ignore level zero: there is no rightlink pointer (freeblocks instead) */
671: while (j < MAXLEV) { /* check last right link pointers (all zero!) */
672:
673: if (rlink[j] != 0) {
674: printf ("\tblock #%ld right link pointer mismatch 0 vs. %ld\012\015", llink[j], rlink[j]);
675: showpath ();
676: }
677:
678: j++;
679:
680: }
681:
682: /* check free blocks */
683: freeblks = UNSIGN (block[0][FREE]) * 65536 +
684: UNSIGN (block[0][FREE + 1]) * 256 +
685: UNSIGN (block[0][FREE + 2]);
686:
687: while (freeblks) {
688:
689: unsigned long free;
690: int i;
691:
692: if (freeblks > no_of_blks) {
693: printf ("\tblock# %ld (free list) greater than number of blocks (%ld)\012\015", freeblks, no_of_blks);
694: showpath ();
695: }
696:
697: lseek (filedes, (long) (freeblks) * BLOCKLEN, 0);
698:
699: if (read (filedes, block[0], BLOCKLEN) < BLOCKLEN) {
700:
701: printf ("\tblock #%ld is (partially) empty\012\015", freeblks);
702: showpath ();
703:
704: exit (exstat);
705:
706: }
707:
708: j = UNSIGN (block[0][OFFS]) * 256 +
709: UNSIGN (block[0][OFFS + 1]); /* offset */
710:
711: freeblks = UNSIGN (block[0][RLPTR]) * 65536 +
712: UNSIGN (block[0][RLPTR + 1]) * 256 +
713: UNSIGN (block[0][RLPTR + 1]);
714:
715: i = 0;
716:
717: while (i < j) {
718:
719: free = UNSIGN (block[0][i++]) * 65536;
720: free += UNSIGN (block[0][i++]) * 256;
721: free += UNSIGN (block[0][i++]);
722:
723: if (free > no_of_blks) {
724: printf ("\tblock #%ld (free) greater than number of blocks (%ld)\012\015", free, no_of_blks);
725: showpath ();
726: }
727:
728: lseek (filedes, free * BLOCKLEN, 0);
729: read (filedes, block[1], BLOCKLEN);
730:
731: if (block[1][BTYP] != EMPTY) {
732:
733: printf ("\tblock #%ld expected block type: EMPTY\012\015", free);
734:
735: if (++exstat >= ERRLIM) {
736: fprintf (stderr, "Error limit exceeded\012\015");
737: return exstat;
738: }
739:
740: }
741:
742: }
743:
744: }
745:
746: return exstat;
747:
748: }
749:
750: void check (unsigned long blknbr)
751: {
752: unsigned long left; /* current left link pointer */
753: unsigned long right; /* current right link pointer */
754: long i;
755: long k;
756:
757: lseek (filedes, blknbr * BLOCKLEN, 0);
758: blck = block[level];
759:
760: if (read (filedes, blck, BLOCKLEN) < BLOCKLEN) {
761: printf ("\tblock #%ld is (partially) empty\012\015", blknbr);
762: showpath ();
763: }
764:
765: type[level] = blck[BTYP];
766:
767: left = UNSIGN (blck[LLPTR]) * 65536 +
768: UNSIGN (blck[LLPTR + 1]) * 256 +
769: UNSIGN (blck[LLPTR + 2]);
770:
771: right = UNSIGN (blck[RLPTR]) * 65536 +
772: UNSIGN (blck[RLPTR + 1]) * 256 +
773: UNSIGN (blck[RLPTR + 2]);
774:
775: if (blknbr == ROOT) {
776: no_of_blks = UNSIGN (block[0][NRBLK]) * 65536 +
777: UNSIGN (block[0][NRBLK + 1]) * 256 +
778: UNSIGN (block[0][NRBLK + 2]);
779: }
780: else {
781:
782: if (blknbr > no_of_blks) {
783: printf ("\tblock# %ld greater than number of blocks (%ld)\012\015", blknbr, no_of_blks);
784: showpath ();
785: }
786:
787: }
788:
789: offsets[level] = UNSIGN (blck[OFFS]) * 256 + UNSIGN (blck[OFFS + 1]);
790:
791: if (rlink[level] != 0L && rlink[level] != blknbr) {
792: printf ("\tblock #%ld right link pointer mismatch %ld vs. %ld\012\015", llink[level], blknbr, rlink[level]);
793: showpath ();
794: }
795:
796: if (llink[level] != 0L && left != llink[level]) {
797: printf ("\tblock #%ld left link pointer mismatch %ld vs. %ld\012\015", blknbr, left, llink[level]);
798: showpath ();
799: }
800:
801: rlink[level] = right;
802: llink[level] = blknbr;
803:
804: if (blknbr != ROOT) {
805:
806: k = UNSIGN (blck[0]);
807: i = 0;
808:
809: while (i < k) {
810: if (blck[i + 2] != key[i]) {
811: printf ("\tblock #%ld first key mismatch to pointer block(%ld)\012\015", blknbr, llink[level - 1]);
812: showpath ();
813: break;
814: }
815: i++;
816: }
817:
818: }
819:
820: switch (type[level]) {
821:
822: case EMPTY:
823: printf ("\tblock #%ld unexpected block type: EMPTY\012\015", blknbr);
824: showpath ();
825: break;
826:
827: case FBLK:
828: printf ("\tblock #%ld unexpected block type: FBLK\012\015", blknbr);
829: showpath ();
830: break;
831:
832: case POINTER:
833: case BOTTOM:
834: /*******************************/
835: /* scan pointer block */
836: {
837: register long i;
838: register long k;
839: short j;
840: short len;
841: char key1[256];
842:
843: key1[0] = g_EOL;
844: i = 0;
845:
846: while (i < offsets[level]) {
847:
848: j = i++; /* save adress of current entry */
849:
850: if ((len = UNSIGN (blck[j]) + (k = UNSIGN (blck[i++]))) > 255) {
851: printf ("\tblock #%ld key too long\012\015", blknbr);
852: showpath ();
853: }
854: else {
855:
856: if (len == 0 && j) {
857: printf ("\tblock #%ld empty key\012\015", blknbr);
858: showpath ();
859: }
860:
861: while (k < len) key[k++] = blck[i++];
862:
863: key[k] = g_EOL;
864:
865: if (key_check (key)) {
866: printf ("\tblock #%ld illegal key\012\015", blknbr);
867: showpath ();
868: }
869:
870: if (g_collate (key1, key) == 0) {
871: printf ("\tblock #%ld collation mismatch\012\015", blknbr);
872: show (key1);
873: show (key);
874: showpath ();
875: }
876:
877: stcpy0 (key1, key, k + 1);
878: level++;
879:
880: check ((long) (UNSIGN (blck[i]) * 65536 +
881: UNSIGN (blck[i + 1]) * 256 +
882: UNSIGN (blck[i + 2])));
883:
884: blck = block[--level];
885:
886: }
887:
888: i += PLEN;
889:
890: if (i > DATALIM) {
891: printf ("\tblock #%ld pointer in status bytes\012\015", blknbr);
892: showpath ();
893: }
894:
895: }
896:
897: if (i > offsets[level]) {
898: printf ("\tblock #%ld offset mismatch %ld vs. %d\012\015", blknbr, i, offsets[level]);
899: showpath ();
900: }
901: }
902: break;
903:
904: case DATA:
905: /* scan data block */
906: {
907: register long i;
908: register long k;
909: short len;
910: char key0[256];
911: char key1[256];
912:
913: if (type[level - 1] != BOTTOM) {
914: printf ("\tblock #%ld unexpected block type: DATA\012\015", blknbr);
915: showpath ();
916: }
917:
918: key1[0] = g_EOL;
919: i = 0;
920:
921: while (i < offsets[level]) {
922:
923: len = UNSIGN (blck[i++]);
924: len += (k = UNSIGN (blck[i++]));
925:
926: if (len > 255) {
927: printf ("\tblock #%ld key too long\012\015", blknbr);
928: showpath ();
929: i += len - k;
930: }
931: else {
932:
933: if (len == 0 && i > 2) {
934: printf ("\tblock #%ld empty key\012\015", blknbr);
935: showpath ();
936: }
937:
938: while (k < len) key0[k++] = blck[i++];
939:
940: key0[k] = g_EOL;
941:
942: if (key_check (key0)) {
943: printf ("\tblock #%ld illegal key\012\015", blknbr);
944: showpath ();
945: }
946:
947: if (g_collate (key1, key0) == 0) {
948: printf ("\tblock #%ld collation mismatch\012\015", blknbr);
949: show (key1);
950: show (key0);
951: showpath ();
952: }
953:
954: stcpy0 (key1, key0, k + 1);
955: }
956:
957: k = i + 1;
958: len = UNSIGN (blck[i]);
959: i += UNSIGN (blck[i]);
960: i++; /* skip data */
961:
962: #ifdef NEVER
963: while (k < i) {
964: if (blck[k++] & ~0177) {
965: printf ("\tblock #%ld illegal character in data string\012\015", blknbr);
966: showpath ();
967: break;
968: }
969: }
970: #endif /* NEVER */
971:
972: if (i > DATALIM) {
973: printf ("\tblock #%ld data in status bytes\012\015", blknbr);
974: showpath ();
975: }
976:
977: }
978:
979: if (i > offsets[level]) {
980: printf ("\tblock #%ld offset mismatch %ld vs. %d\012\015", blknbr, i, offsets[level]);
981: showpath ();
982: }
983: }
984: break;
985:
986: default:
987: printf ("\tblock #%ld illegal type %d\012\015", blknbr, type[level]);
988: showpath ();
989:
990: }
991:
992: return;
993:
994: } /* end check */
995: void
996: showpath ()
997: { /* display path of pointers */
998: int i;
999:
1000: if (level > 1)
1001: for (i = 0; i < level; i++)
1002: printf (" path level(%d)=%ld\012\015", i, llink[i]);
1003: if (++exstat >= ERRLIM) {
1.2 snw 1004: fprintf (stderr, "Error limit exceeded (%hd errors)\012\015", exstat);
1.1 snw 1005: return;
1006: }
1007: return;
1008: }
1009: /******************************************************************************/
1010: int key_check (char *key) /* checks a global key in compressed form */
1011: {
1012: short ch,
1013: typ = 0;
1014:
1015: while ((ch = UNSIGN (*key++)) != g_EOL) {
1016: if (ch == (DEL << 1)) {
1017: if ((ch = UNSIGN (*key++)) == (DEL << 1))
1018: key++;
1019: ch = SP;
1020: }
1021: if (ch >= SP) {
1022: if (typ == 2)
1023: return 1;
1024: typ = 1;
1025: }
1026: /* alphabetics */
1027: else {
1028: if (typ == 1)
1029: return 1;
1030: typ = 2; /* numerics '.' '-' */
1031: if (ch >= 20 && ch != POINT && ch != MINUS)
1032: return 1; /* illegal character */
1033: }
1034: if (ch & 01)
1035: typ = 0; /* comma between two indices */
1036: }
1037: return 0;
1038: } /* end key_check */
1039: /******************************************************************************/
1.2 snw 1040: static short int g_collate (char s[], char t[]) /* if 't' follows 's' in MUMPS collating */
1.1 snw 1041: {
1042: register int chs = *s;
1043: register int cht = *t;
1044: register int tx = 0;
1045: register int sx;
1046: short dif;
1047:
1048: /* the empty one is the leader! */
1049: if (chs == g_EOL) {
1050: if (cht == g_EOL)
1051: return 2;
1052: return 1;
1053: }
1054: if (cht == g_EOL)
1055: return FALSE;
1056:
1057: while (cht == s[tx]) {
1058: if (cht == g_EOL)
1059: return 0;
1060: cht = t[++tx];
1061: } /* (s==t) */
1062: chs = s[tx];
1063: if (chs == OMEGA)
1064: return 0;
1065: if (chs == ALPHA)
1066: return cht != g_EOL;
1067: if (chs == g_EOL && t[tx - 1] & 01)
1068: return 1;
1069: if (cht == g_EOL && s[tx - 1] & 01)
1070: return 0;
1071:
1072: /* vade retro usque ad comma */
1073: if (tx > 0) {
1074: tx--;
1075: while ((t[tx] & 01) == 0)
1076: if (--tx < 0)
1077: break;
1078: tx++;
1079: }
1080: chs = s[tx];
1081: cht = t[tx];
1082: if (UNSIGN (chs) <= POINT) { /* then come numerics */
1083: if (UNSIGN (cht) > POINT)
1084: return UNSIGN (cht) != g_EOL;
1085: /* both are numeric! now compare numeric values */
1086: /*****g_comp()*********************************************************/
1087: if (chs == MINUS) {
1088: if (cht != MINUS)
1089: return 1;
1090: } else {
1091: if (cht == MINUS)
1092: return 0;
1093: }
1094: if (chs == 1 && cht == POINT)
1095: return 1;
1096: if (cht == 1 && chs == POINT)
1097: return 0;
1098: dif = sx = tx;
1099: while (s[sx] != POINT) {
1100: if (s[sx++] & 01)
1101: break;
1102: }
1103: while (t[tx] != POINT) {
1104: if (t[tx++] & 01)
1105: break;
1106: }
1107: if (tx > sx)
1108: return (cht != MINUS);
1109: if (tx < sx)
1110: return (cht == MINUS);
1111: tx = dif;
1112: while ((cht >> 1) == (chs >> 1)) {
1113: if (cht & 01)
1114: return t[dif] == MINUS;
1115: if (chs & 01)
1116: return t[dif] != MINUS;
1117: chs = s[++tx];
1118: cht = t[tx];
1119: }
1120: return (((cht >> 1) > (chs >> 1)) == (t[dif] != MINUS))
1121: && (t[tx] != s[tx]);
1122: /**********************************************************************/
1123: }
1124: if (UNSIGN (cht) <= POINT)
1125: return 0;
1126: while ((dif = (UNSIGN (cht) >> 1) - (UNSIGN (chs) >> 1)) == 0) { /* ASCII collating */
1127: if ((cht & 01) && ((chs & 01) == 0))
1128: return 0;
1129: if ((chs & 01) && ((cht & 01) == 0))
1130: return 1;
1131: chs = s[++tx];
1132: cht = t[tx];
1133: }
1134: if (chs == g_EOL)
1135: return 1;
1136: if (cht == g_EOL)
1137: return 0;
1138: return dif > 0;
1139: } /* end g_collate */
1140: /******************************************************************************/
1.2 snw 1141: void show (char key[])
1.1 snw 1142: {
1143: int k,
1144: ch,
1145: i,
1146: j;
1147: char data[256];
1148:
1149: k = 0;
1150: i = 0;
1151: j = 0;
1152: while ((ch = UNSIGN (key[i++])) != g_EOL) {
1153: if (k) {
1154: k = 0;
1155: if (ch > ' ')
1156: data[j++] = '"';
1157: }
1158: data[j] = (ch > SP ? (ch >> 1) : (ch < 20 ? (ch >> 1) + '0' : (ch >> 1) + ' '));
1159: if (data[j++] == '"')
1160: data[j++] = '"';
1161: if (ch & 01) {
1162: if (ch > SP)
1163: data[j++] = '"';
1164: data[j++] = ',';
1165: k = 1;
1166: }
1167: }
1168: data[j--] = 0;
1169: printf ("(%s);", data);
1170: return;
1171: } /* end show() */
1172: /******************************************************************************/
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>