Skip to content

Commit

Permalink
hybrid boot keyboard
Browse files Browse the repository at this point in the history
Make the Boot Keyboard the only keyboard, and make it have a hybrid
report format in Report Protoocol, while still ending Boot Reports in
Boot Protocol. The hybrid reports have the Boot Report as a prefix,
which might help hosts that don't request Boot Protocol but still
expect to see it.

Adjust USBQuirks to switch between always Boot Protocol and the
hybrid report.

Add visual feedback about the target mode to USBQuirks.

Signed-off-by: Taylor Yu <[email protected]>
  • Loading branch information
tlyu committed Dec 15, 2023
1 parent a0ec7e1 commit 41f02c8
Show file tree
Hide file tree
Showing 14 changed files with 514 additions and 728 deletions.
37 changes: 29 additions & 8 deletions plugins/Kaleidoscope-USB-Quirks/README.md
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
# USB-Quirks

USB-Quirks provides a few methods to deal with more obscure parts of the USB spec, such as changing between `Boot` and `Report` protocols. These are in a separate plugin, because these features are not part of the USB spec, and are often workarounds for various issues. See the provided methods for more information about what they're useful for.
USB-Quirks provides a few methods to deal with more obscure parts of the USB spec, such as changing the behavior around the boot protocol. These are in a separate plugin, because these features are not part of the USB spec, and are often workarounds for various issues. See the provided methods for more information about what they're useful for.

## Using the plugin

Expand Down Expand Up @@ -29,11 +29,32 @@ The plugin provides one object, `USBQuirks`, which provides the following method
### `.toggleKeyboardProtocol()`
> Toggle between `Boot` and `Report` protocol by detaching, and then
> re-attaching the USB devices, and setting the `BootKeyboard` protocol
> in between.
> Toggle whether the keyboard is able to send extended key reports (the
> default), or instead always sends boot reports, regardless of the
> protocol requested by the host. Switching the toggle causes the keyboard
> to detach and then re-attach to the host. (This re-attach is necessary to
> force re-enumeration with a different Report Descriptor.)
>
> This is most useful when one needs to have a boot keyboard, when one's in a
> BIOS, boot loader, or early password prompt or the like, and the host does not
> explicitly request the boot protocol for one reason or the other. With this
> toggle, we can switch between the two on-demand.
> Switching the toggle also lights up a key indicating the mode being
> switched to: by default, `B` for boot reports only, and `N` for extended
> reports enabled.
>
> The extended key report supports n-key rollover (NKRO), and is actually a
> hybrid, having a prefix containing the boot report, for compatibility
> with older hosts. The boot report only supports 6-key rollover (6KRO),
> and is meant to support constrained hosts, such as BIOS, UEFI, or other
> pre-boot environments. The keyboard changes protocols as requested by the
> host.
>
> The USB HID specification requires that hosts explicitly request boot
> protocol if they need it, and that devices default to the non-boot
> protocol. Some hosts do not follow the specification, and expect boot
> protocol, even without requesting it. The backwards compatibility prefix
> of the hybrid extended report should accommodate some of these hosts.
> This toggle helps to work with hosts that neither request boot protocol
> nor tolerate the longer hybrid report.
### `.setKeys(Key boot_led, Key nkro_led)`
> Set which keys to light up to indicate the target mode. Defaults to
> `(Key_B, Key_N)`.
Original file line number Diff line number Diff line change
Expand Up @@ -23,16 +23,62 @@
#include "kaleidoscope/Runtime.h" // for Runtime, Runtime_
#include "kaleidoscope/device/device.h" // for Base<>::HID, VirtualProps::HID
#include "kaleidoscope/driver/hid/keyboardio/Keyboard.h" // for Keyboard
#include "kaleidoscope/key_defs.h" // for Key
#include "kaleidoscope/plugin/LEDControl.h" // for LEDControl

namespace kaleidoscope {
namespace plugin {

KeyAddr USBQuirks::key_boot_addr = KeyAddr::none();
KeyAddr USBQuirks::key_nkro_addr = KeyAddr::none();

static KeyAddr findKey(Key search_key) {
for (auto key_addr : KeyAddr::all()) {
Key k = Layer.lookupOnActiveLayer(key_addr);

if (k == search_key) {
return key_addr;
}
}
return KeyAddr::none();
}

void USBQuirks::setKeys(Key boot_led, Key nkro_led) {
key_boot_addr = findKey(boot_led);
key_nkro_addr = findKey(nkro_led);
}

EventHandlerResult USBQuirks::onSetup() {
setKeys(Key_B, Key_N);
return EventHandlerResult::OK;
}

void USBQuirks::toggleKeyboardProtocol() {
uint8_t new_protocol = !Runtime.hid().keyboard().getProtocol();
KeyAddr key_addr;
uint8_t new_bootonly = !Runtime.hid().keyboard().getBootOnly();

if (new_bootonly) {
key_addr = key_boot_addr;
} else {
key_addr = key_nkro_addr;
}
::LEDControl.disable();
if (key_addr.isValid()) {
::LEDControl.setCrgbAt(key_addr, CRGB(0, 0, 255));
}
Runtime.device().syncLeds();
/*
* Release keys, because after detach, Windows 10 remembers keys that
* were pressed (from the MagicCombo that activated this function).
*/
Runtime.hid().keyboard().releaseAllKeys();
Runtime.hid().keyboard().sendReport();
delay(10);
Runtime.detachFromHost();
Runtime.hid().keyboard().setDefaultProtocol(new_protocol);
Runtime.hid().keyboard().setBootOnly(new_bootonly);
delay(1000);
::LEDControl.set_all_leds_to(CRGB(0, 0, 0));
::LEDControl.enable();
Runtime.attachToHost();
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -17,13 +17,22 @@

#pragma once

#include "kaleidoscope/plugin.h" // for Plugin
#include "kaleidoscope/KeyAddr.h" // for KeyAddr
#include "kaleidoscope/event_handler_result.h" // for EventHandlerResult
#include "kaleidoscope/key_defs.h" // for Key
#include "kaleidoscope/plugin.h" // for Plugin

namespace kaleidoscope {
namespace plugin {
class USBQuirks : public kaleidoscope::Plugin {
public:
void toggleKeyboardProtocol();
EventHandlerResult onSetup();
static void setKeys(Key boot_led, Key nkro_led);

private:
static KeyAddr key_boot_addr;
static KeyAddr key_nkro_addr;
};

} // namespace plugin
Expand Down
Loading

0 comments on commit 41f02c8

Please sign in to comment.