pmu.c 3.0 KB
Newer Older
1 2 3 4
/*
 *  linux/arch/arm/kernel/pmu.c
 *
 *  Copyright (C) 2009 picoChip Designs Ltd, Jamie Iles
5
 *  Copyright (C) 2010 ARM Ltd, Will Deacon
6 7 8 9 10 11 12
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 2 as
 * published by the Free Software Foundation.
 *
 */

13 14
#define pr_fmt(fmt) "PMU: " fmt

15 16 17 18 19
#include <linux/cpumask.h>
#include <linux/err.h>
#include <linux/interrupt.h>
#include <linux/kernel.h>
#include <linux/module.h>
20
#include <linux/platform_device.h>
21 22 23

#include <asm/pmu.h>

24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42
static volatile long pmu_lock;

static struct platform_device *pmu_devices[ARM_NUM_PMU_DEVICES];

static int __devinit pmu_device_probe(struct platform_device *pdev)
{

	if (pdev->id < 0 || pdev->id >= ARM_NUM_PMU_DEVICES) {
		pr_warning("received registration request for unknown "
				"device %d\n", pdev->id);
		return -EINVAL;
	}

	if (pmu_devices[pdev->id])
		pr_warning("registering new PMU device type %d overwrites "
				"previous registration!\n", pdev->id);
	else
		pr_info("registered new PMU device of type %d\n",
				pdev->id);
43

44 45 46 47 48 49 50 51 52
	pmu_devices[pdev->id] = pdev;
	return 0;
}

static struct platform_driver pmu_driver = {
	.driver		= {
		.name	= "arm-pmu",
	},
	.probe		= pmu_device_probe,
53 54
};

55 56 57 58 59
static int __init register_pmu_driver(void)
{
	return platform_driver_register(&pmu_driver);
}
device_initcall(register_pmu_driver);
60

61 62
struct platform_device *
reserve_pmu(enum arm_pmu_type device)
63
{
64 65 66 67 68 69 70 71 72 73 74 75
	struct platform_device *pdev;

	if (test_and_set_bit_lock(device, &pmu_lock)) {
		pdev = ERR_PTR(-EBUSY);
	} else if (pmu_devices[device] == NULL) {
		clear_bit_unlock(device, &pmu_lock);
		pdev = ERR_PTR(-ENODEV);
	} else {
		pdev = pmu_devices[device];
	}

	return pdev;
76 77 78 79
}
EXPORT_SYMBOL_GPL(reserve_pmu);

int
80
release_pmu(struct platform_device *pdev)
81
{
82
	if (WARN_ON(pdev != pmu_devices[pdev->id]))
83
		return -EINVAL;
84
	clear_bit_unlock(pdev->id, &pmu_lock);
85 86 87 88 89 90 91 92 93 94 95 96 97 98 99
	return 0;
}
EXPORT_SYMBOL_GPL(release_pmu);

static int
set_irq_affinity(int irq,
		 unsigned int cpu)
{
#ifdef CONFIG_SMP
	int err = irq_set_affinity(irq, cpumask_of(cpu));
	if (err)
		pr_warning("unable to set irq affinity (irq=%d, cpu=%u)\n",
			   irq, cpu);
	return err;
#else
100
	return -EINVAL;
101 102 103
#endif
}

104 105
static int
init_cpu_pmu(void)
106
{
107
	int i, irqs, err = 0;
108 109
	struct platform_device *pdev = pmu_devices[ARM_PMU_DEVICE_CPU];

110 111 112 113 114 115 116 117 118 119 120
	if (!pdev)
		return -ENODEV;

	irqs = pdev->num_resources;

	/*
	 * If we have a single PMU interrupt that we can't shift, assume that
	 * we're running on a uniprocessor machine and continue.
	 */
	if (irqs == 1 && !irq_can_set_affinity(platform_get_irq(pdev, 0)))
		return 0;
121

122
	for (i = 0; i < irqs; ++i) {
123
		err = set_irq_affinity(platform_get_irq(pdev, i), i);
124 125 126 127
		if (err)
			break;
	}

128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145
	return err;
}

int
init_pmu(enum arm_pmu_type device)
{
	int err = 0;

	switch (device) {
	case ARM_PMU_DEVICE_CPU:
		err = init_cpu_pmu();
		break;
	default:
		pr_warning("attempt to initialise unknown device %d\n",
				device);
		err = -EINVAL;
	}

146 147 148
	return err;
}
EXPORT_SYMBOL_GPL(init_pmu);