|
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index] Re: [PATCH] RFC x86/msr: Use WRMSRNS $imm when available
On 11.08.2025 11:50, Andrew Cooper wrote:
> On 11/08/2025 9:16 am, Jan Beulich wrote:
>> On 09.08.2025 00:20, Andrew Cooper wrote:
>>> --- a/xen/arch/x86/include/asm/msr.h
>>> +++ b/xen/arch/x86/include/asm/msr.h
>>> @@ -38,9 +38,46 @@ static inline void wrmsrl(unsigned int msr, uint64_t val)
>>> wrmsr(msr, lo, hi);
>>> }
>>>
>>> +/*
>>> + * Non-serialising WRMSR with a compile-time constant index, when
>>> available.
>>> + * Falls back to plain WRMSRNS, or to a serialising WRMSR.
>>> + */
>>> +static always_inline void __wrmsrns_imm(uint32_t msr, uint64_t val)
>>> +{
>>> + /*
>>> + * For best performance, WRMSRNS %r64, $msr is recommended. For
>>> + * compatibility, we need to fall back to plain WRMSRNS, or to WRMSR.
>>> + *
>>> + * The combined ABI is awkward, because WRMSRNS $imm takes a single
>>> r64,
>>> + * whereas WRMSR{,NS} takes a split edx:eax pair.
>>> + *
>>> + * Always use WRMSRNS %rax, $imm, because it has the most in common
>>> with
>>> + * the legacy forms. When MSR_IMM isn't available, emit setup logic
>>> for
>>> + * %ecx and %edx too.
>>> + */
>>> + alternative_input_2(
>>> + "mov $%c[msr], %%ecx\n\t"
>> Simply %[msr] here?
>>
>> And then, might it make sense to pull out this and ...
>>
>>> + "mov %%rax, %%rdx\n\t"
>>> + "shr $32, %%rdx\n\t"
>>> + ".byte 0x2e; wrmsr",
>>> +
>>> + /* WRMSRNS %rax, $msr */
>>> + ".byte 0xc4,0xe7,0x7a,0xf6,0xc0; .long %c[msr]",
>>> X86_FEATURE_MSR_IMM,
>>> +
>>> + "mov $%c[msr], %%ecx\n\t"
>> ... this, to ...
>>
>>> + "mov %%rax, %%rdx\n\t"
>>> + "shr $32, %%rdx\n\t"
>>> + ".byte 0x0f,0x01,0xc6", X86_FEATURE_WRMSRNS,
>>> +
>>> + [msr] "i" (msr), "a" (val) : "rcx", "rdx");
>> [msr] "i" (msr), "a" (val), "c" (msr) : "rdx");
>>
>> allowing the compiler to actually know what's put in %ecx? That'll make
>> original and 2nd replacement code 10 bytes, better balancing with the 9
>> bytes of the 1st replacement. And I'd guess that the potentially dead
>> MOV to %ecx would be hidden in the noise as well.
>
> I considered that, but what can the compiler do as a result of knowing %ecx?
For example ...
> That said, we do need an RDMSR form (which I desperately want to make
> foo = rdmsr(MSR_BAR) but my cleanup series from 2019 got nowhere), and
> in a read+write case I suppose the compiler could deduplicate the setup
> of %ecx.
... this. But also simply to use a good pattern (exposing as much as possible
to the compiler), so there are more good instances of code for future cloning
from. (In size-optimizing builds, the compiler could further favor ADD/SUB
over MOV when the two MSRs accessed are relatively close together.)
Jan
|
![]() |
Lists.xenproject.org is hosted with RackSpace, monitoring our |