[BACK]Return to de.c CVS log [TXT][DIR] Up to [local] / OpenXM_contrib / gc / cord

Annotation of OpenXM_contrib/gc/cord/de.c, Revision 1.1

1.1     ! maekawa     1: /*
        !             2:  * Copyright (c) 1993-1994 by Xerox Corporation.  All rights reserved.
        !             3:  *
        !             4:  * THIS MATERIAL IS PROVIDED AS IS, WITH ABSOLUTELY NO WARRANTY EXPRESSED
        !             5:  * OR IMPLIED.  ANY USE IS AT YOUR OWN RISK.
        !             6:  *
        !             7:  * Permission is hereby granted to use or copy this program
        !             8:  * for any purpose,  provided the above notices are retained on all copies.
        !             9:  * Permission to modify the code and to distribute modified code is granted,
        !            10:  * provided the above notices are retained, and a notice that the code was
        !            11:  * modified is included with the above copyright notice.
        !            12:  *
        !            13:  * Author: Hans-J. Boehm (boehm@parc.xerox.com)
        !            14:  */
        !            15: /*
        !            16:  * A really simple-minded text editor based on cords.
        !            17:  * Things it does right:
        !            18:  *     No size bounds.
        !            19:  *     Inbounded undo.
        !            20:  *     Shouldn't crash no matter what file you invoke it on (e.g. /vmunix)
        !            21:  *             (Make sure /vmunix is not writable before you try this.)
        !            22:  *     Scrolls horizontally.
        !            23:  * Things it does wrong:
        !            24:  *     It doesn't handle tabs reasonably (use "expand" first).
        !            25:  *     The command set is MUCH too small.
        !            26:  *     The redisplay algorithm doesn't let curses do the scrolling.
        !            27:  *     The rule for moving the window over the file is suboptimal.
        !            28:  */
        !            29: /* Boehm, February 6, 1995 12:27 pm PST */
        !            30:
        !            31: /* Boehm, May 19, 1994 2:20 pm PDT */
        !            32: #include <stdio.h>
        !            33: #include "gc.h"
        !            34: #include "cord.h"
        !            35:
        !            36: #ifdef THINK_C
        !            37: #define MACINTOSH
        !            38: #include <ctype.h>
        !            39: #endif
        !            40:
        !            41: #if defined(__BORLANDC__) && !defined(WIN32)
        !            42:     /* If this is DOS or win16, we'll fail anyway.     */
        !            43:     /* Might as well assume win32.                     */
        !            44: #   define WIN32
        !            45: #endif
        !            46:
        !            47: #if defined(WIN32)
        !            48: #  include <windows.h>
        !            49: #  include "de_win.h"
        !            50: #elif defined(MACINTOSH)
        !            51: #      include <console.h>
        !            52: /* curses emulation. */
        !            53: #      define initscr()
        !            54: #      define endwin()
        !            55: #      define nonl()
        !            56: #      define noecho() csetmode(C_NOECHO, stdout)
        !            57: #      define cbreak() csetmode(C_CBREAK, stdout)
        !            58: #      define refresh()
        !            59: #      define addch(c) putchar(c)
        !            60: #      define standout() cinverse(1, stdout)
        !            61: #      define standend() cinverse(0, stdout)
        !            62: #      define move(line,col) cgotoxy(col + 1, line + 1, stdout)
        !            63: #      define clrtoeol() ccleol(stdout)
        !            64: #      define de_error(s) { fprintf(stderr, s); getchar(); }
        !            65: #      define LINES 25
        !            66: #      define COLS 80
        !            67: #else
        !            68: #  include <curses.h>
        !            69: #  define de_error(s) { fprintf(stderr, s); sleep(2); }
        !            70: #endif
        !            71: #include "de_cmds.h"
        !            72:
        !            73: /* List of line number to position mappings, in descending order. */
        !            74: /* There may be holes.                                           */
        !            75: typedef struct LineMapRep {
        !            76:     int line;
        !            77:     size_t pos;
        !            78:     struct LineMapRep * previous;
        !            79: } * line_map;
        !            80:
        !            81: /* List of file versions, one per edit operation */
        !            82: typedef struct HistoryRep {
        !            83:     CORD file_contents;
        !            84:     struct HistoryRep * previous;
        !            85:     line_map map;      /* Invalid for first record "now" */
        !            86: } * history;
        !            87:
        !            88: history now = 0;
        !            89: CORD current;          /* == now -> file_contents.     */
        !            90: size_t current_len;    /* Current file length.         */
        !            91: line_map current_map = 0;      /* Current line no. to pos. map  */
        !            92: size_t current_map_size = 0;   /* Number of current_map entries.       */
        !            93:                                /* Not always accurate, but reset       */
        !            94:                                /* by prune_map.                        */
        !            95: # define MAX_MAP_SIZE 3000
        !            96:
        !            97: /* Current display position */
        !            98: int dis_line = 0;
        !            99: int dis_col = 0;
        !           100:
        !           101: # define ALL -1
        !           102: # define NONE - 2
        !           103: int need_redisplay = 0;        /* Line that needs to be redisplayed.   */
        !           104:
        !           105:
        !           106: /* Current cursor position. Always within file. */
        !           107: int line = 0;
        !           108: int col = 0;
        !           109: size_t file_pos = 0;   /* Character position corresponding to cursor.  */
        !           110:
        !           111: /* Invalidate line map for lines > i */
        !           112: void invalidate_map(int i)
        !           113: {
        !           114:     while(current_map -> line > i) {
        !           115:         current_map = current_map -> previous;
        !           116:         current_map_size--;
        !           117:     }
        !           118: }
        !           119:
        !           120: /* Reduce the number of map entries to save space for huge files. */
        !           121: /* This also affects maps in histories.                                  */
        !           122: void prune_map()
        !           123: {
        !           124:     line_map map = current_map;
        !           125:     int start_line = map -> line;
        !           126:
        !           127:     current_map_size = 0;
        !           128:     for(; map != 0; map = map -> previous) {
        !           129:        current_map_size++;
        !           130:        if (map -> line < start_line - LINES && map -> previous != 0) {
        !           131:            map -> previous = map -> previous -> previous;
        !           132:        }
        !           133:     }
        !           134: }
        !           135: /* Add mapping entry */
        !           136: void add_map(int line, size_t pos)
        !           137: {
        !           138:     line_map new_map = GC_NEW(struct LineMapRep);
        !           139:
        !           140:     if (current_map_size >= MAX_MAP_SIZE) prune_map();
        !           141:     new_map -> line = line;
        !           142:     new_map -> pos = pos;
        !           143:     new_map -> previous = current_map;
        !           144:     current_map = new_map;
        !           145:     current_map_size++;
        !           146: }
        !           147:
        !           148:
        !           149:
        !           150: /* Return position of column *c of ith line in   */
        !           151: /* current file. Adjust *c to be within the line.*/
        !           152: /* A 0 pointer is taken as 0 column.            */
        !           153: /* Returns CORD_NOT_FOUND if i is too big.      */
        !           154: /* Assumes i > dis_line.                        */
        !           155: size_t line_pos(int i, int *c)
        !           156: {
        !           157:     int j;
        !           158:     size_t cur;
        !           159:     size_t next;
        !           160:     line_map map = current_map;
        !           161:
        !           162:     while (map -> line > i) map = map -> previous;
        !           163:     if (map -> line < i - 2) /* rebuild */ invalidate_map(i);
        !           164:     for (j = map -> line, cur = map -> pos; j < i;) {
        !           165:        cur = CORD_chr(current, cur, '\n');
        !           166:         if (cur == current_len-1) return(CORD_NOT_FOUND);
        !           167:         cur++;
        !           168:         if (++j > current_map -> line) add_map(j, cur);
        !           169:     }
        !           170:     if (c != 0) {
        !           171:         next = CORD_chr(current, cur, '\n');
        !           172:         if (next == CORD_NOT_FOUND) next = current_len - 1;
        !           173:         if (next < cur + *c) {
        !           174:             *c = next - cur;
        !           175:         }
        !           176:         cur += *c;
        !           177:     }
        !           178:     return(cur);
        !           179: }
        !           180:
        !           181: void add_hist(CORD s)
        !           182: {
        !           183:     history new_file = GC_NEW(struct HistoryRep);
        !           184:
        !           185:     new_file -> file_contents = current = s;
        !           186:     current_len = CORD_len(s);
        !           187:     new_file -> previous = now;
        !           188:     if (now != 0) now -> map = current_map;
        !           189:     now = new_file;
        !           190: }
        !           191:
        !           192: void del_hist(void)
        !           193: {
        !           194:     now = now -> previous;
        !           195:     current = now -> file_contents;
        !           196:     current_map = now -> map;
        !           197:     current_len = CORD_len(current);
        !           198: }
        !           199:
        !           200: /* Current screen_contents; a dynamically allocated array of CORDs     */
        !           201: CORD * screen = 0;
        !           202: int screen_size = 0;
        !           203:
        !           204: # ifndef WIN32
        !           205: /* Replace a line in the curses stdscr.        All control characters are      */
        !           206: /* displayed as upper case characters in standout mode.  This isn't    */
        !           207: /* terribly appropriate for tabs.                                                                      */
        !           208: void replace_line(int i, CORD s)
        !           209: {
        !           210:     register int c;
        !           211:     CORD_pos p;
        !           212:     size_t len = CORD_len(s);
        !           213:
        !           214:     if (screen == 0 || LINES > screen_size) {
        !           215:         screen_size = LINES;
        !           216:        screen = (CORD *)GC_MALLOC(screen_size * sizeof(CORD));
        !           217:     }
        !           218: #   if !defined(MACINTOSH)
        !           219:         /* A gross workaround for an apparent curses bug: */
        !           220:         if (i == LINES-1 && len == COLS) {
        !           221:             s = CORD_substr(s, 0, CORD_len(s) - 1);
        !           222:         }
        !           223: #   endif
        !           224:     if (CORD_cmp(screen[i], s) != 0) {
        !           225:         move(i, 0); clrtoeol(); move(i,0);
        !           226:
        !           227:         CORD_FOR (p, s) {
        !           228:             c = CORD_pos_fetch(p) & 0x7f;
        !           229:             if (iscntrl(c)) {
        !           230:                standout(); addch(c + 0x40); standend();
        !           231:             } else {
        !           232:                addch(c);
        !           233:            }
        !           234:        }
        !           235:        screen[i] = s;
        !           236:     }
        !           237: }
        !           238: #else
        !           239: # define replace_line(i,s) invalidate_line(i)
        !           240: #endif
        !           241:
        !           242: /* Return up to COLS characters of the line of s starting at pos,      */
        !           243: /* returning only characters after the given column.                   */
        !           244: CORD retrieve_line(CORD s, size_t pos, unsigned column)
        !           245: {
        !           246:     CORD candidate = CORD_substr(s, pos, column + COLS);
        !           247:                        /* avoids scanning very long lines      */
        !           248:     int eol = CORD_chr(candidate, 0, '\n');
        !           249:     int len;
        !           250:
        !           251:     if (eol == CORD_NOT_FOUND) eol = CORD_len(candidate);
        !           252:     len = (int)eol - (int)column;
        !           253:     if (len < 0) len = 0;
        !           254:     return(CORD_substr(s, pos + column, len));
        !           255: }
        !           256:
        !           257: # ifdef WIN32
        !           258: #   define refresh();
        !           259:
        !           260:     CORD retrieve_screen_line(int i)
        !           261:     {
        !           262:        register size_t pos;
        !           263:
        !           264:        invalidate_map(dis_line + LINES);       /* Prune search */
        !           265:        pos = line_pos(dis_line + i, 0);
        !           266:        if (pos == CORD_NOT_FOUND) return(CORD_EMPTY);
        !           267:        return(retrieve_line(current, pos, dis_col));
        !           268:     }
        !           269: # endif
        !           270:
        !           271: /* Display the visible section of the current file      */
        !           272: void redisplay(void)
        !           273: {
        !           274:     register int i;
        !           275:
        !           276:     invalidate_map(dis_line + LINES);  /* Prune search */
        !           277:     for (i = 0; i < LINES; i++) {
        !           278:         if (need_redisplay == ALL || need_redisplay == i) {
        !           279:             register size_t pos = line_pos(dis_line + i, 0);
        !           280:
        !           281:             if (pos == CORD_NOT_FOUND) break;
        !           282:             replace_line(i, retrieve_line(current, pos, dis_col));
        !           283:             if (need_redisplay == i) goto done;
        !           284:         }
        !           285:     }
        !           286:     for (; i < LINES; i++) replace_line(i, CORD_EMPTY);
        !           287: done:
        !           288:     refresh();
        !           289:     need_redisplay = NONE;
        !           290: }
        !           291:
        !           292: int dis_granularity;
        !           293:
        !           294: /* Update dis_line, dis_col, and dis_pos to make cursor visible.       */
        !           295: /* Assumes line, col, dis_line, dis_pos are in bounds.                 */
        !           296: void normalize_display()
        !           297: {
        !           298:     int old_line = dis_line;
        !           299:     int old_col = dis_col;
        !           300:
        !           301:     dis_granularity = 1;
        !           302:     if (LINES > 15 && COLS > 15) dis_granularity = 2;
        !           303:     while (dis_line > line) dis_line -= dis_granularity;
        !           304:     while (dis_col > col) dis_col -= dis_granularity;
        !           305:     while (line >= dis_line + LINES) dis_line += dis_granularity;
        !           306:     while (col >= dis_col + COLS) dis_col += dis_granularity;
        !           307:     if (old_line != dis_line || old_col != dis_col) {
        !           308:         need_redisplay = ALL;
        !           309:     }
        !           310: }
        !           311:
        !           312: # if defined(WIN32)
        !           313: # elif defined(MACINTOSH)
        !           314: #              define move_cursor(x,y) cgotoxy(x + 1, y + 1, stdout)
        !           315: # else
        !           316: #              define move_cursor(x,y) move(y,x)
        !           317: # endif
        !           318:
        !           319: /* Adjust display so that cursor is visible; move cursor into position */
        !           320: /* Update screen if necessary.                                         */
        !           321: void fix_cursor(void)
        !           322: {
        !           323:     normalize_display();
        !           324:     if (need_redisplay != NONE) redisplay();
        !           325:     move_cursor(col - dis_col, line - dis_line);
        !           326:     refresh();
        !           327: #   ifndef WIN32
        !           328:       fflush(stdout);
        !           329: #   endif
        !           330: }
        !           331:
        !           332: /* Make sure line, col, and dis_pos are somewhere inside file. */
        !           333: /* Recompute file_pos. Assumes dis_pos is accurate or past eof */
        !           334: void fix_pos()
        !           335: {
        !           336:     int my_col = col;
        !           337:
        !           338:     if ((size_t)line > current_len) line = current_len;
        !           339:     file_pos = line_pos(line, &my_col);
        !           340:     if (file_pos == CORD_NOT_FOUND) {
        !           341:         for (line = current_map -> line, file_pos = current_map -> pos;
        !           342:              file_pos < current_len;
        !           343:              line++, file_pos = CORD_chr(current, file_pos, '\n') + 1);
        !           344:        line--;
        !           345:         file_pos = line_pos(line, &col);
        !           346:     } else {
        !           347:        col = my_col;
        !           348:     }
        !           349: }
        !           350:
        !           351: #if defined(WIN32)
        !           352: #  define beep() Beep(1000 /* Hz */, 300 /* msecs */)
        !           353: #elif defined(MACINTOSH)
        !           354: #      define beep() SysBeep(1)
        !           355: #else
        !           356: /*
        !           357:  * beep() is part of some curses packages and not others.
        !           358:  * We try to match the type of the builtin one, if any.
        !           359:  */
        !           360: #ifdef __STDC__
        !           361:     int beep(void)
        !           362: #else
        !           363:     int beep()
        !           364: #endif
        !           365: {
        !           366:     putc('\007', stderr);
        !           367:     return(0);
        !           368: }
        !           369: #endif
        !           370:
        !           371: #   define NO_PREFIX -1
        !           372: #   define BARE_PREFIX -2
        !           373: int repeat_count = NO_PREFIX;  /* Current command prefix. */
        !           374:
        !           375: int locate_mode = 0;                   /* Currently between 2 ^Ls      */
        !           376: CORD locate_string = CORD_EMPTY;       /* Current search string.       */
        !           377:
        !           378: char * arg_file_name;
        !           379:
        !           380: #ifdef WIN32
        !           381: /* Change the current position to whatever is currently displayed at   */
        !           382: /* the given SCREEN coordinates.                                       */
        !           383: void set_position(int c, int l)
        !           384: {
        !           385:     line = l + dis_line;
        !           386:     col = c + dis_col;
        !           387:     fix_pos();
        !           388:     move_cursor(col - dis_col, line - dis_line);
        !           389: }
        !           390: #endif /* WIN32 */
        !           391:
        !           392: /* Perform the command associated with character c.  C may be an       */
        !           393: /* integer > 256 denoting a windows command, one of the above control  */
        !           394: /* characters, or another ASCII character to be used as either a       */
        !           395: /* character to be inserted, a repeat count, or a search string,       */
        !           396: /* depending on the current state.                                     */
        !           397: void do_command(int c)
        !           398: {
        !           399:     int i;
        !           400:     int need_fix_pos;
        !           401:     FILE * out;
        !           402:
        !           403:     if ( c == '\r') c = '\n';
        !           404:     if (locate_mode) {
        !           405:         size_t new_pos;
        !           406:
        !           407:         if (c == LOCATE) {
        !           408:               locate_mode = 0;
        !           409:               locate_string = CORD_EMPTY;
        !           410:               return;
        !           411:         }
        !           412:         locate_string = CORD_cat_char(locate_string, (char)c);
        !           413:         new_pos = CORD_str(current, file_pos - CORD_len(locate_string) + 1,
        !           414:                           locate_string);
        !           415:         if (new_pos != CORD_NOT_FOUND) {
        !           416:             need_redisplay = ALL;
        !           417:             new_pos += CORD_len(locate_string);
        !           418:             for (;;) {
        !           419:                file_pos = line_pos(line + 1, 0);
        !           420:                if (file_pos > new_pos) break;
        !           421:                line++;
        !           422:             }
        !           423:             col = new_pos - line_pos(line, 0);
        !           424:             file_pos = new_pos;
        !           425:             fix_cursor();
        !           426:         } else {
        !           427:             locate_string = CORD_substr(locate_string, 0,
        !           428:                                        CORD_len(locate_string) - 1);
        !           429:             beep();
        !           430:         }
        !           431:         return;
        !           432:     }
        !           433:     if (c == REPEAT) {
        !           434:        repeat_count = BARE_PREFIX; return;
        !           435:     } else if (c < 0x100 && isdigit(c)){
        !           436:         if (repeat_count == BARE_PREFIX) {
        !           437:           repeat_count = c - '0'; return;
        !           438:         } else if (repeat_count != NO_PREFIX) {
        !           439:           repeat_count = 10 * repeat_count + c - '0'; return;
        !           440:         }
        !           441:     }
        !           442:     if (repeat_count == NO_PREFIX) repeat_count = 1;
        !           443:     if (repeat_count == BARE_PREFIX && (c == UP || c == DOWN)) {
        !           444:        repeat_count = LINES - dis_granularity;
        !           445:     }
        !           446:     if (repeat_count == BARE_PREFIX) repeat_count = 8;
        !           447:     need_fix_pos = 0;
        !           448:     for (i = 0; i < repeat_count; i++) {
        !           449:         switch(c) {
        !           450:           case LOCATE:
        !           451:             locate_mode = 1;
        !           452:             break;
        !           453:           case TOP:
        !           454:             line = col = file_pos = 0;
        !           455:             break;
        !           456:          case UP:
        !           457:            if (line != 0) {
        !           458:                line--;
        !           459:                need_fix_pos = 1;
        !           460:            }
        !           461:            break;
        !           462:          case DOWN:
        !           463:            line++;
        !           464:            need_fix_pos = 1;
        !           465:            break;
        !           466:          case LEFT:
        !           467:            if (col != 0) {
        !           468:                col--; file_pos--;
        !           469:            }
        !           470:            break;
        !           471:          case RIGHT:
        !           472:            if (CORD_fetch(current, file_pos) == '\n') break;
        !           473:            col++; file_pos++;
        !           474:            break;
        !           475:          case UNDO:
        !           476:            del_hist();
        !           477:            need_redisplay = ALL; need_fix_pos = 1;
        !           478:            break;
        !           479:          case BS:
        !           480:            if (col == 0) {
        !           481:                beep();
        !           482:                break;
        !           483:            }
        !           484:            col--; file_pos--;
        !           485:            /* fall through: */
        !           486:          case DEL:
        !           487:            if (file_pos == current_len-1) break;
        !           488:                /* Can't delete trailing newline */
        !           489:            if (CORD_fetch(current, file_pos) == '\n') {
        !           490:                need_redisplay = ALL; need_fix_pos = 1;
        !           491:            } else {
        !           492:                need_redisplay = line - dis_line;
        !           493:            }
        !           494:            add_hist(CORD_cat(
        !           495:                        CORD_substr(current, 0, file_pos),
        !           496:                        CORD_substr(current, file_pos+1, current_len)));
        !           497:            invalidate_map(line);
        !           498:            break;
        !           499:          case WRITE:
        !           500:            {
        !           501:                CORD name = CORD_cat(CORD_from_char_star(arg_file_name),
        !           502:                                     ".new");
        !           503:
        !           504:                if ((out = fopen(CORD_to_const_char_star(name), "wb")) == NULL
        !           505:                    || CORD_put(current, out) == EOF) {
        !           506:                    de_error("Write failed\n");
        !           507:                    need_redisplay = ALL;
        !           508:                 } else {
        !           509:                     fclose(out);
        !           510:                 }
        !           511:            }
        !           512:             break;
        !           513:          default:
        !           514:            {
        !           515:                CORD left_part = CORD_substr(current, 0, file_pos);
        !           516:                CORD right_part = CORD_substr(current, file_pos, current_len);
        !           517:
        !           518:                add_hist(CORD_cat(CORD_cat_char(left_part, (char)c),
        !           519:                                  right_part));
        !           520:                invalidate_map(line);
        !           521:                if (c == '\n') {
        !           522:                    col = 0; line++; file_pos++;
        !           523:                    need_redisplay = ALL;
        !           524:                } else {
        !           525:                    col++; file_pos++;
        !           526:                    need_redisplay = line - dis_line;
        !           527:                }
        !           528:                break;
        !           529:            }
        !           530:         }
        !           531:     }
        !           532:     if (need_fix_pos) fix_pos();
        !           533:     fix_cursor();
        !           534:     repeat_count = NO_PREFIX;
        !           535: }
        !           536:
        !           537: /* OS independent initialization */
        !           538:
        !           539: void generic_init(void)
        !           540: {
        !           541:     FILE * f;
        !           542:     CORD initial;
        !           543:
        !           544:     if ((f = fopen(arg_file_name, "rb")) == NULL) {
        !           545:        initial = "\n";
        !           546:     } else {
        !           547:         initial = CORD_from_file(f);
        !           548:         if (initial == CORD_EMPTY
        !           549:             || CORD_fetch(initial, CORD_len(initial)-1) != '\n') {
        !           550:             initial = CORD_cat(initial, "\n");
        !           551:         }
        !           552:     }
        !           553:     add_map(0,0);
        !           554:     add_hist(initial);
        !           555:     now -> map = current_map;
        !           556:     now -> previous = now;  /* Can't back up further: beginning of the world */
        !           557:     need_redisplay = ALL;
        !           558:     fix_cursor();
        !           559: }
        !           560:
        !           561: #ifndef WIN32
        !           562:
        !           563: main(argc, argv)
        !           564: int argc;
        !           565: char ** argv;
        !           566: {
        !           567:     int c;
        !           568:
        !           569: #if defined(MACINTOSH)
        !           570:        console_options.title = "\pDumb Editor";
        !           571:        cshow(stdout);
        !           572:        GC_init();
        !           573:        argc = ccommand(&argv);
        !           574: #endif
        !           575:
        !           576:     if (argc != 2) goto usage;
        !           577:     arg_file_name = argv[1];
        !           578:     setvbuf(stdout, GC_MALLOC_ATOMIC(8192), _IOFBF, 8192);
        !           579:     initscr();
        !           580:     noecho(); nonl(); cbreak();
        !           581:     generic_init();
        !           582:     while ((c = getchar()) != QUIT) {
        !           583:                if (c == EOF) break;
        !           584:            do_command(c);
        !           585:     }
        !           586: done:
        !           587:     move(LINES-1, 0);
        !           588:     clrtoeol();
        !           589:     refresh();
        !           590:     nl();
        !           591:     echo();
        !           592:     endwin();
        !           593:     exit(0);
        !           594: usage:
        !           595:     fprintf(stderr, "Usage: %s file\n", argv[0]);
        !           596:     fprintf(stderr, "Cursor keys: ^B(left) ^F(right) ^P(up) ^N(down)\n");
        !           597:     fprintf(stderr, "Undo: ^U    Write to <file>.new: ^W");
        !           598:     fprintf(stderr, "Quit:^D  Repeat count: ^R[n]\n");
        !           599:     fprintf(stderr, "Top: ^T   Locate (search, find): ^L text ^L\n");
        !           600:     exit(1);
        !           601: }
        !           602:
        !           603: #endif  /* !WIN32 */

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