ble nano hid over gatt
Dependencies: BLE_API mbed-dev nRF51822
HIDController_BLE.cpp@15:70bf079d3ee1, 2016-08-22 (annotated)
- Committer:
- cho45
- Date:
- Mon Aug 22 02:29:37 2016 +0000
- Revision:
- 15:70bf079d3ee1
- Parent:
- 13:b0ffdf2012b9
- Child:
- 16:345eebc4f259
dfu unused pins
Who changed what in which revision?
User | Revision | Line number | New contents of line |
---|---|---|---|
cho45 |
6:f1c3ea8bc850 | 1 | #include "mbed.h" |
cho45 |
6:f1c3ea8bc850 | 2 | #include "BLE.h" |
cho45 |
6:f1c3ea8bc850 | 3 | #include "KeyboardService.h" |
cho45 |
6:f1c3ea8bc850 | 4 | #include "BatteryService.h" |
cho45 |
6:f1c3ea8bc850 | 5 | #include "DeviceInformationService.h" |
cho45 | 15:70bf079d3ee1 | 6 | #include "DFUService.h" |
cho45 |
6:f1c3ea8bc850 | 7 | #include "HIDController_BLE.h" |
cho45 |
6:f1c3ea8bc850 | 8 | |
cho45 |
6:f1c3ea8bc850 | 9 | static const char MODEL_NAME[] = "keyboard"; |
cho45 |
6:f1c3ea8bc850 | 10 | static const char SERIAL_NUMBER[] = "X00000"; |
cho45 |
6:f1c3ea8bc850 | 11 | static const char HARDWARE_REVISION[] = "0.1"; |
cho45 |
6:f1c3ea8bc850 | 12 | static const char FIRMWARE_REVISION[] = "0.1"; |
cho45 |
6:f1c3ea8bc850 | 13 | static const char SOFTWARE_REVISION[] = "0.0"; |
cho45 |
6:f1c3ea8bc850 | 14 | |
cho45 |
6:f1c3ea8bc850 | 15 | static const uint8_t DEVICE_NAME[] = "my keyboard"; |
cho45 |
6:f1c3ea8bc850 | 16 | static const uint8_t SHORT_DEVICE_NAME[] = "kbd1"; |
cho45 |
6:f1c3ea8bc850 | 17 | |
cho45 |
6:f1c3ea8bc850 | 18 | static const bool ENABLE_BONDING = true; |
cho45 |
6:f1c3ea8bc850 | 19 | static const bool REQUIRE_MITM = true; |
cho45 |
6:f1c3ea8bc850 | 20 | static const uint8_t PASSKEY[6] = {'1','2','3','4','5','6'}; // must be 6-digits number |
cho45 |
6:f1c3ea8bc850 | 21 | |
cho45 |
6:f1c3ea8bc850 | 22 | static const uint16_t uuid16_list[] = { |
cho45 |
6:f1c3ea8bc850 | 23 | GattService::UUID_HUMAN_INTERFACE_DEVICE_SERVICE, |
cho45 |
6:f1c3ea8bc850 | 24 | GattService::UUID_DEVICE_INFORMATION_SERVICE, |
cho45 |
6:f1c3ea8bc850 | 25 | GattService::UUID_BATTERY_SERVICE |
cho45 |
6:f1c3ea8bc850 | 26 | }; |
cho45 |
6:f1c3ea8bc850 | 27 | |
cho45 |
6:f1c3ea8bc850 | 28 | static KeyboardService* keyboardService; |
cho45 |
6:f1c3ea8bc850 | 29 | static BatteryService* batteryService; |
cho45 |
6:f1c3ea8bc850 | 30 | static DeviceInformationService* deviceInformationService; |
cho45 | 15:70bf079d3ee1 | 31 | static DFUService* dfuService; |
cho45 |
6:f1c3ea8bc850 | 32 | |
cho45 | 13:b0ffdf2012b9 | 33 | static BLEProtocol::Address_t peerAddress; |
cho45 | 13:b0ffdf2012b9 | 34 | |
cho45 |
6:f1c3ea8bc850 | 35 | static void updateBatteryLevel() { |
cho45 |
6:f1c3ea8bc850 | 36 | if (!batteryService) return; |
cho45 |
6:f1c3ea8bc850 | 37 | static const float BATTERY_MAX = 2.4; |
cho45 |
6:f1c3ea8bc850 | 38 | static const float REFERNECE = 1.2; |
cho45 |
6:f1c3ea8bc850 | 39 | static const float PRESCALE = 3; |
cho45 |
6:f1c3ea8bc850 | 40 | |
cho45 |
6:f1c3ea8bc850 | 41 | NRF_ADC->ENABLE = ADC_ENABLE_ENABLE_Enabled; |
cho45 |
6:f1c3ea8bc850 | 42 | |
cho45 |
6:f1c3ea8bc850 | 43 | // Use internal 1.2V reference for batteryInput |
cho45 |
6:f1c3ea8bc850 | 44 | // 1/3 pre-scaled input and 1.2V internal band gap reference |
cho45 |
6:f1c3ea8bc850 | 45 | // ref. mbed-src/targets/hal/TARGET_NORDIC/TARGET_MCU_NRF51822/analogin_api.c |
cho45 |
6:f1c3ea8bc850 | 46 | NRF_ADC->CONFIG = |
cho45 |
6:f1c3ea8bc850 | 47 | (ADC_CONFIG_RES_10bit << ADC_CONFIG_RES_Pos) | |
cho45 |
6:f1c3ea8bc850 | 48 | // Use VDD 1/3 for input |
cho45 |
6:f1c3ea8bc850 | 49 | (ADC_CONFIG_INPSEL_SupplyOneThirdPrescaling << ADC_CONFIG_INPSEL_Pos) | |
cho45 |
6:f1c3ea8bc850 | 50 | // Use internal band gap for reference |
cho45 |
6:f1c3ea8bc850 | 51 | (ADC_CONFIG_REFSEL_VBG << ADC_CONFIG_REFSEL_Pos) | |
cho45 |
6:f1c3ea8bc850 | 52 | (ADC_CONFIG_EXTREFSEL_None << ADC_CONFIG_EXTREFSEL_Pos); |
cho45 |
6:f1c3ea8bc850 | 53 | |
cho45 |
6:f1c3ea8bc850 | 54 | // Start ADC |
cho45 |
6:f1c3ea8bc850 | 55 | NRF_ADC->TASKS_START = 1; |
cho45 |
6:f1c3ea8bc850 | 56 | while (((NRF_ADC->BUSY & ADC_BUSY_BUSY_Msk) >> ADC_BUSY_BUSY_Pos) == ADC_BUSY_BUSY_Busy) { |
cho45 |
6:f1c3ea8bc850 | 57 | // busy loop |
cho45 |
6:f1c3ea8bc850 | 58 | } |
cho45 |
6:f1c3ea8bc850 | 59 | |
cho45 |
6:f1c3ea8bc850 | 60 | // Read ADC result |
cho45 |
6:f1c3ea8bc850 | 61 | uint16_t raw10bit = static_cast<uint16_t>(NRF_ADC->RESULT); |
cho45 |
6:f1c3ea8bc850 | 62 | float ratio = raw10bit / static_cast<float>(1<<10); |
cho45 |
6:f1c3ea8bc850 | 63 | |
cho45 |
6:f1c3ea8bc850 | 64 | float batteryVoltage = ratio * (REFERNECE * PRESCALE); |
cho45 |
6:f1c3ea8bc850 | 65 | float percentage = (batteryVoltage / BATTERY_MAX) * 100; |
cho45 |
6:f1c3ea8bc850 | 66 | if (percentage > 100) { |
cho45 |
6:f1c3ea8bc850 | 67 | percentage = 100; |
cho45 |
6:f1c3ea8bc850 | 68 | } |
cho45 |
6:f1c3ea8bc850 | 69 | printf("updateBatteryLevel %f V : %d/100\r\n", batteryVoltage, static_cast<uint8_t>(percentage)); |
cho45 |
6:f1c3ea8bc850 | 70 | batteryService->updateBatteryLevel(static_cast<uint8_t>(percentage)); |
cho45 |
6:f1c3ea8bc850 | 71 | } |
cho45 |
6:f1c3ea8bc850 | 72 | |
cho45 |
6:f1c3ea8bc850 | 73 | |
cho45 |
6:f1c3ea8bc850 | 74 | static void onConnect(const Gap::ConnectionCallbackParams_t *params) { |
cho45 |
6:f1c3ea8bc850 | 75 | printf("onConnect\r\n"); |
cho45 | 13:b0ffdf2012b9 | 76 | peerAddress.type = params->peerAddrType; |
cho45 | 13:b0ffdf2012b9 | 77 | memcpy(peerAddress.address, params->peerAddr, Gap::ADDR_LEN); |
cho45 |
6:f1c3ea8bc850 | 78 | } |
cho45 |
6:f1c3ea8bc850 | 79 | |
cho45 |
6:f1c3ea8bc850 | 80 | static void onDisconnect(const Gap::DisconnectionCallbackParams_t *params) { |
cho45 |
6:f1c3ea8bc850 | 81 | printf("onDisconnect\r\n"); |
cho45 |
6:f1c3ea8bc850 | 82 | BLE::Instance(BLE::DEFAULT_INSTANCE).gap().startAdvertising(); |
cho45 |
6:f1c3ea8bc850 | 83 | } |
cho45 |
6:f1c3ea8bc850 | 84 | |
cho45 |
6:f1c3ea8bc850 | 85 | static void onTimeout(const Gap::TimeoutSource_t source) { |
cho45 |
6:f1c3ea8bc850 | 86 | printf("onTimeout\r\n"); |
cho45 |
6:f1c3ea8bc850 | 87 | BLE::Instance(BLE::DEFAULT_INSTANCE).gap().startAdvertising(); |
cho45 |
6:f1c3ea8bc850 | 88 | } |
cho45 |
6:f1c3ea8bc850 | 89 | |
cho45 |
6:f1c3ea8bc850 | 90 | static void passkeyDisplayCallback(Gap::Handle_t handle, const SecurityManager::Passkey_t passkey) { |
cho45 |
6:f1c3ea8bc850 | 91 | printf("Input passKey: "); |
cho45 |
6:f1c3ea8bc850 | 92 | for (unsigned i = 0; i < Gap::ADDR_LEN; i++) { |
cho45 |
6:f1c3ea8bc850 | 93 | printf("%c", passkey[i]); |
cho45 |
6:f1c3ea8bc850 | 94 | } |
cho45 |
6:f1c3ea8bc850 | 95 | printf("\r\n"); |
cho45 |
6:f1c3ea8bc850 | 96 | } |
cho45 |
6:f1c3ea8bc850 | 97 | |
cho45 |
6:f1c3ea8bc850 | 98 | static void securitySetupCompletedCallback(Gap::Handle_t handle, SecurityManager::SecurityCompletionStatus_t status) { |
cho45 |
6:f1c3ea8bc850 | 99 | if (status == SecurityManager::SEC_STATUS_SUCCESS) { |
cho45 |
6:f1c3ea8bc850 | 100 | printf("Security success %d\r\n", status); |
cho45 | 13:b0ffdf2012b9 | 101 | |
cho45 | 13:b0ffdf2012b9 | 102 | printf("Set whitelist\r\n"); |
cho45 | 13:b0ffdf2012b9 | 103 | Gap::Whitelist_t whitelist; |
cho45 | 13:b0ffdf2012b9 | 104 | whitelist.size = 1; |
cho45 | 13:b0ffdf2012b9 | 105 | whitelist.capacity = 1; |
cho45 | 13:b0ffdf2012b9 | 106 | whitelist.addresses = &peerAddress; |
cho45 | 13:b0ffdf2012b9 | 107 | |
cho45 | 13:b0ffdf2012b9 | 108 | BLE::Instance(BLE::DEFAULT_INSTANCE).gap().setWhitelist(whitelist); |
cho45 | 13:b0ffdf2012b9 | 109 | printf("Set Advertising Policy Mode\r\n"); |
cho45 | 13:b0ffdf2012b9 | 110 | // BLE::Instance(BLE::DEFAULT_INSTANCE).gap().setAdvertisingPolicyMode(Gap::ADV_POLICY_FILTER_SCAN_REQS); |
cho45 | 13:b0ffdf2012b9 | 111 | // BLE::Instance(BLE::DEFAULT_INSTANCE).gap().setAdvertisingPolicyMode(Gap::ADV_POLICY_FILTER_CONN_REQS); |
cho45 | 13:b0ffdf2012b9 | 112 | BLE::Instance(BLE::DEFAULT_INSTANCE).gap().setAdvertisingPolicyMode(Gap::ADV_POLICY_FILTER_ALL_REQS); |
cho45 | 13:b0ffdf2012b9 | 113 | |
cho45 |
6:f1c3ea8bc850 | 114 | } else { |
cho45 |
6:f1c3ea8bc850 | 115 | printf("Security failed %d\r\n", status); |
cho45 |
6:f1c3ea8bc850 | 116 | } |
cho45 |
6:f1c3ea8bc850 | 117 | } |
cho45 |
6:f1c3ea8bc850 | 118 | |
cho45 |
6:f1c3ea8bc850 | 119 | static void securitySetupInitiatedCallback(Gap::Handle_t, bool allowBonding, bool requireMITM, SecurityManager::SecurityIOCapabilities_t iocaps) { |
cho45 |
6:f1c3ea8bc850 | 120 | printf("Security setup initiated\r\n"); |
cho45 |
6:f1c3ea8bc850 | 121 | } |
cho45 |
6:f1c3ea8bc850 | 122 | |
cho45 |
6:f1c3ea8bc850 | 123 | static void bleInitComplete(BLE::InitializationCompleteCallbackContext *params) { |
cho45 |
6:f1c3ea8bc850 | 124 | // https://developer.mbed.org/compiler/#nav:/keyboard/BLE_API/ble/blecommon.h; |
cho45 |
6:f1c3ea8bc850 | 125 | ble_error_t error; |
cho45 |
6:f1c3ea8bc850 | 126 | BLE &ble = params->ble; |
cho45 |
6:f1c3ea8bc850 | 127 | |
cho45 |
6:f1c3ea8bc850 | 128 | error = params->error; |
cho45 |
6:f1c3ea8bc850 | 129 | if (error != BLE_ERROR_NONE) { |
cho45 |
6:f1c3ea8bc850 | 130 | printf("error on ble.init() \r\n"); |
cho45 |
6:f1c3ea8bc850 | 131 | goto return_error; |
cho45 |
6:f1c3ea8bc850 | 132 | } |
cho45 |
6:f1c3ea8bc850 | 133 | |
cho45 |
6:f1c3ea8bc850 | 134 | ble.gap().onDisconnection(onDisconnect); |
cho45 |
6:f1c3ea8bc850 | 135 | ble.gap().onConnection(onConnect); |
cho45 |
6:f1c3ea8bc850 | 136 | ble.gap().onTimeout(onTimeout); |
cho45 |
6:f1c3ea8bc850 | 137 | |
cho45 |
6:f1c3ea8bc850 | 138 | printf("setup ble security manager\r\n"); |
cho45 |
6:f1c3ea8bc850 | 139 | ble.securityManager().onSecuritySetupInitiated(securitySetupInitiatedCallback); |
cho45 |
6:f1c3ea8bc850 | 140 | ble.securityManager().onPasskeyDisplay(passkeyDisplayCallback); |
cho45 |
6:f1c3ea8bc850 | 141 | ble.securityManager().onSecuritySetupCompleted(securitySetupCompletedCallback); |
cho45 |
6:f1c3ea8bc850 | 142 | // bonding with hard-coded passkey. |
cho45 |
6:f1c3ea8bc850 | 143 | error = ble.securityManager().init(ENABLE_BONDING, REQUIRE_MITM, SecurityManager::IO_CAPS_DISPLAY_ONLY, PASSKEY); |
cho45 |
6:f1c3ea8bc850 | 144 | if (error != BLE_ERROR_NONE) { |
cho45 |
6:f1c3ea8bc850 | 145 | printf("error on ble.securityManager().init()"); |
cho45 |
6:f1c3ea8bc850 | 146 | goto return_error; |
cho45 |
6:f1c3ea8bc850 | 147 | } |
cho45 |
6:f1c3ea8bc850 | 148 | |
cho45 |
6:f1c3ea8bc850 | 149 | keyboardService = new KeyboardService(ble); |
cho45 |
6:f1c3ea8bc850 | 150 | deviceInformationService = new DeviceInformationService(ble, "lowreal.net", MODEL_NAME, SERIAL_NUMBER, HARDWARE_REVISION, FIRMWARE_REVISION, SOFTWARE_REVISION); |
cho45 |
6:f1c3ea8bc850 | 151 | batteryService = new BatteryService(ble, 100); |
cho45 |
6:f1c3ea8bc850 | 152 | updateBatteryLevel(); |
cho45 | 15:70bf079d3ee1 | 153 | dfuService = new DFUService(ble, NULL); |
cho45 |
6:f1c3ea8bc850 | 154 | |
cho45 |
6:f1c3ea8bc850 | 155 | printf("general setup\r\n"); |
cho45 |
6:f1c3ea8bc850 | 156 | error = ble.gap().accumulateAdvertisingPayload( |
cho45 |
6:f1c3ea8bc850 | 157 | GapAdvertisingData::BREDR_NOT_SUPPORTED | |
cho45 |
6:f1c3ea8bc850 | 158 | GapAdvertisingData::LE_GENERAL_DISCOVERABLE |
cho45 |
6:f1c3ea8bc850 | 159 | ); |
cho45 |
9:d1daefbf1fbd | 160 | // shoud be LE_LIMITED_DISCOVERABLE |
cho45 |
9:d1daefbf1fbd | 161 | // error = ble.gap().accumulateAdvertisingPayload( |
cho45 |
9:d1daefbf1fbd | 162 | // GapAdvertisingData::BREDR_NOT_SUPPORTED | |
cho45 |
9:d1daefbf1fbd | 163 | // GapAdvertisingData::LE_LIMITED_DISCOVERABLE |
cho45 |
9:d1daefbf1fbd | 164 | // ); |
cho45 |
6:f1c3ea8bc850 | 165 | if (error != BLE_ERROR_NONE) goto return_error; |
cho45 |
6:f1c3ea8bc850 | 166 | |
cho45 |
6:f1c3ea8bc850 | 167 | printf("set COMPLETE_LIST_16BIT_SERVICE_IDS\r\n"); |
cho45 |
6:f1c3ea8bc850 | 168 | error = ble.gap().accumulateAdvertisingPayload( |
cho45 |
6:f1c3ea8bc850 | 169 | GapAdvertisingData::COMPLETE_LIST_16BIT_SERVICE_IDS, |
cho45 |
6:f1c3ea8bc850 | 170 | (uint8_t*)uuid16_list, sizeof(uuid16_list) |
cho45 |
6:f1c3ea8bc850 | 171 | ); |
cho45 |
6:f1c3ea8bc850 | 172 | if (error != BLE_ERROR_NONE) goto return_error; |
cho45 |
6:f1c3ea8bc850 | 173 | |
cho45 |
6:f1c3ea8bc850 | 174 | printf("set advertising\r\n"); |
cho45 |
6:f1c3ea8bc850 | 175 | // see 5.1.2: HID over GATT Specification (pg. 25) |
cho45 |
6:f1c3ea8bc850 | 176 | ble.gap().setAdvertisingType(GapAdvertisingParams::ADV_CONNECTABLE_UNDIRECTED); |
cho45 |
6:f1c3ea8bc850 | 177 | |
cho45 |
6:f1c3ea8bc850 | 178 | printf("set advertising interval\r\n"); |
cho45 |
9:d1daefbf1fbd | 179 | ble.gap().setAdvertisingInterval(30); |
cho45 |
9:d1daefbf1fbd | 180 | printf("set advertising timeout\r\n"); |
cho45 |
9:d1daefbf1fbd | 181 | ble.gap().setAdvertisingTimeout(180); |
cho45 |
6:f1c3ea8bc850 | 182 | |
cho45 |
6:f1c3ea8bc850 | 183 | printf("set keyboard\r\n"); |
cho45 |
6:f1c3ea8bc850 | 184 | error = ble.gap().accumulateAdvertisingPayload(GapAdvertisingData::KEYBOARD); |
cho45 |
6:f1c3ea8bc850 | 185 | if (error != BLE_ERROR_NONE) goto return_error; |
cho45 |
6:f1c3ea8bc850 | 186 | |
cho45 |
6:f1c3ea8bc850 | 187 | printf("set complete local name\r\n"); |
cho45 |
6:f1c3ea8bc850 | 188 | error = ble.gap().accumulateAdvertisingPayload( |
cho45 |
6:f1c3ea8bc850 | 189 | GapAdvertisingData::COMPLETE_LOCAL_NAME, |
cho45 |
6:f1c3ea8bc850 | 190 | DEVICE_NAME, sizeof(DEVICE_NAME) |
cho45 |
6:f1c3ea8bc850 | 191 | ); |
cho45 |
6:f1c3ea8bc850 | 192 | if (error != BLE_ERROR_NONE) goto return_error; |
cho45 |
6:f1c3ea8bc850 | 193 | |
cho45 |
6:f1c3ea8bc850 | 194 | printf("set device name\r\n"); |
cho45 |
6:f1c3ea8bc850 | 195 | error = ble.gap().setDeviceName(DEVICE_NAME); |
cho45 |
6:f1c3ea8bc850 | 196 | if (error != BLE_ERROR_NONE) goto return_error; |
cho45 |
6:f1c3ea8bc850 | 197 | // ble.gap().setTxPower(-12); |
cho45 |
6:f1c3ea8bc850 | 198 | |
cho45 |
6:f1c3ea8bc850 | 199 | |
cho45 |
6:f1c3ea8bc850 | 200 | printf("advertising\r\n"); |
cho45 |
6:f1c3ea8bc850 | 201 | error = ble.gap().startAdvertising(); |
cho45 |
6:f1c3ea8bc850 | 202 | if (error != BLE_ERROR_NONE) goto return_error; |
cho45 |
6:f1c3ea8bc850 | 203 | return; |
cho45 |
6:f1c3ea8bc850 | 204 | |
cho45 |
6:f1c3ea8bc850 | 205 | return_error: |
cho45 |
6:f1c3ea8bc850 | 206 | printf("error with %d\r\n", error); |
cho45 |
6:f1c3ea8bc850 | 207 | return; |
cho45 |
6:f1c3ea8bc850 | 208 | } |
cho45 |
6:f1c3ea8bc850 | 209 | |
cho45 |
6:f1c3ea8bc850 | 210 | void HIDController::init() { |
cho45 |
6:f1c3ea8bc850 | 211 | // https://github.com/jpbrucker/BLE_HID/blob/master/examples/examples_common.cpp |
cho45 |
6:f1c3ea8bc850 | 212 | printf("ble.init\r\n"); |
cho45 |
6:f1c3ea8bc850 | 213 | |
cho45 |
6:f1c3ea8bc850 | 214 | BLE& ble = BLE::Instance(BLE::DEFAULT_INSTANCE); |
cho45 |
6:f1c3ea8bc850 | 215 | ble.init(bleInitComplete); |
cho45 |
6:f1c3ea8bc850 | 216 | |
cho45 |
6:f1c3ea8bc850 | 217 | while (!ble.hasInitialized()) { } |
cho45 |
6:f1c3ea8bc850 | 218 | |
cho45 |
6:f1c3ea8bc850 | 219 | printf("ble.hasIntialized\r\n"); |
cho45 |
6:f1c3ea8bc850 | 220 | } |
cho45 |
6:f1c3ea8bc850 | 221 | |
cho45 |
10:1aed2481a743 | 222 | void HIDController::waitForEvent() { |
cho45 |
6:f1c3ea8bc850 | 223 | BLE& ble = BLE::Instance(BLE::DEFAULT_INSTANCE); |
cho45 |
6:f1c3ea8bc850 | 224 | ble.waitForEvent(); |
cho45 |
6:f1c3ea8bc850 | 225 | } |
cho45 |
6:f1c3ea8bc850 | 226 | |
cho45 |
6:f1c3ea8bc850 | 227 | void HIDController::appendReportData(uint8_t key) { |
cho45 |
6:f1c3ea8bc850 | 228 | if (keyboardService) { |
cho45 |
6:f1c3ea8bc850 | 229 | keyboardService->appendReportData(key); |
cho45 |
6:f1c3ea8bc850 | 230 | } |
cho45 |
6:f1c3ea8bc850 | 231 | } |
cho45 |
6:f1c3ea8bc850 | 232 | |
cho45 |
6:f1c3ea8bc850 | 233 | void HIDController::deleteReportData(uint8_t key) { |
cho45 |
6:f1c3ea8bc850 | 234 | if (keyboardService) { |
cho45 |
6:f1c3ea8bc850 | 235 | keyboardService->deleteReportData(key); |
cho45 |
6:f1c3ea8bc850 | 236 | } |
cho45 |
6:f1c3ea8bc850 | 237 | } |
cho45 |
9:d1daefbf1fbd | 238 | |
cho45 |
9:d1daefbf1fbd | 239 | void HIDController::queueCurrentReportData() { |
cho45 |
9:d1daefbf1fbd | 240 | if (keyboardService) { |
cho45 |
9:d1daefbf1fbd | 241 | keyboardService->queueCurrentReportData(); |
cho45 |
9:d1daefbf1fbd | 242 | } |
cho45 |
9:d1daefbf1fbd | 243 | } |