181 lines
		
	
	
	
		
			3.6 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			181 lines
		
	
	
	
		
			3.6 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
| 
								 | 
							
								/*
							 | 
						||
| 
								 | 
							
								 * 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);
							 | 
						||
| 
								 | 
							
								}
							 |