linux/arch/mips/kernel/vpe-cmp.c
<<
>>
Prefs
   1/*
   2 * This file is subject to the terms and conditions of the GNU General Public
   3 * License.  See the file "COPYING" in the main directory of this archive
   4 * for more details.
   5 *
   6 * Copyright (C) 2004, 2005 MIPS Technologies, Inc.  All rights reserved.
   7 * Copyright (C) 2013 Imagination Technologies Ltd.
   8 */
   9#include <linux/kernel.h>
  10#include <linux/device.h>
  11#include <linux/fs.h>
  12#include <linux/slab.h>
  13#include <linux/export.h>
  14
  15#include <asm/vpe.h>
  16
  17static int major;
  18
  19void cleanup_tc(struct tc *tc)
  20{
  21
  22}
  23
  24static ssize_t store_kill(struct device *dev, struct device_attribute *attr,
  25                          const char *buf, size_t len)
  26{
  27        struct vpe *vpe = get_vpe(aprp_cpu_index());
  28        struct vpe_notifications *notifier;
  29
  30        list_for_each_entry(notifier, &vpe->notify, list)
  31                notifier->stop(aprp_cpu_index());
  32
  33        release_progmem(vpe->load_addr);
  34        vpe->state = VPE_STATE_UNUSED;
  35
  36        return len;
  37}
  38static DEVICE_ATTR(kill, S_IWUSR, NULL, store_kill);
  39
  40static ssize_t ntcs_show(struct device *cd, struct device_attribute *attr,
  41                         char *buf)
  42{
  43        struct vpe *vpe = get_vpe(aprp_cpu_index());
  44
  45        return sprintf(buf, "%d\n", vpe->ntcs);
  46}
  47
  48static ssize_t ntcs_store(struct device *dev, struct device_attribute *attr,
  49                          const char *buf, size_t len)
  50{
  51        struct vpe *vpe = get_vpe(aprp_cpu_index());
  52        unsigned long new;
  53        int ret;
  54
  55        ret = kstrtoul(buf, 0, &new);
  56        if (ret < 0)
  57                return ret;
  58
  59        /* APRP can only reserve one TC in a VPE and no more. */
  60        if (new != 1)
  61                return -EINVAL;
  62
  63        vpe->ntcs = new;
  64
  65        return len;
  66}
  67static DEVICE_ATTR_RW(ntcs);
  68
  69static struct attribute *vpe_attrs[] = {
  70        &dev_attr_kill.attr,
  71        &dev_attr_ntcs.attr,
  72        NULL,
  73};
  74ATTRIBUTE_GROUPS(vpe);
  75
  76static void vpe_device_release(struct device *cd)
  77{
  78        kfree(cd);
  79}
  80
  81static struct class vpe_class = {
  82        .name = "vpe",
  83        .owner = THIS_MODULE,
  84        .dev_release = vpe_device_release,
  85        .dev_groups = vpe_groups,
  86};
  87
  88static struct device vpe_device;
  89
  90int __init vpe_module_init(void)
  91{
  92        struct vpe *v = NULL;
  93        struct tc *t;
  94        int err;
  95
  96        if (!cpu_has_mipsmt) {
  97                pr_warn("VPE loader: not a MIPS MT capable processor\n");
  98                return -ENODEV;
  99        }
 100
 101        if (num_possible_cpus() - aprp_cpu_index() < 1) {
 102                pr_warn("No VPEs reserved for AP/SP, not initialize VPE loader\n"
 103                        "Pass maxcpus=<n> argument as kernel argument\n");
 104                return -ENODEV;
 105        }
 106
 107        major = register_chrdev(0, VPE_MODULE_NAME, &vpe_fops);
 108        if (major < 0) {
 109                pr_warn("VPE loader: unable to register character device\n");
 110                return major;
 111        }
 112
 113        err = class_register(&vpe_class);
 114        if (err) {
 115                pr_err("vpe_class registration failed\n");
 116                goto out_chrdev;
 117        }
 118
 119        device_initialize(&vpe_device);
 120        vpe_device.class        = &vpe_class,
 121        vpe_device.parent       = NULL,
 122        dev_set_name(&vpe_device, "vpe_sp");
 123        vpe_device.devt = MKDEV(major, VPE_MODULE_MINOR);
 124        err = device_add(&vpe_device);
 125        if (err) {
 126                pr_err("Adding vpe_device failed\n");
 127                goto out_class;
 128        }
 129
 130        t = alloc_tc(aprp_cpu_index());
 131        if (!t) {
 132                pr_warn("VPE: unable to allocate TC\n");
 133                err = -ENOMEM;
 134                goto out_dev;
 135        }
 136
 137        /* VPE */
 138        v = alloc_vpe(aprp_cpu_index());
 139        if (v == NULL) {
 140                pr_warn("VPE: unable to allocate VPE\n");
 141                kfree(t);
 142                err = -ENOMEM;
 143                goto out_dev;
 144        }
 145
 146        v->ntcs = 1;
 147
 148        /* add the tc to the list of this vpe's tc's. */
 149        list_add(&t->tc, &v->tc);
 150
 151        /* TC */
 152        t->pvpe = v;    /* set the parent vpe */
 153
 154        return 0;
 155
 156out_dev:
 157        device_del(&vpe_device);
 158
 159out_class:
 160        class_unregister(&vpe_class);
 161
 162out_chrdev:
 163        unregister_chrdev(major, VPE_MODULE_NAME);
 164
 165        return err;
 166}
 167
 168void __exit vpe_module_exit(void)
 169{
 170        struct vpe *v, *n;
 171
 172        device_del(&vpe_device);
 173        class_unregister(&vpe_class);
 174        unregister_chrdev(major, VPE_MODULE_NAME);
 175
 176        /* No locking needed here */
 177        list_for_each_entry_safe(v, n, &vpecontrol.vpe_list, list)
 178                if (v->state != VPE_STATE_UNUSED)
 179                        release_vpe(v);
 180}
 181