Shadowtrance 5c78d55b04
Bluetooth (#518)
* Bluetooth LE addition

* fixes

* use the psram!

helps a little on S3 (t-deck)

* custom device name

* Update symbols.c

* Feedback + fixes

Fixes external app start/stop server (child devices)
Fixes BtManage causing a full system hang upon disabling bt when a device is connected to the host.

* updoot

* more updoot

* move back!

* Revert "move back!"

This reverts commit d3694365c634acc5db62ac59771c496cb971a727.

* fix some of the things

* Addressing feedback? hmm

* Fixes

Bug 1 — Reconnect loop / Reconnect not working fixed
Bug 2 — Name-only advertising overwrites HID advertising
Bug 3 — BleHidDeviceCtx leak on re-enable
Enhancement — HID device auto-start on radio re-enable

* stuff...

* update for consistency with others

* fix crashes

and some bonus symbols

* a few symbols, i2c speed, cdn message

100kHz i2c speed seems to be more compatible with m5stack modules...and probably in general.
cdn message no longer applies

* Hide BT Settings when bt not enabled

* Addressing things and device fixes

* Missed one!

* stuff
2026-05-22 22:49:37 +02:00

308 lines
10 KiB
C

#pragma once
#include <stdbool.h>
#include <stddef.h>
#include <stdint.h>
#include <tactility/error.h>
#ifdef __cplusplus
extern "C" {
#endif
struct Device;
struct DeviceType;
// ---- Device name ----
/**
* Maximum BLE device name length in bytes, excluding the NUL terminator.
* Must match CONFIG_BT_NIMBLE_GAP_DEVICE_NAME_MAX_LEN (set in device.py for BT devices).
* ble_svc_gap_device_name_set() returns BLE_HS_EINVAL for names longer than this.
*/
#define BLE_DEVICE_NAME_MAX 64
// ---- Address ----
#define BT_ADDR_LEN 6
typedef uint8_t BtAddr[BT_ADDR_LEN];
// ---- Radio ----
enum BtRadioState {
BT_RADIO_STATE_OFF,
BT_RADIO_STATE_ON_PENDING,
BT_RADIO_STATE_ON,
BT_RADIO_STATE_OFF_PENDING,
};
// ---- Peer record ----
#define BT_NAME_MAX 248
struct BtPeerRecord {
BtAddr addr;
/** BLE address type (BLE_ADDR_PUBLIC=0, BLE_ADDR_RANDOM=1, etc.) */
uint8_t addr_type;
char name[BT_NAME_MAX + 1];
int8_t rssi;
bool paired;
bool connected;
};
// ---- Profile identifiers ----
enum BtProfileId {
/** Connect to a BLE HID device (keyboard, mouse, gamepad) */
BT_PROFILE_HID_HOST,
/** Present this device as a BLE HID peripheral (keyboard, gamepad) */
BT_PROFILE_HID_DEVICE,
/** BLE SPP serial port (Nordic UART Service / custom GATT) */
BT_PROFILE_SPP,
/** BLE MIDI (GATT-based) */
BT_PROFILE_MIDI,
};
enum BtProfileState {
BT_PROFILE_STATE_IDLE,
BT_PROFILE_STATE_CONNECTING,
BT_PROFILE_STATE_CONNECTED,
BT_PROFILE_STATE_DISCONNECTING,
};
// ---- Events ----
enum BtEventType {
/** Radio state changed */
BT_EVENT_RADIO_STATE_CHANGED,
/** Started scanning for peers */
BT_EVENT_SCAN_STARTED,
/** Finished scanning for peers */
BT_EVENT_SCAN_FINISHED,
/** A new peer was found during scan */
BT_EVENT_PEER_FOUND,
/** Pairing requires user confirmation (passkey displayed or entry required) */
BT_EVENT_PAIR_REQUEST,
/** Pairing attempt completed */
BT_EVENT_PAIR_RESULT,
/** A peer's connection state changed */
BT_EVENT_CONNECT_STATE_CHANGED,
/** A profile's state changed */
BT_EVENT_PROFILE_STATE_CHANGED,
/** Data was received on the BLE SPP (NUS) RX characteristic */
BT_EVENT_SPP_DATA_RECEIVED,
/** Data was received on the BLE MIDI I/O characteristic */
BT_EVENT_MIDI_DATA_RECEIVED,
};
enum BtPairResult {
BT_PAIR_RESULT_SUCCESS,
BT_PAIR_RESULT_FAILED,
BT_PAIR_RESULT_REJECTED,
/** Stale bond detected and removed; fresh pairing will follow */
BT_PAIR_RESULT_BOND_LOST,
};
struct BtPairRequestData {
BtAddr addr;
uint32_t passkey; /**< Passkey to display (0 if not applicable) */
bool needs_confirmation; /**< true: just confirm, false: user must enter passkey */
};
struct BtPairResultData {
BtAddr addr;
enum BtPairResult result;
/** Profile active when pairing completed (BtProfileId value) */
int profile;
};
struct BtProfileStateData {
BtAddr addr;
enum BtProfileId profile;
enum BtProfileState state;
};
struct BtEvent {
enum BtEventType type;
union {
enum BtRadioState radio_state;
struct BtPeerRecord peer;
struct BtPairRequestData pair_request;
struct BtPairResultData pair_result;
struct BtProfileStateData profile_state;
};
};
typedef void (*BtEventCallback)(struct Device* device, void* context, struct BtEvent event);
// ---- Top-level Bluetooth API ----
struct BluetoothApi {
/**
* Get the radio state.
* @param[in] device the bluetooth device
* @param[out] state the current radio state
* @return ERROR_NONE on success
*/
error_t (*get_radio_state)(struct Device* device, enum BtRadioState* state);
/**
* Enable or disable the Bluetooth radio.
* @param[in] device the bluetooth device
* @param[in] enabled true to enable, false to disable
* @return ERROR_NONE on success
*/
error_t (*set_radio_enabled)(struct Device* device, bool enabled);
/**
* Start scanning for nearby BLE devices.
* @param[in] device the bluetooth device
* @return ERROR_NONE on success
*/
error_t (*scan_start)(struct Device* device);
/**
* Stop an active scan.
* @param[in] device the bluetooth device
* @return ERROR_NONE on success
*/
error_t (*scan_stop)(struct Device* device);
/**
* @param[in] device the bluetooth device
* @return true when a scan is in progress
*/
bool (*is_scanning)(struct Device* device);
/**
* Initiate pairing with a peer.
* @param[in] device the bluetooth device
* @param[in] addr the peer address
* @return ERROR_NONE on success
*/
error_t (*pair)(struct Device* device, const BtAddr addr);
/**
* Remove a previously paired peer.
* @param[in] device the bluetooth device
* @param[in] addr the peer address
* @return ERROR_NONE on success
*/
error_t (*unpair)(struct Device* device, const BtAddr addr);
/**
* Get the list of currently paired peers.
* @param[in] device the bluetooth device
* @param[out] out the buffer to write records into (may be NULL to query count only)
* @param[in, out] count in: capacity of out, out: actual number of paired peers
* @return ERROR_NONE on success
*/
error_t (*get_paired_peers)(struct Device* device, struct BtPeerRecord* out, size_t* count);
/**
* Connect to a peer using the specified profile.
* @param[in] device the bluetooth device
* @param[in] addr the peer address
* @param[in] profile the profile to connect with
* @return ERROR_NONE on success
*/
error_t (*connect)(struct Device* device, const BtAddr addr, enum BtProfileId profile);
/**
* Disconnect a peer from the specified profile.
* @param[in] device the bluetooth device
* @param[in] addr the peer address
* @param[in] profile the profile to disconnect from
* @return ERROR_NONE on success
*/
error_t (*disconnect)(struct Device* device, const BtAddr addr, enum BtProfileId profile);
/**
* Add an event callback.
* @param[in] device the bluetooth device
* @param[in] context context pointer passed to the callback
* @param[in] callback the callback function
* @return ERROR_NONE on success
*/
error_t (*add_event_callback)(struct Device* device, void* context, BtEventCallback callback);
/**
* Remove a previously added event callback.
* @param[in] device the bluetooth device
* @param[in] callback the callback to remove
* @return ERROR_NONE on success
*/
error_t (*remove_event_callback)(struct Device* device, BtEventCallback callback);
/**
* Set the BLE device name used in advertising and the GAP service.
* Can be called before or after the radio is enabled.
* If called while advertising is active, advertising restarts with the new name.
* @param[in] device the bluetooth device
* @param[in] name NUL-terminated name (max BLE_DEVICE_NAME_MAX bytes)
* @return ERROR_NONE on success, ERROR_INVALID_ARGUMENT if name is too long or NULL
*/
error_t (*set_device_name)(struct Device* device, const char* name);
/**
* Get the current BLE device name.
* @param[in] device the bluetooth device
* @param[out] buf buffer to write the name into
* @param[in] buf_len size of buf (must be >= BLE_DEVICE_NAME_MAX + 1)
* @return ERROR_NONE on success
*/
error_t (*get_device_name)(struct Device* device, char* buf, size_t buf_len);
/**
* Notify the driver that a HID host connection is in progress or complete.
* Called by the Tactility HID host module to prevent name resolution from
* initiating a simultaneous central connection (BLE_HS_EALREADY).
* @param[in] device the bluetooth device
* @param[in] active true when HID host is connecting/connected, false when idle
*/
void (*set_hid_host_active)(struct Device* device, bool active);
/**
* Fire an event through all registered event callbacks.
* Used by the Tactility HID host module to inject profile-state events that
* originate outside the platform driver (e.g. HID host connect/disconnect).
*/
void (*fire_event)(struct Device* device, struct BtEvent event);
};
extern const struct DeviceType BLUETOOTH_TYPE;
// ---- Public C API ----
// These are the only functions external code should call.
// The BluetoothApi struct above is the internal driver interface only.
/**
* Find the first ready Bluetooth device.
* Use this instead of referencing BLUETOOTH_TYPE directly from external apps,
* since data symbols may not be exported by the ELF loader.
* @return the first ready Device of BLUETOOTH_TYPE, or NULL if none found.
*/
struct Device* bluetooth_find_first_ready_device(void);
error_t bluetooth_get_radio_state(struct Device* device, enum BtRadioState* state);
error_t bluetooth_set_radio_enabled(struct Device* device, bool enabled);
error_t bluetooth_scan_start(struct Device* device);
error_t bluetooth_scan_stop(struct Device* device);
bool bluetooth_is_scanning(struct Device* device);
error_t bluetooth_pair(struct Device* device, const BtAddr addr);
error_t bluetooth_unpair(struct Device* device, const BtAddr addr);
error_t bluetooth_get_paired_peers(struct Device* device, struct BtPeerRecord* out, size_t* count);
error_t bluetooth_connect(struct Device* device, const BtAddr addr, enum BtProfileId profile);
error_t bluetooth_disconnect(struct Device* device, const BtAddr addr, enum BtProfileId profile);
error_t bluetooth_add_event_callback(struct Device* device, void* context, BtEventCallback callback);
error_t bluetooth_remove_event_callback(struct Device* device, BtEventCallback callback);
error_t bluetooth_set_device_name(struct Device* device, const char* name);
error_t bluetooth_get_device_name(struct Device* device, char* buf, size_t buf_len);
void bluetooth_set_hid_host_active(struct Device* device, bool active);
void bluetooth_fire_event(struct Device* device, struct BtEvent event);
#ifdef __cplusplus
}
#endif