Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Support remote wakeup from suspend #30

Open
wants to merge 3 commits into
base: master
Choose a base branch
from
Open
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
45 changes: 40 additions & 5 deletions src/bus.rs
Original file line number Diff line number Diff line change
Expand Up @@ -71,6 +71,44 @@ impl<USB: UsbPeripheral> UsbBus<USB> {
regs.cntr.modify(|_, w| w.pdwn().bit(pdwn));
});
}

/// Enter suspend low-power mode
///
/// This should be used when in suspend mode if low-power mode needs to be entered during
/// suspend (bus-powered device). Application should call this when it is ready to decrease
/// power consumption to meet power consumption requirements of the USB suspend condition
/// (e.g. disable system clocks or reduce their frequency). When wake up event is received
/// low power mode will be automatically disabled.
///
/// Will not enter low-power mode if not in suspend state. Returns `true` if entered.
pub fn suspend_low_power_mode(&self) -> bool {
interrupt::free(|cs| {
let regs = self.regs.borrow(cs);
if regs.cntr.read().fsusp().is_suspend() {
regs.cntr.modify(|_, w| w.lpmode().set_bit());
true
} else {
false
}
})
}

/// Set/clear remote wakeup bit
///
/// Allows to modify CNTR.RESUME bit from device to perform remote wake up from suspend
/// (e.g. on keyboard/mouse input). To perform remote wake up: when a condition is met
/// during suspend mode, use this method with `true` to set the RESUME bit. Then, after
/// waiting between 1-15 ms (see reference manual) call it again with `false` to clear
/// the bit.
///
/// This method will not set the bit if device is not suspended because performing remote
/// wake up in other mode is invalid and host will most likely disable such a device.
pub fn remote_wakeup(&self, resume: bool) {
interrupt::free(|cs| {
self.regs.borrow(cs)
.cntr.modify(|r, w| w.resume().bit(resume && r.fsusp().is_suspend()));
})
}
}

impl<USB: UsbPeripheral> usb_device::bus::UsbBus for UsbBus<USB> {
Expand Down Expand Up @@ -218,10 +256,7 @@ impl<USB: UsbPeripheral> usb_device::bus::UsbBus for UsbBus<USB> {

if v.ctr_tx().bit_is_set() {
ep_in_complete |= bit;

interrupt::free(|cs| {
ep.clear_ctr_tx(cs);
});
ep.clear_ctr_tx(cs);
}

bit <<= 1;
Expand Down Expand Up @@ -288,7 +323,7 @@ impl<USB: UsbPeripheral> usb_device::bus::UsbBus for UsbBus<USB> {
self.regs
.borrow(cs)
.cntr
.modify(|_, w| w.fsusp().set_bit().lpmode().set_bit());
.modify(|_, w| w.fsusp().set_bit());
});
}

Expand Down