Prev: A share can be mounted multiple times on the same mountpoint when using the option 'noac'
Next: parport_pc.c and parport_serial.c need improvements
From: Alan Stern on 23 Apr 2010 12:40 On Thu, 22 Apr 2010, [UTF-8] Arve Hjønnevåg wrote: > Adds /sys/power/policy that selects the behaviour of /sys/power/state. > After setting the policy to opportunistic, writes to /sys/power/state > become non-blocking requests that specify which suspend state to enter > when no suspend blockers are active. A special state, "on", stops the > process by activating the "main" suspend blocker. > --- /dev/null > +++ b/Documentation/power/suspend-blockers.txt > @@ -0,0 +1,97 @@ > +Suspend blockers > +================ > + > +Suspend blockers provide a mechanism for device drivers and user-space processes > +to prevent the system from entering suspend. By default writing to > +/sys/power/state will ignore suspend blockers. Writing "opportunistic" to > +/sys/power/policy will change the behaviour of /sys/power/state to repeatedly > +enter the requested state when no suspend blockers are active. Writing "on" to > +/sys/power/state will cancel the automatic sleep request. Suspend blockers do > +not affect sleep states entered from idle. You should document that writing "forced" to /sys/power/policy causes the /sys/power/state to return to normal operation (i.e., ignoring suspend blockers). > +In cell phones or other embedded systems where powering the screen is a > +significant drain on the battery, suspend blockers can be used to allow > +user-space to decide whether a keystroke received while the system is suspended > +should cause the screen to be turned back on or allow the system to go back into > +suspend. The description below is incomplete and consequently doesn't make sense. > Use set_irq_wake or a platform specific api to make sure the keypad > +interrupt wakes up the cpu. And as part of waking up the CPU, the screen driver's resume method is called. Presumably this will turn the screen back on. If it doesn't, you need to explain why not. > Once the keypad driver has resumed, the sequence of > +events can look like this: > + > +- The Keypad driver gets an interrupt. It then calls suspend_block on the > + keypad-scan suspend_blocker and starts scanning the keypad matrix. > +- The keypad-scan code detects a key change and reports it to the input-event > + driver. > +- The input-event driver sees the key change, enqueues an event, and calls > + suspend_block on the input-event-queue suspend_blocker. > +- The keypad-scan code detects that no keys are held and calls suspend_unblock > + on the keypad-scan suspend_blocker. > +- The user-space input-event thread returns from select/poll, calls > + suspend_block on the process-input-events suspend_blocker and then calls read > + on the input-event device. > +- The input-event driver dequeues the key-event and, since the queue is now > + empty, it calls suspend_unblock on the input-event-queue suspend_blocker. > +- The user-space input-event thread returns from read. If it determines that > + the key should leave the screen off, it calls suspend_unblock on the > + process_input_events suspend_blocker and then calls select or poll. The > + system will automatically suspend again, since now no suspend blockers are > + active. This is highly misleading. Although you _say_ this decision is about whether or not to leave the screen off, it _really_ is about whether or not to let the system automatically suspend again. In other words, the exposition confuses suspending the system with turning off the screen. Unless you can keep the two notions separate and explain more clearly what's going on, this whole section should be removed from the documentation. > +Driver API > +========== > + > +A driver can use the suspend block api by adding a suspend_blocker variable to > +its state and calling suspend_blocker_init. For instance: > +struct state { > + struct suspend_blocker suspend_blocker; > +} > + > +init() { > + suspend_blocker_init(&state->suspend_blocker, "suspend-blocker-name"); > +} > + > +Before freeing the memory, suspend_blocker_destroy must be called: > + > +uninit() { > + suspend_blocker_destroy(&state->suspend_blocker); > +} > + > +When the driver determines that it needs to run (usually in an interrupt > +handler) it calls suspend_block: > + suspend_block(&state->suspend_blocker); > + > +When it no longer needs to run it calls suspend_unblock: > + suspend_unblock(&state->suspend_blocker); > + > +Calling suspend_block when the suspend blocker is active or suspend_unblock when > +it is not active has no effect. This allows drivers to update their state and > +call suspend suspend_block or suspend_unblock based on the result. But suspend_block() and suspend_unblock() don't nest. You should mention this. > --- /dev/null > +++ b/include/linux/suspend_blocker.h > @@ -0,0 +1,60 @@ .... > +/** > + * struct suspend_blocker - the basic suspend_blocker structure > + * @flags: Tracks initialized and active state. > + * @name: Name used for debugging. > + * > + * When a suspend_blocker is active it prevents the system from entering > + * suspend. > + * > + * The suspend_blocker structure must be initialized by suspend_blocker_init() > + */ > + > +struct suspend_blocker { > +#ifdef CONFIG_SUSPEND_BLOCKERS > + atomic_t flags; > + const char *name; > +#endif Why is flags an atomic_t? Are you worried that drivers might try to activate a suspend_blocker at the same time that it is being destroyed? If this happens, does the code do the right thing? I don't think it does -- if a race occurs, suspend_block() will leave flags set to the wrong value. The same goes for suspend_unblock(). Since these routines don't nest, there is also the possibility of a race between suspend_block() and suspend_unblock(). If the race goes one way the blocker is active; the other way it isn't. Given that such problems already exist, why worry about what happens when the suspend blocker is destroyed? Alan Stern -- To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to majordomo(a)vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/
From: Arve Hjønnevåg on 23 Apr 2010 22:20 2010/4/23 Alan Stern <stern(a)rowland.harvard.edu>: > On Thu, 22 Apr 2010, [UTF-8] Arve Hj�nnev�g wrote: .... >> +Calling suspend_block when the suspend blocker is active or suspend_unblock when >> +it is not active has no effect. This allows drivers to update their state and >> +call suspend suspend_block or suspend_unblock based on the result. > > But suspend_block() and suspend_unblock() don't nest. �You should > mention this. > I'm not sure what you mean by this? I think the first sentence dictates nesting is not supported. I think the rest of your comments will be addressed in the next version of the document, that I worked with Rafael on. -- Arve Hj�nnev�g -- To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to majordomo(a)vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/
From: Alan Stern on 23 Apr 2010 22:40 On Fri, 23 Apr 2010, Arve Hj�nnev�g wrote: > 2010/4/23 Alan Stern <stern(a)rowland.harvard.edu>: > > On Thu, 22 Apr 2010, [UTF-8] Arve Hj�nnev�g wrote: > ... > >> +Calling suspend_block when the suspend blocker is active or suspend_unblock when > >> +it is not active has no effect. This allows drivers to update their state and > >> +call suspend suspend_block or suspend_unblock based on the result. > > > > But suspend_block() and suspend_unblock() don't nest. �You should > > mention this. > > > > I'm not sure what you mean by this? I think the first sentence > dictates nesting is not supported. That fact is implicit from the first sentence. Mentioning it _explicitly_ will help people to understand more easily. You don't have to add much; a parenthetical remark would be enough: Calling suspend_block when the suspend blocker is active or suspend_unblock when it is not active has no effect (i.e., these functions don't nest). This allows drivers to ... Alan Stern -- To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to majordomo(a)vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/
From: Arve Hjønnevåg on 23 Apr 2010 23:20 2010/4/23 Alan Stern <stern(a)rowland.harvard.edu>: > On Fri, 23 Apr 2010, Arve Hj�nnev�g wrote: > >> 2010/4/23 Alan Stern <stern(a)rowland.harvard.edu>: >> > On Thu, 22 Apr 2010, [UTF-8] Arve Hj�nnev�g wrote: >> ... >> >> +Calling suspend_block when the suspend blocker is active or suspend_unblock when >> >> +it is not active has no effect. This allows drivers to update their state and >> >> +call suspend suspend_block or suspend_unblock based on the result. >> > >> > But suspend_block() and suspend_unblock() don't nest. �You should >> > mention this. >> > >> >> I'm not sure what you mean by this? I think the first sentence >> dictates nesting is not supported. > > That fact is implicit from the first sentence. �Mentioning it > _explicitly_ will help people to understand more easily. �You don't > have to add much; a parenthetical remark would be enough: > > � � � �Calling suspend_block when the suspend blocker is active or > � � � �suspend_unblock when it is not active has no effect (i.e., > � � � �these functions don't nest). �This allows drivers to ... > OK. -- Arve Hj�nnev�g -- To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to majordomo(a)vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/
From: Alan Stern on 28 Apr 2010 15:20
On Tue, 27 Apr 2010, [UTF-8] Arve Hjønnevåg wrote: > +For example, in cell phones or other embedded systems, where powering the screen > +is a significant drain on the battery, suspend blockers can be used to allow > +user-space to decide whether a keystroke received while the system is suspended > +should cause the screen to be turned back on or allow the system to go back into > +suspend. Use set_irq_wake or a platform specific api to make sure the keypad > +interrupt wakes up the cpu. Once the keypad driver has resumed, the sequence of > +events can look like this: > + > +- The Keypad driver gets an interrupt. It then calls suspend_block on the > + keypad-scan suspend_blocker and starts scanning the keypad matrix. > +- The keypad-scan code detects a key change and reports it to the input-event > + driver. > +- The input-event driver sees the key change, enqueues an event, and calls > + suspend_block on the input-event-queue suspend_blocker. > +- The keypad-scan code detects that no keys are held and calls suspend_unblock > + on the keypad-scan suspend_blocker. > +- The user-space input-event thread returns from select/poll, calls > + suspend_block on the process-input-events suspend_blocker and then calls read > + on the input-event device. > +- The input-event driver dequeues the key-event and, since the queue is now > + empty, it calls suspend_unblock on the input-event-queue suspend_blocker. > +- The user-space input-event thread returns from read. If it determines that > + the key should leave the screen off, it calls suspend_unblock on the > + process_input_events suspend_blocker and then calls select or poll. The > + system will automatically suspend again, since now no suspend blockers are > + active. > + > + Key pressed Key released > + | | > +keypad-scan ++++++++++++++++++ > +input-event-queue +++ +++ > +process-input-events +++ +++ This is better than before, but it still isn't ideal. Here's what I mean: > suspend blockers can be used to allow > +user-space to decide whether a keystroke received while the system is suspended > +should cause the screen to be turned back on or allow the system to go back into > +suspend. That's not right. Handling the screen doesn't need suspend blockers: The program decides what to do and then either turns on the screen or else writes "mem" to /sys/power/state. What suspend blockers add is the ability to resolve races and satisfy multiple constraints when going into suspend -- which has nothing to do with operating the screen. I _think_ what you're trying to get at can be expressed this way: Here's an example showing how a cell phone or other embedded system can handle keystrokes (or other input events) in the presence of suspend blockers. Use set_irq_wake... ... - The user-space input-event thread returns from read. It carries out whatever activities are appropriate (for example, powering up the display screen, running other programs, and so on). When it is finished, it calls suspend_unblock on the process_input_events suspend_blocker and then calls select or poll. The system will automatically suspend again when it is idle and no suspend blockers remain active. > +/** > + * suspend_block() - Block suspend > + * @blocker: The suspend blocker to use > + * > + * It is safe to call this function from interrupt context. > + */ > +void suspend_block(struct suspend_blocker *blocker) > +{ > + unsigned long irqflags; > + > + if (WARN_ON(!(blocker->flags & SB_INITIALIZED))) > + return; > + > + spin_lock_irqsave(&list_lock, irqflags); > + blocker->flags |= SB_ACTIVE; > + list_del(&blocker->link); > + > + if (debug_mask & DEBUG_SUSPEND_BLOCKER) > + pr_info("suspend_block: %s\n", blocker->name); > + > + list_add(&blocker->link, &active_blockers); Here and in suspend_unblock(), you can use list_move() in place of list_del() followed by list_add(). Alan Stern -- To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to majordomo(a)vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/ |