mirror of
https://github.com/edk2-porting/linux-next.git
synced 2025-01-15 00:54:03 +08:00
1acda878e2
The new ring buffer implementation allows the storage of samples with different size. This patch implements the usage of the new sample format to store ibs samples in the cpu buffer. Until now, writing to the cpu buffer could lead to incomplete sampling sequences since IBS samples were transfered in multiple samples. Due to a full buffer, data could be lost at any time. This can't happen any more since the complete data is reserved in advance and then stored in a single sample. Signed-off-by: Robert Richter <robert.richter@amd.com>
514 lines
12 KiB
C
514 lines
12 KiB
C
/*
|
|
* @file op_model_amd.c
|
|
* athlon / K7 / K8 / Family 10h model-specific MSR operations
|
|
*
|
|
* @remark Copyright 2002-2009 OProfile authors
|
|
* @remark Read the file COPYING
|
|
*
|
|
* @author John Levon
|
|
* @author Philippe Elie
|
|
* @author Graydon Hoare
|
|
* @author Robert Richter <robert.richter@amd.com>
|
|
* @author Barry Kasindorf
|
|
*/
|
|
|
|
#include <linux/oprofile.h>
|
|
#include <linux/device.h>
|
|
#include <linux/pci.h>
|
|
|
|
#include <asm/ptrace.h>
|
|
#include <asm/msr.h>
|
|
#include <asm/nmi.h>
|
|
|
|
#include "op_x86_model.h"
|
|
#include "op_counter.h"
|
|
#include "../../../drivers/oprofile/cpu_buffer.h"
|
|
|
|
#define NUM_COUNTERS 4
|
|
#define NUM_CONTROLS 4
|
|
|
|
#define CTR_IS_RESERVED(msrs, c) (msrs->counters[(c)].addr ? 1 : 0)
|
|
#define CTR_READ(l, h, msrs, c) do {rdmsr(msrs->counters[(c)].addr, (l), (h)); } while (0)
|
|
#define CTR_WRITE(l, msrs, c) do {wrmsr(msrs->counters[(c)].addr, -(unsigned int)(l), -1); } while (0)
|
|
#define CTR_OVERFLOWED(n) (!((n) & (1U<<31)))
|
|
|
|
#define CTRL_IS_RESERVED(msrs, c) (msrs->controls[(c)].addr ? 1 : 0)
|
|
#define CTRL_READ(l, h, msrs, c) do {rdmsr(msrs->controls[(c)].addr, (l), (h)); } while (0)
|
|
#define CTRL_WRITE(l, h, msrs, c) do {wrmsr(msrs->controls[(c)].addr, (l), (h)); } while (0)
|
|
#define CTRL_SET_ACTIVE(n) (n |= (1<<22))
|
|
#define CTRL_SET_INACTIVE(n) (n &= ~(1<<22))
|
|
#define CTRL_CLEAR_LO(x) (x &= (1<<21))
|
|
#define CTRL_CLEAR_HI(x) (x &= 0xfffffcf0)
|
|
#define CTRL_SET_ENABLE(val) (val |= 1<<20)
|
|
#define CTRL_SET_USR(val, u) (val |= ((u & 1) << 16))
|
|
#define CTRL_SET_KERN(val, k) (val |= ((k & 1) << 17))
|
|
#define CTRL_SET_UM(val, m) (val |= (m << 8))
|
|
#define CTRL_SET_EVENT_LOW(val, e) (val |= (e & 0xff))
|
|
#define CTRL_SET_EVENT_HIGH(val, e) (val |= ((e >> 8) & 0xf))
|
|
#define CTRL_SET_HOST_ONLY(val, h) (val |= ((h & 1) << 9))
|
|
#define CTRL_SET_GUEST_ONLY(val, h) (val |= ((h & 1) << 8))
|
|
|
|
static unsigned long reset_value[NUM_COUNTERS];
|
|
|
|
#ifdef CONFIG_OPROFILE_IBS
|
|
|
|
/* IbsFetchCtl bits/masks */
|
|
#define IBS_FETCH_HIGH_VALID_BIT (1UL << 17) /* bit 49 */
|
|
#define IBS_FETCH_HIGH_ENABLE (1UL << 16) /* bit 48 */
|
|
#define IBS_FETCH_LOW_MAX_CNT_MASK 0x0000FFFFUL /* MaxCnt mask */
|
|
|
|
/*IbsOpCtl bits */
|
|
#define IBS_OP_LOW_VALID_BIT (1ULL<<18) /* bit 18 */
|
|
#define IBS_OP_LOW_ENABLE (1ULL<<17) /* bit 17 */
|
|
|
|
/*
|
|
* The function interface needs to be fixed, something like add
|
|
* data. Should then be added to linux/oprofile.h.
|
|
*/
|
|
extern
|
|
void oprofile_add_data(struct op_entry *entry, struct pt_regs * const regs,
|
|
unsigned long pc, int code, int size);
|
|
|
|
#define IBS_FETCH_SIZE 6
|
|
#define IBS_OP_SIZE 12
|
|
|
|
static int has_ibs; /* AMD Family10h and later */
|
|
|
|
struct op_ibs_config {
|
|
unsigned long op_enabled;
|
|
unsigned long fetch_enabled;
|
|
unsigned long max_cnt_fetch;
|
|
unsigned long max_cnt_op;
|
|
unsigned long rand_en;
|
|
unsigned long dispatched_ops;
|
|
};
|
|
|
|
static struct op_ibs_config ibs_config;
|
|
|
|
#endif
|
|
|
|
/* functions for op_amd_spec */
|
|
|
|
static void op_amd_fill_in_addresses(struct op_msrs * const msrs)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < NUM_COUNTERS; i++) {
|
|
if (reserve_perfctr_nmi(MSR_K7_PERFCTR0 + i))
|
|
msrs->counters[i].addr = MSR_K7_PERFCTR0 + i;
|
|
else
|
|
msrs->counters[i].addr = 0;
|
|
}
|
|
|
|
for (i = 0; i < NUM_CONTROLS; i++) {
|
|
if (reserve_evntsel_nmi(MSR_K7_EVNTSEL0 + i))
|
|
msrs->controls[i].addr = MSR_K7_EVNTSEL0 + i;
|
|
else
|
|
msrs->controls[i].addr = 0;
|
|
}
|
|
}
|
|
|
|
|
|
static void op_amd_setup_ctrs(struct op_msrs const * const msrs)
|
|
{
|
|
unsigned int low, high;
|
|
int i;
|
|
|
|
/* clear all counters */
|
|
for (i = 0 ; i < NUM_CONTROLS; ++i) {
|
|
if (unlikely(!CTRL_IS_RESERVED(msrs, i)))
|
|
continue;
|
|
CTRL_READ(low, high, msrs, i);
|
|
CTRL_CLEAR_LO(low);
|
|
CTRL_CLEAR_HI(high);
|
|
CTRL_WRITE(low, high, msrs, i);
|
|
}
|
|
|
|
/* avoid a false detection of ctr overflows in NMI handler */
|
|
for (i = 0; i < NUM_COUNTERS; ++i) {
|
|
if (unlikely(!CTR_IS_RESERVED(msrs, i)))
|
|
continue;
|
|
CTR_WRITE(1, msrs, i);
|
|
}
|
|
|
|
/* enable active counters */
|
|
for (i = 0; i < NUM_COUNTERS; ++i) {
|
|
if ((counter_config[i].enabled) && (CTR_IS_RESERVED(msrs, i))) {
|
|
reset_value[i] = counter_config[i].count;
|
|
|
|
CTR_WRITE(counter_config[i].count, msrs, i);
|
|
|
|
CTRL_READ(low, high, msrs, i);
|
|
CTRL_CLEAR_LO(low);
|
|
CTRL_CLEAR_HI(high);
|
|
CTRL_SET_ENABLE(low);
|
|
CTRL_SET_USR(low, counter_config[i].user);
|
|
CTRL_SET_KERN(low, counter_config[i].kernel);
|
|
CTRL_SET_UM(low, counter_config[i].unit_mask);
|
|
CTRL_SET_EVENT_LOW(low, counter_config[i].event);
|
|
CTRL_SET_EVENT_HIGH(high, counter_config[i].event);
|
|
CTRL_SET_HOST_ONLY(high, 0);
|
|
CTRL_SET_GUEST_ONLY(high, 0);
|
|
|
|
CTRL_WRITE(low, high, msrs, i);
|
|
} else {
|
|
reset_value[i] = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
#ifdef CONFIG_OPROFILE_IBS
|
|
|
|
static inline int
|
|
op_amd_handle_ibs(struct pt_regs * const regs,
|
|
struct op_msrs const * const msrs)
|
|
{
|
|
u32 low, high;
|
|
u64 msr;
|
|
struct op_entry entry;
|
|
|
|
if (!has_ibs)
|
|
return 1;
|
|
|
|
if (ibs_config.fetch_enabled) {
|
|
rdmsr(MSR_AMD64_IBSFETCHCTL, low, high);
|
|
if (high & IBS_FETCH_HIGH_VALID_BIT) {
|
|
rdmsrl(MSR_AMD64_IBSFETCHLINAD, msr);
|
|
oprofile_add_data(&entry, regs, msr, IBS_FETCH_CODE,
|
|
IBS_FETCH_SIZE);
|
|
op_cpu_buffer_add_data(&entry, (u32)msr);
|
|
op_cpu_buffer_add_data(&entry, (u32)(msr >> 32));
|
|
op_cpu_buffer_add_data(&entry, low);
|
|
op_cpu_buffer_add_data(&entry, high);
|
|
rdmsrl(MSR_AMD64_IBSFETCHPHYSAD, msr);
|
|
op_cpu_buffer_add_data(&entry, (u32)msr);
|
|
op_cpu_buffer_add_data(&entry, (u32)(msr >> 32));
|
|
op_cpu_buffer_write_commit(&entry);
|
|
|
|
/* reenable the IRQ */
|
|
high &= ~IBS_FETCH_HIGH_VALID_BIT;
|
|
high |= IBS_FETCH_HIGH_ENABLE;
|
|
low &= IBS_FETCH_LOW_MAX_CNT_MASK;
|
|
wrmsr(MSR_AMD64_IBSFETCHCTL, low, high);
|
|
}
|
|
}
|
|
|
|
if (ibs_config.op_enabled) {
|
|
rdmsr(MSR_AMD64_IBSOPCTL, low, high);
|
|
if (low & IBS_OP_LOW_VALID_BIT) {
|
|
rdmsrl(MSR_AMD64_IBSOPRIP, msr);
|
|
oprofile_add_data(&entry, regs, msr, IBS_OP_CODE,
|
|
IBS_OP_SIZE);
|
|
op_cpu_buffer_add_data(&entry, (u32)msr);
|
|
op_cpu_buffer_add_data(&entry, (u32)(msr >> 32));
|
|
rdmsrl(MSR_AMD64_IBSOPDATA, msr);
|
|
op_cpu_buffer_add_data(&entry, (u32)msr);
|
|
op_cpu_buffer_add_data(&entry, (u32)(msr >> 32));
|
|
rdmsrl(MSR_AMD64_IBSOPDATA2, msr);
|
|
op_cpu_buffer_add_data(&entry, (u32)msr);
|
|
op_cpu_buffer_add_data(&entry, (u32)(msr >> 32));
|
|
rdmsrl(MSR_AMD64_IBSOPDATA3, msr);
|
|
op_cpu_buffer_add_data(&entry, (u32)msr);
|
|
op_cpu_buffer_add_data(&entry, (u32)(msr >> 32));
|
|
rdmsrl(MSR_AMD64_IBSDCLINAD, msr);
|
|
op_cpu_buffer_add_data(&entry, (u32)msr);
|
|
op_cpu_buffer_add_data(&entry, (u32)(msr >> 32));
|
|
rdmsrl(MSR_AMD64_IBSDCPHYSAD, msr);
|
|
op_cpu_buffer_add_data(&entry, (u32)msr);
|
|
op_cpu_buffer_add_data(&entry, (u32)(msr >> 32));
|
|
op_cpu_buffer_write_commit(&entry);
|
|
|
|
/* reenable the IRQ */
|
|
high = 0;
|
|
low &= ~IBS_OP_LOW_VALID_BIT;
|
|
low |= IBS_OP_LOW_ENABLE;
|
|
wrmsr(MSR_AMD64_IBSOPCTL, low, high);
|
|
}
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
#endif
|
|
|
|
static int op_amd_check_ctrs(struct pt_regs * const regs,
|
|
struct op_msrs const * const msrs)
|
|
{
|
|
unsigned int low, high;
|
|
int i;
|
|
|
|
for (i = 0 ; i < NUM_COUNTERS; ++i) {
|
|
if (!reset_value[i])
|
|
continue;
|
|
CTR_READ(low, high, msrs, i);
|
|
if (CTR_OVERFLOWED(low)) {
|
|
oprofile_add_sample(regs, i);
|
|
CTR_WRITE(reset_value[i], msrs, i);
|
|
}
|
|
}
|
|
|
|
#ifdef CONFIG_OPROFILE_IBS
|
|
op_amd_handle_ibs(regs, msrs);
|
|
#endif
|
|
|
|
/* See op_model_ppro.c */
|
|
return 1;
|
|
}
|
|
|
|
static void op_amd_start(struct op_msrs const * const msrs)
|
|
{
|
|
unsigned int low, high;
|
|
int i;
|
|
for (i = 0 ; i < NUM_COUNTERS ; ++i) {
|
|
if (reset_value[i]) {
|
|
CTRL_READ(low, high, msrs, i);
|
|
CTRL_SET_ACTIVE(low);
|
|
CTRL_WRITE(low, high, msrs, i);
|
|
}
|
|
}
|
|
|
|
#ifdef CONFIG_OPROFILE_IBS
|
|
if (has_ibs && ibs_config.fetch_enabled) {
|
|
low = (ibs_config.max_cnt_fetch >> 4) & 0xFFFF;
|
|
high = ((ibs_config.rand_en & 0x1) << 25) /* bit 57 */
|
|
+ IBS_FETCH_HIGH_ENABLE;
|
|
wrmsr(MSR_AMD64_IBSFETCHCTL, low, high);
|
|
}
|
|
|
|
if (has_ibs && ibs_config.op_enabled) {
|
|
low = ((ibs_config.max_cnt_op >> 4) & 0xFFFF)
|
|
+ ((ibs_config.dispatched_ops & 0x1) << 19) /* bit 19 */
|
|
+ IBS_OP_LOW_ENABLE;
|
|
high = 0;
|
|
wrmsr(MSR_AMD64_IBSOPCTL, low, high);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
|
|
static void op_amd_stop(struct op_msrs const * const msrs)
|
|
{
|
|
unsigned int low, high;
|
|
int i;
|
|
|
|
/*
|
|
* Subtle: stop on all counters to avoid race with setting our
|
|
* pm callback
|
|
*/
|
|
for (i = 0 ; i < NUM_COUNTERS ; ++i) {
|
|
if (!reset_value[i])
|
|
continue;
|
|
CTRL_READ(low, high, msrs, i);
|
|
CTRL_SET_INACTIVE(low);
|
|
CTRL_WRITE(low, high, msrs, i);
|
|
}
|
|
|
|
#ifdef CONFIG_OPROFILE_IBS
|
|
if (has_ibs && ibs_config.fetch_enabled) {
|
|
/* clear max count and enable */
|
|
low = 0;
|
|
high = 0;
|
|
wrmsr(MSR_AMD64_IBSFETCHCTL, low, high);
|
|
}
|
|
|
|
if (has_ibs && ibs_config.op_enabled) {
|
|
/* clear max count and enable */
|
|
low = 0;
|
|
high = 0;
|
|
wrmsr(MSR_AMD64_IBSOPCTL, low, high);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
static void op_amd_shutdown(struct op_msrs const * const msrs)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0 ; i < NUM_COUNTERS ; ++i) {
|
|
if (CTR_IS_RESERVED(msrs, i))
|
|
release_perfctr_nmi(MSR_K7_PERFCTR0 + i);
|
|
}
|
|
for (i = 0 ; i < NUM_CONTROLS ; ++i) {
|
|
if (CTRL_IS_RESERVED(msrs, i))
|
|
release_evntsel_nmi(MSR_K7_EVNTSEL0 + i);
|
|
}
|
|
}
|
|
|
|
#ifdef CONFIG_OPROFILE_IBS
|
|
|
|
static u8 ibs_eilvt_off;
|
|
|
|
static inline void apic_init_ibs_nmi_per_cpu(void *arg)
|
|
{
|
|
ibs_eilvt_off = setup_APIC_eilvt_ibs(0, APIC_EILVT_MSG_NMI, 0);
|
|
}
|
|
|
|
static inline void apic_clear_ibs_nmi_per_cpu(void *arg)
|
|
{
|
|
setup_APIC_eilvt_ibs(0, APIC_EILVT_MSG_FIX, 1);
|
|
}
|
|
|
|
static int init_ibs_nmi(void)
|
|
{
|
|
#define IBSCTL_LVTOFFSETVAL (1 << 8)
|
|
#define IBSCTL 0x1cc
|
|
struct pci_dev *cpu_cfg;
|
|
int nodes;
|
|
u32 value = 0;
|
|
|
|
/* per CPU setup */
|
|
on_each_cpu(apic_init_ibs_nmi_per_cpu, NULL, 1);
|
|
|
|
nodes = 0;
|
|
cpu_cfg = NULL;
|
|
do {
|
|
cpu_cfg = pci_get_device(PCI_VENDOR_ID_AMD,
|
|
PCI_DEVICE_ID_AMD_10H_NB_MISC,
|
|
cpu_cfg);
|
|
if (!cpu_cfg)
|
|
break;
|
|
++nodes;
|
|
pci_write_config_dword(cpu_cfg, IBSCTL, ibs_eilvt_off
|
|
| IBSCTL_LVTOFFSETVAL);
|
|
pci_read_config_dword(cpu_cfg, IBSCTL, &value);
|
|
if (value != (ibs_eilvt_off | IBSCTL_LVTOFFSETVAL)) {
|
|
pci_dev_put(cpu_cfg);
|
|
printk(KERN_DEBUG "Failed to setup IBS LVT offset, "
|
|
"IBSCTL = 0x%08x", value);
|
|
return 1;
|
|
}
|
|
} while (1);
|
|
|
|
if (!nodes) {
|
|
printk(KERN_DEBUG "No CPU node configured for IBS");
|
|
return 1;
|
|
}
|
|
|
|
#ifdef CONFIG_NUMA
|
|
/* Sanity check */
|
|
/* Works only for 64bit with proper numa implementation. */
|
|
if (nodes != num_possible_nodes()) {
|
|
printk(KERN_DEBUG "Failed to setup CPU node(s) for IBS, "
|
|
"found: %d, expected %d",
|
|
nodes, num_possible_nodes());
|
|
return 1;
|
|
}
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
/* uninitialize the APIC for the IBS interrupts if needed */
|
|
static void clear_ibs_nmi(void)
|
|
{
|
|
if (has_ibs)
|
|
on_each_cpu(apic_clear_ibs_nmi_per_cpu, NULL, 1);
|
|
}
|
|
|
|
/* initialize the APIC for the IBS interrupts if available */
|
|
static void ibs_init(void)
|
|
{
|
|
has_ibs = boot_cpu_has(X86_FEATURE_IBS);
|
|
|
|
if (!has_ibs)
|
|
return;
|
|
|
|
if (init_ibs_nmi()) {
|
|
has_ibs = 0;
|
|
return;
|
|
}
|
|
|
|
printk(KERN_INFO "oprofile: AMD IBS detected\n");
|
|
}
|
|
|
|
static void ibs_exit(void)
|
|
{
|
|
if (!has_ibs)
|
|
return;
|
|
|
|
clear_ibs_nmi();
|
|
}
|
|
|
|
static int (*create_arch_files)(struct super_block *sb, struct dentry *root);
|
|
|
|
static int setup_ibs_files(struct super_block *sb, struct dentry *root)
|
|
{
|
|
struct dentry *dir;
|
|
int ret = 0;
|
|
|
|
/* architecture specific files */
|
|
if (create_arch_files)
|
|
ret = create_arch_files(sb, root);
|
|
|
|
if (ret)
|
|
return ret;
|
|
|
|
if (!has_ibs)
|
|
return ret;
|
|
|
|
/* model specific files */
|
|
|
|
/* setup some reasonable defaults */
|
|
ibs_config.max_cnt_fetch = 250000;
|
|
ibs_config.fetch_enabled = 0;
|
|
ibs_config.max_cnt_op = 250000;
|
|
ibs_config.op_enabled = 0;
|
|
ibs_config.dispatched_ops = 1;
|
|
|
|
dir = oprofilefs_mkdir(sb, root, "ibs_fetch");
|
|
oprofilefs_create_ulong(sb, dir, "enable",
|
|
&ibs_config.fetch_enabled);
|
|
oprofilefs_create_ulong(sb, dir, "max_count",
|
|
&ibs_config.max_cnt_fetch);
|
|
oprofilefs_create_ulong(sb, dir, "rand_enable",
|
|
&ibs_config.rand_en);
|
|
|
|
dir = oprofilefs_mkdir(sb, root, "ibs_op");
|
|
oprofilefs_create_ulong(sb, dir, "enable",
|
|
&ibs_config.op_enabled);
|
|
oprofilefs_create_ulong(sb, dir, "max_count",
|
|
&ibs_config.max_cnt_op);
|
|
oprofilefs_create_ulong(sb, dir, "dispatched_ops",
|
|
&ibs_config.dispatched_ops);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int op_amd_init(struct oprofile_operations *ops)
|
|
{
|
|
ibs_init();
|
|
create_arch_files = ops->create_files;
|
|
ops->create_files = setup_ibs_files;
|
|
return 0;
|
|
}
|
|
|
|
static void op_amd_exit(void)
|
|
{
|
|
ibs_exit();
|
|
}
|
|
|
|
#else
|
|
|
|
/* no IBS support */
|
|
|
|
static int op_amd_init(struct oprofile_operations *ops)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static void op_amd_exit(void) {}
|
|
|
|
#endif /* CONFIG_OPROFILE_IBS */
|
|
|
|
struct op_x86_model_spec const op_amd_spec = {
|
|
.init = op_amd_init,
|
|
.exit = op_amd_exit,
|
|
.num_counters = NUM_COUNTERS,
|
|
.num_controls = NUM_CONTROLS,
|
|
.fill_in_addresses = &op_amd_fill_in_addresses,
|
|
.setup_ctrs = &op_amd_setup_ctrs,
|
|
.check_ctrs = &op_amd_check_ctrs,
|
|
.start = &op_amd_start,
|
|
.stop = &op_amd_stop,
|
|
.shutdown = &op_amd_shutdown
|
|
};
|