CodexPad Arduino 库 3.0.0
载入中...
搜索中...
未找到
codex_pad.cpp
1#include "codex_pad.h"
2
3#include <Arduino.h>
4
5#include <memory>
6
7#include "WString.h"
8
9namespace {
10constexpr uint16_t kGapServiceUuid{0x1800};
11constexpr uint16_t kGapDeviceNameUuid{0x2A00};
12
13constexpr uint16_t kInputsServiceUuid{0xFFA0};
14constexpr uint16_t kInputsCharacteristicUuid{0xFFA1};
15
16constexpr uint16_t kBatteryServiceUuid{0x180F};
17constexpr uint16_t kBatteryLevelCharacteristicUuid{0x2A19};
18
19constexpr uint16_t kDeviceInfoServiceUuid{0x180A};
20constexpr uint16_t kModelNumberCharacteristicUuid{0x2A24};
21constexpr uint16_t kSerialNumberCharacteristicUuid{0x2A25};
22constexpr uint16_t kFirmwareRevisionCharacteristicUuid{0x2A26};
23constexpr uint16_t kManufacturerNameCharacteristicUuid{0x2A29};
24} // namespace
25
26CodexPad::CodexPad() noexcept {}
27
28CodexPad::~CodexPad() { Reset(); }
29
30void CodexPad::Init() noexcept {
31 if (!NimBLEDevice::isInitialized()) {
32 NimBLEDevice::init("CodexPadClient");
33 }
34}
35
36bool CodexPad::Connect(const std::string& bluetooth_device_address, const uint32_t timeout_ms) noexcept {
37 // check mac address is valid
38 if (bluetooth_device_address.length() != 17 || bluetooth_device_address[2] != ':' || bluetooth_device_address[5] != ':' ||
39 bluetooth_device_address[8] != ':' || bluetooth_device_address[11] != ':' || bluetooth_device_address[14] != ':') {
40 abort();
41 return false;
42 }
43
44 return Connect(NimBLEAddress(bluetooth_device_address, 0), false, timeout_ms);
45}
46
47bool CodexPad::ScanAndConnect(const gamepad::input::Button buttons) noexcept {
48 auto scanner = NimBLEDevice::getScan();
49 scanner->setActiveScan(true); // active scan uses more power, but get results faster
50 scanner->setInterval(1000);
51 scanner->setWindow(999); // less or equal setInterval value
52
53 if (!scanner->start(1000)) {
54 scanner->stop();
55 scanner->clearResults();
56 // printf("Scan failed\n");
57 return false;
58 }
59
60 while (scanner->isScanning()) {
61 delay(100);
62 }
63
64 // printf("Scan done, device count: %d\n", scanner->getResults().getCount());
65
66 int8_t rssi = INT8_MIN;
67 NimBLEAddress address;
68
69#pragma pack(push, 1)
70 struct ManufacturerSpecificData {
71 uint16_t company_id = 0xFFFF;
72 uint8_t header[8] = {'C', 'o', 'd', 'e', 'x', 'P', 'a', 'd'};
73 uint8_t version_major = 0;
74 uint8_t version_minor = 0;
75 uint8_t version_patch = 0;
76 uint32_t button_state = 0;
77 uint8_t button_states_duration_seconds = 0;
78 };
79#pragma pack(pop)
80
81 for (const auto device : scanner->getResults()) {
82 if (device->haveName() && String(device->getName().c_str()).startsWith("CodexPad-") && device->haveManufacturerData()) {
83 // printf("Name: %s\n", device->getName().c_str());
84 const auto manufacturer_data = device->getManufacturerData();
85 if (manufacturer_data.length() >= sizeof(ManufacturerSpecificData)) {
86 const auto data = reinterpret_cast<const ManufacturerSpecificData*>(manufacturer_data.c_str());
87 if (data->company_id == 0xFFFF // company id
88 && memcmp(data->header, "CodexPad", 8) == 0 // header
89 && data->button_state == static_cast<uint32_t>(buttons) // button mask
90 && device->getRSSI() > rssi // rssi
91 && (data->version_major < 2 || data->button_states_duration_seconds >= 1) // button states duration
92 ) {
93 rssi = device->getRSSI();
94 address = device->getAddress();
95 // printf("Found device, rssi: %d, address: %s\n", rssi, address.toString().c_str());
96 }
97 }
98 }
99 }
100
101 scanner->clearResults();
102
103 return address.isNull() ? false : Connect(address, 2000);
104}
105
106const gamepad::input::Tracker& CodexPad::Update() noexcept {
107 if (ble_client_ == nullptr) {
108 return input_tracker_;
109 }
110
111 if (!ble_client_->isConnected()) {
112 Reset();
113 return input_tracker_;
114 }
115
116 input_tracker_.Tick();
117
118 do {
119 std::lock_guard<std::mutex> l(mutex_);
120 if (inputs_queue_.empty()) {
121 break;
122 }
123
124 input_tracker_.Update(inputs_queue_.front());
125 inputs_queue_.pop();
126 } while (false);
127
128 return input_tracker_;
129}
130
131bool CodexPad::is_connected() const noexcept { return ble_client_ != nullptr && ble_client_->isConnected(); }
132
134 if (ble_client_ == nullptr) {
135 return false;
136 }
137
138 if (!ble_client_->isConnected()) {
139 return false;
140 }
141
142 auto remote_service = ble_client_->getService(uint16_t{0x1804});
143 if (remote_service == nullptr) {
144 return false;
145 }
146
147 auto remote_characteristic = remote_service->getCharacteristic(uint16_t{0x2A07});
148 if (remote_characteristic == nullptr) {
149 return false;
150 }
151
152 return remote_characteristic->writeValue(static_cast<uint8_t>(tx_power));
153}
154
155bool CodexPad::Connect(const NimBLEAddress& address, bool async_connect, const uint32_t timeout_ms) {
156 Reset();
157 assert(ble_client_ == nullptr);
158 ble_client_ = NimBLEDevice::createClient(address);
159 ble_client_->setConnectTimeout(timeout_ms);
160 auto ret = ble_client_->connect(true, async_connect, true);
161
162 if (!ret || !ble_client_->isConnected()) {
163 goto FAILED;
164 }
165
166 remote_device_name_ = ble_client_->getValue(kGapServiceUuid, kGapDeviceNameUuid);
167 remote_model_number_ = ble_client_->getValue(kDeviceInfoServiceUuid, kModelNumberCharacteristicUuid);
168 {
169 auto firmware_revision = ble_client_->getValue(kDeviceInfoServiceUuid, kFirmwareRevisionCharacteristicUuid);
170 if (firmware_revision.length() == remote_firmware_version_.size()) {
171 memcpy(remote_firmware_version_.data(), firmware_revision.data(), firmware_revision.length());
172 }
173 }
174
175 {
176 auto remote_service = ble_client_->getService(kInputsServiceUuid);
177 if (remote_service == nullptr) {
178 goto FAILED;
179 }
180
181 auto remote_characteristic = remote_service->getCharacteristic(kInputsCharacteristicUuid);
182 if (remote_characteristic == nullptr) {
183 goto FAILED;
184 }
185
186 if (!remote_characteristic->canNotify()) {
187 goto FAILED;
188 }
189
190 if (!remote_characteristic->subscribe(
191 true, std::bind(&CodexPad::OnNotify, this, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3,
192 std::placeholders::_4))) {
193 goto FAILED;
194 }
195 }
196
197 return ret;
198
199FAILED:
200 Reset();
201 return false;
202}
203
204void CodexPad::OnNotify(const NimBLERemoteCharacteristic* remote_characteristic, const uint8_t* data, const size_t length,
205 const bool is_notify) {
206 if (remote_characteristic != nullptr && remote_characteristic->getUUID().equals(kInputsCharacteristicUuid)) {
207 if (length != sizeof(gamepad::input::State)) {
208 printf("WARNING: length != sizeof(Inputs)\n");
209 return;
210 }
211
212 std::lock_guard<std::mutex> l(mutex_);
213 if (inputs_queue_.size() > kInputsQueueMax) {
214 inputs_queue_.pop();
215 }
216 inputs_queue_.emplace(gamepad::input::State::FromBytes(data));
217 }
218}
219
220void CodexPad::Reset() {
221 if (ble_client_ != nullptr) {
222 ble_client_->cancelConnect();
223 ble_client_->disconnect();
224 NimBLEDevice::deleteClient(ble_client_);
225 ble_client_ = nullptr;
226 }
227
228 remote_device_name_.clear();
229 remote_model_number_.clear();
230 remote_firmware_version_.fill(0);
231 input_tracker_.Reset();
232 std::lock_guard<std::mutex> l(mutex_);
233 inputs_queue_ = {};
234}
~CodexPad() noexcept
析构函数
CodexPad() noexcept
构造函数
bool Connect(const std::string &bluetooth_device_address, uint32_t timeout_ms=5000) noexcept
连接
bool is_connected() const noexcept
是否连接
const gamepad::input::Tracker & Update() noexcept
更新,需要在Loop中不断调用
void Init() noexcept
初始化
bool set_remote_tx_power(TxPower power) noexcept
设置发射功率,连接状态下调用,立即生效于当前连接,下次连接生效
bool ScanAndConnect(gamepad::input::Button buttons) noexcept
扫描附近的 CodexPad 设备,并自动连接到一个按键状态与指定掩码匹配的设备。