/* * Copyright (C) 1996-2005 Paul Mackerras. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version * 2 of the License, or (at your option) any later version. */ #include <linux/string.h> #include <asm/udbg.h> #include <asm/time.h> #include "nonstdio.h" static bool paginating, paginate_skipping; static unsigned long paginate_lpp; /* Lines Per Page */ static unsigned long paginate_pos; void xmon_start_pagination(void) { paginating = true; paginate_skipping = false; paginate_pos = 0; } void xmon_end_pagination(void) { paginating = false; } void xmon_set_pagination_lpp(unsigned long lpp) { paginate_lpp = lpp; } static int xmon_readchar(void) { if (udbg_getc) return udbg_getc(); return -1; } static int xmon_write(const char *ptr, int nb) { int rv = 0; const char *p = ptr, *q; const char msg[] = "[Hit a key (a:all, q:truncate, any:next page)]"; if (nb <= 0) return rv; if (paginating && paginate_skipping) return nb; if (paginate_lpp) { while (paginating && (q = strchr(p, '\n'))) { rv += udbg_write(p, q - p + 1); p = q + 1; paginate_pos++; if (paginate_pos >= paginate_lpp) { udbg_write(msg, strlen(msg)); switch (xmon_readchar()) { case 'a': paginating = false; break; case 'q': paginate_skipping = true; break; default: /* nothing */ break; } paginate_pos = 0; udbg_write("\r\n", 2); if (paginate_skipping) return nb; } } } return rv + udbg_write(p, nb - (p - ptr)); } int xmon_putchar(int c) { char ch = c; if (c == '\n') xmon_putchar('\r'); return xmon_write(&ch, 1) == 1? c: -1; } static char line[256]; static char *lineptr; static int lineleft; static int xmon_getchar(void) { int c; if (lineleft == 0) { lineptr = line; for (;;) { c = xmon_readchar(); if (c == -1 || c == 4) break; if (c == '\r' || c == '\n') { *lineptr++ = '\n'; xmon_putchar('\n'); break; } switch (c) { case 0177: case '\b': if (lineptr > line) { xmon_putchar('\b'); xmon_putchar(' '); xmon_putchar('\b'); --lineptr; } break; case 'U' & 0x1F: while (lineptr > line) { xmon_putchar('\b'); xmon_putchar(' '); xmon_putchar('\b'); --lineptr; } break; default: if (lineptr >= &line[sizeof(line) - 1]) xmon_putchar('\a'); else { xmon_putchar(c); *lineptr++ = c; } } } lineleft = lineptr - line; lineptr = line; } if (lineleft == 0) return -1; --lineleft; return *lineptr++; } char *xmon_gets(char *str, int nb) { char *p; int c; for (p = str; p < str + nb - 1; ) { c = xmon_getchar(); if (c == -1) { if (p == str) return NULL; break; } *p++ = c; if (c == '\n') break; } *p = 0; return str; } void xmon_printf(const char *format, ...) { va_list args; static char xmon_outbuf[1024]; int rc, n; va_start(args, format); n = vsnprintf(xmon_outbuf, sizeof(xmon_outbuf), format, args); va_end(args); rc = xmon_write(xmon_outbuf, n); if (n && rc == 0) { /* No udbg hooks, fallback to printk() - dangerous */ printk("%s", xmon_outbuf); } } void xmon_puts(const char *str) { xmon_write(str, strlen(str)); }