mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-01 10:42:11 +00:00
c3c1250e93
hwspinlock devices provide system-wide hardware locks that are used by remote processors that have no other way to achieve synchronization. To achieve that, each physical lock must have a system-wide id number that is agreed upon, otherwise remote processors can't possibly assume they're using the same hardware lock. Usually boards have a single hwspinlock device, which provides several hwspinlocks, and in this case, they can be trivially numbered 0 to (num-of-locks - 1). In case boards have several hwspinlocks devices, a different base id should be used for each hwspinlock device (they can't all use 0 as a starting id!). While this is certainly not common, it's just plain wrong to just silently use 0 as a base id whenever the hwspinlock driver is probed. This patch provides a hwspinlock_pdata structure, that boards can use to set a different base id for each of the hwspinlock devices they may have, and demonstrates how to use it with the omap hwspinlock driver. While we're at it, make sure the hwspinlock core prints an explicit error message in case an hwspinlock is registered with an id number that already exists; this will help users catch such base id issues. Reported-by: Arnd Bergmann <arnd@arndb.de> Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com> Acked-by: Tony Lindgren <tony@atomide.com>
219 lines
5.5 KiB
C
219 lines
5.5 KiB
C
/*
|
|
* OMAP hardware spinlock driver
|
|
*
|
|
* Copyright (C) 2010 Texas Instruments Incorporated - http://www.ti.com
|
|
*
|
|
* Contact: Simon Que <sque@ti.com>
|
|
* Hari Kanigeri <h-kanigeri2@ti.com>
|
|
* Ohad Ben-Cohen <ohad@wizery.com>
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* version 2 as published by the Free Software Foundation.
|
|
*
|
|
* This program 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.
|
|
*/
|
|
|
|
#include <linux/kernel.h>
|
|
#include <linux/module.h>
|
|
#include <linux/device.h>
|
|
#include <linux/delay.h>
|
|
#include <linux/io.h>
|
|
#include <linux/bitops.h>
|
|
#include <linux/pm_runtime.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/spinlock.h>
|
|
#include <linux/hwspinlock.h>
|
|
#include <linux/platform_device.h>
|
|
|
|
#include "hwspinlock_internal.h"
|
|
|
|
/* Spinlock register offsets */
|
|
#define SYSSTATUS_OFFSET 0x0014
|
|
#define LOCK_BASE_OFFSET 0x0800
|
|
|
|
#define SPINLOCK_NUMLOCKS_BIT_OFFSET (24)
|
|
|
|
/* Possible values of SPINLOCK_LOCK_REG */
|
|
#define SPINLOCK_NOTTAKEN (0) /* free */
|
|
#define SPINLOCK_TAKEN (1) /* locked */
|
|
|
|
#define to_omap_hwspinlock(lock) \
|
|
container_of(lock, struct omap_hwspinlock, lock)
|
|
|
|
struct omap_hwspinlock {
|
|
struct hwspinlock lock;
|
|
void __iomem *addr;
|
|
};
|
|
|
|
struct omap_hwspinlock_state {
|
|
int num_locks; /* Total number of locks in system */
|
|
void __iomem *io_base; /* Mapped base address */
|
|
struct omap_hwspinlock lock[0]; /* Array of 'num_locks' locks */
|
|
};
|
|
|
|
static int omap_hwspinlock_trylock(struct hwspinlock *lock)
|
|
{
|
|
struct omap_hwspinlock *omap_lock = to_omap_hwspinlock(lock);
|
|
|
|
/* attempt to acquire the lock by reading its value */
|
|
return (SPINLOCK_NOTTAKEN == readl(omap_lock->addr));
|
|
}
|
|
|
|
static void omap_hwspinlock_unlock(struct hwspinlock *lock)
|
|
{
|
|
struct omap_hwspinlock *omap_lock = to_omap_hwspinlock(lock);
|
|
|
|
/* release the lock by writing 0 to it */
|
|
writel(SPINLOCK_NOTTAKEN, omap_lock->addr);
|
|
}
|
|
|
|
/*
|
|
* relax the OMAP interconnect while spinning on it.
|
|
*
|
|
* The specs recommended that the retry delay time will be
|
|
* just over half of the time that a requester would be
|
|
* expected to hold the lock.
|
|
*
|
|
* The number below is taken from an hardware specs example,
|
|
* obviously it is somewhat arbitrary.
|
|
*/
|
|
static void omap_hwspinlock_relax(struct hwspinlock *lock)
|
|
{
|
|
ndelay(50);
|
|
}
|
|
|
|
static const struct hwspinlock_ops omap_hwspinlock_ops = {
|
|
.trylock = omap_hwspinlock_trylock,
|
|
.unlock = omap_hwspinlock_unlock,
|
|
.relax = omap_hwspinlock_relax,
|
|
};
|
|
|
|
static int __devinit omap_hwspinlock_probe(struct platform_device *pdev)
|
|
{
|
|
struct hwspinlock_pdata *pdata = pdev->dev.platform_data;
|
|
struct omap_hwspinlock *omap_lock;
|
|
struct omap_hwspinlock_state *state;
|
|
struct resource *res;
|
|
void __iomem *io_base;
|
|
int i, ret;
|
|
|
|
if (!pdata)
|
|
return -ENODEV;
|
|
|
|
res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
|
if (!res)
|
|
return -ENODEV;
|
|
|
|
io_base = ioremap(res->start, resource_size(res));
|
|
if (!io_base)
|
|
return -ENOMEM;
|
|
|
|
/* Determine number of locks */
|
|
i = readl(io_base + SYSSTATUS_OFFSET);
|
|
i >>= SPINLOCK_NUMLOCKS_BIT_OFFSET;
|
|
|
|
/* one of the four lsb's must be set, and nothing else */
|
|
if (hweight_long(i & 0xf) != 1 || i > 8) {
|
|
ret = -EINVAL;
|
|
goto iounmap_base;
|
|
}
|
|
|
|
i *= 32; /* actual number of locks in this device */
|
|
|
|
state = kzalloc(sizeof(*state) + i * sizeof(*omap_lock), GFP_KERNEL);
|
|
if (!state) {
|
|
ret = -ENOMEM;
|
|
goto iounmap_base;
|
|
}
|
|
|
|
state->num_locks = i;
|
|
state->io_base = io_base;
|
|
|
|
platform_set_drvdata(pdev, state);
|
|
|
|
/*
|
|
* runtime PM will make sure the clock of this module is
|
|
* enabled iff at least one lock is requested
|
|
*/
|
|
pm_runtime_enable(&pdev->dev);
|
|
|
|
for (i = 0; i < state->num_locks; i++) {
|
|
omap_lock = &state->lock[i];
|
|
|
|
omap_lock->lock.dev = &pdev->dev;
|
|
omap_lock->lock.id = pdata->base_id + i;
|
|
omap_lock->lock.ops = &omap_hwspinlock_ops;
|
|
omap_lock->addr = io_base + LOCK_BASE_OFFSET + sizeof(u32) * i;
|
|
|
|
ret = hwspin_lock_register(&omap_lock->lock);
|
|
if (ret)
|
|
goto free_locks;
|
|
}
|
|
|
|
return 0;
|
|
|
|
free_locks:
|
|
while (--i >= 0)
|
|
hwspin_lock_unregister(i);
|
|
pm_runtime_disable(&pdev->dev);
|
|
kfree(state);
|
|
iounmap_base:
|
|
iounmap(io_base);
|
|
return ret;
|
|
}
|
|
|
|
static int omap_hwspinlock_remove(struct platform_device *pdev)
|
|
{
|
|
struct omap_hwspinlock_state *state = platform_get_drvdata(pdev);
|
|
struct hwspinlock *lock;
|
|
int i;
|
|
|
|
for (i = 0; i < state->num_locks; i++) {
|
|
lock = hwspin_lock_unregister(i);
|
|
/* this shouldn't happen at this point. if it does, at least
|
|
* don't continue with the remove */
|
|
if (!lock) {
|
|
dev_err(&pdev->dev, "%s: failed on %d\n", __func__, i);
|
|
return -EBUSY;
|
|
}
|
|
}
|
|
|
|
pm_runtime_disable(&pdev->dev);
|
|
iounmap(state->io_base);
|
|
kfree(state);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static struct platform_driver omap_hwspinlock_driver = {
|
|
.probe = omap_hwspinlock_probe,
|
|
.remove = omap_hwspinlock_remove,
|
|
.driver = {
|
|
.name = "omap_hwspinlock",
|
|
.owner = THIS_MODULE,
|
|
},
|
|
};
|
|
|
|
static int __init omap_hwspinlock_init(void)
|
|
{
|
|
return platform_driver_register(&omap_hwspinlock_driver);
|
|
}
|
|
/* board init code might need to reserve hwspinlocks for predefined purposes */
|
|
postcore_initcall(omap_hwspinlock_init);
|
|
|
|
static void __exit omap_hwspinlock_exit(void)
|
|
{
|
|
platform_driver_unregister(&omap_hwspinlock_driver);
|
|
}
|
|
module_exit(omap_hwspinlock_exit);
|
|
|
|
MODULE_LICENSE("GPL v2");
|
|
MODULE_DESCRIPTION("Hardware spinlock driver for OMAP");
|
|
MODULE_AUTHOR("Simon Que <sque@ti.com>");
|
|
MODULE_AUTHOR("Hari Kanigeri <h-kanigeri2@ti.com>");
|
|
MODULE_AUTHOR("Ohad Ben-Cohen <ohad@wizery.com>");
|