|
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index] Re: [PATCH 1/5] xen/domain: Remove function pointers from domain pause helpers
Hi Andrew,
> On 11 Nov 2021, at 17:57, Andrew Cooper <andrew.cooper3@xxxxxxxxxx> wrote:
>
> Retpolines are expensive, and all these do are select between the sync and
> nosync helpers. Pass a boolean instead, and use direct calls everywhere.
>
> Pause/unpause operations on behalf of dom0 are not fastpaths, so avoid
> exposing the __domain_pause_by_systemcontroller() internal.
>
> This actually compiles smaller than before:
>
> $ ../scripts/bloat-o-meter xen-syms-before xen-syms-after
> add/remove: 3/1 grow/shrink: 0/5 up/down: 250/-273 (-23)
> Function old new delta
> _domain_pause - 115 +115
> domain_pause_by_systemcontroller - 69 +69
> domain_pause_by_systemcontroller_nosync - 66 +66
> domain_kill 426 398 -28
> domain_resume 246 214 -32
> domain_pause_except_self 189 141 -48
> domain_pause 59 10 -49
> domain_pause_nosync 59 7 -52
> __domain_pause_by_systemcontroller 64 - -64
>
> despite GCC's best efforts. The new _domain_pause_by_systemcontroller()
> really should not be inlined, considering that the difference is only the
> setup of the sync boolean to pass to _domain_pause(), and there are plenty of
> registers to spare.
To add an argument to the discussion I would say that preventing to use function
pointers is something that is good for FuSa so I am in favour here.
>
> Signed-off-by: Andrew Cooper <andrew.cooper3@xxxxxxxxxx>
> ---
> CC: Jan Beulich <JBeulich@xxxxxxxx>
> CC: Roger Pau Monné <roger.pau@xxxxxxxxxx>
> CC: Wei Liu <wl@xxxxxxx>
> CC: Stefano Stabellini <sstabellini@xxxxxxxxxx>
> CC: Julien Grall <julien@xxxxxxx>
> CC: Volodymyr Babchuk <Volodymyr_Babchuk@xxxxxxxx>
> CC: Bertrand Marquis <bertrand.marquis@xxxxxxx>
> ---
> xen/common/domain.c | 31 ++++++++++++++++++++++---------
> xen/include/xen/sched.h | 15 +++++----------
> 2 files changed, 27 insertions(+), 19 deletions(-)
>
> diff --git a/xen/common/domain.c b/xen/common/domain.c
> index 56d47dd66478..562872cdf87a 100644
> --- a/xen/common/domain.c
> +++ b/xen/common/domain.c
> @@ -1234,15 +1234,18 @@ int vcpu_unpause_by_systemcontroller(struct vcpu *v)
> return 0;
> }
>
> -static void do_domain_pause(struct domain *d,
> - void (*sleep_fn)(struct vcpu *v))
> +static void _domain_pause(struct domain *d, bool sync /* or nosync */)
Here you use comments inside the function definition.
I think this is something that should be avoided and in this specific case a
boolean sync is clear enough not to need to explain that false is nosing.
> {
> struct vcpu *v;
>
> atomic_inc(&d->pause_count);
>
> - for_each_vcpu( d, v )
> - sleep_fn(v);
> + if ( sync )
> + for_each_vcpu ( d, v )
> + vcpu_sleep_sync(v);
> + else
> + for_each_vcpu ( d, v )
> + vcpu_sleep_nosync(v);
>
> arch_domain_pause(d);
> }
> @@ -1250,12 +1253,12 @@ static void do_domain_pause(struct domain *d,
> void domain_pause(struct domain *d)
> {
> ASSERT(d != current->domain);
> - do_domain_pause(d, vcpu_sleep_sync);
> + _domain_pause(d, true /* sync */);
Same here.
> }
>
> void domain_pause_nosync(struct domain *d)
> {
> - do_domain_pause(d, vcpu_sleep_nosync);
> + _domain_pause(d, false /* nosync */);
Same here.
> }
>
> void domain_unpause(struct domain *d)
> @@ -1269,8 +1272,8 @@ void domain_unpause(struct domain *d)
> vcpu_wake(v);
> }
>
> -int __domain_pause_by_systemcontroller(struct domain *d,
> - void (*pause_fn)(struct domain *d))
> +static int _domain_pause_by_systemcontroller(
> + struct domain *d, bool sync /* or nosync */)
Same here.
> {
> int old, new, prev = d->controller_pause_count;
>
> @@ -1289,11 +1292,21 @@ int __domain_pause_by_systemcontroller(struct domain
> *d,
> prev = cmpxchg(&d->controller_pause_count, old, new);
> } while ( prev != old );
>
> - pause_fn(d);
> + _domain_pause(d, sync);
>
> return 0;
> }
>
> +int domain_pause_by_systemcontroller(struct domain *d)
> +{
> + return _domain_pause_by_systemcontroller(d, true /* sync */);
Same here.
> +}
> +
> +int domain_pause_by_systemcontroller_nosync(struct domain *d)
> +{
> + return _domain_pause_by_systemcontroller(d, false /* nosync */);
Same here.
Cheers
Bertrand
> +}
> +
> int domain_unpause_by_systemcontroller(struct domain *d)
> {
> int old, new, prev = d->controller_pause_count;
> diff --git a/xen/include/xen/sched.h b/xen/include/xen/sched.h
> index 28146ee404e6..37f78cc4c4c9 100644
> --- a/xen/include/xen/sched.h
> +++ b/xen/include/xen/sched.h
> @@ -920,26 +920,21 @@ static inline bool vcpu_cpu_dirty(const struct vcpu *v)
>
> void vcpu_block(void);
> void vcpu_unblock(struct vcpu *v);
> +
> void vcpu_pause(struct vcpu *v);
> void vcpu_pause_nosync(struct vcpu *v);
> void vcpu_unpause(struct vcpu *v);
> +
> int vcpu_pause_by_systemcontroller(struct vcpu *v);
> int vcpu_unpause_by_systemcontroller(struct vcpu *v);
>
> void domain_pause(struct domain *d);
> void domain_pause_nosync(struct domain *d);
> void domain_unpause(struct domain *d);
> +
> +int domain_pause_by_systemcontroller(struct domain *d);
> +int domain_pause_by_systemcontroller_nosync(struct domain *d);
> int domain_unpause_by_systemcontroller(struct domain *d);
> -int __domain_pause_by_systemcontroller(struct domain *d,
> - void (*pause_fn)(struct domain *d));
> -static inline int domain_pause_by_systemcontroller(struct domain *d)
> -{
> - return __domain_pause_by_systemcontroller(d, domain_pause);
> -}
> -static inline int domain_pause_by_systemcontroller_nosync(struct domain *d)
> -{
> - return __domain_pause_by_systemcontroller(d, domain_pause_nosync);
> -}
>
> /* domain_pause() but safe against trying to pause current. */
> int __must_check domain_pause_except_self(struct domain *d);
> --
> 2.11.0
>
|
![]() |
Lists.xenproject.org is hosted with RackSpace, monitoring our |