315cd55dba
It is always a good idea. In one case, attempting to allocate N+1 PMCs from a class with N hardware counters would incorrectly attempt to allocate from the next class in the list. Without this validation, this can lead to all kinds of strange behaviour. Since powerpc_allocate_pmc() is used by both the mpc7xxx and ppc970 classes, add a new global to track which is active (it will never be both). Reviewed by: luporl, ray MFC after: 1 week Sponsored by: The FreeBSD Foundation Differential Revision: https://reviews.freebsd.org/D31387
244 lines
6.8 KiB
C
244 lines
6.8 KiB
C
/*-
|
|
* SPDX-License-Identifier: BSD-2-Clause-FreeBSD
|
|
*
|
|
* Copyright (c) 2013 Justin Hibbits
|
|
* Copyright (c) 2020 Leandro Lupori
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
* SUCH DAMAGE.
|
|
*/
|
|
|
|
#include <sys/cdefs.h>
|
|
__FBSDID("$FreeBSD$");
|
|
|
|
#include <sys/param.h>
|
|
#include <sys/pmc.h>
|
|
#include <sys/pmckern.h>
|
|
#include <sys/systm.h>
|
|
|
|
#include <machine/pmc_mdep.h>
|
|
#include <machine/spr.h>
|
|
#include <machine/cpu.h>
|
|
|
|
#include "hwpmc_powerpc.h"
|
|
|
|
#define POWER8_MAX_PMCS 6
|
|
|
|
#define PM_EVENT_CODE(pe) (pe & 0xffff)
|
|
#define PM_EVENT_COUNTER(pe) ((pe >> 16) & 0xffff)
|
|
|
|
#define PM_CYC 0x1e
|
|
#define PM_INST_CMPL 0x02
|
|
|
|
static void
|
|
power8_set_pmc(int cpu, int ri, int config)
|
|
{
|
|
register_t mmcr;
|
|
|
|
/* Select event */
|
|
switch (ri) {
|
|
case 0:
|
|
case 1:
|
|
case 2:
|
|
case 3:
|
|
mmcr = mfspr(SPR_MMCR1);
|
|
mmcr &= ~SPR_MMCR1_P8_PMCNSEL_MASK(ri);
|
|
mmcr |= SPR_MMCR1_P8_PMCNSEL(ri, config & ~POWERPC_PMC_ENABLE);
|
|
mtspr(SPR_MMCR1, mmcr);
|
|
break;
|
|
}
|
|
|
|
/*
|
|
* By default, freeze counter in all states.
|
|
* If counter is being started, unfreeze it in selected states.
|
|
*/
|
|
mmcr = mfspr(SPR_MMCR2) | SPR_MMCR2_FCNHSP(ri);
|
|
if (config != PMCN_NONE) {
|
|
if (config & POWERPC_PMC_USER_ENABLE)
|
|
mmcr &= ~(SPR_MMCR2_FCNP0(ri) |
|
|
SPR_MMCR2_FCNP1(ri));
|
|
if (config & POWERPC_PMC_KERNEL_ENABLE)
|
|
mmcr &= ~(SPR_MMCR2_FCNH(ri) |
|
|
SPR_MMCR2_FCNS(ri));
|
|
}
|
|
mtspr(SPR_MMCR2, mmcr);
|
|
}
|
|
|
|
static int
|
|
power8_pcpu_init(struct pmc_mdep *md, int cpu)
|
|
{
|
|
register_t mmcr0;
|
|
int i;
|
|
|
|
powerpc_pcpu_init(md, cpu);
|
|
|
|
/* Freeze all counters before modifying PMC registers */
|
|
mmcr0 = mfspr(SPR_MMCR0) | SPR_MMCR0_FC;
|
|
mtspr(SPR_MMCR0, mmcr0);
|
|
|
|
/*
|
|
* Now setup MMCR0:
|
|
* - PMAO=0: clear alerts
|
|
* - FCPC=0, FCP=0: don't freeze counters in problem state
|
|
* - FCECE: Freeze Counters on Enabled Condition or Event
|
|
* - PMC1CE/PMCNCE: PMC1/N Condition Enable
|
|
*/
|
|
mmcr0 &= ~(SPR_MMCR0_PMAO | SPR_MMCR0_FCPC | SPR_MMCR0_FCP);
|
|
mmcr0 |= SPR_MMCR0_FCECE | SPR_MMCR0_PMC1CE | SPR_MMCR0_PMCNCE;
|
|
mtspr(SPR_MMCR0, mmcr0);
|
|
|
|
/* Clear all PMCs to prevent enabled condition interrupts */
|
|
for (i = 0; i < POWER8_MAX_PMCS; i++)
|
|
powerpc_pmcn_write(i, 0);
|
|
|
|
/* Disable events in PMCs 1-4 */
|
|
mtspr(SPR_MMCR1, mfspr(SPR_MMCR1) & ~SPR_MMCR1_P8_PMCSEL_ALL);
|
|
|
|
/* Freeze each counter, in all states */
|
|
mtspr(SPR_MMCR2, mfspr(SPR_MMCR2) |
|
|
SPR_MMCR2_FCNHSP(0) | SPR_MMCR2_FCNHSP(1) | SPR_MMCR2_FCNHSP(2) |
|
|
SPR_MMCR2_FCNHSP(3) | SPR_MMCR2_FCNHSP(4) | SPR_MMCR2_FCNHSP(5));
|
|
|
|
/* Enable interrupts, unset global freeze */
|
|
mmcr0 &= ~SPR_MMCR0_FC;
|
|
mmcr0 |= SPR_MMCR0_PMAE;
|
|
mtspr(SPR_MMCR0, mmcr0);
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
power8_pcpu_fini(struct pmc_mdep *md, int cpu)
|
|
{
|
|
register_t mmcr0;
|
|
|
|
/* Freeze counters, disable interrupts */
|
|
mmcr0 = mfspr(SPR_MMCR0);
|
|
mmcr0 &= ~SPR_MMCR0_PMAE;
|
|
mmcr0 |= SPR_MMCR0_FC;
|
|
mtspr(SPR_MMCR0, mmcr0);
|
|
|
|
return (powerpc_pcpu_fini(md, cpu));
|
|
}
|
|
|
|
static void
|
|
power8_resume_pmc(bool ie)
|
|
{
|
|
register_t mmcr0;
|
|
|
|
/* Unfreeze counters and re-enable PERF exceptions if requested. */
|
|
mmcr0 = mfspr(SPR_MMCR0);
|
|
mmcr0 &= ~(SPR_MMCR0_FC | SPR_MMCR0_PMAO | SPR_MMCR0_PMAE);
|
|
if (ie)
|
|
mmcr0 |= SPR_MMCR0_PMAE;
|
|
mtspr(SPR_MMCR0, mmcr0);
|
|
}
|
|
|
|
static int
|
|
power8_allocate_pmc(int cpu, int ri, struct pmc *pm,
|
|
const struct pmc_op_pmcallocate *a)
|
|
{
|
|
uint32_t caps, config, counter, pe;
|
|
|
|
KASSERT(cpu >= 0 && cpu < pmc_cpu_max(),
|
|
("[powerpc,%d] illegal CPU value %d", __LINE__, cpu));
|
|
KASSERT(ri >= 0 && ri < ppc_max_pmcs,
|
|
("[powerpc,%d] illegal row index %d", __LINE__, ri));
|
|
|
|
pe = a->pm_md.pm_event;
|
|
counter = PM_EVENT_COUNTER(pe);
|
|
config = PM_EVENT_CODE(pe);
|
|
|
|
if (a->pm_class != PMC_CLASS_POWER8)
|
|
return (EINVAL);
|
|
|
|
/*
|
|
* PMC5 and PMC6 are not programmable and always count instructions
|
|
* completed and cycles, respectively.
|
|
*
|
|
* When counter is 0 any of the 4 programmable PMCs may be used for
|
|
* the specified event, otherwise it must match ri + 1.
|
|
*/
|
|
if (counter == 0 && config == PM_INST_CMPL)
|
|
counter = 5;
|
|
else if (counter == 0 && config == PM_CYC)
|
|
counter = 6;
|
|
else if (counter > 4)
|
|
return (EINVAL);
|
|
|
|
if (counter != 0 && counter != ri + 1)
|
|
return (EINVAL);
|
|
|
|
caps = a->pm_caps;
|
|
|
|
if (caps & PMC_CAP_SYSTEM)
|
|
config |= POWERPC_PMC_KERNEL_ENABLE;
|
|
if (caps & PMC_CAP_USER)
|
|
config |= POWERPC_PMC_USER_ENABLE;
|
|
if ((caps & (PMC_CAP_USER | PMC_CAP_SYSTEM)) == 0)
|
|
config |= POWERPC_PMC_ENABLE;
|
|
|
|
pm->pm_md.pm_powerpc.pm_powerpc_evsel = config;
|
|
|
|
PMCDBG3(MDP,ALL,1,"powerpc-allocate cpu=%d ri=%d -> config=0x%x",
|
|
cpu, ri, config);
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
pmc_power8_initialize(struct pmc_mdep *pmc_mdep)
|
|
{
|
|
struct pmc_classdep *pcd;
|
|
|
|
pmc_mdep->pmd_cputype = PMC_CPU_PPC_POWER8;
|
|
|
|
pcd = &pmc_mdep->pmd_classdep[PMC_MDEP_CLASS_INDEX_POWERPC];
|
|
pcd->pcd_caps = POWERPC_PMC_CAPS;
|
|
pcd->pcd_class = PMC_CLASS_POWER8;
|
|
pcd->pcd_num = POWER8_MAX_PMCS;
|
|
pcd->pcd_ri = pmc_mdep->pmd_npmc;
|
|
pcd->pcd_width = 32;
|
|
|
|
pcd->pcd_pcpu_init = power8_pcpu_init;
|
|
pcd->pcd_pcpu_fini = power8_pcpu_fini;
|
|
pcd->pcd_allocate_pmc = power8_allocate_pmc;
|
|
pcd->pcd_release_pmc = powerpc_release_pmc;
|
|
pcd->pcd_start_pmc = powerpc_start_pmc;
|
|
pcd->pcd_stop_pmc = powerpc_stop_pmc;
|
|
pcd->pcd_get_config = powerpc_get_config;
|
|
pcd->pcd_config_pmc = powerpc_config_pmc;
|
|
pcd->pcd_describe = powerpc_describe;
|
|
pcd->pcd_read_pmc = powerpc_read_pmc;
|
|
pcd->pcd_write_pmc = powerpc_write_pmc;
|
|
|
|
pmc_mdep->pmd_npmc += POWER8_MAX_PMCS;
|
|
pmc_mdep->pmd_intr = powerpc_pmc_intr;
|
|
|
|
ppc_max_pmcs = POWER8_MAX_PMCS;
|
|
|
|
powerpc_set_pmc = power8_set_pmc;
|
|
powerpc_pmcn_read = powerpc_pmcn_read_default;
|
|
powerpc_pmcn_write = powerpc_pmcn_write_default;
|
|
powerpc_resume_pmc = power8_resume_pmc;
|
|
|
|
return (0);
|
|
}
|