25edd6946a
This is based upon a report by Meelis Roos showing that it's possible that we'll try to fetch a property that is 32K in size with some devices. With the current fixed 3K buffer we use for moving data in and out of the firmware during PROM calls, that simply won't work. In fact, it will scramble random kernel data during bootup. The reasoning behind the temporary buffer is entirely historical. It used to be the case that we had problems referencing dynamic kernel memory (including the stack) early in the boot process before we explicitly told the firwmare to switch us over to the kernel trap table. So what we did was always give the firmware buffers that were locked into the main kernel image. But we no longer have problems like that, so get rid of all of this indirect bounce buffering. Besides fixing Meelis's bug, this also makes the kernel data about 3K smaller. It was also discovered during these conversions that the implementation of prom_retain() was completely wrong, so that was fixed here as well. Currently that interface is not in use. Reported-by: Meelis Roos <mroos@linux.ee> Tested-by: Meelis Roos <mroos@linux.ee> Signed-off-by: David S. Miller <davem@davemloft.net>
100 lines
1.9 KiB
C
100 lines
1.9 KiB
C
/* console.c: Routines that deal with sending and receiving IO
|
|
* to/from the current console device using the PROM.
|
|
*
|
|
* Copyright (C) 1995 David S. Miller (davem@davemloft.net)
|
|
* Copyright (C) 1996,1997 Jakub Jelinek (jj@sunsite.mff.cuni.cz)
|
|
*/
|
|
|
|
#include <linux/types.h>
|
|
#include <linux/kernel.h>
|
|
#include <linux/sched.h>
|
|
#include <asm/openprom.h>
|
|
#include <asm/oplib.h>
|
|
#include <asm/system.h>
|
|
#include <linux/string.h>
|
|
|
|
extern int prom_stdin, prom_stdout;
|
|
|
|
/* Non blocking get character from console input device, returns -1
|
|
* if no input was taken. This can be used for polling.
|
|
*/
|
|
inline int
|
|
prom_nbgetchar(void)
|
|
{
|
|
unsigned long args[7];
|
|
char inc;
|
|
|
|
args[0] = (unsigned long) "read";
|
|
args[1] = 3;
|
|
args[2] = 1;
|
|
args[3] = (unsigned int) prom_stdin;
|
|
args[4] = (unsigned long) &inc;
|
|
args[5] = 1;
|
|
args[6] = (unsigned long) -1;
|
|
|
|
p1275_cmd_direct(args);
|
|
|
|
if (args[6] == 1)
|
|
return inc;
|
|
return -1;
|
|
}
|
|
|
|
/* Non blocking put character to console device, returns -1 if
|
|
* unsuccessful.
|
|
*/
|
|
inline int
|
|
prom_nbputchar(char c)
|
|
{
|
|
unsigned long args[7];
|
|
char outc;
|
|
|
|
outc = c;
|
|
|
|
args[0] = (unsigned long) "write";
|
|
args[1] = 3;
|
|
args[2] = 1;
|
|
args[3] = (unsigned int) prom_stdout;
|
|
args[4] = (unsigned long) &outc;
|
|
args[5] = 1;
|
|
args[6] = (unsigned long) -1;
|
|
|
|
p1275_cmd_direct(args);
|
|
|
|
if (args[6] == 1)
|
|
return 0;
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
/* Blocking version of get character routine above. */
|
|
char
|
|
prom_getchar(void)
|
|
{
|
|
int character;
|
|
while((character = prom_nbgetchar()) == -1) ;
|
|
return (char) character;
|
|
}
|
|
|
|
/* Blocking version of put character routine above. */
|
|
void
|
|
prom_putchar(char c)
|
|
{
|
|
prom_nbputchar(c);
|
|
}
|
|
|
|
void
|
|
prom_puts(const char *s, int len)
|
|
{
|
|
unsigned long args[7];
|
|
|
|
args[0] = (unsigned long) "write";
|
|
args[1] = 3;
|
|
args[2] = 1;
|
|
args[3] = (unsigned int) prom_stdout;
|
|
args[4] = (unsigned long) s;
|
|
args[5] = len;
|
|
args[6] = (unsigned long) -1;
|
|
|
|
p1275_cmd_direct(args);
|
|
}
|