123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180 |
- /*
- * 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);
- }
|