sparc: Write to prom console using indirect buffer.
sparc64 systems have a restriction in that passing in buffer addressses above 4GB to prom calls is not reliable. We end up violating this when we do prom console writes, because we use an on-stack buffer to translate '\n' into '\r\n'. So instead, do this translation into an intermediate buffer, which is in the kernel image and thus below 4GB, then pass that to the PROM console write calls. On the 32-bit side we don't have to deal with any of these issues, so the new prom_console_write_buf() uses the existing prom_nbputchar() implementation. However we can now mark those routines static. Since the 64-bit side completely uses new code we can delete the putchar bits as they are now completely unused. Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
12c7a35ee6
commit
595a251c07
5 changed files with 52 additions and 29 deletions
|
@ -102,8 +102,8 @@ extern int prom_getrev(void);
|
||||||
/* Get the prom firmware revision. */
|
/* Get the prom firmware revision. */
|
||||||
extern int prom_getprev(void);
|
extern int prom_getprev(void);
|
||||||
|
|
||||||
/* Blocking put character to console. */
|
/* Write a buffer of characters to the console. */
|
||||||
extern void prom_putchar(const char *buf);
|
extern void prom_console_write_buf(const char *buf, int len);
|
||||||
|
|
||||||
/* Prom's internal routines, don't use in kernel/boot code. */
|
/* Prom's internal routines, don't use in kernel/boot code. */
|
||||||
extern void prom_printf(const char *fmt, ...);
|
extern void prom_printf(const char *fmt, ...);
|
||||||
|
|
|
@ -94,8 +94,8 @@ extern void prom_halt_power_off(void) __attribute__ ((noreturn));
|
||||||
*/
|
*/
|
||||||
extern unsigned char prom_get_idprom(char *idp_buffer, int idpbuf_size);
|
extern unsigned char prom_get_idprom(char *idp_buffer, int idpbuf_size);
|
||||||
|
|
||||||
/* Blocking put character to console. */
|
/* Write a buffer of characters to the console. */
|
||||||
extern void prom_putchar(const char *buf);
|
extern void prom_console_write_buf(const char *buf, int len);
|
||||||
|
|
||||||
/* Prom's internal routines, don't use in kernel/boot code. */
|
/* Prom's internal routines, don't use in kernel/boot code. */
|
||||||
extern void prom_printf(const char *fmt, ...);
|
extern void prom_printf(const char *fmt, ...);
|
||||||
|
|
|
@ -43,12 +43,14 @@ static int prom_nbputchar(const char *buf)
|
||||||
return i; /* Ugh, we could spin forever on unsupported proms ;( */
|
return i; /* Ugh, we could spin forever on unsupported proms ;( */
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Blocking version of put character routine above. */
|
void prom_console_write_buf(const char *buf, int len)
|
||||||
void prom_putchar(const char *buf)
|
|
||||||
{
|
{
|
||||||
while (1) {
|
while (len) {
|
||||||
int err = prom_nbputchar(buf);
|
int n = prom_nbputchar(buf);
|
||||||
if (!err)
|
if (n)
|
||||||
break;
|
continue;
|
||||||
|
len--;
|
||||||
|
buf++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -15,35 +15,34 @@
|
||||||
|
|
||||||
extern int prom_stdin, prom_stdout;
|
extern int prom_stdin, prom_stdout;
|
||||||
|
|
||||||
/* Non blocking put character to console device, returns -1 if
|
static int __prom_console_write_buf(const char *buf, int len)
|
||||||
* unsuccessful.
|
|
||||||
*/
|
|
||||||
static int prom_nbputchar(const char *buf)
|
|
||||||
{
|
{
|
||||||
unsigned long args[7];
|
unsigned long args[7];
|
||||||
|
int ret;
|
||||||
|
|
||||||
args[0] = (unsigned long) "write";
|
args[0] = (unsigned long) "write";
|
||||||
args[1] = 3;
|
args[1] = 3;
|
||||||
args[2] = 1;
|
args[2] = 1;
|
||||||
args[3] = (unsigned int) prom_stdout;
|
args[3] = (unsigned int) prom_stdout;
|
||||||
args[4] = (unsigned long) buf;
|
args[4] = (unsigned long) buf;
|
||||||
args[5] = 1;
|
args[5] = (unsigned int) len;
|
||||||
args[6] = (unsigned long) -1;
|
args[6] = (unsigned long) -1;
|
||||||
|
|
||||||
p1275_cmd_direct(args);
|
p1275_cmd_direct(args);
|
||||||
|
|
||||||
if (args[6] == 1)
|
ret = (int) args[6];
|
||||||
return 0;
|
if (ret < 0)
|
||||||
else
|
|
||||||
return -1;
|
return -1;
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Blocking version of put character routine above. */
|
void prom_console_write_buf(const char *buf, int len)
|
||||||
void prom_putchar(const char *buf)
|
|
||||||
{
|
{
|
||||||
while (1) {
|
while (len) {
|
||||||
int err = prom_nbputchar(buf);
|
int n = __prom_console_write_buf(buf, len);
|
||||||
if (!err)
|
if (n < 0)
|
||||||
break;
|
continue;
|
||||||
|
len -= n;
|
||||||
|
buf += len;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,23 +15,45 @@
|
||||||
|
|
||||||
#include <linux/kernel.h>
|
#include <linux/kernel.h>
|
||||||
#include <linux/compiler.h>
|
#include <linux/compiler.h>
|
||||||
|
#include <linux/spinlock.h>
|
||||||
|
|
||||||
#include <asm/openprom.h>
|
#include <asm/openprom.h>
|
||||||
#include <asm/oplib.h>
|
#include <asm/oplib.h>
|
||||||
|
|
||||||
|
#define CONSOLE_WRITE_BUF_SIZE 1024
|
||||||
|
|
||||||
static char ppbuf[1024];
|
static char ppbuf[1024];
|
||||||
|
static char console_write_buf[CONSOLE_WRITE_BUF_SIZE];
|
||||||
|
static DEFINE_RAW_SPINLOCK(console_write_lock);
|
||||||
|
|
||||||
void notrace prom_write(const char *buf, unsigned int n)
|
void notrace prom_write(const char *buf, unsigned int n)
|
||||||
{
|
{
|
||||||
|
unsigned int dest_len;
|
||||||
|
unsigned long flags;
|
||||||
|
char *dest;
|
||||||
|
|
||||||
|
dest = console_write_buf;
|
||||||
|
raw_spin_lock_irqsave(&console_write_lock, flags);
|
||||||
|
|
||||||
|
dest_len = 0;
|
||||||
while (n-- != 0) {
|
while (n-- != 0) {
|
||||||
char ch = *buf;
|
char ch = *buf++;
|
||||||
if (ch == '\n') {
|
if (ch == '\n') {
|
||||||
char tmp = '\r';
|
*dest++ = '\r';
|
||||||
prom_putchar(&tmp);
|
dest_len++;
|
||||||
|
}
|
||||||
|
*dest++ = ch;
|
||||||
|
dest_len++;
|
||||||
|
if (dest_len >= CONSOLE_WRITE_BUF_SIZE - 1) {
|
||||||
|
prom_console_write_buf(console_write_buf, dest_len);
|
||||||
|
dest = console_write_buf;
|
||||||
|
dest_len = 0;
|
||||||
}
|
}
|
||||||
prom_putchar(buf);
|
|
||||||
buf++;
|
|
||||||
}
|
}
|
||||||
|
if (dest_len)
|
||||||
|
prom_console_write_buf(console_write_buf, dest_len);
|
||||||
|
|
||||||
|
raw_spin_unlock_irqrestore(&console_write_lock, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
void notrace prom_printf(const char *fmt, ...)
|
void notrace prom_printf(const char *fmt, ...)
|
||||||
|
|
Loading…
Reference in a new issue