p->basic.raw[ctl->input[0]] = leaf;
break;
+ case 0x40000000:
+ p->hv_limit = ctl->eax;
+ break;
+
+ case 0x40000100:
+ p->hv2_limit = ctl->eax;
+ break;
+
case 0x80000000 ... 0x80000000 + ARRAY_SIZE(p->extd.raw) - 1:
p->extd.raw[ctl->input[0] - 0x80000000] = leaf;
break;
uint32_t subleaf, struct cpuid_leaf *res)
{
const struct domain *d = v->domain;
+ const struct cpuid_policy *p = d->arch.cpuid;
uint32_t base = is_viridian_domain(d) ? 0x40000100 : 0x40000000;
uint32_t idx = leaf - base;
- uint32_t limit, dummy;
+ unsigned int limit = is_viridian_domain(d) ? p->hv2_limit : p->hv_limit;
- if ( idx > XEN_CPUID_MAX_NUM_LEAVES )
- return; /* Avoid unnecessary pass through domain_cpuid() */
-
- domain_cpuid(d, base, 0, &limit, &dummy, &dummy, &dummy);
if ( limit == 0 )
/* Default number of leaves */
limit = XEN_CPUID_MAX_NUM_LEAVES;
else
- {
- /* User-specified number of leaves */
- limit &= 0xff;
- if ( limit < 2 )
- limit = 2;
- else if ( limit > XEN_CPUID_MAX_NUM_LEAVES )
- limit = XEN_CPUID_MAX_NUM_LEAVES;
- }
+ /* Clamp toolstack value between 2 and MAX_NUM_LEAVES. */
+ limit = min(max(limit, 2u), XEN_CPUID_MAX_NUM_LEAVES + 0u);
if ( idx > limit )
return;
break;
default:
- BUG();
+ ASSERT_UNREACHABLE();
}
}
#undef __DECL_BITFIELD
#undef _DECL_BITFIELD
#undef DECL_BITFIELD
+
+ /* Toolstack selected Hypervisor max_leaf (if non-zero). */
+ uint8_t hv_limit, hv2_limit;
};
/* Fill in a featureset bitmap from a CPUID policy. */