aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--drivers/i2c/busses/scx200_acb.c57
1 files changed, 43 insertions, 14 deletions
diff --git a/drivers/i2c/busses/scx200_acb.c b/drivers/i2c/busses/scx200_acb.c
index 02150bb25d43..0ae77f84ffe8 100644
--- a/drivers/i2c/busses/scx200_acb.c
+++ b/drivers/i2c/busses/scx200_acb.c
@@ -2,6 +2,7 @@
2 Copyright (c) 2001,2002 Christer Weinigel <wingel@nano-system.com> 2 Copyright (c) 2001,2002 Christer Weinigel <wingel@nano-system.com>
3 3
4 National Semiconductor SCx200 ACCESS.bus support 4 National Semiconductor SCx200 ACCESS.bus support
5 Also supports the AMD CS5535 and AMD CS5536
5 6
6 Based on i2c-keywest.c which is: 7 Based on i2c-keywest.c which is:
7 Copyright (c) 2001 Benjamin Herrenschmidt <benh@kernel.crashing.org> 8 Copyright (c) 2001 Benjamin Herrenschmidt <benh@kernel.crashing.org>
@@ -31,6 +32,7 @@
31#include <linux/pci.h> 32#include <linux/pci.h>
32#include <linux/delay.h> 33#include <linux/delay.h>
33#include <asm/io.h> 34#include <asm/io.h>
35#include <asm/msr.h>
34 36
35#include <linux/scx200.h> 37#include <linux/scx200.h>
36 38
@@ -416,7 +418,7 @@ static int scx200_acb_probe(struct scx200_acb_iface *iface)
416 return 0; 418 return 0;
417} 419}
418 420
419static int __init scx200_acb_create(int base, int index) 421static int __init scx200_acb_create(const char *text, int base, int index)
420{ 422{
421 struct scx200_acb_iface *iface; 423 struct scx200_acb_iface *iface;
422 struct i2c_adapter *adapter; 424 struct i2c_adapter *adapter;
@@ -432,7 +434,7 @@ static int __init scx200_acb_create(int base, int index)
432 434
433 adapter = &iface->adapter; 435 adapter = &iface->adapter;
434 i2c_set_adapdata(adapter, iface); 436 i2c_set_adapdata(adapter, iface);
435 snprintf(adapter->name, I2C_NAME_SIZE, "SCx200 ACB%d", index); 437 snprintf(adapter->name, I2C_NAME_SIZE, "%s ACB%d", text, index);
436 adapter->owner = THIS_MODULE; 438 adapter->owner = THIS_MODULE;
437 adapter->id = I2C_HW_SMBUS_SCX200; 439 adapter->id = I2C_HW_SMBUS_SCX200;
438 adapter->algo = &scx200_acb_algorithm; 440 adapter->algo = &scx200_acb_algorithm;
@@ -440,8 +442,9 @@ static int __init scx200_acb_create(int base, int index)
440 442
441 init_MUTEX(&iface->sem); 443 init_MUTEX(&iface->sem);
442 444
443 snprintf(description, sizeof(description), 445 snprintf(description, sizeof(description), "%s ACCESS.bus [%s]",
444 "NatSemi SCx200 ACCESS.bus [%s]", adapter->name); 446 text, adapter->name);
447
445 if (request_region(base, 8, description) == 0) { 448 if (request_region(base, 8, description) == 0) {
446 printk(KERN_ERR NAME ": can't allocate io 0x%x-0x%x\n", 449 printk(KERN_ERR NAME ": can't allocate io 0x%x-0x%x\n",
447 base, base + 8-1); 450 base, base + 8-1);
@@ -485,24 +488,50 @@ static struct pci_device_id scx200[] = {
485 { }, 488 { },
486}; 489};
487 490
491static struct pci_device_id divil_pci[] = {
492 { PCI_DEVICE(PCI_VENDOR_ID_NS, PCI_DEVICE_ID_NS_CS5535_ISA) },
493 { PCI_DEVICE(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_CS5536_ISA) },
494 { } /* NULL entry */
495};
496
497#define MSR_LBAR_SMB 0x5140000B
498
499static int scx200_add_cs553x(void)
500{
501 u32 low, hi;
502 u32 smb_base;
503
504 /* Grab & reserve the SMB I/O range */
505 rdmsr(MSR_LBAR_SMB, low, hi);
506
507 /* Check the IO mask and whether SMB is enabled */
508 if (hi != 0x0000F001) {
509 printk(KERN_WARNING NAME ": SMBus not enabled\n");
510 return -ENODEV;
511 }
512
513 /* SMBus IO size is 8 bytes */
514 smb_base = low & 0x0000FFF8;
515
516 return scx200_acb_create("CS5535", smb_base, 0);
517}
518
488static int __init scx200_acb_init(void) 519static int __init scx200_acb_init(void)
489{ 520{
490 int i; 521 int i;
491 int rc; 522 int rc = -ENODEV;
492 523
493 pr_debug(NAME ": NatSemi SCx200 ACCESS.bus Driver\n"); 524 pr_debug(NAME ": NatSemi SCx200 ACCESS.bus Driver\n");
494 525
495 /* Verify that this really is a SCx200 processor */ 526 /* Verify that this really is a SCx200 processor */
496 if (pci_dev_present(scx200) == 0) 527 if (pci_dev_present(scx200)) {
497 return -ENODEV; 528 for (i = 0; i < MAX_DEVICES; ++i) {
529 if (base[i] > 0)
530 rc = scx200_acb_create("SCx200", base[i], i);
531 }
532 } else if (pci_dev_present(divil_pci))
533 rc = scx200_add_cs553x();
498 534
499 rc = -ENXIO;
500 for (i = 0; i < MAX_DEVICES; ++i) {
501 if (base[i] > 0)
502 rc = scx200_acb_create(base[i], i);
503 }
504 if (scx200_acb_list)
505 return 0;
506 return rc; 535 return rc;
507} 536}
508 537