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 _______________________________________________ linux-pm mailing list linux-pm@xxxxxxxxxxxxxxxxxxxxxxxxxx https://lists.linux-foundation.org/mailman/listinfo/linux-pm