diff options
Diffstat (limited to 'drivers/net/macsonic.c')
| -rw-r--r-- | drivers/net/macsonic.c | 117 |
1 files changed, 60 insertions, 57 deletions
diff --git a/drivers/net/macsonic.c b/drivers/net/macsonic.c index 61eabcac734c..b3d7d8d77f46 100644 --- a/drivers/net/macsonic.c +++ b/drivers/net/macsonic.c | |||
| @@ -223,69 +223,73 @@ static int __devinit macsonic_init(struct net_device *dev) | |||
| 223 | return 0; | 223 | return 0; |
| 224 | } | 224 | } |
| 225 | 225 | ||
| 226 | static int __devinit mac_onboard_sonic_ethernet_addr(struct net_device *dev) | 226 | #define INVALID_MAC(mac) (memcmp(mac, "\x08\x00\x07", 3) && \ |
| 227 | memcmp(mac, "\x00\xA0\x40", 3) && \ | ||
| 228 | memcmp(mac, "\x00\x80\x19", 3) && \ | ||
| 229 | memcmp(mac, "\x00\x05\x02", 3)) | ||
| 230 | |||
| 231 | static void __devinit mac_onboard_sonic_ethernet_addr(struct net_device *dev) | ||
| 227 | { | 232 | { |
| 228 | struct sonic_local *lp = netdev_priv(dev); | 233 | struct sonic_local *lp = netdev_priv(dev); |
| 229 | const int prom_addr = ONBOARD_SONIC_PROM_BASE; | 234 | const int prom_addr = ONBOARD_SONIC_PROM_BASE; |
| 230 | int i; | 235 | unsigned short val; |
| 231 | 236 | ||
| 232 | /* On NuBus boards we can sometimes look in the ROM resources. | 237 | /* |
| 233 | No such luck for comm-slot/onboard. */ | 238 | * On NuBus boards we can sometimes look in the ROM resources. |
| 234 | for(i = 0; i < 6; i++) | 239 | * No such luck for comm-slot/onboard. |
| 235 | dev->dev_addr[i] = SONIC_READ_PROM(i); | 240 | * On the PowerBook 520, the PROM base address is a mystery. |
| 241 | */ | ||
| 242 | if (hwreg_present((void *)prom_addr)) { | ||
| 243 | int i; | ||
| 244 | |||
| 245 | for (i = 0; i < 6; i++) | ||
| 246 | dev->dev_addr[i] = SONIC_READ_PROM(i); | ||
| 247 | if (!INVALID_MAC(dev->dev_addr)) | ||
| 248 | return; | ||
| 236 | 249 | ||
| 237 | /* Most of the time, the address is bit-reversed. The NetBSD | 250 | /* |
| 238 | source has a rather long and detailed historical account of | 251 | * Most of the time, the address is bit-reversed. The NetBSD |
| 239 | why this is so. */ | 252 | * source has a rather long and detailed historical account of |
| 240 | if (memcmp(dev->dev_addr, "\x08\x00\x07", 3) && | 253 | * why this is so. |
| 241 | memcmp(dev->dev_addr, "\x00\xA0\x40", 3) && | 254 | */ |
| 242 | memcmp(dev->dev_addr, "\x00\x80\x19", 3) && | ||
| 243 | memcmp(dev->dev_addr, "\x00\x05\x02", 3)) | ||
| 244 | bit_reverse_addr(dev->dev_addr); | 255 | bit_reverse_addr(dev->dev_addr); |
| 245 | else | 256 | if (!INVALID_MAC(dev->dev_addr)) |
| 246 | return 0; | 257 | return; |
| 247 | 258 | ||
| 248 | /* If we still have what seems to be a bogus address, we'll | ||
| 249 | look in the CAM. The top entry should be ours. */ | ||
| 250 | /* Danger! This only works if MacOS has already initialized | ||
| 251 | the card... */ | ||
| 252 | if (memcmp(dev->dev_addr, "\x08\x00\x07", 3) && | ||
| 253 | memcmp(dev->dev_addr, "\x00\xA0\x40", 3) && | ||
| 254 | memcmp(dev->dev_addr, "\x00\x80\x19", 3) && | ||
| 255 | memcmp(dev->dev_addr, "\x00\x05\x02", 3)) | ||
| 256 | { | ||
| 257 | unsigned short val; | ||
| 258 | |||
| 259 | printk(KERN_INFO "macsonic: PROM seems to be wrong, trying CAM entry 15\n"); | ||
| 260 | |||
| 261 | SONIC_WRITE(SONIC_CMD, SONIC_CR_RST); | ||
| 262 | SONIC_WRITE(SONIC_CEP, 15); | ||
| 263 | |||
| 264 | val = SONIC_READ(SONIC_CAP2); | ||
| 265 | dev->dev_addr[5] = val >> 8; | ||
| 266 | dev->dev_addr[4] = val & 0xff; | ||
| 267 | val = SONIC_READ(SONIC_CAP1); | ||
| 268 | dev->dev_addr[3] = val >> 8; | ||
| 269 | dev->dev_addr[2] = val & 0xff; | ||
| 270 | val = SONIC_READ(SONIC_CAP0); | ||
| 271 | dev->dev_addr[1] = val >> 8; | ||
| 272 | dev->dev_addr[0] = val & 0xff; | ||
| 273 | |||
| 274 | printk(KERN_INFO "HW Address from CAM 15: %pM\n", | ||
| 275 | dev->dev_addr); | ||
| 276 | } else return 0; | ||
| 277 | |||
| 278 | if (memcmp(dev->dev_addr, "\x08\x00\x07", 3) && | ||
| 279 | memcmp(dev->dev_addr, "\x00\xA0\x40", 3) && | ||
| 280 | memcmp(dev->dev_addr, "\x00\x80\x19", 3) && | ||
| 281 | memcmp(dev->dev_addr, "\x00\x05\x02", 3)) | ||
| 282 | { | ||
| 283 | /* | 259 | /* |
| 284 | * Still nonsense ... messed up someplace! | 260 | * If we still have what seems to be a bogus address, we'll |
| 261 | * look in the CAM. The top entry should be ours. | ||
| 285 | */ | 262 | */ |
| 286 | printk(KERN_ERR "macsonic: ERROR (INVALID MAC)\n"); | 263 | printk(KERN_WARNING "macsonic: MAC address in PROM seems " |
| 287 | return -EIO; | 264 | "to be invalid, trying CAM\n"); |
| 288 | } else return 0; | 265 | } else { |
| 266 | printk(KERN_WARNING "macsonic: cannot read MAC address from " | ||
| 267 | "PROM, trying CAM\n"); | ||
| 268 | } | ||
| 269 | |||
| 270 | /* This only works if MacOS has already initialized the card. */ | ||
| 271 | |||
| 272 | SONIC_WRITE(SONIC_CMD, SONIC_CR_RST); | ||
| 273 | SONIC_WRITE(SONIC_CEP, 15); | ||
| 274 | |||
| 275 | val = SONIC_READ(SONIC_CAP2); | ||
| 276 | dev->dev_addr[5] = val >> 8; | ||
| 277 | dev->dev_addr[4] = val & 0xff; | ||
| 278 | val = SONIC_READ(SONIC_CAP1); | ||
| 279 | dev->dev_addr[3] = val >> 8; | ||
| 280 | dev->dev_addr[2] = val & 0xff; | ||
| 281 | val = SONIC_READ(SONIC_CAP0); | ||
| 282 | dev->dev_addr[1] = val >> 8; | ||
| 283 | dev->dev_addr[0] = val & 0xff; | ||
| 284 | |||
| 285 | if (!INVALID_MAC(dev->dev_addr)) | ||
| 286 | return; | ||
| 287 | |||
| 288 | /* Still nonsense ... messed up someplace! */ | ||
| 289 | |||
| 290 | printk(KERN_WARNING "macsonic: MAC address in CAM entry 15 " | ||
| 291 | "seems invalid, will use a random MAC\n"); | ||
| 292 | random_ether_addr(dev->dev_addr); | ||
| 289 | } | 293 | } |
| 290 | 294 | ||
| 291 | static int __devinit mac_onboard_sonic_probe(struct net_device *dev) | 295 | static int __devinit mac_onboard_sonic_probe(struct net_device *dev) |
| @@ -402,8 +406,7 @@ static int __devinit mac_onboard_sonic_probe(struct net_device *dev) | |||
| 402 | SONIC_WRITE(SONIC_ISR, 0x7fff); | 406 | SONIC_WRITE(SONIC_ISR, 0x7fff); |
| 403 | 407 | ||
| 404 | /* Now look for the MAC address. */ | 408 | /* Now look for the MAC address. */ |
| 405 | if (mac_onboard_sonic_ethernet_addr(dev) != 0) | 409 | mac_onboard_sonic_ethernet_addr(dev); |
| 406 | return -ENODEV; | ||
| 407 | 410 | ||
| 408 | /* Shared init code */ | 411 | /* Shared init code */ |
| 409 | return macsonic_init(dev); | 412 | return macsonic_init(dev); |
