mirror of
https://github.com/torvalds/linux.git
synced 2024-12-15 23:51:46 +00:00
62c83cde92
GPIO SUPPORT FOR SCx200 & PC8736x The patch-set reworks the 2.4 vintage scx200_gpio driver for modern 2.6, and refactors GPIO support to reuse it in a new driver for the GPIO on PC-8736x chips. Its handy for the Soekris.com net-4801, which has both chips. These patches have been seen recently on Kernel-Mentors, and then Kernel-Newbies ML, where Jesper Juhl kindly reviewed it. His feedback has been incorporated. Thanks Jesper ! Its also gone to soekris-tech@soekris.com for possible testing by linux folks, I've gotten 1 promise so far. Theyre mostly BSD folk over there, but we'll see.. Device-file & Sysfs The driver preserves the existing device-file interface, including the write/cmd set, but adds v to 'view' the pin-settings & configs by inducing, via gpio_dump(), a dev_info() call. Its a fairly crappy way to get status, but it sticks to the syslog approach, conservatively. Allowing users to voluntarily trigger logging is good, it gives them a familiar way to confirm their app's control & use of the pins, and I've thus reduced the pin-mode-updates from dev_info to dev_dbg. I've recently bolted on a proto sysfs interface for both new drivers. Im not including those patches here; they (the patch + doc-pre-patch) are still quite raw (and unreviewed on KNML), and since they 'invent' a convention for GPIO, a proper vetting is needed. Since this patchset is much bigger than my previous ones, Id like to keep things simpler, and address it 1st, before bolting on more stuff. The driver-split The Geode CPU and the PC-87366 Super-IO chip have GPIO units which share a common pin-architecture (same pin features, with same bits controlling), but with different addressing mechanics and port organizations. The vintage driver expresses the pin capabilities with pin-mode commands [OoPpTt],etc that change the pin configurations, and since the 2 chips share pin-arch, we can reuse the read(), write() commands, once the implementation is suitably adjusted. The patchset adds a vtable: struct nsc_gpio_ops, to abstract the existing gpio operations, then adjusts fileops.write() code to invoke operations via that vtable. Driver specific open()s set private_data to the vtable so its available for use by write(). The vtable gets the gpio_dump() too, since its user-friendly, and (could be construed as) part of the current device-file interface. To support use of dev_dbg() in write() & _dump(), the vtable gets a dev ptr too, set by both scx200 & pc8736x _gpio drivers. heres how the pins are presented in syslog: [ 1890.176223] scx200_gpio.0: io00: 0x0044 TS OD PUE EDGE LO DEBOUNCE [ 1890.287223] scx200_gpio.0: io01: 0x0003 OE PP PUD EDGE LO nsc_gpio.c: new file is new home of several file-ops methods, which are modified to get their vtable from filp->private_data, and use it where needed. scx200_gpio.c: keeps some of its existing gpio routines, but now wires them up via the vtable (they're invoked by nsc_gpio.c:nsc_gpio_write() thru this vtable). A driver-spcific open() initializes filp->private_data with the vtable. Once the split is clean, and the scx200_gpio driver is working, we copy and modify the function and variable names, and rework the access-method bodies for the different addressing scheme. Heres a working overview of the patchset: # series file for GPIO # Spring Cleaning gpio-scx/patch.preclean # scripts/Lindent fixes, editor-ctrl comments # API Modernization gpio-scx/patch.api26 # what I learned from LDD3 gpio-scx/patch.platform-dev-2 # get pdev, support for dev_dbg() gpio-scx/patch.unsigned-minor # fix to match std practice # Debuggability gpio-scx/patch.dump-diet # shrink gpio_dump() gpio-scx/patch.viewpins # add new 'command' to call dump() gpio-scx/patch.init-refactor # pull shadow-register init to sub # Access-Abstraction (add vtable) gpio-scx/patch.access-vtable # introduce nsg_gpio_ops vtable, w dump gpio-scx/patch.vtable-calls # add & use the vtable in scx200_gpio gpio-scx/patch.nscgpio-shell # add empty driver for common-fops # move code under abstraction gpio-scx/patch.migrate-fops # move file-ops methods from scx200_gpio gpio-scx/patch.common-dump # mv scx200.c:scx200_gpio_dump() to nsc_gpio.c gpio-scx/patch.add-pc8736x-gpio # add new driver, like old, w chip adapt # gpio-scx/patch.add-DEBUG # enable all dev_dbg()s # Cleanups # finish printk -> dev_dbg() etc gpio-scx/patch.pdev-pc8736x # new drvr needs pdev too, gpio-scx/patch.devdbg-nscgpio # add device to 'vtable', use in dev_dbg() # gpio-scx/patch.pin-config-view # another 'c' 'command' # gpio-scx/quiet-getset # take out excess dbg stuff (pretty quiet now) gpio-scx/patch.shadow-current # imitate scx200_gpio's shadow regs in pc87* # post KMentors-post patches .. gpio-scx/patch.mutexes # use mutexes for config-locks gpio-scx/patch.viewpins-values # extend dump to obsolete separate 'c' cmd gpio-scx/patch.kconfig # add stuff for kbuild # TBC # combine api26 with pdev, which is just one step. # merge c&v commands to single do-all-fn # delay viewpins, dump-diet should also un-ifdef it too. diff.sys-gpio-rollup-1 This patch: Removed editor format-control comments, and used scripts/Lindent to clean up whitespace, then deleted the bogus chunks :-( Signed-off-by: Jim Cromie <jim.cromie@gmail.com> Signed-off-by: Andrew Morton <akpm@osdl.org> Signed-off-by: Linus Torvalds <torvalds@osdl.org>
162 lines
4.1 KiB
C
162 lines
4.1 KiB
C
/* linux/arch/i386/kernel/scx200.c
|
|
|
|
Copyright (c) 2001,2002 Christer Weinigel <wingel@nano-system.com>
|
|
|
|
National Semiconductor SCx200 support. */
|
|
|
|
#include <linux/config.h>
|
|
#include <linux/module.h>
|
|
#include <linux/errno.h>
|
|
#include <linux/kernel.h>
|
|
#include <linux/init.h>
|
|
#include <linux/pci.h>
|
|
|
|
#include <linux/scx200.h>
|
|
#include <linux/scx200_gpio.h>
|
|
|
|
/* Verify that the configuration block really is there */
|
|
#define scx200_cb_probe(base) (inw((base) + SCx200_CBA) == (base))
|
|
|
|
#define NAME "scx200"
|
|
|
|
MODULE_AUTHOR("Christer Weinigel <wingel@nano-system.com>");
|
|
MODULE_DESCRIPTION("NatSemi SCx200 Driver");
|
|
MODULE_LICENSE("GPL");
|
|
|
|
unsigned scx200_gpio_base = 0;
|
|
long scx200_gpio_shadow[2];
|
|
|
|
unsigned scx200_cb_base = 0;
|
|
|
|
static struct pci_device_id scx200_tbl[] = {
|
|
{ PCI_DEVICE(PCI_VENDOR_ID_NS, PCI_DEVICE_ID_NS_SCx200_BRIDGE) },
|
|
{ PCI_DEVICE(PCI_VENDOR_ID_NS, PCI_DEVICE_ID_NS_SC1100_BRIDGE) },
|
|
{ PCI_DEVICE(PCI_VENDOR_ID_NS, PCI_DEVICE_ID_NS_SCx200_XBUS) },
|
|
{ PCI_DEVICE(PCI_VENDOR_ID_NS, PCI_DEVICE_ID_NS_SC1100_XBUS) },
|
|
{ },
|
|
};
|
|
MODULE_DEVICE_TABLE(pci,scx200_tbl);
|
|
|
|
static int __devinit scx200_probe(struct pci_dev *, const struct pci_device_id *);
|
|
|
|
static struct pci_driver scx200_pci_driver = {
|
|
.name = "scx200",
|
|
.id_table = scx200_tbl,
|
|
.probe = scx200_probe,
|
|
};
|
|
|
|
static DEFINE_SPINLOCK(scx200_gpio_config_lock);
|
|
|
|
static int __devinit scx200_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
|
|
{
|
|
int bank;
|
|
unsigned base;
|
|
|
|
if (pdev->device == PCI_DEVICE_ID_NS_SCx200_BRIDGE ||
|
|
pdev->device == PCI_DEVICE_ID_NS_SC1100_BRIDGE) {
|
|
base = pci_resource_start(pdev, 0);
|
|
printk(KERN_INFO NAME ": GPIO base 0x%x\n", base);
|
|
|
|
if (request_region(base, SCx200_GPIO_SIZE, "NatSemi SCx200 GPIO") == 0) {
|
|
printk(KERN_ERR NAME ": can't allocate I/O for GPIOs\n");
|
|
return -EBUSY;
|
|
}
|
|
|
|
scx200_gpio_base = base;
|
|
|
|
/* read the current values driven on the GPIO signals */
|
|
for (bank = 0; bank < 2; ++bank)
|
|
scx200_gpio_shadow[bank] = inl(scx200_gpio_base + 0x10 * bank);
|
|
|
|
} else {
|
|
/* find the base of the Configuration Block */
|
|
if (scx200_cb_probe(SCx200_CB_BASE_FIXED)) {
|
|
scx200_cb_base = SCx200_CB_BASE_FIXED;
|
|
} else {
|
|
pci_read_config_dword(pdev, SCx200_CBA_SCRATCH, &base);
|
|
if (scx200_cb_probe(base)) {
|
|
scx200_cb_base = base;
|
|
} else {
|
|
printk(KERN_WARNING NAME ": Configuration Block not found\n");
|
|
return -ENODEV;
|
|
}
|
|
}
|
|
printk(KERN_INFO NAME ": Configuration Block base 0x%x\n", scx200_cb_base);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
u32 scx200_gpio_configure(int index, u32 mask, u32 bits)
|
|
{
|
|
u32 config, new_config;
|
|
unsigned long flags;
|
|
|
|
spin_lock_irqsave(&scx200_gpio_config_lock, flags);
|
|
|
|
outl(index, scx200_gpio_base + 0x20);
|
|
config = inl(scx200_gpio_base + 0x24);
|
|
|
|
new_config = (config & mask) | bits;
|
|
outl(new_config, scx200_gpio_base + 0x24);
|
|
|
|
spin_unlock_irqrestore(&scx200_gpio_config_lock, flags);
|
|
|
|
return config;
|
|
}
|
|
|
|
#if 0
|
|
void scx200_gpio_dump(unsigned index)
|
|
{
|
|
u32 config = scx200_gpio_configure(index, ~0, 0);
|
|
printk(KERN_DEBUG "GPIO%02u: 0x%08lx", index, (unsigned long)config);
|
|
|
|
if (config & 1)
|
|
printk(" OE"); /* output enabled */
|
|
else
|
|
printk(" TS"); /* tristate */
|
|
if (config & 2)
|
|
printk(" PP"); /* push pull */
|
|
else
|
|
printk(" OD"); /* open drain */
|
|
if (config & 4)
|
|
printk(" PUE"); /* pull up enabled */
|
|
else
|
|
printk(" PUD"); /* pull up disabled */
|
|
if (config & 8)
|
|
printk(" LOCKED"); /* locked */
|
|
if (config & 16)
|
|
printk(" LEVEL"); /* level input */
|
|
else
|
|
printk(" EDGE"); /* edge input */
|
|
if (config & 32)
|
|
printk(" HI"); /* trigger on rising edge */
|
|
else
|
|
printk(" LO"); /* trigger on falling edge */
|
|
if (config & 64)
|
|
printk(" DEBOUNCE"); /* debounce */
|
|
printk("\n");
|
|
}
|
|
#endif /* 0 */
|
|
|
|
static int __init scx200_init(void)
|
|
{
|
|
printk(KERN_INFO NAME ": NatSemi SCx200 Driver\n");
|
|
|
|
return pci_register_driver(&scx200_pci_driver);
|
|
}
|
|
|
|
static void __exit scx200_cleanup(void)
|
|
{
|
|
pci_unregister_driver(&scx200_pci_driver);
|
|
release_region(scx200_gpio_base, SCx200_GPIO_SIZE);
|
|
}
|
|
|
|
module_init(scx200_init);
|
|
module_exit(scx200_cleanup);
|
|
|
|
EXPORT_SYMBOL(scx200_gpio_base);
|
|
EXPORT_SYMBOL(scx200_gpio_shadow);
|
|
EXPORT_SYMBOL(scx200_gpio_configure);
|
|
EXPORT_SYMBOL(scx200_cb_base);
|