Commit 1d64093f authored by Jeremy Kerr's avatar Jeremy Kerr Committed by Paul Mackerras

[POWERPC] cell: register SPUs as sysdevs

SPUs are registered as system devices, exposing attributes through
sysfs. Since the sysdev includes a kref, we can remove the one in
struct spu (it isn't used at the moment anyway).

Currently only the interrupt source and numa node attributes are added.
Signed-off-by: default avatarArnd Bergmann <arnd.bergmann@de.ibm.com>
Signed-off-by: default avatarPaul Mackerras <paulus@samba.org>
parent 0f0f90c3
...@@ -649,6 +649,46 @@ static int __init spu_map_device(struct spu *spu, struct device_node *node) ...@@ -649,6 +649,46 @@ static int __init spu_map_device(struct spu *spu, struct device_node *node)
return ret; return ret;
} }
struct sysdev_class spu_sysdev_class = {
set_kset_name("spu")
};
static ssize_t spu_show_isrc(struct sys_device *sysdev, char *buf)
{
struct spu *spu = container_of(sysdev, struct spu, sysdev);
return sprintf(buf, "%d\n", spu->isrc);
}
static SYSDEV_ATTR(isrc, 0400, spu_show_isrc, NULL);
extern int attach_sysdev_to_node(struct sys_device *dev, int nid);
static int spu_create_sysdev(struct spu *spu)
{
int ret;
spu->sysdev.id = spu->number;
spu->sysdev.cls = &spu_sysdev_class;
ret = sysdev_register(&spu->sysdev);
if (ret) {
printk(KERN_ERR "Can't register SPU %d with sysfs\n",
spu->number);
return ret;
}
sysdev_create_file(&spu->sysdev, &attr_isrc);
sysfs_add_device_to_node(&spu->sysdev, spu->nid);
return 0;
}
static void spu_destroy_sysdev(struct spu *spu)
{
sysdev_remove_file(&spu->sysdev, &attr_isrc);
sysfs_remove_device_from_node(&spu->sysdev, spu->nid);
sysdev_unregister(&spu->sysdev);
}
static int __init create_spu(struct device_node *spe) static int __init create_spu(struct device_node *spe)
{ {
struct spu *spu; struct spu *spu;
...@@ -695,6 +735,10 @@ static int __init create_spu(struct device_node *spe) ...@@ -695,6 +735,10 @@ static int __init create_spu(struct device_node *spe)
if (ret) if (ret)
goto out_unmap; goto out_unmap;
ret = spu_create_sysdev(spu);
if (ret)
goto out_free_irqs;
list_add(&spu->list, &spu_list); list_add(&spu->list, &spu_list);
mutex_unlock(&spu_mutex); mutex_unlock(&spu_mutex);
...@@ -703,6 +747,9 @@ static int __init create_spu(struct device_node *spe) ...@@ -703,6 +747,9 @@ static int __init create_spu(struct device_node *spe)
spu->problem, spu->priv1, spu->priv2, spu->number); spu->problem, spu->priv1, spu->priv2, spu->number);
goto out; goto out;
out_free_irqs:
spu_free_irqs(spu);
out_unmap: out_unmap:
mutex_unlock(&spu_mutex); mutex_unlock(&spu_mutex);
spu_unmap(spu); spu_unmap(spu);
...@@ -716,6 +763,7 @@ static void destroy_spu(struct spu *spu) ...@@ -716,6 +763,7 @@ static void destroy_spu(struct spu *spu)
{ {
list_del_init(&spu->list); list_del_init(&spu->list);
spu_destroy_sysdev(spu);
spu_free_irqs(spu); spu_free_irqs(spu);
spu_unmap(spu); spu_unmap(spu);
kfree(spu); kfree(spu);
...@@ -728,6 +776,7 @@ static void cleanup_spu_base(void) ...@@ -728,6 +776,7 @@ static void cleanup_spu_base(void)
list_for_each_entry_safe(spu, tmp, &spu_list, list) list_for_each_entry_safe(spu, tmp, &spu_list, list)
destroy_spu(spu); destroy_spu(spu);
mutex_unlock(&spu_mutex); mutex_unlock(&spu_mutex);
sysdev_class_unregister(&spu_sysdev_class);
} }
module_exit(cleanup_spu_base); module_exit(cleanup_spu_base);
...@@ -736,6 +785,11 @@ static int __init init_spu_base(void) ...@@ -736,6 +785,11 @@ static int __init init_spu_base(void)
struct device_node *node; struct device_node *node;
int ret; int ret;
/* create sysdev class for spus */
ret = sysdev_class_register(&spu_sysdev_class);
if (ret)
return ret;
ret = -ENODEV; ret = -ENODEV;
for (node = of_find_node_by_type(NULL, "spe"); for (node = of_find_node_by_type(NULL, "spe");
node; node = of_find_node_by_type(node, "spe")) { node; node = of_find_node_by_type(node, "spe")) {
......
...@@ -25,8 +25,8 @@ ...@@ -25,8 +25,8 @@
#ifdef __KERNEL__ #ifdef __KERNEL__
#include <linux/config.h> #include <linux/config.h>
#include <linux/kref.h>
#include <linux/workqueue.h> #include <linux/workqueue.h>
#include <linux/sysdev.h>
#define LS_SIZE (256 * 1024) #define LS_SIZE (256 * 1024)
#define LS_ADDR_MASK (LS_SIZE - 1) #define LS_ADDR_MASK (LS_SIZE - 1)
...@@ -123,7 +123,6 @@ struct spu { ...@@ -123,7 +123,6 @@ struct spu {
u64 flags; u64 flags;
u64 dar; u64 dar;
u64 dsisr; u64 dsisr;
struct kref kref;
size_t ls_size; size_t ls_size;
unsigned int slb_replace; unsigned int slb_replace;
struct mm_struct *mm; struct mm_struct *mm;
...@@ -144,6 +143,8 @@ struct spu { ...@@ -144,6 +143,8 @@ struct spu {
char irq_c0[8]; char irq_c0[8];
char irq_c1[8]; char irq_c1[8];
char irq_c2[8]; char irq_c2[8];
struct sys_device sysdev;
}; };
struct spu *spu_alloc(void); struct spu *spu_alloc(void);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment