Annotation of freem/src/gfix.c, revision 1.1
1.1 ! snw 1: /*
! 2: * *
! 3: * * *
! 4: * * *
! 5: * ***************
! 6: * * * * *
! 7: * * MUMPS *
! 8: * * * * *
! 9: * ***************
! 10: * * *
! 11: * * *
! 12: * *
! 13: *
! 14: * gfix.c
! 15: * display freem database blocks on-screen
! 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: #include <setjmp.h>
! 42: #include <signal.h>
! 43: #include <stdio.h>
! 44: #include "mpsdef0.h"
! 45: #include "errmsg.h"
! 46: #include <fcntl.h>
! 47: #include <unistd.h>
! 48: #include <string.h>
! 49: #include <sys/types.h>
! 50: #include <sys/wait.h>
! 51:
! 52: /* needed if byte data are to be interpreted as unsigned integer */
! 53: #define UNSIGN(A) ((A)&0377)
! 54:
! 55: #define g_EOL 30
! 56: #define POINT 28
! 57: #define MINUS 26
! 58:
! 59: #define ROOT 0L
! 60: /* length of blocks. status bytes defined as offset to blocklength */
! 61: #define BLOCKLEN 1024
! 62: #define DATALIM (BLOCKLEN-11)
! 63: #define LLPTR (BLOCKLEN-10)
! 64: #define NRBLK LLPTR
! 65: #define RLPTR (BLOCKLEN- 6)
! 66: #define FREE RLPTR
! 67: #define BTYP (BLOCKLEN- 3)
! 68: #define OFFS (BLOCKLEN- 2)
! 69:
! 70: #define PLEN 3
! 71:
! 72: #define EMPTY 0
! 73: #define FBLK 1
! 74: #define POINTER 2
! 75: #define BOTTOM 6
! 76: #define DATA 8
! 77:
! 78: #define LF 10
! 79: #define CR 13
! 80: #define SPC ' '
! 81: #define NUL 0
! 82: #define DEL 127
! 83:
! 84: #ifndef SYSFIVE
! 85: #define FreeM_timezone -3600
! 86: #else
! 87:
! 88: #ifdef __CYGWIN__
! 89: #define FreeM_timezone _timezone
! 90: #else
! 91: extern long FreeM_timezone;
! 92: #endif /* __CYGWIN__ */
! 93:
! 94: #endif /* SYSFIVE */
! 95:
! 96: /* mumps commands */
! 97: #define BREAK 'b'
! 98: #define CLOSE 'c'
! 99: #define DO 'd'
! 100: #define DO_BLOCK 2
! 101: #define ELSE 'e'
! 102: #define FOR 'f'
! 103: #define GOTO 'g'
! 104: #define HA 'h'
! 105: #define HALT '0'
! 106: #define HANG '1'
! 107: #define IF 'i'
! 108: #define JOB 'j'
! 109: #define KILL 'k'
! 110: #define LOCK 'l'
! 111: #define NEW 'n'
! 112: #define OPEN 'o'
! 113: #define QUIT 'q'
! 114: #define READ 'r'
! 115: #define SET 's'
! 116: #define USE 'u'
! 117: #define VIEW 'v'
! 118: #define WRITE 'w'
! 119: #define XECUTE 'x'
! 120:
! 121: #define ZALLOCATE 'A'
! 122: #define ZBREAK 'B'
! 123: #define ZDEALLOCATE 'D'
! 124: #define ZGO 'G'
! 125: #define ZHALT 'H'
! 126: #define ZINSERT 'I'
! 127: #define ZJOB 'J'
! 128: #define ZLOAD 'L'
! 129: #define ZNEW 'N'
! 130: #define ZPRINT 'P'
! 131: #define ZQUIT 'Q'
! 132: #define ZREMOVE 'R'
! 133: #define ZSAVE 'S'
! 134: #define ZTRAP 'T'
! 135: #define ZWRITE 'W'
! 136: #define PRIVATE SP
! 137:
! 138:
! 139:
! 140: int
! 141: main (argc, argv)
! 142: int argc; /* arguments count */
! 143: char *argv[]; /* arguments string */
! 144:
! 145: {
! 146: static char ASCII[] = "NULSOHSTXETXEOTENQACKBELBS TABLF VT FF CR SO SI DLEDC1DC2DC3DC4NAKSYNETBCANEM SUBESCFS GS RS US ";
! 147: char filnam[40];
! 148: short filedes;
! 149: char block[BLOCKLEN];
! 150: char key[512];
! 151: char data[1024];
! 152: long blknbr;
! 153: short offset;
! 154: short type;
! 155: unsigned long pointer;
! 156: short length;
! 157: short koffs;
! 158:
! 159: register int i,
! 160: j,
! 161: k,
! 162: ch;
! 163:
! 164: filnam[0] = '^';
! 165: filnam[1] = 0;
! 166:
! 167: if (argc > 1) {
! 168: j = 0;
! 169: while (--argc > 0) {
! 170: j++; /* accept it with or without '^' sign */
! 171: if (**(argv + j) == '-') {
! 172: fprintf (stderr, "usage is: %s [^]global\012\015", *argv);
! 173: exit (0);
! 174: }
! 175: if (**(argv + j) == '^')
! 176: strcpy (filnam, *(argv + j));
! 177: else
! 178: strcpy (&filnam[1], *(argv + j));
! 179: }
! 180: } else {
! 181: printf ("\012\015display global ^");
! 182: scanf ("%s", &filnam[1]);
! 183: }
! 184: if ((filedes = open (filnam, 0)) == -1) {
! 185: printf ("cannot open file %s\007\012\015", filnam);
! 186: exit (0);
! 187: }
! 188: again:;
! 189:
! 190: printf ("\012\015display block #");
! 191: scanf ("%ld", &blknbr);
! 192: if (blknbr < 0) {
! 193: printf ("\012\015*** done ***\012\015\033[r\033[24H");
! 194: exit (0);
! 195: }
! 196: lseek (filedes, blknbr * 1024L, 0);
! 197: if (read (filedes, block, BLOCKLEN) == 0) {
! 198: printf ("block #%ld does not exist\007\012\015-1 will terminate", blknbr);
! 199: goto again;
! 200: }
! 201: printf ("\033[r\033[2Jglobal\033[;20H%s\033[;40Hblock\033[;60H%ld",
! 202: filnam,
! 203: blknbr);
! 204: printf ("\033[2Hblock type\033[2;20H");
! 205: type = block[BTYP];
! 206: switch (type) {
! 207: case DATA:
! 208: printf ("DATA");
! 209: break;
! 210: case POINTER:
! 211: printf ("POINTER");
! 212: break;
! 213: case BOTTOM:
! 214: printf ("BOTTOM POINTER");
! 215: break;
! 216: case EMPTY:
! 217: printf ("EMPTY");
! 218: break;
! 219: case FBLK:
! 220: printf ("FBLK");
! 221: break;
! 222: default:
! 223: printf ("ILLEGAL TYPE");
! 224: type = DATA;
! 225: }
! 226: if (blknbr == ROOT)
! 227: printf (", ROOT");
! 228:
! 229: offset = UNSIGN (block[OFFS]) * 256 +
! 230: UNSIGN (block[OFFS + 1]);
! 231: printf ("\033[2;40Hoffset\033[2;60H%d", offset);
! 232: if (offset > DATALIM || (type == FBLK && offset % PLEN)) {
! 233: printf (" ???");
! 234: offset = DATALIM;
! 235: }
! 236: pointer = UNSIGN (block[LLPTR]) * 65536 +
! 237: UNSIGN (block[LLPTR + 1]) * 256 +
! 238: UNSIGN (block[LLPTR + 2]);
! 239: printf ("\033[3H%s\033[3;20H%ld",
! 240: blknbr != ROOT ? "LEFT LINK POINTER" : "NR_OF_BLKS",
! 241: pointer);
! 242:
! 243: pointer = UNSIGN (block[RLPTR]) * 65536 +
! 244: UNSIGN (block[RLPTR + 1]) * 256 +
! 245: UNSIGN (block[RLPTR + 2]);
! 246: printf ("\033[3;40H%s\033[3;60H%ld",
! 247: blknbr != ROOT ? "RIGHT LINK POINTER" : "FREE",
! 248: pointer);
! 249:
! 250: printf ("\033[4;24r\033[4H"); /* define scrolling area */
! 251:
! 252: if (type == FBLK) {
! 253: i = 0;
! 254: while (i < offset) {
! 255: k = UNSIGN (block[i]) * 65536 +
! 256: UNSIGN (block[i + 1]) * 256 +
! 257: UNSIGN (block[i + 2]);
! 258: i += PLEN;
! 259: printf ("%8d", k);
! 260: }
! 261: goto again;
! 262: }
! 263: i = 0;
! 264: while (i < offset) {
! 265: printf ("\012\015%3d", i);
! 266: length = UNSIGN (block[i++]);
! 267: k = koffs = UNSIGN (block[i++]);
! 268: if ((i + length) > offset)
! 269: break;
! 270: for (j = 0; j < length; j++)
! 271: key[k++] = block[i++];
! 272: key[k] = g_EOL;
! 273: /*----------------------*/
! 274: {
! 275: short ch0,
! 276: i,
! 277: j,
! 278: k,
! 279: typ;
! 280:
! 281: j = 0;
! 282: i = 0;
! 283: data[j++] = '(';
! 284: k = 1;
! 285: while ((ch = UNSIGN (key[i++])) != g_EOL) {
! 286: if (k) {
! 287: k = 0;
! 288: if ((typ = (ch > SPC)))
! 289: data[j++] = '"';
! 290: }
! 291: ch0 = (ch >= SPC ? (ch >> 1) : /* 'string' chars */
! 292: (ch < 20 ? (ch >> 1) + '0' : /* 0...9 */
! 293: (ch >> 1) + SPC)); /* '.' or '-' */
! 294: if (ch0 == DEL) {
! 295: if (((ch = UNSIGN (key[i++])) >> 1) == DEL) {
! 296: ch0 += DEL;
! 297: ch = UNSIGN (key[i++]);
! 298: }
! 299: ch0 += (ch >> 1);
! 300: data[j] = '<';
! 301: data[++j] = '0' + ch0 / 100;
! 302: data[++j] = '0' + (ch0 % 100) / 10;
! 303: data[++j] = '0' + ch0 % 10;
! 304: data[++j] = '>';
! 305: } else
! 306: data[j] = ch0;
! 307: if (data[j++] == '"')
! 308: data[j++] = '"';
! 309: if (ch & 01) {
! 310: if (typ)
! 311: data[j++] = '"';
! 312: data[j++] = ',';
! 313: k = 1;
! 314: }
! 315: }
! 316: data[j--] = 0;
! 317: data[j] = ')';
! 318: if (j == 0)
! 319: data[0] = 0;
! 320: while (j >= 0) {
! 321: if ((ch = data[--j]) < SPC || ch >= DEL)
! 322: break;
! 323: }
! 324: if (j < 0)
! 325: printf ("[%d][%d] %s ", length, koffs, data);
! 326: else
! 327: printf ("[%d][%d] <illegal subscipt>", length, koffs);
! 328: /*----------------------*/
! 329: }
! 330: if (type == DATA) {
! 331: length = UNSIGN (block[i++]);
! 332: k = 0;
! 333: if ((i + length) > offset)
! 334: break;
! 335: while (length-- > 0) {
! 336: ch = UNSIGN (block[i++]);
! 337: if ((ch >= SPC) && (ch < DEL))
! 338: data[k++] = ch;
! 339: else {
! 340: data[k++] = '<';
! 341: if ((ch >= NUL) && (ch < SPC)) {
! 342: ch = ch * 3;
! 343: data[k++] = ASCII[ch++];
! 344: data[k++] = ASCII[ch++];
! 345: if ((data[k++] = ASCII[ch++]) == SPC)
! 346: k--;
! 347: } else if (ch == DEL) {
! 348: data[k++] = 'D';
! 349: data[k++] = 'E';
! 350: data[k++] = 'L';
! 351: } else {
! 352: if (ch > 99) {
! 353: data[k++] = '0' + (ch / 100);
! 354: ch = ch % 100;
! 355: }
! 356: if (ch > 9) {
! 357: data[k++] = '0' + (ch / 10);
! 358: ch = ch % 10;
! 359: }
! 360: data[k++] = '0' + ch;
! 361: }
! 362: data[k++] = '>';
! 363: }
! 364: }
! 365: data[k] = 0;
! 366: printf ("= %s", data);
! 367: } else {
! 368: pointer = UNSIGN (block[i]) * 65536 +
! 369: UNSIGN (block[i + 1]) * 256 +
! 370: UNSIGN (block[i + 2]);
! 371: i += PLEN;
! 372: printf ("-> %ld", pointer);
! 373: }
! 374: }
! 375: if (i != offset)
! 376: printf ("\012\015wrong offset %d vs. %d\012\015", offset, i);
! 377: goto again;
! 378: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>