Ignore:
File:
1 edited

Legend:

Unmodified
Added
Removed
  • uspace/app/bdsh/cmds/modules/cat/cat.c

    rafe1d1e r28a3e74  
    11/* Copyright (c) 2008, Tim Post <tinkertim@gmail.com>
     2 * Copyright (c) 2011, Martin Sucha
    23 * All rights reserved.
    34 *
     
    3536#include <str.h>
    3637#include <fcntl.h>
     38#include <io/console.h>
     39#include <io/color.h>
     40#include <io/style.h>
     41#include <io/keycode.h>
     42#include <errno.h>
     43#include <vfs/vfs.h>
     44#include <assert.h>
    3745
    3846#include "config.h"
     
    4856
    4957static const char *cat_oops = "That option is not yet supported\n";
     58static const char *hexchars = "0123456789abcdef";
     59
     60static bool paging_enabled = false;
     61static size_t chars_remaining = 0;
     62static size_t lines_remaining = 0;
     63static sysarg_t console_cols = 0;
     64static sysarg_t console_rows = 0;
     65static bool should_quit = false;
    5066
    5167static struct option const long_options[] = {
     
    5672        { "buffer", required_argument, 0, 'b' },
    5773        { "more", no_argument, 0, 'm' },
     74        { "hex", no_argument, 0, 'x' },
    5875        { 0, 0, 0, 0 }
    5976};
     
    7592                "  -b, --buffer ##  Set the read buffer size to ##\n"
    7693                "  -m, --more       Pause after each screen full\n"
     94                "  -x, --hex        Print bytes as hex values\n"
    7795                "Currently, %s is under development, some options don't work.\n",
    7896                cmdname, cmdname);
     
    82100}
    83101
    84 static unsigned int cat_file(const char *fname, size_t blen)
     102static void waitprompt()
     103{
     104        console_set_pos(fphone(stdout), 0, console_rows-1);
     105        console_set_color(fphone(stdout), COLOR_BLUE, COLOR_WHITE, 0);
     106        printf("ENTER/SPACE/PAGE DOWN - next page, "
     107               "ESC/Q - quit, C - continue unpaged");
     108        fflush(stdout);
     109        console_set_style(fphone(stdout), STYLE_NORMAL);
     110}
     111
     112static void waitkey()
     113{
     114        console_event_t ev;
     115       
     116        while (true) {
     117                if (!console_get_event(fphone(stdin), &ev)) {
     118                        return;
     119                }
     120                if (ev.type == KEY_PRESS) {
     121                        if (ev.key == KC_ESCAPE || ev.key == KC_Q) {
     122                                should_quit = true;
     123                                return;
     124                        }
     125                        if (ev.key == KC_C) {
     126                                paging_enabled = false;
     127                                return;
     128                        }
     129                        if (ev.key == KC_ENTER || ev.key == KC_SPACE ||
     130                            ev.key == KC_PAGE_DOWN) {
     131                                return;
     132                        }
     133                }
     134        }
     135        assert(false);
     136}
     137
     138static void newpage()
     139{
     140        console_clear(fphone(stdout));
     141        chars_remaining = console_cols;
     142        lines_remaining = console_rows-1;
     143}
     144
     145static void paged_char(wchar_t c)
     146{
     147        putchar(c);
     148        if (paging_enabled) {
     149                chars_remaining--;
     150                if (c == '\n' || chars_remaining == 0) {
     151                        chars_remaining = console_cols;
     152                        lines_remaining--;
     153                }
     154                if (lines_remaining == 0) {
     155                        fflush(stdout);
     156                        waitprompt();
     157                        waitkey();
     158                        newpage();
     159                }
     160        }
     161}
     162
     163static unsigned int cat_file(const char *fname, size_t blen, bool hex)
    85164{
    86165        int fd, bytes = 0, count = 0, reads = 0;
    87         off64_t total = 0;
    88166        char *buff = NULL;
     167        int i;
     168        size_t offset = 0;
    89169
    90170        fd = open(fname, O_RDONLY);
     
    93173                return 1;
    94174        }
    95 
    96         total = lseek(fd, 0, SEEK_END);
    97         lseek(fd, 0, SEEK_SET);
    98175
    99176        if (NULL == (buff = (char *) malloc(blen + 1))) {
     
    109186                        count += bytes;
    110187                        buff[bytes] = '\0';
    111                         printf("%s", buff);
     188                        offset = 0;
     189                        for (i = 0; i < bytes && !should_quit; i++) {
     190                                if (hex) {
     191                                        paged_char(hexchars[((uint8_t)buff[i])/16]);
     192                                        paged_char(hexchars[((uint8_t)buff[i])%16]);
     193                                }
     194                                else {
     195                                        wchar_t c = str_decode(buff, &offset, bytes);
     196                                        if (c == 0) {
     197                                                /* Reached end of string */
     198                                                break;
     199                                        }
     200                                        paged_char(c);
     201                                }
     202                               
     203                        }
    112204                        reads++;
    113205                }
    114         } while (bytes > 0);
     206        } while (bytes > 0 && !should_quit);
    115207
    116208        close(fd);
     
    131223        unsigned int argc, i, ret = 0, buffer = 0;
    132224        int c, opt_ind;
     225        bool hex = false;
     226        bool more = false;
     227        sysarg_t rows, cols;
     228        int rc;
     229       
     230        /*
     231         * reset global state
     232         * TODO: move to structure?
     233         */
     234        paging_enabled = false;
     235        chars_remaining = 0;
     236        lines_remaining = 0;
     237        console_cols = 0;
     238        console_rows = 0;
     239        should_quit = false;
    133240
    134241        argc = cli_count_args(argv);
    135242
    136243        for (c = 0, optind = 0, opt_ind = 0; c != -1;) {
    137                 c = getopt_long(argc, argv, "hvmH:t:b:", long_options, &opt_ind);
     244                c = getopt_long(argc, argv, "xhvmH:t:b:", long_options, &opt_ind);
    138245                switch (c) {
    139246                case 'h':
     
    153260                        break;
    154261                case 'm':
    155                         printf("%s", cat_oops);
    156                         return CMD_FAILURE;
     262                        more = true;
     263                        break;
     264                case 'x':
     265                        hex = true;
     266                        break;
    157267                }
    158268        }
     
    168278        if (buffer <= 0)
    169279                buffer = CAT_DEFAULT_BUFLEN;
    170 
    171         for (i = optind; argv[i] != NULL; i++)
    172                 ret += cat_file(argv[i], buffer);
     280       
     281        if (more) {
     282                rc = console_get_size(fphone(stdout), &cols, &rows);
     283                if (rc != EOK) {
     284                        printf("%s - cannot get console size\n", cmdname);
     285                        return CMD_FAILURE;
     286                }
     287                console_cols = cols;
     288                console_rows = rows;
     289                paging_enabled = true;
     290                newpage();
     291        }
     292
     293        for (i = optind; argv[i] != NULL && !should_quit; i++)
     294                ret += cat_file(argv[i], buffer, hex);
    173295
    174296        if (ret)
Note: See TracChangeset for help on using the changeset viewer.