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