rumcake

rumcake/hw/
mod.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
//! Utilities for interfacing with hardware.

#[cfg(all(not(feature = "stm32"), not(feature = "nrf"), not(feature = "rp")))]
compile_error!("Please enable the appropriate feature flag for the chip you're using.");

#[cfg(any(
    all(feature = "stm32", feature = "nrf"),
    all(feature = "nrf", feature = "rp"),
    all(feature = "rp", feature = "stm32")
))]
compile_error!("Please enable only one chip feature flag.");

#[cfg_attr(feature = "stm32", path = "mcu/stm32.rs")]
#[cfg_attr(feature = "nrf", path = "mcu/nrf.rs")]
#[cfg_attr(feature = "rp", path = "mcu/rp.rs")]
pub mod platform;

use crate::hw::platform::jump_to_bootloader;
use crate::State;
use core::cell::UnsafeCell;
use core::mem::MaybeUninit;
use core::ptr::read_volatile;
use core::ptr::write_volatile;
use defmt::info;
use embassy_futures::select;
use embassy_sync::channel::Channel;
use embassy_sync::signal::Signal;
use embassy_time::Timer;
use embedded_hal::digital::v2::OutputPin;

use platform::RawMutex;
use usbd_human_interface_device::device::consumer::MultipleConsumerReport;
use usbd_human_interface_device::device::keyboard::NKROBootKeyboardReport;

/// State that contains the current battery level. `rumcake` may or may not use this static
/// internally, depending on what MCU is being used. The contents of this state is usually set by a
/// task in the [`platform`] module. For example, on nRF5x-based MCUs, this is controlled by a task
/// called `adc_task`.
pub static BATTERY_LEVEL_STATE: State<u8> = State::new(
    100,
    &[
        #[cfg(feature = "display")]
        &crate::display::BATTERY_LEVEL_LISTENER,
        #[cfg(feature = "bluetooth")]
        &crate::bluetooth::BATTERY_LEVEL_LISTENER,
    ],
);

/// Possible settings used to determine how the firmware will choose the destination for HID
/// reports.
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
#[non_exhaustive]
pub enum OutputMode {
    Usb,
    Bluetooth,
}

/// State that contains the desired output mode. This configures how the firmware will decide to
/// send HID reports. This doesn't not represent the actual destination of HID reports. Use
/// [`CURRENT_OUTPUT_STATE`] for that.
pub static OUTPUT_MODE_STATE: State<OutputMode> = State::new(
    if cfg!(feature = "bluetooth") {
        OutputMode::Bluetooth
    } else {
        OutputMode::Usb
    },
    &[
        &OUTPUT_MODE_STATE_LISTENER,
        #[cfg(feature = "display")]
        &crate::display::OUTPUT_MODE_STATE_LISTENER,
    ],
);

/// Possible destinations for HID reports.
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum HIDOutput {
    Usb,
    Bluetooth,
}

/// State that contains the current destination of HID reports.
pub static CURRENT_OUTPUT_STATE: State<Option<HIDOutput>> = State::new(
    None,
    &[
        #[cfg(feature = "usb")]
        &crate::usb::KB_CURRENT_OUTPUT_STATE_LISTENER,
        #[cfg(feature = "usb")]
        &crate::usb::CONSUMER_CURRENT_OUTPUT_STATE_LISTENER,
        #[cfg(all(feature = "usb", feature = "via"))]
        &crate::usb::VIA_CURRENT_OUTPUT_STATE_LISTENER,
        #[cfg(feature = "bluetooth")]
        &crate::bluetooth::CURRENT_OUTPUT_STATE_LISTENER,
    ],
);

pub(crate) static OUTPUT_MODE_STATE_LISTENER: Signal<RawMutex, ()> = Signal::new();
pub(crate) static USB_RUNNING_STATE_LISTENER: Signal<RawMutex, ()> = Signal::new();
pub(crate) static BLUETOOTH_CONNECTED_STATE_LISTENER: Signal<RawMutex, ()> = Signal::new();

pub(crate) static HARDWARE_COMMAND_CHANNEL: Channel<RawMutex, HardwareCommand, 2> = Channel::new();

#[derive(Debug, Clone, Copy)]
#[non_exhaustive]
/// An enumeration of possible commands that will be processed by the bluetooth task.
pub enum HardwareCommand {
    /// Switch between USB and Bluetooth operation.
    ///
    /// This will **NOT** disconnect your keyboard from your host device. It
    /// will simply determine which device the HID reports get sent to.
    ToggleOutput = 0,
    /// Switch to USB operation.
    ///
    /// If your keyboard is connected to a bluetooth device, this will **NOT** disconnect your
    /// keyboard from it. It will simply output the HID reports to the connected USB device.
    OutputUSB = 1,
    /// Switch to bluetooth operation.
    ///
    /// If your keyboard is connected to a USB device, this will **NOT** disconnect your keyboard
    /// from it. It will simply output the HID reports to the connected bluetooth device.
    OutputBluetooth = 2,
}

