target/ppc: Increment PMC5 with inline insns
Profiling QEMU during Fedora 35 for PPC64 boot revealed that 6.39% of total time was being spent in helper_insns_inc(), on a POWER9 machine. To avoid calling this helper every time PMCs had to be incremented, an inline implementation of PMC5 increment and check for overflow was developed. This led to a reduction of about 12% in Fedora's boot time. Signed-off-by: Leandro Lupori <leandro.lupori@eldorado.org.br> Reviewed-by: Daniel Henrique Barboza <danielhb413@gmail.com> Message-Id: <20221025202424.195984-4-leandro.lupori@eldorado.org.br> Signed-off-by: Daniel Henrique Barboza <danielhb413@gmail.com>
This commit is contained in:
parent
8b3d1c49a9
commit
eeaaefe9fa
@ -30,6 +30,7 @@ DEF_HELPER_2(store_mmcr1, void, env, tl)
|
|||||||
DEF_HELPER_3(store_pmc, void, env, i32, i64)
|
DEF_HELPER_3(store_pmc, void, env, i32, i64)
|
||||||
DEF_HELPER_2(read_pmc, tl, env, i32)
|
DEF_HELPER_2(read_pmc, tl, env, i32)
|
||||||
DEF_HELPER_2(insns_inc, void, env, i32)
|
DEF_HELPER_2(insns_inc, void, env, i32)
|
||||||
|
DEF_HELPER_1(handle_pmc5_overflow, void, env)
|
||||||
#endif
|
#endif
|
||||||
DEF_HELPER_1(check_tlb_flush_local, void, env)
|
DEF_HELPER_1(check_tlb_flush_local, void, env)
|
||||||
DEF_HELPER_1(check_tlb_flush_global, void, env)
|
DEF_HELPER_1(check_tlb_flush_global, void, env)
|
||||||
|
@ -22,8 +22,6 @@
|
|||||||
|
|
||||||
#if defined(TARGET_PPC64) && !defined(CONFIG_USER_ONLY)
|
#if defined(TARGET_PPC64) && !defined(CONFIG_USER_ONLY)
|
||||||
|
|
||||||
#define PMC_COUNTER_NEGATIVE_VAL 0x80000000UL
|
|
||||||
|
|
||||||
static bool pmc_has_overflow_enabled(CPUPPCState *env, int sprn)
|
static bool pmc_has_overflow_enabled(CPUPPCState *env, int sprn)
|
||||||
{
|
{
|
||||||
if (sprn == SPR_POWER_PMC1) {
|
if (sprn == SPR_POWER_PMC1) {
|
||||||
@ -88,7 +86,6 @@ static bool pmu_increment_insns(CPUPPCState *env, uint32_t num_insns)
|
|||||||
bool overflow_triggered = false;
|
bool overflow_triggered = false;
|
||||||
target_ulong tmp;
|
target_ulong tmp;
|
||||||
|
|
||||||
if (unlikely(ins_cnt & 0x1e)) {
|
|
||||||
if (ins_cnt & (1 << 1)) {
|
if (ins_cnt & (1 << 1)) {
|
||||||
tmp = env->spr[SPR_POWER_PMC1];
|
tmp = env->spr[SPR_POWER_PMC1];
|
||||||
tmp += num_insns;
|
tmp += num_insns;
|
||||||
@ -132,7 +129,6 @@ static bool pmu_increment_insns(CPUPPCState *env, uint32_t num_insns)
|
|||||||
env->spr[SPR_POWER_PMC4] = tmp;
|
env->spr[SPR_POWER_PMC4] = tmp;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if (ins_cnt & (1 << 5)) {
|
if (ins_cnt & (1 << 5)) {
|
||||||
tmp = env->spr[SPR_POWER_PMC5];
|
tmp = env->spr[SPR_POWER_PMC5];
|
||||||
@ -310,6 +306,12 @@ static void fire_PMC_interrupt(PowerPCCPU *cpu)
|
|||||||
raise_ebb_perfm_exception(env);
|
raise_ebb_perfm_exception(env);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void helper_handle_pmc5_overflow(CPUPPCState *env)
|
||||||
|
{
|
||||||
|
env->spr[SPR_POWER_PMC5] = PMC_COUNTER_NEGATIVE_VAL;
|
||||||
|
fire_PMC_interrupt(env_archcpu(env));
|
||||||
|
}
|
||||||
|
|
||||||
/* This helper assumes that the PMC is running. */
|
/* This helper assumes that the PMC is running. */
|
||||||
void helper_insns_inc(CPUPPCState *env, uint32_t num_insns)
|
void helper_insns_inc(CPUPPCState *env, uint32_t num_insns)
|
||||||
{
|
{
|
||||||
|
@ -14,6 +14,9 @@
|
|||||||
#define POWER8_PMU_H
|
#define POWER8_PMU_H
|
||||||
|
|
||||||
#if defined(TARGET_PPC64) && !defined(CONFIG_USER_ONLY)
|
#if defined(TARGET_PPC64) && !defined(CONFIG_USER_ONLY)
|
||||||
|
|
||||||
|
#define PMC_COUNTER_NEGATIVE_VAL 0x80000000UL
|
||||||
|
|
||||||
void cpu_ppc_pmu_init(CPUPPCState *env);
|
void cpu_ppc_pmu_init(CPUPPCState *env);
|
||||||
void pmu_update_summaries(CPUPPCState *env);
|
void pmu_update_summaries(CPUPPCState *env);
|
||||||
#else
|
#else
|
||||||
|
@ -36,6 +36,7 @@
|
|||||||
#include "exec/log.h"
|
#include "exec/log.h"
|
||||||
#include "qemu/atomic128.h"
|
#include "qemu/atomic128.h"
|
||||||
#include "spr_common.h"
|
#include "spr_common.h"
|
||||||
|
#include "power8-pmu.h"
|
||||||
|
|
||||||
#include "qemu/qemu-print.h"
|
#include "qemu/qemu-print.h"
|
||||||
#include "qapi/error.h"
|
#include "qapi/error.h"
|
||||||
@ -4271,6 +4272,9 @@ static void pmu_count_insns(DisasContext *ctx)
|
|||||||
}
|
}
|
||||||
|
|
||||||
#if !defined(CONFIG_USER_ONLY)
|
#if !defined(CONFIG_USER_ONLY)
|
||||||
|
TCGLabel *l;
|
||||||
|
TCGv t0;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The PMU insns_inc() helper stops the internal PMU timer if a
|
* The PMU insns_inc() helper stops the internal PMU timer if a
|
||||||
* counter overflows happens. In that case, if the guest is
|
* counter overflows happens. In that case, if the guest is
|
||||||
@ -4279,7 +4283,25 @@ static void pmu_count_insns(DisasContext *ctx)
|
|||||||
*/
|
*/
|
||||||
gen_icount_io_start(ctx);
|
gen_icount_io_start(ctx);
|
||||||
|
|
||||||
|
/* Avoid helper calls when only PMC5-6 are enabled. */
|
||||||
|
if (!ctx->pmc_other) {
|
||||||
|
l = gen_new_label();
|
||||||
|
t0 = tcg_temp_new();
|
||||||
|
|
||||||
|
gen_load_spr(t0, SPR_POWER_PMC5);
|
||||||
|
tcg_gen_addi_tl(t0, t0, ctx->base.num_insns);
|
||||||
|
gen_store_spr(SPR_POWER_PMC5, t0);
|
||||||
|
/* Check for overflow, if it's enabled */
|
||||||
|
if (ctx->mmcr0_pmcjce) {
|
||||||
|
tcg_gen_brcondi_tl(TCG_COND_LT, t0, PMC_COUNTER_NEGATIVE_VAL, l);
|
||||||
|
gen_helper_handle_pmc5_overflow(cpu_env);
|
||||||
|
}
|
||||||
|
|
||||||
|
gen_set_label(l);
|
||||||
|
tcg_temp_free(t0);
|
||||||
|
} else {
|
||||||
gen_helper_insns_inc(cpu_env, tcg_constant_i32(ctx->base.num_insns));
|
gen_helper_insns_inc(cpu_env, tcg_constant_i32(ctx->base.num_insns));
|
||||||
|
}
|
||||||
#else
|
#else
|
||||||
/*
|
/*
|
||||||
* User mode can read (but not write) PMC5 and start/stop
|
* User mode can read (but not write) PMC5 and start/stop
|
||||||
|
Loading…
Reference in New Issue
Block a user