Annotation of freem/src/fma_gedit.c, revision 1.3

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

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