pub async fn output_switcher() {
    // This task doesn't need to run if only one of USB or Bluetooth is enabled, and there are no
    // HardwareCommand members on the user's layout.
    let switcher_fut = async {
        loop {
            let output = match OUTPUT_MODE_STATE.get().await {
                #[cfg(feature = "usb")]
                OutputMode::Usb => {
                    if crate::usb::USB_RUNNING_STATE.get().await {
                        Some(HIDOutput::Usb)
                    } else {
                        None
                    }
                }
                #[cfg(feature = "bluetooth")]
                OutputMode::Bluetooth => {
                    if crate::bluetooth::BLUETOOTH_CONNECTED_STATE.get().await {
                        Some(HIDOutput::Bluetooth)
                    } else {
                        None
                    }
                }
                #[allow(unreachable_patterns)]
                _ => None,
            };

            CURRENT_OUTPUT_STATE.set(output).await;
            info!("[HW] Output updated: {:?}", defmt::Debug2Format(&output));

            // Wait for a change in state before attempting to update the output again.
            select::select3(
                USB_RUNNING_STATE_LISTENER.wait(),
                BLUETOOTH_CONNECTED_STATE_LISTENER.wait(),
                OUTPUT_MODE_STATE_LISTENER.wait(),
            )
            .await;
        }
    };

    let command_fut = async {
        loop {
            match HARDWARE_COMMAND_CHANNEL.receive().await {
                HardwareCommand::ToggleOutput => {
                    OUTPUT_MODE_STATE.set(match OUTPUT_MODE_STATE.get().await {
                        OutputMode::Usb => OutputMode::Bluetooth,
                        OutputMode::Bluetooth => OutputMode::Usb,
                    });
                }
                HardwareCommand::OutputUSB => {
                    OUTPUT_MODE_STATE.set(OutputMode::Usb);
                }
                HardwareCommand::OutputBluetooth => {
                    OUTPUT_MODE_STATE.set(OutputMode::Bluetooth);
                }
            }
        }
    };

    select::select(switcher_fut, command_fut).await;
    info!("[HW] Output switching task has failed. This should not happen.");
}

const BOOTLOADER_MAGIC: u32 = 0xDEADBEEF;

#[link_section = ".uninit.FLAG"]
static mut FLAG: UnsafeCell<MaybeUninit<u32>> = UnsafeCell::new(MaybeUninit::uninit());

pub async unsafe fn check_double_tap_bootloader(timeout: u64) {
    if read_volatile(FLAG.get().cast::<u32>()) == BOOTLOADER_MAGIC {
        write_volatile(FLAG.get().cast(), 0);

        jump_to_bootloader();
    }

    write_volatile(FLAG.get().cast(), BOOTLOADER_MAGIC);

    Timer::after_millis(timeout).await;

    write_volatile(FLAG.get().cast(), 0);
}

extern "C" {
    /// This static value will have an address equal to the `__config_start` address in your
    /// `memory.x` file. You must set this, along with [`__config_end`], if you're using on-chip
    /// flash with the storage task (which is the default). Keep in mind that the start and end
    /// address must be relative to the address of your chip's flash. For example, on STM32F072CBx,
    /// flash memory is located at `0x08000000`, so if you want your config data to start at
    /// `0x08100000`, your start address must be `0x00100000`.
    pub static __config_start: u32;
    /// This static value will have an address equal to the `__config_end` address in your
    /// `memory.x` file. If you want to know what value to set this to in `memory.x`, take
    /// [`__config_start`], and add the size of your config section, in bytes.
    pub static __config_end: u32;
}

pub struct Multiplexer<T, const P: usize> {
    cur_channel: u8,
    pins: [Option<T>; P],
    en: Option<T>,
}

impl<E, T: OutputPin<Error = E>, const P: usize> Multiplexer<T, P> {
    pub fn new(pins: [Option<T>; P], en: Option<T>) -> Self {
        let mut multiplexer = Self {
            pins,
            en,
            cur_channel: 0,
        };
        let _ = multiplexer.select_channel(0);
        multiplexer
    }

    pub fn select_channel(&mut self, mut channel: u8) -> Result<(), E> {
        for i in 0..P {
            if let Some(ref mut pin) = self.pins[i] {
                if channel & 0x01 == 0x01 {
                    pin.set_high()?;
                } else {
                    pin.set_low()?;
                }
            }
            channel >>= 1;
        }
        self.cur_channel = channel;
        Ok(())
    }

    pub fn enable(&mut self) -> Result<(), E> {
        if let Some(ref mut en) = self.en {
            en.set_low()?;
        }
        Ok(())
    }

    pub fn disable(&mut self) -> Result<(), E> {
        if let Some(ref mut en) = self.en {
            en.set_high()?;
        }
        Ok(())
    }
}

pub trait HIDDevice {
    fn get_keyboard_report_send_channel() -> &'static Channel<RawMutex, NKROBootKeyboardReport, 1> {
        static KEYBOARD_REPORT_HID_SEND_CHANNEL: Channel<RawMutex, NKROBootKeyboardReport, 1> =
            Channel::new();
        &KEYBOARD_REPORT_HID_SEND_CHANNEL
    }

    fn get_consumer_report_send_channel() -> &'static Channel<RawMutex, MultipleConsumerReport, 1> {
        static CONSUMER_REPORT_HID_SEND_CHANNEL: Channel<RawMutex, MultipleConsumerReport, 1> =
            Channel::new();
        &CONSUMER_REPORT_HID_SEND_CHANNEL
    }

    #[cfg(feature = "via")]
    fn get_via_hid_send_channel() -> &'static Channel<RawMutex, [u8; 32], 1> {
        static VIA_REPORT_HID_SEND_CHANNEL: Channel<RawMutex, [u8; 32], 1> = Channel::new();
        &VIA_REPORT_HID_SEND_CHANNEL
    }

    #[cfg(feature = "via")]
    fn get_via_hid_receive_channel() -> &'static Channel<RawMutex, [u8; 32], 1> {
        static VIA_REPORT_HID_RECEIVE_CHANNEL: Channel<RawMutex, [u8; 32], 1> = Channel::new();
        &VIA_REPORT_HID_RECEIVE_CHANNEL
    }
}