Re: Async suspend-resume patch w/ completions (was: Re: Asyncsuspend-resume patch w/ rwsems)

From: Alan Stern
Date: Tue Dec 15 2009 - 15:26:16 EST


On Tue, 15 Dec 2009, Linus Torvalds wrote:

> It's a very subtle theory, and it's not necessarily always 100% true. For
> example, a cardbus bridge is strictly speaking very much a PCI bridge, but
> for cardbus bridges we _do_ have a suspend/resume function.
>
> And perhaps worse than that, cardbus bridges are one of the canonical
> examples where two different PCI devices actually share registers. It's
> quite common that some of the control registers are shared across the two
> subfunctions of a two-slot cardbus controller (and we generally don't even
> have full docs for them!)

Okay. This obviously implies that if/when cardbus bridges are
converted to async suspend/resume, the driver should make sure that the
lower-numbered devices wait for their sibling higher-numbered devices
to suspend (and vice versa for resume). Awkward though it may be.

> > The same goes for devices that don't have suspend or resume methods.
>
> Yes and no.
>
> Again, the "async_suspend" flag is done at the generic device layer, but
> 99% of all suspend/resume methods are _not_ done at that level: they are
> bus-specific functions, where the bus has a generic suspend-resume
> function that it exposes to the generic device layer, and that knows about
> the bus-specific rules.
>
> So if you are a PCI device (to take just that example - but it's true of
> just about all other buses too), and you don't have any suspend or resume
> methods, it's actually impossible to see that fact from the generic device
> layer.

Sure. That's why the async_suspend flag is set at the bus/driver
level.

> And even when you know it's PCI, our rules are actually not simple at all.
> Our rules for PCI devices (and this strictly speaking is true for bridges
> too) are rather complex:
>
> - do we have _any_ legacy PM support (ie the "direct" driver
> suspend/resume functions in the driver ops, rather than having a
> "struct dev_pm_ops" pointer)? If so, call "->suspend()"
>
> - If not - do we have that "dev_pm_ops" thing? If so, call it.
>
> - If not - just disable the device entirely _UNLESS_ you're a PCI bridge.
>
> Notice? The way things are set up, if you have no suspend routine, you'll
> not get suspended, but you will get disabled.
>
> So it's _not_ actually safe to asynchronously suspend a PCI device if that
> device has no driver or no suspend routines - because even in the absense
> of a driver and suspend routines, we'll still least disable it. And if
> there is some subtle dependency on that device that isn't obvious (say, it
> might be used indirectly for some ACPI thing), then that async suspend is
> the wrong thing to do.
>
> Subtle? Hell yes.

I don't disagree. However the subtlety lies mainly in the matter of
non-obvious dependencies. (The other stuff is all known to the PCI
core.) AFAICS there's otherwise little difference between an async
routine that does nothing and one that disables the device -- both
operations are very fast.

The ACPI relations are definitely something to worry about. It would
be a good idea, at an early stage, to add those dependencies
explicitly. I don't know enough about them to say more; perhaps Rafael
does.

As for other non-obvious dependencies... Who knows? Probably the only
way to find them is by experimentation. My guess is that they will
turn out to be connected mostly with "high-level" devices: system
devices, things on the motherboard -- generally speaking, stuff close
to the CPU. Relatively few will be associated with devices below the
level of a PCI device or equivalent.

Ideally we would figure out how to do the slow devices in parallel
without interference from fast devices having unknown dependencies.
Unfortunately this may not be possible.

> So the whole thing about "we can do PCI bridges asynchronously because
> they are obviously no-op" is kind of true - except for the "obviously"
> part. It's not obvious at all. It's rather subtle.
>
> As an example of this kind of subtlety - iirc PCIE bridges used to have
> suspend and resume bugs when we initially switched over to the "new world"
> suspend/resume exactly because they actually did things at "suspend" time
> (rather than suspend_late), and that broke devices behind them (this was
> not related to async, of course, but the point is that even when you look
> like a PCI bridge, you might be doing odd things).
>
> So just saying "let's do it asynchronously" is _not_ always guaranteed to
> be the right thing at all. It's _probably_ safe for at least regular PCI
> bridges. Cardbus bridges? Probably not, but since most modern laptop have
> just a single slot - and people who have multiple slots seldom use them
> all - most people will probably never see the problems that it _could_
> introduce.
>
> And PCIE bridges? Should be safe these days, but it wasn't quite as
> obvious, because a PCIE bridge actually has a driver unlike a regular
> plain PCI-PCI bridge.
>
> Subtle, subtle.

Indeed. Perhaps you were too hasty in suggesting that PCI bridges
should be async.

It would help a lot to see some device lists for typical machines. (If
there are such things.) Otherwise we are just blowing gas.

> > There remains a separate question: Should async devices also be forced
> > to wait for their children? I don't see why not. For PCI bridges it
> > won't make any significant difference. As long as the async code
> > doesn't have to do anything, who cares when it runs?
>
> That's why I just set the "async_resume = 1" thing.
>
> But there might actually be reasons why we care. Like the fact that we
> actually throttle the amount of parallel work we do in async_schedule().
> So doing even a "no-op" asynchronously isn't actually a no-op: while it is
> pending (and those things can be pending for a long time, since they have
> to wait for those slow devices underneath them), it can cause _other_
> async work - that isn't necessarily a no-op at all - to be then done
> synchronously.
>
> Now, admittedly our async throttling limits are high enough that the above
> kind of detail will probably never ever realy matter (default 256 worker
> threads etc). But it's an example of how practice is different from theory
> - in _theory_ it doesn't make any difference if you wait for something
> asynchronously, but in practice it could make a difference under some
> circumstances.

We certainly shouldn't be worried about side effects of async
throttling as this stage. KISS works both ways: Don't overdesign, and
don't worry about things that might crop up when you expand the design.


We have strayed off the point of your original objection: not providing
a way for devices to skip waiting for their children. This really is a
separate issue from deciding whether or not to go async. For example,
your proposed patch makes PCI bridges async but doesn't allow them to
avoid waiting for children. IMO that's a good thing.

The real issue is "blockage": synchronous devices preventing
possible concurrency among async devices. That's what you thought
making PCI bridges async would help.

In general, blockage arises in suspend when you have an async child
with a synchronous parent. The parent has to wait for the child, which
might take a long time, thereby delaying other unrelated devices.
(This explains why you wanted to make PCI bridges async -- they are the
parents of USB controllers.) For resume it's the opposite: an async
parent with synchronous children. Thus, while making PCI bridges async
might make suspend faster, it probably won't help much with resume
speed. You'd have to make the children of USB devices (SCSI hosts,
TTYs, and so on) async. Depending on the order of device registration,
of course.


Apart from all this, there's a glaring hole in the discussion so far.
You and Arjan may not have noticed it, but those of us still using
rotating media have to put up with disk resume times that are a factor
of 100 (!) larger than USB resume times. That's where the greatest
gains are to be found.

Alan Stern

--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/