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