|
@@ -0,0 +1,180 @@
|
|
|
+/*
|
|
|
+ * This file is subject to the terms and conditions of the GNU General Public
|
|
|
+ * License. See the file "COPYING" in the main directory of this archive
|
|
|
+ * for more details.
|
|
|
+ *
|
|
|
+ * Copyright (C) 2004, 2005 MIPS Technologies, Inc. All rights reserved.
|
|
|
+ * Copyright (C) 2013 Imagination Technologies Ltd.
|
|
|
+ */
|
|
|
+#include <linux/kernel.h>
|
|
|
+#include <linux/device.h>
|
|
|
+#include <linux/fs.h>
|
|
|
+#include <linux/slab.h>
|
|
|
+#include <linux/export.h>
|
|
|
+
|
|
|
+#include <asm/vpe.h>
|
|
|
+
|
|
|
+static int major;
|
|
|
+
|
|
|
+void cleanup_tc(struct tc *tc)
|
|
|
+{
|
|
|
+
|
|
|
+}
|
|
|
+
|
|
|
+static ssize_t store_kill(struct device *dev, struct device_attribute *attr,
|
|
|
+ const char *buf, size_t len)
|
|
|
+{
|
|
|
+ struct vpe *vpe = get_vpe(aprp_cpu_index());
|
|
|
+ struct vpe_notifications *notifier;
|
|
|
+
|
|
|
+ list_for_each_entry(notifier, &vpe->notify, list)
|
|
|
+ notifier->stop(aprp_cpu_index());
|
|
|
+
|
|
|
+ release_progmem(vpe->load_addr);
|
|
|
+ vpe->state = VPE_STATE_UNUSED;
|
|
|
+
|
|
|
+ return len;
|
|
|
+}
|
|
|
+static DEVICE_ATTR(kill, S_IWUSR, NULL, store_kill);
|
|
|
+
|
|
|
+static ssize_t ntcs_show(struct device *cd, struct device_attribute *attr,
|
|
|
+ char *buf)
|
|
|
+{
|
|
|
+ struct vpe *vpe = get_vpe(aprp_cpu_index());
|
|
|
+
|
|
|
+ return sprintf(buf, "%d\n", vpe->ntcs);
|
|
|
+}
|
|
|
+
|
|
|
+static ssize_t ntcs_store(struct device *dev, struct device_attribute *attr,
|
|
|
+ const char *buf, size_t len)
|
|
|
+{
|
|
|
+ struct vpe *vpe = get_vpe(aprp_cpu_index());
|
|
|
+ unsigned long new;
|
|
|
+ int ret;
|
|
|
+
|
|
|
+ ret = kstrtoul(buf, 0, &new);
|
|
|
+ if (ret < 0)
|
|
|
+ return ret;
|
|
|
+
|
|
|
+ /* APRP can only reserve one TC in a VPE and no more. */
|
|
|
+ if (new != 1)
|
|
|
+ return -EINVAL;
|
|
|
+
|
|
|
+ vpe->ntcs = new;
|
|
|
+
|
|
|
+ return len;
|
|
|
+}
|
|
|
+static DEVICE_ATTR_RW(ntcs);
|
|
|
+
|
|
|
+static struct attribute *vpe_attrs[] = {
|
|
|
+ &dev_attr_kill.attr,
|
|
|
+ &dev_attr_ntcs.attr,
|
|
|
+ NULL,
|
|
|
+};
|
|
|
+ATTRIBUTE_GROUPS(vpe);
|
|
|
+
|
|
|
+static void vpe_device_release(struct device *cd)
|
|
|
+{
|
|
|
+ kfree(cd);
|
|
|
+}
|
|
|
+
|
|
|
+static struct class vpe_class = {
|
|
|
+ .name = "vpe",
|
|
|
+ .owner = THIS_MODULE,
|
|
|
+ .dev_release = vpe_device_release,
|
|
|
+ .dev_groups = vpe_groups,
|
|
|
+};
|
|
|
+
|
|
|
+static struct device vpe_device;
|
|
|
+
|
|
|
+int __init vpe_module_init(void)
|
|
|
+{
|
|
|
+ struct vpe *v = NULL;
|
|
|
+ struct tc *t;
|
|
|
+ int err;
|
|
|
+
|
|
|
+ if (!cpu_has_mipsmt) {
|
|
|
+ pr_warn("VPE loader: not a MIPS MT capable processor\n");
|
|
|
+ return -ENODEV;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (num_possible_cpus() - aprp_cpu_index() < 1) {
|
|
|
+ pr_warn("No VPEs reserved for AP/SP, not initialize VPE loader\n"
|
|
|
+ "Pass maxcpus=<n> argument as kernel argument\n");
|
|
|
+ return -ENODEV;
|
|
|
+ }
|
|
|
+
|
|
|
+ major = register_chrdev(0, VPE_MODULE_NAME, &vpe_fops);
|
|
|
+ if (major < 0) {
|
|
|
+ pr_warn("VPE loader: unable to register character device\n");
|
|
|
+ return major;
|
|
|
+ }
|
|
|
+
|
|
|
+ err = class_register(&vpe_class);
|
|
|
+ if (err) {
|
|
|
+ pr_err("vpe_class registration failed\n");
|
|
|
+ goto out_chrdev;
|
|
|
+ }
|
|
|
+
|
|
|
+ device_initialize(&vpe_device);
|
|
|
+ vpe_device.class = &vpe_class,
|
|
|
+ vpe_device.parent = NULL,
|
|
|
+ dev_set_name(&vpe_device, "vpe_sp");
|
|
|
+ vpe_device.devt = MKDEV(major, VPE_MODULE_MINOR);
|
|
|
+ err = device_add(&vpe_device);
|
|
|
+ if (err) {
|
|
|
+ pr_err("Adding vpe_device failed\n");
|
|
|
+ goto out_class;
|
|
|
+ }
|
|
|
+
|
|
|
+ t = alloc_tc(aprp_cpu_index());
|
|
|
+ if (!t) {
|
|
|
+ pr_warn("VPE: unable to allocate TC\n");
|
|
|
+ err = -ENOMEM;
|
|
|
+ goto out_dev;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* VPE */
|
|
|
+ v = alloc_vpe(aprp_cpu_index());
|
|
|
+ if (v == NULL) {
|
|
|
+ pr_warn("VPE: unable to allocate VPE\n");
|
|
|
+ kfree(t);
|
|
|
+ err = -ENOMEM;
|
|
|
+ goto out_dev;
|
|
|
+ }
|
|
|
+
|
|
|
+ v->ntcs = 1;
|
|
|
+
|
|
|
+ /* add the tc to the list of this vpe's tc's. */
|
|
|
+ list_add(&t->tc, &v->tc);
|
|
|
+
|
|
|
+ /* TC */
|
|
|
+ t->pvpe = v; /* set the parent vpe */
|
|
|
+
|
|
|
+ return 0;
|
|
|
+
|
|
|
+out_dev:
|
|
|
+ device_del(&vpe_device);
|
|
|
+
|
|
|
+out_class:
|
|
|
+ class_unregister(&vpe_class);
|
|
|
+
|
|
|
+out_chrdev:
|
|
|
+ unregister_chrdev(major, VPE_MODULE_NAME);
|
|
|
+
|
|
|
+ return err;
|
|
|
+}
|
|
|
+
|
|
|
+void __exit vpe_module_exit(void)
|
|
|
+{
|
|
|
+ struct vpe *v, *n;
|
|
|
+
|
|
|
+ device_del(&vpe_device);
|
|
|
+ class_unregister(&vpe_class);
|
|
|
+ unregister_chrdev(major, VPE_MODULE_NAME);
|
|
|
+
|
|
|
+ /* No locking needed here */
|
|
|
+ list_for_each_entry_safe(v, n, &vpecontrol.vpe_list, list)
|
|
|
+ if (v->state != VPE_STATE_UNUSED)
|
|
|
+ release_vpe(v);
|
|
|
+}
|