Re: [PATCH v4 04/17] x86/acrn: Introduce hypercall interfaces
From: Segher Boessenkool
Date: Wed Sep 30 2020 - 12:17:28 EST
Hi!
On Wed, Sep 30, 2020 at 01:16:12PM +0200, Peter Zijlstra wrote:
> On Sun, Sep 27, 2020 at 08:38:03AM -0700, Dave Hansen wrote:
> > On 9/27/20 3:51 AM, Greg Kroah-Hartman wrote:
> > >> +static inline long acrn_hypercall0(unsigned long hcall_id)
> > >> +{
> > >> + register long r8 asm("r8");
> > >> + long result;
> > >> +
> > >> + /* Nothing can come between the r8 assignment and the asm: */
> > >> + r8 = hcall_id;
> > >> + asm volatile("vmcall\n\t"
> > >> + : "=a" (result)
> > >> + : "r" (r8)
> > >> + : );
> > > What keeps an interrupt from happening between the r8 assignment and the
> > > asm: ?
> >
> > It's probably better phrased something like: "No other C code can come
> > between this r8 assignment and the inline asm". An interrupt would
> > actually be fine in there because interrupts save and restore all
> > register state, including r8.
> >
> > The problem (mentioned in the changelog) is that gcc does not let you
> > place data directly into r8. But, it does allow you to declare a
> > register variable that you can assign to use r8. There might be a
> > problem if a function calls was in between and clobber the register,
> > thus the "nothing can come between" comment.
> >
> > The comment is really intended to scare away anyone from adding printk()'s.
> >
> > More information about these register variables is here:
> >
> > > https://gcc.gnu.org/onlinedocs/gcc/Local-Register-Variables.html#Local-Register-Variables
> >
> > Any better ideas for comments would be greatly appreciated. It has 4 or
> > 5 copies so I wanted it to be succinct.
>
> This is disguisting.. Segher, does this actually work? Nick, does clang
> also support this?
The C variable "r8" is just a variable like any other; it can live in
memory, or in any register, and different in all places, too. It can be
moved around too; where "the assignment to it" happens is a
philosophical question more than anything (the assignment there can be
optimised away completely, for example; it is just a C variable, there
is no magic).
Since this variable is a local register asm, on entry to the asm the
compiler guarantees that the value lives in the assigned register (the
"r8" hardware register in this case). This all works completely fine.
This is the only guaranteed behaviour for local register asm (well,
together with analogous behaviour for outputs).
If you want to *always* have it live in the hardware reg "r8", you have
to use a global register asm, and almost certainly do that for all
translation units, and use -ffixed-r8 as well. This of course is
extremely costly.
Segher