mirror of https://github.com/hak5/openwrt.git
289 lines
7.3 KiB
C
289 lines
7.3 KiB
C
|
/*
|
||
|
* drivers/uio/uio_ubicom32ring.c
|
||
|
*
|
||
|
* Userspace I/O platform driver for Ubicom32 ring buffers
|
||
|
*
|
||
|
* (C) Copyright 2009, Ubicom, Inc.
|
||
|
*
|
||
|
* This file is part of the Ubicom32 Linux Kernel Port.
|
||
|
*
|
||
|
* Based on uio_ubicom32ring.c by Magnus Damm
|
||
|
*
|
||
|
* The Ubicom32 Linux Kernel Port 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.
|
||
|
*
|
||
|
* The Ubicom32 Linux Kernel Port is distributed in the hope that it
|
||
|
* will be useful, but WITHOUT ANY WARRANTY; without even the implied
|
||
|
* warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
|
||
|
* the GNU General Public License for more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU General Public License
|
||
|
* along with the Ubicom32 Linux Kernel Port. If not,
|
||
|
* see <http://www.gnu.org/licenses/>.
|
||
|
*/
|
||
|
|
||
|
#include <linux/platform_device.h>
|
||
|
#include <linux/uio_driver.h>
|
||
|
#include <linux/spinlock.h>
|
||
|
#include <linux/bitops.h>
|
||
|
#include <linux/interrupt.h>
|
||
|
#include <linux/stringify.h>
|
||
|
|
||
|
#include <asm/ip5000.h>
|
||
|
#include <asm/ubicom32ring.h>
|
||
|
|
||
|
#define DRIVER_NAME "uio_ubicom32ring"
|
||
|
|
||
|
struct uio_ubicom32ring_data {
|
||
|
struct uio_info *uioinfo;
|
||
|
|
||
|
struct uio_ubicom32ring_regs *regs;
|
||
|
|
||
|
/*
|
||
|
* IRQ used to kick the ring buffer
|
||
|
*/
|
||
|
int irq_tx;
|
||
|
int irq_rx;
|
||
|
|
||
|
spinlock_t lock;
|
||
|
|
||
|
unsigned long flags;
|
||
|
|
||
|
char name[0];
|
||
|
};
|
||
|
|
||
|
static irqreturn_t uio_ubicom32ring_handler(int irq, struct uio_info *dev_info)
|
||
|
{
|
||
|
struct uio_ubicom32ring_data *priv = dev_info->priv;
|
||
|
|
||
|
/* Just disable the interrupt in the interrupt controller, and
|
||
|
* remember the state so we can allow user space to enable it later.
|
||
|
*/
|
||
|
|
||
|
if (!test_and_set_bit(0, &priv->flags))
|
||
|
disable_irq_nosync(irq);
|
||
|
|
||
|
return IRQ_HANDLED;
|
||
|
}
|
||
|
|
||
|
static int uio_ubicom32ring_irqcontrol(struct uio_info *dev_info, s32 irq_on)
|
||
|
{
|
||
|
struct uio_ubicom32ring_data *priv = dev_info->priv;
|
||
|
unsigned long flags;
|
||
|
|
||
|
/* Allow user space to enable and disable the interrupt
|
||
|
* in the interrupt controller, but keep track of the
|
||
|
* state to prevent per-irq depth damage.
|
||
|
*
|
||
|
* Serialize this operation to support multiple tasks.
|
||
|
*/
|
||
|
|
||
|
spin_lock_irqsave(&priv->lock, flags);
|
||
|
|
||
|
if (irq_on & 2) {
|
||
|
/*
|
||
|
* Kick the ring buffer (if we can)
|
||
|
*/
|
||
|
if (priv->irq_tx != 0xFF) {
|
||
|
ubicom32_set_interrupt(priv->irq_tx);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (priv->irq_rx != 0xFF) {
|
||
|
if (irq_on & 1) {
|
||
|
if (test_and_clear_bit(0, &priv->flags))
|
||
|
enable_irq(dev_info->irq);
|
||
|
} else {
|
||
|
if (!test_and_set_bit(0, &priv->flags))
|
||
|
disable_irq(dev_info->irq);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
spin_unlock_irqrestore(&priv->lock, flags);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int uio_ubicom32ring_probe(struct platform_device *pdev)
|
||
|
{
|
||
|
struct uio_info *uioinfo;
|
||
|
struct uio_mem *uiomem;
|
||
|
struct uio_ubicom32ring_data *priv;
|
||
|
struct uio_ubicom32ring_regs *regs;
|
||
|
struct resource *mem_resource;
|
||
|
struct resource *irqtx_resource;
|
||
|
struct resource *irqrx_resource;
|
||
|
int ret = -EINVAL;
|
||
|
int i;
|
||
|
|
||
|
uioinfo = kzalloc(sizeof(struct uio_info), GFP_KERNEL);
|
||
|
if (!uioinfo) {
|
||
|
dev_err(&pdev->dev, "unable to kmalloc\n");
|
||
|
return -ENOMEM;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Allocate private data with some string space after
|
||
|
*/
|
||
|
i = sizeof(DRIVER_NAME) + 1;
|
||
|
i += pdev->dev.platform_data ? strlen(pdev->dev.platform_data) : 0;
|
||
|
priv = kzalloc(sizeof(struct uio_ubicom32ring_data) + i, GFP_KERNEL);
|
||
|
if (!priv) {
|
||
|
dev_err(&pdev->dev, "unable to kmalloc\n");
|
||
|
kfree(uioinfo);
|
||
|
return -ENOMEM;
|
||
|
}
|
||
|
|
||
|
strcpy(priv->name, DRIVER_NAME ":");
|
||
|
if (pdev->dev.platform_data) {
|
||
|
strcat(priv->name, pdev->dev.platform_data);
|
||
|
}
|
||
|
uioinfo->priv = priv;
|
||
|
uioinfo->name = priv->name;
|
||
|
uioinfo->version = "0.1";
|
||
|
|
||
|
priv->uioinfo = uioinfo;
|
||
|
spin_lock_init(&priv->lock);
|
||
|
priv->flags = 0; /* interrupt is enabled to begin with */
|
||
|
|
||
|
/*
|
||
|
* Get our resources, the IRQ_TX and IRQ_RX are optional.
|
||
|
*/
|
||
|
priv->irq_tx = 0xFF;
|
||
|
irqtx_resource = platform_get_resource(pdev, IORESOURCE_IRQ, 0);
|
||
|
if (irqtx_resource) {
|
||
|
priv->irq_tx = irqtx_resource->start;
|
||
|
}
|
||
|
|
||
|
uioinfo->irq = -1;
|
||
|
priv->irq_rx = 0xFF;
|
||
|
irqrx_resource = platform_get_resource(pdev, IORESOURCE_IRQ, 1);
|
||
|
if (irqrx_resource) {
|
||
|
priv->irq_rx = irqrx_resource->start;
|
||
|
uioinfo->irq = priv->irq_rx;
|
||
|
uioinfo->handler = uio_ubicom32ring_handler;
|
||
|
}
|
||
|
|
||
|
mem_resource = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
||
|
if (!mem_resource || !mem_resource->start) {
|
||
|
dev_err(&pdev->dev, "No valid memory resource found\n");
|
||
|
ret = -ENODEV;
|
||
|
goto fail;
|
||
|
}
|
||
|
regs = (struct uio_ubicom32ring_regs *)mem_resource->start;
|
||
|
priv->regs = regs;
|
||
|
|
||
|
if (regs->version != UIO_UBICOM32RING_REG_VERSION) {
|
||
|
dev_err(&pdev->dev, "version %d not supported\n", regs->version);
|
||
|
ret = -ENODEV;
|
||
|
goto fail;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* First range is the shared register space, if we have any
|
||
|
*/
|
||
|
uiomem = &uioinfo->mem[0];
|
||
|
if (regs->regs_size) {
|
||
|
uiomem->memtype = UIO_MEM_PHYS;
|
||
|
uiomem->addr = (u32_t)regs->regs;
|
||
|
uiomem->size = regs->regs_size;
|
||
|
++uiomem;
|
||
|
dev_info(&pdev->dev, "regs:%p (%u) / rings: %d found\n", regs->regs, regs->regs_size, regs->num_rings);
|
||
|
} else {
|
||
|
dev_info(&pdev->dev, "rings: %d found\n", regs->num_rings);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* The rest of the range correspond to the rings
|
||
|
*/
|
||
|
for (i = 0; i < regs->num_rings; i++) {
|
||
|
dev_info(&pdev->dev, "\t%d: entries:%d ring:%p\n",
|
||
|
i, regs->rings[i]->entries, &(regs->rings[i]->ring));
|
||
|
if (uiomem >= &uioinfo->mem[MAX_UIO_MAPS]) {
|
||
|
dev_warn(&pdev->dev, "device has more than "
|
||
|
__stringify(MAX_UIO_MAPS)
|
||
|
" I/O memory resources.\n");
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
uiomem->memtype = UIO_MEM_PHYS;
|
||
|
uiomem->addr = (u32_t)&(regs->rings[i]->head);
|
||
|
uiomem->size = (regs->rings[i]->entries * sizeof(u32_t)) +
|
||
|
sizeof(struct uio_ubicom32ring_desc);
|
||
|
++uiomem;
|
||
|
}
|
||
|
|
||
|
while (uiomem < &uioinfo->mem[MAX_UIO_MAPS]) {
|
||
|
uiomem->size = 0;
|
||
|
++uiomem;
|
||
|
}
|
||
|
|
||
|
/* This driver requires no hardware specific kernel code to handle
|
||
|
* interrupts. Instead, the interrupt handler simply disables the
|
||
|
* interrupt in the interrupt controller. User space is responsible
|
||
|
* for performing hardware specific acknowledge and re-enabling of
|
||
|
* the interrupt in the interrupt controller.
|
||
|
*
|
||
|
* Interrupt sharing is not supported.
|
||
|
*/
|
||
|
uioinfo->irq_flags = IRQF_DISABLED;
|
||
|
uioinfo->irqcontrol = uio_ubicom32ring_irqcontrol;
|
||
|
|
||
|
ret = uio_register_device(&pdev->dev, priv->uioinfo);
|
||
|
if (ret) {
|
||
|
dev_err(&pdev->dev, "unable to register uio device\n");
|
||
|
goto fail;
|
||
|
}
|
||
|
|
||
|
platform_set_drvdata(pdev, priv);
|
||
|
|
||
|
dev_info(&pdev->dev, "'%s' using irq: rx %d tx %d, regs %p\n",
|
||
|
priv->name, priv->irq_rx, priv->irq_tx, priv->regs);
|
||
|
|
||
|
return 0;
|
||
|
|
||
|
fail:
|
||
|
kfree(uioinfo);
|
||
|
kfree(priv);
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
static int uio_ubicom32ring_remove(struct platform_device *pdev)
|
||
|
{
|
||
|
struct uio_ubicom32ring_data *priv = platform_get_drvdata(pdev);
|
||
|
|
||
|
uio_unregister_device(priv->uioinfo);
|
||
|
kfree(priv->uioinfo);
|
||
|
kfree(priv);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static struct platform_driver uio_ubicom32ring = {
|
||
|
.probe = uio_ubicom32ring_probe,
|
||
|
.remove = uio_ubicom32ring_remove,
|
||
|
.driver = {
|
||
|
.name = DRIVER_NAME,
|
||
|
.owner = THIS_MODULE,
|
||
|
},
|
||
|
};
|
||
|
|
||
|
static int __init uio_ubicom32ring_init(void)
|
||
|
{
|
||
|
return platform_driver_register(&uio_ubicom32ring);
|
||
|
}
|
||
|
|
||
|
static void __exit uio_ubicom32ring_exit(void)
|
||
|
{
|
||
|
platform_driver_unregister(&uio_ubicom32ring);
|
||
|
}
|
||
|
|
||
|
module_init(uio_ubicom32ring_init);
|
||
|
module_exit(uio_ubicom32ring_exit);
|
||
|
|
||
|
MODULE_AUTHOR("Patrick Tjin");
|
||
|
MODULE_DESCRIPTION("Userspace I/O driver for Ubicom32 ring buffers");
|
||
|
MODULE_LICENSE("GPL v2");
|
||
|
MODULE_ALIAS("platform:" DRIVER_NAME);
|