551 lines
9.7 KiB
C
551 lines
9.7 KiB
C
/*
|
|
* Misc. bootloader code (almost) all platforms can use
|
|
*
|
|
* Author: Johnnie Peters <jpeters@mvista.com>
|
|
* Editor: Tom Rini <trini@mvista.com>
|
|
*
|
|
* Derived from arch/ppc/boot/prep/misc.c
|
|
*
|
|
* 2000-2001 (c) MontaVista, Software, Inc. This file is licensed under
|
|
* the terms of the GNU General Public License version 2. This program
|
|
* is licensed "as is" without any warranty of any kind, whether express
|
|
* or implied.
|
|
*/
|
|
|
|
#include <stdarg.h> /* for va_ bits */
|
|
#include <linux/string.h>
|
|
#include <linux/zlib.h>
|
|
#include "nonstdio.h"
|
|
|
|
/* If we're on a PReP, assume we have a keyboard controller
|
|
* Also note, if we're not PReP, we assume you are a serial
|
|
* console - Tom */
|
|
#if defined(CONFIG_PPC_PREP) && defined(CONFIG_VGA_CONSOLE)
|
|
extern void cursor(int x, int y);
|
|
extern void scroll(void);
|
|
extern char *vidmem;
|
|
extern int lines, cols;
|
|
extern int orig_x, orig_y;
|
|
extern int keyb_present;
|
|
extern int CRT_tstc(void);
|
|
extern int CRT_getc(void);
|
|
#else
|
|
int cursor(int x, int y) {return 0;}
|
|
void scroll(void) {}
|
|
char vidmem[1];
|
|
#define lines 0
|
|
#define cols 0
|
|
int orig_x = 0;
|
|
int orig_y = 0;
|
|
#define keyb_present 0
|
|
int CRT_tstc(void) {return 0;}
|
|
int CRT_getc(void) {return 0;}
|
|
#endif
|
|
|
|
extern char *avail_ram;
|
|
extern char *end_avail;
|
|
extern char _end[];
|
|
|
|
void puts(const char *);
|
|
void putc(const char c);
|
|
void puthex(unsigned long val);
|
|
void gunzip(void *, int, unsigned char *, int *);
|
|
static int _cvt(unsigned long val, char *buf, long radix, char *digits);
|
|
|
|
void _vprintk(void(*putc)(const char), const char *fmt0, va_list ap);
|
|
unsigned char *ISA_io = NULL;
|
|
|
|
#if defined(CONFIG_SERIAL_CPM_CONSOLE) || defined(CONFIG_SERIAL_8250_CONSOLE) \
|
|
|| defined(CONFIG_SERIAL_MPC52xx_CONSOLE) \
|
|
|| defined(CONFIG_SERIAL_MPSC_CONSOLE)
|
|
extern unsigned long com_port;
|
|
|
|
extern int serial_tstc(unsigned long com_port);
|
|
extern unsigned char serial_getc(unsigned long com_port);
|
|
extern void serial_putc(unsigned long com_port, unsigned char c);
|
|
#endif
|
|
|
|
void pause(void)
|
|
{
|
|
puts("pause\n");
|
|
}
|
|
|
|
void exit(void)
|
|
{
|
|
puts("exit\n");
|
|
while(1);
|
|
}
|
|
|
|
int tstc(void)
|
|
{
|
|
#if defined(CONFIG_SERIAL_CPM_CONSOLE) || defined(CONFIG_SERIAL_8250_CONSOLE) \
|
|
|| defined(CONFIG_SERIAL_MPC52xx_CONSOLE) \
|
|
|| defined(CONFIG_SERIAL_MPSC_CONSOLE)
|
|
if(keyb_present)
|
|
return (CRT_tstc() || serial_tstc(com_port));
|
|
else
|
|
return (serial_tstc(com_port));
|
|
#else
|
|
return CRT_tstc();
|
|
#endif
|
|
}
|
|
|
|
int getc(void)
|
|
{
|
|
while (1) {
|
|
#if defined(CONFIG_SERIAL_CPM_CONSOLE) || defined(CONFIG_SERIAL_8250_CONSOLE) \
|
|
|| defined(CONFIG_SERIAL_MPC52xx_CONSOLE) \
|
|
|| defined(CONFIG_SERIAL_MPSC_CONSOLE)
|
|
if (serial_tstc(com_port))
|
|
return (serial_getc(com_port));
|
|
#endif /* serial console */
|
|
if (keyb_present)
|
|
if(CRT_tstc())
|
|
return (CRT_getc());
|
|
}
|
|
}
|
|
|
|
void
|
|
putc(const char c)
|
|
{
|
|
int x,y;
|
|
|
|
#if defined(CONFIG_SERIAL_CPM_CONSOLE) || defined(CONFIG_SERIAL_8250_CONSOLE) \
|
|
|| defined(CONFIG_SERIAL_MPC52xx_CONSOLE) \
|
|
|| defined(CONFIG_SERIAL_MPSC_CONSOLE)
|
|
serial_putc(com_port, c);
|
|
if ( c == '\n' )
|
|
serial_putc(com_port, '\r');
|
|
#endif /* serial console */
|
|
|
|
x = orig_x;
|
|
y = orig_y;
|
|
|
|
if ( c == '\n' ) {
|
|
x = 0;
|
|
if ( ++y >= lines ) {
|
|
scroll();
|
|
y--;
|
|
}
|
|
} else if (c == '\r') {
|
|
x = 0;
|
|
} else if (c == '\b') {
|
|
if (x > 0) {
|
|
x--;
|
|
}
|
|
} else {
|
|
vidmem [ ( x + cols * y ) * 2 ] = c;
|
|
if ( ++x >= cols ) {
|
|
x = 0;
|
|
if ( ++y >= lines ) {
|
|
scroll();
|
|
y--;
|
|
}
|
|
}
|
|
}
|
|
|
|
cursor(x, y);
|
|
|
|
orig_x = x;
|
|
orig_y = y;
|
|
}
|
|
|
|
void puts(const char *s)
|
|
{
|
|
int x,y;
|
|
char c;
|
|
|
|
x = orig_x;
|
|
y = orig_y;
|
|
|
|
while ( ( c = *s++ ) != '\0' ) {
|
|
#if defined(CONFIG_SERIAL_CPM_CONSOLE) || defined(CONFIG_SERIAL_8250_CONSOLE) \
|
|
|| defined(CONFIG_SERIAL_MPC52xx_CONSOLE) \
|
|
|| defined(CONFIG_SERIAL_MPSC_CONSOLE)
|
|
serial_putc(com_port, c);
|
|
if ( c == '\n' ) serial_putc(com_port, '\r');
|
|
#endif /* serial console */
|
|
|
|
if ( c == '\n' ) {
|
|
x = 0;
|
|
if ( ++y >= lines ) {
|
|
scroll();
|
|
y--;
|
|
}
|
|
} else if (c == '\b') {
|
|
if (x > 0) {
|
|
x--;
|
|
}
|
|
} else {
|
|
vidmem [ ( x + cols * y ) * 2 ] = c;
|
|
if ( ++x >= cols ) {
|
|
x = 0;
|
|
if ( ++y >= lines ) {
|
|
scroll();
|
|
y--;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
cursor(x, y);
|
|
|
|
orig_x = x;
|
|
orig_y = y;
|
|
}
|
|
|
|
void error(char *x)
|
|
{
|
|
puts("\n\n");
|
|
puts(x);
|
|
puts("\n\n -- System halted");
|
|
|
|
while(1); /* Halt */
|
|
}
|
|
|
|
static void *zalloc(unsigned size)
|
|
{
|
|
void *p = avail_ram;
|
|
|
|
size = (size + 7) & -8;
|
|
avail_ram += size;
|
|
if (avail_ram > end_avail) {
|
|
puts("oops... out of memory\n");
|
|
pause();
|
|
}
|
|
return p;
|
|
}
|
|
|
|
#define HEAD_CRC 2
|
|
#define EXTRA_FIELD 4
|
|
#define ORIG_NAME 8
|
|
#define COMMENT 0x10
|
|
#define RESERVED 0xe0
|
|
|
|
void gunzip(void *dst, int dstlen, unsigned char *src, int *lenp)
|
|
{
|
|
z_stream s;
|
|
int r, i, flags;
|
|
|
|
/* skip header */
|
|
i = 10;
|
|
flags = src[3];
|
|
if (src[2] != Z_DEFLATED || (flags & RESERVED) != 0) {
|
|
puts("bad gzipped data\n");
|
|
exit();
|
|
}
|
|
if ((flags & EXTRA_FIELD) != 0)
|
|
i = 12 + src[10] + (src[11] << 8);
|
|
if ((flags & ORIG_NAME) != 0)
|
|
while (src[i++] != 0)
|
|
;
|
|
if ((flags & COMMENT) != 0)
|
|
while (src[i++] != 0)
|
|
;
|
|
if ((flags & HEAD_CRC) != 0)
|
|
i += 2;
|
|
if (i >= *lenp) {
|
|
puts("gunzip: ran out of data in header\n");
|
|
exit();
|
|
}
|
|
|
|
/* Initialize ourself. */
|
|
s.workspace = zalloc(zlib_inflate_workspacesize());
|
|
r = zlib_inflateInit2(&s, -MAX_WBITS);
|
|
if (r != Z_OK) {
|
|
puts("zlib_inflateInit2 returned "); puthex(r); puts("\n");
|
|
exit();
|
|
}
|
|
s.next_in = src + i;
|
|
s.avail_in = *lenp - i;
|
|
s.next_out = dst;
|
|
s.avail_out = dstlen;
|
|
r = zlib_inflate(&s, Z_FINISH);
|
|
if (r != Z_OK && r != Z_STREAM_END) {
|
|
puts("inflate returned "); puthex(r); puts("\n");
|
|
exit();
|
|
}
|
|
*lenp = s.next_out - (unsigned char *) dst;
|
|
zlib_inflateEnd(&s);
|
|
}
|
|
|
|
void
|
|
puthex(unsigned long val)
|
|
{
|
|
|
|
unsigned char buf[10];
|
|
int i;
|
|
for (i = 7; i >= 0; i--)
|
|
{
|
|
buf[i] = "0123456789ABCDEF"[val & 0x0F];
|
|
val >>= 4;
|
|
}
|
|
buf[8] = '\0';
|
|
puts(buf);
|
|
}
|
|
|
|
#define FALSE 0
|
|
#define TRUE 1
|
|
|
|
void
|
|
_printk(char const *fmt, ...)
|
|
{
|
|
va_list ap;
|
|
|
|
va_start(ap, fmt);
|
|
_vprintk(putc, fmt, ap);
|
|
va_end(ap);
|
|
return;
|
|
}
|
|
|
|
#define is_digit(c) ((c >= '0') && (c <= '9'))
|
|
|
|
void
|
|
_vprintk(void(*putc)(const char), const char *fmt0, va_list ap)
|
|
{
|
|
char c, sign, *cp = 0;
|
|
int left_prec, right_prec, zero_fill, length = 0, pad, pad_on_right;
|
|
char buf[32];
|
|
long val;
|
|
while ((c = *fmt0++))
|
|
{
|
|
if (c == '%')
|
|
{
|
|
c = *fmt0++;
|
|
left_prec = right_prec = pad_on_right = 0;
|
|
if (c == '-')
|
|
{
|
|
c = *fmt0++;
|
|
pad_on_right++;
|
|
}
|
|
if (c == '0')
|
|
{
|
|
zero_fill = TRUE;
|
|
c = *fmt0++;
|
|
} else
|
|
{
|
|
zero_fill = FALSE;
|
|
}
|
|
while (is_digit(c))
|
|
{
|
|
left_prec = (left_prec * 10) + (c - '0');
|
|
c = *fmt0++;
|
|
}
|
|
if (c == '.')
|
|
{
|
|
c = *fmt0++;
|
|
zero_fill++;
|
|
while (is_digit(c))
|
|
{
|
|
right_prec = (right_prec * 10) + (c - '0');
|
|
c = *fmt0++;
|
|
}
|
|
} else
|
|
{
|
|
right_prec = left_prec;
|
|
}
|
|
sign = '\0';
|
|
switch (c)
|
|
{
|
|
case 'd':
|
|
case 'x':
|
|
case 'X':
|
|
val = va_arg(ap, long);
|
|
switch (c)
|
|
{
|
|
case 'd':
|
|
if (val < 0)
|
|
{
|
|
sign = '-';
|
|
val = -val;
|
|
}
|
|
length = _cvt(val, buf, 10, "0123456789");
|
|
break;
|
|
case 'x':
|
|
length = _cvt(val, buf, 16, "0123456789abcdef");
|
|
break;
|
|
case 'X':
|
|
length = _cvt(val, buf, 16, "0123456789ABCDEF");
|
|
break;
|
|
}
|
|
cp = buf;
|
|
break;
|
|
case 's':
|
|
cp = va_arg(ap, char *);
|
|
length = strlen(cp);
|
|
break;
|
|
case 'c':
|
|
c = va_arg(ap, long /*char*/);
|
|
(*putc)(c);
|
|
continue;
|
|
default:
|
|
(*putc)('?');
|
|
}
|
|
pad = left_prec - length;
|
|
if (sign != '\0')
|
|
{
|
|
pad--;
|
|
}
|
|
if (zero_fill)
|
|
{
|
|
c = '0';
|
|
if (sign != '\0')
|
|
{
|
|
(*putc)(sign);
|
|
sign = '\0';
|
|
}
|
|
} else
|
|
{
|
|
c = ' ';
|
|
}
|
|
if (!pad_on_right)
|
|
{
|
|
while (pad-- > 0)
|
|
{
|
|
(*putc)(c);
|
|
}
|
|
}
|
|
if (sign != '\0')
|
|
{
|
|
(*putc)(sign);
|
|
}
|
|
while (length-- > 0)
|
|
{
|
|
(*putc)(c = *cp++);
|
|
if (c == '\n')
|
|
{
|
|
(*putc)('\r');
|
|
}
|
|
}
|
|
if (pad_on_right)
|
|
{
|
|
while (pad-- > 0)
|
|
{
|
|
(*putc)(c);
|
|
}
|
|
}
|
|
} else
|
|
{
|
|
(*putc)(c);
|
|
if (c == '\n')
|
|
{
|
|
(*putc)('\r');
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
int
|
|
_cvt(unsigned long val, char *buf, long radix, char *digits)
|
|
{
|
|
char temp[80];
|
|
char *cp = temp;
|
|
int length = 0;
|
|
if (val == 0)
|
|
{ /* Special case */
|
|
*cp++ = '0';
|
|
} else
|
|
while (val)
|
|
{
|
|
*cp++ = digits[val % radix];
|
|
val /= radix;
|
|
}
|
|
while (cp != temp)
|
|
{
|
|
*buf++ = *--cp;
|
|
length++;
|
|
}
|
|
*buf = '\0';
|
|
return (length);
|
|
}
|
|
|
|
void
|
|
_dump_buf_with_offset(unsigned char *p, int s, unsigned char *base)
|
|
{
|
|
int i, c;
|
|
if ((unsigned int)s > (unsigned int)p)
|
|
{
|
|
s = (unsigned int)s - (unsigned int)p;
|
|
}
|
|
while (s > 0)
|
|
{
|
|
if (base)
|
|
{
|
|
_printk("%06X: ", (int)p - (int)base);
|
|
} else
|
|
{
|
|
_printk("%06X: ", p);
|
|
}
|
|
for (i = 0; i < 16; i++)
|
|
{
|
|
if (i < s)
|
|
{
|
|
_printk("%02X", p[i] & 0xFF);
|
|
} else
|
|
{
|
|
_printk(" ");
|
|
}
|
|
if ((i % 2) == 1) _printk(" ");
|
|
if ((i % 8) == 7) _printk(" ");
|
|
}
|
|
_printk(" |");
|
|
for (i = 0; i < 16; i++)
|
|
{
|
|
if (i < s)
|
|
{
|
|
c = p[i] & 0xFF;
|
|
if ((c < 0x20) || (c >= 0x7F)) c = '.';
|
|
} else
|
|
{
|
|
c = ' ';
|
|
}
|
|
_printk("%c", c);
|
|
}
|
|
_printk("|\n");
|
|
s -= 16;
|
|
p += 16;
|
|
}
|
|
}
|
|
|
|
void
|
|
_dump_buf(unsigned char *p, int s)
|
|
{
|
|
_printk("\n");
|
|
_dump_buf_with_offset(p, s, 0);
|
|
}
|
|
|
|
/* Very simple inb/outb routines. We declare ISA_io to be 0 above, and
|
|
* then modify it on platforms which need to. We do it like this
|
|
* because on some platforms we give inb/outb an exact location, and
|
|
* on others it's an offset from a given location. -- Tom
|
|
*/
|
|
|
|
void ISA_init(unsigned long base)
|
|
{
|
|
ISA_io = (unsigned char *)base;
|
|
}
|
|
|
|
void
|
|
outb(int port, unsigned char val)
|
|
{
|
|
/* Ensure I/O operations complete */
|
|
__asm__ volatile("eieio");
|
|
ISA_io[port] = val;
|
|
}
|
|
|
|
unsigned char
|
|
inb(int port)
|
|
{
|
|
/* Ensure I/O operations complete */
|
|
__asm__ volatile("eieio");
|
|
return (ISA_io[port]);
|
|
}
|
|
|
|
/*
|
|
* Local variables:
|
|
* c-indent-level: 8
|
|
* c-basic-offset: 8
|
|
* tab-width: 8
|
|
* End:
|
|
*/
|