Commit 85cc313a authored by Finn Thain's avatar Finn Thain Committed by Geert Uytterhoeven

nubus: Fix pointer validation

Fix bounds checking on slot-space pointer movement.
Remove redundant test for zero byte-lanes value.
Fix broken byte-lanes vs. address validation.

This patch changes the circumstances under which an error is printed to
the console and fixes the address validation.

The validation code should work correctly now: the broken test for a
valid bytelanes value is replaced with a working test (which eliminates
false negatives) and the 24-bit directory offset bounds check is fixed
(which eliminates false positives).

Please see "Designing Cards and Drivers for the Macintosh Family"
ch. 8, "NuBus Card Firmware" for an explanation of the bytelanes check
and directory offset value.
Signed-off-by: default avatarFinn Thain <fthain@telegraphics.com.au>
Reviewed-by: default avatarMichael Schmitz <schmitzmic@gmail.com>
Signed-off-by: default avatarGeert Uytterhoeven <geert@linux-m68k.org>
parent 475e6e15
...@@ -92,9 +92,6 @@ static void nubus_rewind(unsigned char **ptr, int len, int map) ...@@ -92,9 +92,6 @@ static void nubus_rewind(unsigned char **ptr, int len, int map)
{ {
unsigned char *p = *ptr; unsigned char *p = *ptr;
/* Sanity check */
if (len > 65536)
pr_err("rewind of 0x%08x!\n", len);
while (len) { while (len) {
do { do {
p--; p--;
...@@ -108,8 +105,6 @@ static void nubus_advance(unsigned char **ptr, int len, int map) ...@@ -108,8 +105,6 @@ static void nubus_advance(unsigned char **ptr, int len, int map)
{ {
unsigned char *p = *ptr; unsigned char *p = *ptr;
if (len > 65536)
pr_err("advance of 0x%08x!\n", len);
while (len) { while (len) {
while (not_useful(p, map)) while (not_useful(p, map))
p++; p++;
...@@ -121,10 +116,15 @@ static void nubus_advance(unsigned char **ptr, int len, int map) ...@@ -121,10 +116,15 @@ static void nubus_advance(unsigned char **ptr, int len, int map)
static void nubus_move(unsigned char **ptr, int len, int map) static void nubus_move(unsigned char **ptr, int len, int map)
{ {
unsigned long slot_space = (unsigned long)*ptr & 0xFF000000;
if (len > 0) if (len > 0)
nubus_advance(ptr, len, map); nubus_advance(ptr, len, map);
else if (len < 0) else if (len < 0)
nubus_rewind(ptr, -len, map); nubus_rewind(ptr, -len, map);
if (((unsigned long)*ptr & 0xFF000000) != slot_space)
pr_err("%s: moved out of slot address space!\n", __func__);
} }
/* Now, functions to read the sResource tree */ /* Now, functions to read the sResource tree */
...@@ -808,8 +808,6 @@ void __init nubus_probe_slot(int slot) ...@@ -808,8 +808,6 @@ void __init nubus_probe_slot(int slot)
continue; continue;
dp = *rp; dp = *rp;
if(dp == 0)
continue;
/* The last byte of the format block consists of two /* The last byte of the format block consists of two
nybbles which are "mirror images" of each other. nybbles which are "mirror images" of each other.
...@@ -818,7 +816,7 @@ void __init nubus_probe_slot(int slot) ...@@ -818,7 +816,7 @@ void __init nubus_probe_slot(int slot)
continue; continue;
/* Check that this value is actually *on* one of the /* Check that this value is actually *on* one of the
bytelanes it claims are valid! */ bytelanes it claims are valid! */
if ((dp & 0x0F) >= (1 << i)) if (not_useful(rp, dp))
continue; continue;
/* Looks promising. Let's put it on the list. */ /* Looks promising. Let's put it on the list. */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment