MIPS: vpe-cmp: remove module owner pointer from struct class usage.
[platform/kernel/linux-starfive.git] / arch / mips / kernel / vpe-cmp.c
1 /*
2  * This file is subject to the terms and conditions of the GNU General Public
3  * License.  See the file "COPYING" in the main directory of this archive
4  * for more details.
5  *
6  * Copyright (C) 2004, 2005 MIPS Technologies, Inc.  All rights reserved.
7  * Copyright (C) 2013 Imagination Technologies Ltd.
8  */
9 #include <linux/kernel.h>
10 #include <linux/device.h>
11 #include <linux/fs.h>
12 #include <linux/slab.h>
13 #include <linux/export.h>
14
15 #include <asm/vpe.h>
16
17 static int major;
18
19 void cleanup_tc(struct tc *tc)
20 {
21
22 }
23
24 static ssize_t store_kill(struct device *dev, struct device_attribute *attr,
25                           const char *buf, size_t len)
26 {
27         struct vpe *vpe = get_vpe(aprp_cpu_index());
28         struct vpe_notifications *notifier;
29
30         list_for_each_entry(notifier, &vpe->notify, list)
31                 notifier->stop(aprp_cpu_index());
32
33         release_progmem(vpe->load_addr);
34         vpe->state = VPE_STATE_UNUSED;
35
36         return len;
37 }
38 static DEVICE_ATTR(kill, S_IWUSR, NULL, store_kill);
39
40 static ssize_t ntcs_show(struct device *cd, struct device_attribute *attr,
41                          char *buf)
42 {
43         struct vpe *vpe = get_vpe(aprp_cpu_index());
44
45         return sprintf(buf, "%d\n", vpe->ntcs);
46 }
47
48 static ssize_t ntcs_store(struct device *dev, struct device_attribute *attr,
49                           const char *buf, size_t len)
50 {
51         struct vpe *vpe = get_vpe(aprp_cpu_index());
52         unsigned long new;
53         int ret;
54
55         ret = kstrtoul(buf, 0, &new);
56         if (ret < 0)
57                 return ret;
58
59         /* APRP can only reserve one TC in a VPE and no more. */
60         if (new != 1)
61                 return -EINVAL;
62
63         vpe->ntcs = new;
64
65         return len;
66 }
67 static DEVICE_ATTR_RW(ntcs);
68
69 static struct attribute *vpe_attrs[] = {
70         &dev_attr_kill.attr,
71         &dev_attr_ntcs.attr,
72         NULL,
73 };
74 ATTRIBUTE_GROUPS(vpe);
75
76 static void vpe_device_release(struct device *cd)
77 {
78 }
79
80 static struct class vpe_class = {
81         .name = "vpe",
82         .dev_release = vpe_device_release,
83         .dev_groups = vpe_groups,
84 };
85
86 static struct device vpe_device;
87
88 int __init vpe_module_init(void)
89 {
90         struct vpe *v = NULL;
91         struct tc *t;
92         int err;
93
94         if (!cpu_has_mipsmt) {
95                 pr_warn("VPE loader: not a MIPS MT capable processor\n");
96                 return -ENODEV;
97         }
98
99         if (num_possible_cpus() - aprp_cpu_index() < 1) {
100                 pr_warn("No VPEs reserved for AP/SP, not initialize VPE loader\n"
101                         "Pass maxcpus=<n> argument as kernel argument\n");
102                 return -ENODEV;
103         }
104
105         major = register_chrdev(0, VPE_MODULE_NAME, &vpe_fops);
106         if (major < 0) {
107                 pr_warn("VPE loader: unable to register character device\n");
108                 return major;
109         }
110
111         err = class_register(&vpe_class);
112         if (err) {
113                 pr_err("vpe_class registration failed\n");
114                 goto out_chrdev;
115         }
116
117         device_initialize(&vpe_device);
118         vpe_device.class        = &vpe_class;
119         vpe_device.parent       = NULL;
120         dev_set_name(&vpe_device, "vpe_sp");
121         vpe_device.devt = MKDEV(major, VPE_MODULE_MINOR);
122         err = device_add(&vpe_device);
123         if (err) {
124                 pr_err("Adding vpe_device failed\n");
125                 goto out_class;
126         }
127
128         t = alloc_tc(aprp_cpu_index());
129         if (!t) {
130                 pr_warn("VPE: unable to allocate TC\n");
131                 err = -ENOMEM;
132                 goto out_dev;
133         }
134
135         /* VPE */
136         v = alloc_vpe(aprp_cpu_index());
137         if (v == NULL) {
138                 pr_warn("VPE: unable to allocate VPE\n");
139                 kfree(t);
140                 err = -ENOMEM;
141                 goto out_dev;
142         }
143
144         v->ntcs = 1;
145
146         /* add the tc to the list of this vpe's tc's. */
147         list_add(&t->tc, &v->tc);
148
149         /* TC */
150         t->pvpe = v;    /* set the parent vpe */
151
152         return 0;
153
154 out_dev:
155         device_del(&vpe_device);
156
157 out_class:
158         put_device(&vpe_device);
159         class_unregister(&vpe_class);
160
161 out_chrdev:
162         unregister_chrdev(major, VPE_MODULE_NAME);
163
164         return err;
165 }
166
167 void __exit vpe_module_exit(void)
168 {
169         struct vpe *v, *n;
170
171         device_unregister(&vpe_device);
172         class_unregister(&vpe_class);
173         unregister_chrdev(major, VPE_MODULE_NAME);
174
175         /* No locking needed here */
176         list_for_each_entry_safe(v, n, &vpecontrol.vpe_list, list)
177                 if (v->state != VPE_STATE_UNUSED)
178                         release_vpe(v);
179 }