|
|
|
/*
|
|
|
|
* arch/sh/kernel/cpu/bus.c
|
|
|
|
*
|
|
|
|
* Virtual bus for SuperH.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2004 Paul Mundt
|
|
|
|
*
|
|
|
|
* Shamelessly cloned from arch/arm/mach-omap/bus.c, which was written
|
|
|
|
* by:
|
|
|
|
*
|
|
|
|
* Copyright (C) 2003 - 2004 Nokia Corporation
|
|
|
|
* Written by Tony Lindgren <tony@atomide.com>
|
|
|
|
* Portions of code based on sa1111.c.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms of the GNU General Public License as published by the
|
|
|
|
* Free Software Foundation; either version 2 of the License, or (at your
|
|
|
|
* option) any later version.
|
|
|
|
*/
|
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <linux/device.h>
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <asm/bus-sh.h>
|
|
|
|
|
|
|
|
static int sh_bus_match(struct device *dev, struct device_driver *drv)
|
|
|
|
{
|
|
|
|
struct sh_driver *shdrv = to_sh_driver(drv);
|
|
|
|
struct sh_dev *shdev = to_sh_dev(dev);
|
|
|
|
|
|
|
|
return shdev->dev_id == shdrv->dev_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int sh_bus_suspend(struct device *dev, pm_message_t state)
|
|
|
|
{
|
|
|
|
struct sh_dev *shdev = to_sh_dev(dev);
|
|
|
|
struct sh_driver *shdrv = to_sh_driver(dev->driver);
|
|
|
|
|
|
|
|
if (shdrv && shdrv->suspend)
|
|
|
|
return shdrv->suspend(shdev, state);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int sh_bus_resume(struct device *dev)
|
|
|
|
{
|
|
|
|
struct sh_dev *shdev = to_sh_dev(dev);
|
|
|
|
struct sh_driver *shdrv = to_sh_driver(dev->driver);
|
|
|
|
|
|
|
|
if (shdrv && shdrv->resume)
|
|
|
|
return shdrv->resume(shdev);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct device sh_bus_devices[SH_NR_BUSES] = {
|
|
|
|
{
|
|
|
|
.bus_id = SH_BUS_NAME_VIRT,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
struct bus_type sh_bus_types[SH_NR_BUSES] = {
|
|
|
|
{
|
|
|
|
.name = SH_BUS_NAME_VIRT,
|
|
|
|
.match = sh_bus_match,
|
|
|
|
.suspend = sh_bus_suspend,
|
|
|
|
.resume = sh_bus_resume,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
static int sh_device_probe(struct device *dev)
|
|
|
|
{
|
|
|
|
struct sh_dev *shdev = to_sh_dev(dev);
|
|
|
|
struct sh_driver *shdrv = to_sh_driver(dev->driver);
|
|
|
|
|
|
|
|
if (shdrv && shdrv->probe)
|
|
|
|
return shdrv->probe(shdev);
|
|
|
|
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int sh_device_remove(struct device *dev)
|
|
|
|
{
|
|
|
|
struct sh_dev *shdev = to_sh_dev(dev);
|
|
|
|
struct sh_driver *shdrv = to_sh_driver(dev->driver);
|
|
|
|
|
|
|
|
if (shdrv && shdrv->remove)
|
|
|
|
return shdrv->remove(shdev);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int sh_device_register(struct sh_dev *dev)
|
|
|
|
{
|
|
|
|
if (!dev)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if (dev->bus_id < 0 || dev->bus_id >= SH_NR_BUSES) {
|
|
|
|
printk(KERN_ERR "%s: bus_id invalid: %s bus: %d\n",
|
|
|
|
__FUNCTION__, dev->name, dev->bus_id);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
dev->dev.parent = &sh_bus_devices[dev->bus_id];
|
|
|
|
dev->dev.bus = &sh_bus_types[dev->bus_id];
|
|
|
|
|
|
|
|
/* This is needed for USB OHCI to work */
|
|
|
|
if (dev->dma_mask)
|
|
|
|
dev->dev.dma_mask = dev->dma_mask;
|
|
|
|
|
|
|
|
snprintf(dev->dev.bus_id, BUS_ID_SIZE, "%s%u",
|
|
|
|
dev->name, dev->dev_id);
|
|
|
|
|
|
|
|
printk(KERN_INFO "Registering SH device '%s'. Parent at %s\n",
|
|
|
|
dev->dev.bus_id, dev->dev.parent->bus_id);
|
|
|
|
|
|
|
|
return device_register(&dev->dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
void sh_device_unregister(struct sh_dev *dev)
|
|
|
|
{
|
|
|
|
device_unregister(&dev->dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
int sh_driver_register(struct sh_driver *drv)
|
|
|
|
{
|
|
|
|
if (!drv)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if (drv->bus_id < 0 || drv->bus_id >= SH_NR_BUSES) {
|
|
|
|
printk(KERN_ERR "%s: bus_id invalid: bus: %d device %d\n",
|
|
|
|
__FUNCTION__, drv->bus_id, drv->dev_id);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
drv->drv.probe = sh_device_probe;
|
|
|
|
drv->drv.remove = sh_device_remove;
|
|
|
|
drv->drv.bus = &sh_bus_types[drv->bus_id];
|
|
|
|
|
|
|
|
return driver_register(&drv->drv);
|
|
|
|
}
|
|
|
|
|
|
|
|
void sh_driver_unregister(struct sh_driver *drv)
|
|
|
|
{
|
|
|
|
driver_unregister(&drv->drv);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __init sh_bus_init(void)
|
|
|
|
{
|
|
|
|
int i, ret = 0;
|
|
|
|
|
|
|
|
for (i = 0; i < SH_NR_BUSES; i++) {
|
|
|
|
ret = device_register(&sh_bus_devices[i]);
|
|
|
|
if (ret != 0) {
|
|
|
|
printk(KERN_ERR "Unable to register bus device %s\n",
|
|
|
|
sh_bus_devices[i].bus_id);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = bus_register(&sh_bus_types[i]);
|
|
|
|
if (ret != 0) {
|
|
|
|
printk(KERN_ERR "Unable to register bus %s\n",
|
|
|
|
sh_bus_types[i].name);
|
|
|
|
device_unregister(&sh_bus_devices[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
printk(KERN_INFO "SH Virtual Bus initialized\n");
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __exit sh_bus_exit(void)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < SH_NR_BUSES; i++) {
|
|
|
|
bus_unregister(&sh_bus_types[i]);
|
|
|
|
device_unregister(&sh_bus_devices[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
module_init(sh_bus_init);
|
|
|
|
module_exit(sh_bus_exit);
|
|
|
|
|
|
|
|
MODULE_AUTHOR("Paul Mundt <lethal@linux-sh.org>");
|
|
|
|
MODULE_DESCRIPTION("SH Virtual Bus");
|
|
|
|
MODULE_LICENSE("GPL");
|
|
|
|
|
|
|
|
EXPORT_SYMBOL(sh_bus_types);
|
|
|
|
EXPORT_SYMBOL(sh_device_register);
|
|
|
|
EXPORT_SYMBOL(sh_device_unregister);
|
|
|
|
EXPORT_SYMBOL(sh_driver_register);
|
|
|
|
EXPORT_SYMBOL(sh_driver_unregister);
|
|
|
|
|