Annotation of freem/src/fma_gedit.c, revision 1.4
1.1 snw 1: /*
1.4 ! snw 2: * $Id: fma_gedit.c,v 1.3 2025/03/09 19:14:24 snw Exp $
1.1 snw 3: * FreeM global editor
4: *
5: *
1.2 snw 6: * Author: Serena Willis <snw@coherent-logic.com>
1.1 snw 7: * Copyright (C) 1998 MUG Deutschland
1.3 snw 8: * Copyright (C) 2023, 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: fma_gedit.c,v $
! 27: * Revision 1.3 2025/03/09 19:14:24 snw
! 28: * First phase of REUSE compliance and header reformat
! 29: *
1.3 snw 30: *
31: * SPDX-FileCopyrightText: (C) 2025 Coherent Logic Development LLC
32: * SPDX-License-Identifier: AGPL-3.0-or-later
1.1 snw 33: **/
34:
35: #include <stdlib.h>
36: #include <stdio.h>
37: #include <stdlib.h>
38: #include <string.h>
39: #include <dirent.h>
40: #include <time.h>
41: #include <unistd.h>
42: #include <sys/types.h>
43: #include <sys/stat.h>
44: #include <fcntl.h>
45:
46: #include "fmadm.h"
47: #include "fma_globals.h"
48:
49: #if defined HAVE_NCURSESW_CURSES_H
50: # include <ncursesw/curses.h>
51: #elif defined HAVE_NCURSESW_H
52: # include <ncursesw.h>
53: #elif defined HAVE_NCURSES_CURSES_H
54: # include <ncurses/curses.h>
55: #elif defined HAVE_NCURSES_H
56: # include <ncurses.h>
57: #elif defined HAVE_CURSES_H
58: # include <curses.h>
59: #else
60: # error "SysV or X/Open-compatible Curses header file required"
61: #endif
62:
63: #define GE_MXGBL 100
64:
65: typedef struct ge_key {
66: char key[256];
67: char data[256];
68:
69: struct ge_key *next;
70: } ge_key;
71:
72: typedef struct ge_blockinfo {
73:
74: int keylen;
75: int keyoffs;
76: char key[STRLEN];
77: int datalen;
78: char data[STRLEN];
79:
80: long llptr;
81: long rlptr;
82: long offset;
83:
84: long blockcount;
85: int collation;
86:
87: int btype;
88: char bt_desc[40];
89: long free_offset;
90:
91: long keyct;
92: ge_key *key_head;
93:
94: } ge_blockinfo;
95:
96:
97: typedef struct ge_buf {
98: char name[256];
99: char namespace[256];
100: char pth[4096];
101:
102: short fd;
103:
104: long blockct;
105: long gblsize;
106:
107: int blocknum;
108: char block_r[BLOCKLEN];
109: ge_blockinfo block;
110:
111: short block_dirty;
112: short is_free;
113: } ge_buf;
114:
115:
116: ge_buf ge_buffers[GE_MXGBL];
117:
118: int ge_curbuf;
119: int ge_nextbuf;
120:
121: int ge_rows;
122: int ge_columns;
123:
124: WINDOW *wge_key;
125: WINDOW *wge_key_border;
126: WINDOW *wge_node;
127: WINDOW *wge_node_border;
128: WINDOW *wge_msg_border;
129: WINDOW *wge_msg;
130: WINDOW *wge_block_border;
131: WINDOW *wge_block;
132: WINDOW *wge_global_border;
133: WINDOW *wge_global;
134: WINDOW *wge_command;
135:
136: WINDOW *wge_cur;
137:
138: void ge_set_wintitle(WINDOW *border_window, char *title);
139: void ge_wininit(void);
140: void ge_update_gbl(void);
141: void ge_select_buffer(int buf);
142: short ge_select_block(int buf, long blocknum);
143: void ge_update_block(void);
144: int ge_open_global(char *gblname);
145: void ge_init_buffers(void);
146: void ge_eventloop(void);
147: void ge_decode_key(const char *key, char *buf);
148: void ge_win_next(void);
149: void ge_win_previous(void);
150:
151: #define SPC ' '
152:
153: int fma_globals_edit(int optc, char **opts)
154: {
155: int editbuf;
156:
157: ge_curbuf = 0;
158: ge_nextbuf = 0;
159:
160: ge_init_buffers ();
161: initscr ();
162: ge_wininit ();
163:
164: wge_cur = wge_command;
165:
166: wprintw (wge_msg, "FreeM Global Editor\n");
167: wprintw (wge_msg, " Copyright (C) 2023 Coherent Logic Development LLC\n\n");
168:
1.4 ! snw 169: wprintw(wge_msg, "optc = %d\n\n", optc);
! 170:
1.1 snw 171: wattron (wge_msg, A_BOLD);
172: wprintw (wge_msg, "Left and right arrow keys navigate blocks; 'q' to quit.\n\n");
173: wattroff (wge_msg, A_BOLD);
174: wrefresh (wge_msg);
175:
176: if (optc == 2) {
177:
178: if ((editbuf = ge_open_global (opts[fma_base_opt])) == -1) {
179: wprintw (wge_msg, "fmadm: cannot open global %s\n", opts[fma_base_opt]);
180: }
181: else {
182: ge_select_buffer (editbuf);
183: }
184:
185: }
186:
187: ge_eventloop ();
188:
189: endwin ();
190: return 0;
191: }
192:
193: void ge_win_next(void)
194: {
195:
196: /*
197: switch (wge_cur) {
198:
199: case wge_key:
200: wge_cur = wge_node;
201: break;
202:
203: case wge_node:
204: wge_cur = wge_global;
205: break;
206:
207: case wge_global:
208: wge_cur = wge_block;
209: break;
210:
211: case wge_block:
212: wge_cur = wge_command;
213: break;
214:
215: case wge_command:
216: wge_cur = wge_msg;
217: break;
218:
219: case wge_msg:
220: wge_cur = wge_key;
221: break;
222:
223: }
224: */
225:
226: }
227:
228:
229: void ge_win_previous(void)
230: {
231:
232: }
233:
234:
235: void ge_eventloop(void)
236: {
237: int c;
238: short quit_flag;
239:
240: quit_flag = FALSE;
241:
242: while (!quit_flag) {
243:
244: noecho ();
245: c = wgetch (wge_command);
246: echo ();
247:
248: switch (c) {
249:
250: case 'q':
251: case 'Q':
252: quit_flag = TRUE;
253: break;
254:
255: case KEY_LEFT:
256: if (ge_buffers[ge_curbuf].blocknum != 0) {
257: ge_select_block (ge_curbuf, ge_buffers[ge_curbuf].blocknum - 1);
258: }
259: else {
260: ge_select_block (ge_curbuf, ge_buffers[ge_curbuf].blockct - 1);
261: }
262: break;
263:
264: case KEY_RIGHT:
265: if (ge_buffers[ge_curbuf].blocknum + 1 < ge_buffers[ge_curbuf].blockct) {
266: ge_select_block (ge_curbuf, ge_buffers[ge_curbuf].blocknum + 1);
267: }
268: else {
269: ge_select_block (ge_curbuf, 0);
270: }
271: break;
272:
273: }
274:
275: }
276:
277: }
278:
279: void ge_init_buffers(void)
280: {
281: register int i;
282: register int j;
283:
284: for (i = 0; i < GE_MXGBL; i++) {
285: ge_buffers[i].name[0] = '\0';
286: ge_buffers[i].namespace[0] = '\0';
287: ge_buffers[i].pth[0] = '\0';
288: ge_buffers[i].fd = 0;
289: ge_buffers[i].blocknum = 0;
290: ge_buffers[i].block_dirty = FALSE;
291: ge_buffers[i].is_free = TRUE;
292: ge_buffers[i].blockct = 0;
293: ge_buffers[i].gblsize = 0;
294:
295: for (j = 0; j < BLOCKLEN; j++) {
296: ge_buffers[i].block_r[j] = '\0';
297: }
298: }
299: }
300:
301: void ge_wininit(void)
302: {
303: int half;
304:
305: getmaxyx (stdscr, ge_rows, ge_columns);
306:
307: half = ge_rows / 2;
308:
309: /* messages window */
310: wge_msg_border = newwin (10, ge_columns - 41, ge_rows - 12, 0);
311: wge_msg = newwin (8, ge_columns - 43, ge_rows - 11, 1);
312: ge_set_wintitle (wge_msg_border, "Messages");
313: wrefresh (wge_msg_border);
314: wrefresh (wge_msg);
315:
316: /* global window */
317: wge_global_border = newwin (half, 40, 0, ge_columns - 40);
318: wge_global = newwin (half - 2, 38, 1, ge_columns - 39);
319: ge_set_wintitle (wge_global_border, "No Global Selected");
320: wrefresh (wge_global_border);
321:
322: /* block window */
323: wge_block_border = newwin (half - 1, 40, half, ge_columns - 40);
324: wge_block = newwin (half - 3, 37, half + 1, ge_columns - 38);
325: ge_set_wintitle (wge_block_border, "Block");
326: wrefresh (wge_block_border);
327:
328: /* command window */
329: wge_command = newwin (1, ge_columns, ge_rows - 1, 0);
330: wrefresh (wge_command);
331:
332: scrollok (wge_msg, TRUE);
333: keypad (wge_command, TRUE);
334: keypad (wge_msg, TRUE);
335: keypad (wge_global, TRUE);
336: keypad (wge_block, TRUE);
337:
338: curs_set (0);
339: }
340:
341: void ge_update_gbl(void)
342: {
343: char wintit[4096];
344:
345: wclear (wge_global);
346: snprintf (wintit, 4095, "%s [#%d]", ge_buffers[ge_curbuf].name, ge_curbuf);
347: ge_set_wintitle (wge_global_border, wintit);
348:
349: wattron (wge_global, A_BOLD);
350: wprintw (wge_global, "GLOBAL: ");
351: wattroff (wge_global, A_BOLD);
352: wprintw (wge_global, "%s\n", ge_buffers[ge_curbuf].name);
353:
354: wattron (wge_global, A_BOLD);
355: wprintw (wge_global, "NAMESPACE: ");
356: wattroff (wge_global, A_BOLD);
357: wprintw (wge_global, "%s\n", ge_buffers[ge_curbuf].namespace);
358:
359: wattron (wge_global, A_BOLD);
360: wprintw (wge_global, "BLOCK SIZE: ");
361: wattroff (wge_global, A_BOLD);
362: wprintw (wge_global, "%d bytes\n", BLOCKLEN);
363:
364: wattron (wge_global, A_BOLD);
365: wprintw (wge_global, "BLOCK COUNT: ");
366: wattroff (wge_global, A_BOLD);
367: wprintw (wge_global, "%d\n", ge_buffers[ge_curbuf].blockct);
368:
369: wattron (wge_global, A_BOLD);
370: wprintw (wge_global, "FILE SIZE: ");
371: wattroff (wge_global, A_BOLD);
372: wprintw (wge_global, "%d bytes\n", ge_buffers[ge_curbuf].gblsize);
373:
374: wrefresh (wge_global);
375: }
376:
377:
378: void ge_select_buffer(int buf)
379: {
380: wprintw (wge_msg, "Selected buffer %d\n", buf);
381: wrefresh (wge_msg);
382:
383: ge_curbuf = buf;
384:
385: ge_update_gbl ();
386: }
387:
388: short ge_select_block(int buf, long blocknum)
389: {
390: ge_blockinfo *b;
391: char *br;
392: char key[256];
393: char decoded[4096];
394: long i;
395: long j;
396: long k;
397: long length;
398:
399: if ((blocknum < 0) || (blocknum > (ge_buffers[buf].blockct - 1))) {
400: wprintw (wge_msg, "Block number for global %s must be between 0 and %d\n", ge_buffers[buf].name, ge_buffers[buf].blockct - 1);
401: return FALSE;
402: }
403:
404: b = &(ge_buffers[buf].block);
405: br = ge_buffers[buf].block_r;
406:
407: lseek (ge_buffers[buf].fd, blocknum * BLOCKLEN, 0);
408: read (ge_buffers[buf].fd, br, BLOCKLEN);
409:
410: ge_buffers[buf].blocknum = blocknum;
411: ge_buffers[buf].block_dirty = FALSE;
412:
413: b->btype = br[BTYP];
414: switch (b->btype) {
415:
416: case DATA:
417: snprintf (b->bt_desc, 39, "DATA");
418: break;
419:
420: case POINTER:
421: snprintf (b->bt_desc, 39, "POINTER");
422: break;
423:
424: case BOTTOM:
425: snprintf (b->bt_desc, 39, "BTM PTR");
426: break;
427:
428: case EMPTY:
429: snprintf (b->bt_desc, 39, "EMPTY");
430: break;
431:
432: case FBLK:
433: snprintf (b->bt_desc, 39, "FBLK");
434: break;
435:
436: default:
437: snprintf (b->bt_desc, 39, "ILLEGAL TYPE");
438: break;
439:
440: }
441:
442: if (blocknum == ROOT) strcat (b->bt_desc, " [ROOT]");
443:
444: if (blocknum != ROOT) {
445: b->llptr = UNSIGN (br[LLPTR]) * 65536 + UNSIGN (br[LLPTR + 1]) * 256 + UNSIGN(br[LLPTR + 2]);
446: b->rlptr = UNSIGN (br[RLPTR]) * 65536 + UNSIGN (br[RLPTR + 1]) * 256 + UNSIGN(br[RLPTR + 2]);
447: }
448: else {
449: b->blockcount = UNSIGN (br[LLPTR]) * 65536 + UNSIGN (br[LLPTR + 1]) * 256 + UNSIGN(br[LLPTR + 2]);
450: b->free_offset = UNSIGN (br[RLPTR]) * 65536 + UNSIGN (br[RLPTR + 1]) * 256 + UNSIGN(br[RLPTR + 2]);
451: }
452:
453: b->offset = UNSIGN (br[OFFS]) * 256 + UNSIGN (br[OFFS + 1]);
454: b->keyct = 0;
455:
456: if (b->btype == FBLK) goto skip_keydec;
457:
458: i = 0;
459: while (i < b->offset) {
460:
461: length = UNSIGN (br[i++]);
462: k = UNSIGN (br[i++]);
463:
464: if ((i + length) > b->offset) break;
465:
466: for (j = 0; j < length; j++) {
467: key[k++] = br[i++];
468: }
469:
470: key[k] = g_EOL;
471:
472: ge_decode_key (key, decoded);
473: decoded[stlen (decoded)] = '\0';
474: wprintw (wge_msg, "found key %s\n", decoded);
475:
476: b->keyct++;
477: }
478:
479: skip_keydec:
480:
481: ge_update_block ();
482:
483: return TRUE;
484: }
485:
486: void ge_decode_key(const char *key, char *buf)
487: {
488: int ch;
489: short ch0;
490: short i;
491: short j;
492: short k;
493: short typ;
494:
495: j = 0;
496: i = 0;
497: k = 1;
498:
499: buf[j++] = '(';
500:
501: while ((ch = UNSIGN (key[i++])) != g_EOL) {
502:
503: if (k) {
504:
505: k = 0;
506:
507: if ((typ = (ch > SPC))) {
508: buf[j++] = '"';
509: }
510:
511: }
512:
513: ch0 = (ch >= SPC ? (ch >> 1) : (ch < 20 ? (ch >> 1) + '0' : (ch >> 1) + SPC));
514:
515: if (ch0 == DEL) {
516: if (((ch = UNSIGN (key[i++])) >> 1) == DEL) {
517: ch0 += DEL;
518: ch = UNSIGN (key[i++]);
519: }
520:
521: ch0 += (ch >> 1);
522: buf[j] = '<';
523: buf[++j] = '0' + ch0 / 100;
524: buf[++j] = '0' + (ch0 % 100) / 10;
525: buf[++j] = '0' + ch0 % 10;
526: buf[++j] = '>';
527: }
528: else {
529: buf[j] = ch0;
530: }
531:
532: if (buf[j++] == '"') {
533: buf[j++] = '"';
534: }
535:
536: if (ch & 01) {
537: if (typ) buf[j++] = '"';
538: buf[j++] = ',';
539: k = 1;
540: }
541: }
542:
543: buf[j--] = 0;
544: buf[j] = ')';
545: if (j == 0) buf[0] = 0;
546:
547: while (j >= 0) {
548: if ((ch = buf[--j]) < SPC || ch >= DEL) break;
549: }
550:
551: }
552:
553: void ge_update_block(void)
554: {
555: char wintit[4096];
556:
557: wclear (wge_block);
558: snprintf (wintit, 4095, "Block %d of %d", ge_buffers[ge_curbuf].blocknum, ge_buffers[ge_curbuf].blockct);
559: ge_set_wintitle (wge_block_border, wintit);
560:
561: wattron (wge_block, A_BOLD);
562: wprintw (wge_block, "TYPE: ");
563: wattroff (wge_block, A_BOLD);
564: wprintw (wge_block, "%s\n", ge_buffers[ge_curbuf].block.bt_desc);
565:
566: if (ge_buffers[ge_curbuf].blocknum != ROOT) {
567: wattron (wge_block, A_BOLD);
568: wprintw (wge_block, "LEFT LINK POINTER: ");
569: wattroff (wge_block, A_BOLD);
570: wprintw (wge_block, "%d\n", ge_buffers[ge_curbuf].block.llptr);
571:
572: wattron (wge_block, A_BOLD);
573: wprintw (wge_block, "RIGHT LINK POINTER: ");
574: wattroff (wge_block, A_BOLD);
575: wprintw (wge_block, "%d\n", ge_buffers[ge_curbuf].block.rlptr);
576: }
577: else {
578: wattron (wge_block, A_BOLD);
579: wprintw (wge_block, "BLOCK COUNT: ");
580: wattroff (wge_block, A_BOLD);
581: wprintw (wge_block, "%d\n", ge_buffers[ge_curbuf].block.blockcount);
582:
583: wattron (wge_block, A_BOLD);
584: wprintw (wge_block, "FREE OFFSET: ");
585: wattroff (wge_block, A_BOLD);
586: wprintw (wge_block, "%d\n", ge_buffers[ge_curbuf].block.free_offset);
587: }
588:
589: wattron (wge_block, A_BOLD);
590: wprintw (wge_block, "KEY COUNT: ");
591: wattroff (wge_block, A_BOLD);
592: wprintw (wge_block, "%d\n", ge_buffers[ge_curbuf].block.keyct);
593:
594: wrefresh (wge_block);
595: }
596:
597: int ge_open_global(char *gblname)
598: {
599: char gpath[4096];
600: int buf;
601: struct stat sb;
602:
603: buf = ge_nextbuf++;
604:
605: snprintf (gpath, 4095, "%s/%s", fma_global_path, gblname);
606:
607: wprintw (wge_msg, "Opening global %s [path %s, namespace %s]... ", gblname, gpath, fma_namespace);
608:
609: wrefresh (wge_msg);
610:
611: if ((ge_buffers[buf].fd = open (gpath, 0)) == -1) {
612: wprintw (wge_msg, "[FAIL]\n");
613: wrefresh (wge_msg);
614: return -1;
615: }
616: else {
617: wprintw (wge_msg, "[OK]\n");
618: wrefresh (wge_msg);
619: }
620:
621: fstat (ge_buffers[buf].fd, &sb);
622:
623: ge_buffers[buf].gblsize = sb.st_size;
624: ge_buffers[buf].blockct = sb.st_size / BLOCKLEN;
625:
626: strcpy (ge_buffers[buf].name, gblname);
627: strcpy (ge_buffers[buf].namespace, fma_namespace);
628: strcpy (ge_buffers[buf].pth, gpath);
629:
630: ge_buffers[buf].blocknum = 0;
631: ge_buffers[buf].block_dirty = FALSE;
632:
633: ge_curbuf = buf;
634:
635: ge_select_block (buf, 0);
636:
637: return buf;
638: }
639:
640: void ge_set_wintitle(WINDOW *border_window, char *title)
641: {
642: box (border_window, 0, 0);
643: wattron (border_window, A_BOLD);
644: mvwprintw (border_window, 0, 3, title);
645: wattroff (border_window, A_BOLD);
646: wrefresh (border_window);
647: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>