#ifdef CONFIG_X86_LOCAL_APIC
 
+static inline int get_possible_num_counters(void)
+{
+       int i, num_counters = x86_pmu.num_counters;
+
+       if (!is_hybrid())
+               return num_counters;
+
+       for (i = 0; i < x86_pmu.num_hybrid_pmus; i++)
+               num_counters = max_t(int, num_counters, x86_pmu.hybrid_pmu[i].num_counters);
+
+       return num_counters;
+}
+
 static bool reserve_pmc_hardware(void)
 {
-       int i;
+       int i, num_counters = get_possible_num_counters();
 
-       for (i = 0; i < x86_pmu.num_counters; i++) {
+       for (i = 0; i < num_counters; i++) {
                if (!reserve_perfctr_nmi(x86_pmu_event_addr(i)))
                        goto perfctr_fail;
        }
 
-       for (i = 0; i < x86_pmu.num_counters; i++) {
+       for (i = 0; i < num_counters; i++) {
                if (!reserve_evntsel_nmi(x86_pmu_config_addr(i)))
                        goto eventsel_fail;
        }
        for (i--; i >= 0; i--)
                release_evntsel_nmi(x86_pmu_config_addr(i));
 
-       i = x86_pmu.num_counters;
+       i = num_counters;
 
 perfctr_fail:
        for (i--; i >= 0; i--)
 
 static void release_pmc_hardware(void)
 {
-       int i;
+       int i, num_counters = get_possible_num_counters();
 
-       for (i = 0; i < x86_pmu.num_counters; i++) {
+       for (i = 0; i < num_counters; i++) {
                release_perfctr_nmi(x86_pmu_event_addr(i));
                release_evntsel_nmi(x86_pmu_config_addr(i));
        }
 
 int x86_schedule_events(struct cpu_hw_events *cpuc, int n, int *assign)
 {
+       int num_counters = hybrid(cpuc->pmu, num_counters);
        struct event_constraint *c;
        struct perf_event *e;
        int n0, i, wmin, wmax, unsched = 0;
 
        /* slow path */
        if (i != n) {
-               int gpmax = x86_pmu.num_counters;
+               int gpmax = num_counters;
 
                /*
                 * Do not allow scheduling of more than half the available
                 * the extra Merge events needed by large increment events.
                 */
                if (x86_pmu.flags & PMU_FL_PAIR) {
-                       gpmax = x86_pmu.num_counters - cpuc->n_pair;
+                       gpmax = num_counters - cpuc->n_pair;
                        WARN_ON(gpmax <= 0);
                }
 
  */
 static int collect_events(struct cpu_hw_events *cpuc, struct perf_event *leader, bool dogrp)
 {
+       int num_counters = hybrid(cpuc->pmu, num_counters);
+       int num_counters_fixed = hybrid(cpuc->pmu, num_counters_fixed);
        struct perf_event *event;
        int n, max_count;
 
-       max_count = x86_pmu.num_counters + x86_pmu.num_counters_fixed;
+       max_count = num_counters + num_counters_fixed;
 
        /* current number of events already accepted */
        n = cpuc->n_events;
 {
        u64 ctrl, status, overflow, pmc_ctrl, pmc_count, prev_left, fixed;
        u64 pebs, debugctl;
-       struct cpu_hw_events *cpuc;
+       int cpu = smp_processor_id();
+       struct cpu_hw_events *cpuc = &per_cpu(cpu_hw_events, cpu);
+       int num_counters = hybrid(cpuc->pmu, num_counters);
+       int num_counters_fixed = hybrid(cpuc->pmu, num_counters_fixed);
        unsigned long flags;
-       int cpu, idx;
+       int idx;
 
-       if (!x86_pmu.num_counters)
+       if (!num_counters)
                return;
 
        local_irq_save(flags);
 
-       cpu = smp_processor_id();
-       cpuc = &per_cpu(cpu_hw_events, cpu);
-
        if (x86_pmu.version >= 2) {
                rdmsrl(MSR_CORE_PERF_GLOBAL_CTRL, ctrl);
                rdmsrl(MSR_CORE_PERF_GLOBAL_STATUS, status);
        }
        pr_info("CPU#%d: active:     %016llx\n", cpu, *(u64 *)cpuc->active_mask);
 
-       for (idx = 0; idx < x86_pmu.num_counters; idx++) {
+       for (idx = 0; idx < num_counters; idx++) {
                rdmsrl(x86_pmu_config_addr(idx), pmc_ctrl);
                rdmsrl(x86_pmu_event_addr(idx), pmc_count);
 
                pr_info("CPU#%d:   gen-PMC%d left:  %016llx\n",
                        cpu, idx, prev_left);
        }
-       for (idx = 0; idx < x86_pmu.num_counters_fixed; idx++) {
+       for (idx = 0; idx < num_counters_fixed; idx++) {
                if (fixed_counter_disabled(idx, cpuc->pmu))
                        continue;
                rdmsrl(MSR_ARCH_PERFMON_FIXED_CTR0 + idx, pmc_count);
 void perf_get_x86_pmu_capability(struct x86_pmu_capability *cap)
 {
        cap->version            = x86_pmu.version;
+       /*
+        * KVM doesn't support the hybrid PMU yet.
+        * Return the common value in global x86_pmu,
+        * which available for all cores.
+        */
        cap->num_counters_gp    = x86_pmu.num_counters;
        cap->num_counters_fixed = x86_pmu.num_counters_fixed;
        cap->bit_width_gp       = x86_pmu.cntval_bits;
 
 {
        struct debug_store *ds = __this_cpu_read(cpu_hw_events.ds);
        struct cpu_hw_events *cpuc = this_cpu_ptr(&cpu_hw_events);
+       int num_counters_fixed = hybrid(cpuc->pmu, num_counters_fixed);
+       int num_counters = hybrid(cpuc->pmu, num_counters);
        unsigned long flags;
        int idx;
 
-       if (!x86_pmu.num_counters)
+       if (!num_counters)
                return;
 
        local_irq_save(flags);
 
        pr_info("clearing PMU state on CPU#%d\n", smp_processor_id());
 
-       for (idx = 0; idx < x86_pmu.num_counters; idx++) {
+       for (idx = 0; idx < num_counters; idx++) {
                wrmsrl_safe(x86_pmu_config_addr(idx), 0ull);
                wrmsrl_safe(x86_pmu_event_addr(idx),  0ull);
        }
-       for (idx = 0; idx < x86_pmu.num_counters_fixed; idx++) {
+       for (idx = 0; idx < num_counters_fixed; idx++) {
                if (fixed_counter_disabled(idx, cpuc->pmu))
                        continue;
                wrmsrl_safe(MSR_ARCH_PERFMON_FIXED_CTR0 + idx, 0ull);
 
 static inline void pebs_update_threshold(struct cpu_hw_events *cpuc)
 {
        struct debug_store *ds = cpuc->ds;
+       int max_pebs_events = hybrid(cpuc->pmu, max_pebs_events);
+       int num_counters_fixed = hybrid(cpuc->pmu, num_counters_fixed);
        u64 threshold;
        int reserved;
 
                return;
 
        if (x86_pmu.flags & PMU_FL_PEBS_ALL)
-               reserved = x86_pmu.max_pebs_events + x86_pmu.num_counters_fixed;
+               reserved = max_pebs_events + num_counters_fixed;
        else
-               reserved = x86_pmu.max_pebs_events;
+               reserved = max_pebs_events;
 
        if (cpuc->n_pebs == cpuc->n_large_pebs) {
                threshold = ds->pebs_absolute_maximum -
 {
        short counts[INTEL_PMC_IDX_FIXED + MAX_FIXED_PEBS_EVENTS] = {};
        struct cpu_hw_events *cpuc = this_cpu_ptr(&cpu_hw_events);
+       int max_pebs_events = hybrid(cpuc->pmu, max_pebs_events);
+       int num_counters_fixed = hybrid(cpuc->pmu, num_counters_fixed);
        struct debug_store *ds = cpuc->ds;
        struct perf_event *event;
        void *base, *at, *top;
 
        ds->pebs_index = ds->pebs_buffer_base;
 
-       mask = ((1ULL << x86_pmu.max_pebs_events) - 1) |
-              (((1ULL << x86_pmu.num_counters_fixed) - 1) << INTEL_PMC_IDX_FIXED);
-       size = INTEL_PMC_IDX_FIXED + x86_pmu.num_counters_fixed;
+       mask = ((1ULL << max_pebs_events) - 1) |
+              (((1ULL << num_counters_fixed) - 1) << INTEL_PMC_IDX_FIXED);
+       size = INTEL_PMC_IDX_FIXED + num_counters_fixed;
 
        if (unlikely(base >= top)) {
                intel_pmu_pebs_event_update_no_drain(cpuc, size);