11f504dbdSmatthias.ringwald /* 21f504dbdSmatthias.ringwald * hci.h 31f504dbdSmatthias.ringwald * 41f504dbdSmatthias.ringwald * Created by Matthias Ringwald on 4/29/09. 51f504dbdSmatthias.ringwald * 61f504dbdSmatthias.ringwald */ 71f504dbdSmatthias.ringwald 81f504dbdSmatthias.ringwald #pragma once 91f504dbdSmatthias.ringwald 1093b8dc03Smatthias.ringwald #include <stdint.h> 1102ea9861Smatthias.ringwald #include <stdlib.h> 121cd208adSmatthias.ringwald #include <stdarg.h> 1393b8dc03Smatthias.ringwald 14*d8905019Smatthias.ringwald #include "hci_cmds.h" 151f504dbdSmatthias.ringwald #include "hci_transport.h" 1611e23e5fSmatthias.ringwald #include "bt_control.h" 171f504dbdSmatthias.ringwald 18ea6387dfSmatthias.ringwald // helper for BT little endian format 19b9fa6cb5Smatthias.ringwald #define READ_BT_16( buffer, pos) ( ((uint16_t) buffer[pos]) | (((uint16_t)buffer[pos+1]) << 8)) 20ea6387dfSmatthias.ringwald #define READ_BT_24( buffer, pos) ( ((uint32_t) buffer[pos]) | (((uint32_t)buffer[pos+1]) << 8) | (((uint32_t)buffer[pos+2]) << 16)) 21ea6387dfSmatthias.ringwald #define READ_BT_32( buffer, pos) ( ((uint32_t) buffer[pos]) | (((uint32_t)buffer[pos+1]) << 8) | (((uint32_t)buffer[pos+2]) << 16) | (((uint32_t) buffer[pos+3])) << 24) 22ea6387dfSmatthias.ringwald 2397addcc5Smatthias.ringwald // packet header lengh 24ea6387dfSmatthias.ringwald #define HCI_CMD_DATA_PKT_HDR 0x03 25ea6387dfSmatthias.ringwald #define HCI_ACL_DATA_PKT_HDR 0x04 26ea6387dfSmatthias.ringwald #define HCI_SCO_DATA_PKT_HDR 0x03 27ea6387dfSmatthias.ringwald #define HCI_EVENT_PKT_HDR 0x02 28ea6387dfSmatthias.ringwald 293429f56bSmatthias.ringwald // Events from host controller to host 303429f56bSmatthias.ringwald #define HCI_EVENT_INQUIRY_COMPLETE 0x01 313429f56bSmatthias.ringwald #define HCI_EVENT_INQUIRY_RESULT 0x02 323429f56bSmatthias.ringwald #define HCI_EVENT_CONNECTION_COMPLETE 0x03 333429f56bSmatthias.ringwald #define HCI_EVENT_CONNECTION_REQUEST 0x04 343429f56bSmatthias.ringwald #define HCI_EVENT_DISCONNECTION_COMPLETE 0x05 353429f56bSmatthias.ringwald #define HCI_EVENT_AUTHENTICATION_COMPLETE_EVENT 0x06 363429f56bSmatthias.ringwald #define HCI_EVENT_REMOTE_NAME_REQUEST_COMPLETE 0x07 373429f56bSmatthias.ringwald #define HCI_EVENT_ENCRIPTION_CHANGE 0x08 383429f56bSmatthias.ringwald #define HCI_EVENT_CHANGE_CONNECTION_LINK_KEY_COMPLETE 0x09 393429f56bSmatthias.ringwald #define HCI_EVENT_MASTER_LINK_KEY_COMPLETE 0x0A 403429f56bSmatthias.ringwald #define HCI_EVENT_READ_REMOTE_SUPPORTED_FEATURES_COMPLETE 0x0B 413429f56bSmatthias.ringwald #define HCI_EVENT_READ_REMOTE_VERSION_INFORMATION_COMPLETE 0x0C 423429f56bSmatthias.ringwald #define HCI_EVENT_QOS_SETUP_COMPLETE 0x0D 433429f56bSmatthias.ringwald #define HCI_EVENT_COMMAND_COMPLETE 0x0E 443429f56bSmatthias.ringwald #define HCI_EVENT_COMMAND_STATUS 0x0F 453429f56bSmatthias.ringwald #define HCI_EVENT_HARDWARE_ERROR 0x10 463429f56bSmatthias.ringwald #define HCI_EVENT_FLUSH_OCCURED 0x11 473429f56bSmatthias.ringwald #define HCI_EVENT_ROLE_CHANGE 0x12 483429f56bSmatthias.ringwald #define HCI_EVENT_NUMBER_OF_COMPLETED_PACKETS 0x13 493429f56bSmatthias.ringwald #define HCI_EVENT_MODE_CHANGE_EVENT 0x14 503429f56bSmatthias.ringwald #define HCI_EVENT_RETURN_LINK_KEYS 0x15 513429f56bSmatthias.ringwald #define HCI_EVENT_PIN_CODE_REQUEST 0x16 523429f56bSmatthias.ringwald #define HCI_EVENT_LINK_KEY_REQUEST 0x17 533429f56bSmatthias.ringwald #define HCI_EVENT_LINK_KEY_NOTIFICATION 0x18 543429f56bSmatthias.ringwald #define HCI_EVENT_DATA_BUFFER_OVERFLOW 0x1A 553429f56bSmatthias.ringwald #define HCI_EVENT_MAX_SLOTS_CHANGED 0x1B 563429f56bSmatthias.ringwald #define HCI_EVENT_READ_CLOCK_OFFSET_COMPLETE 0x1C 573429f56bSmatthias.ringwald #define NECTEVENT_ION_PACKET_TYPE_CHANGED 0x1D 583429f56bSmatthias.ringwald #define HCI_EVENT_INQUIRY_RESULT_WITH_RSSI 0x22 593429f56bSmatthias.ringwald #define HCI_EVENT_VENDOR_SPECIFIC 0xFF 603429f56bSmatthias.ringwald 613429f56bSmatthias.ringwald // events from BTstack for application/client lib 628adf0ddaSmatthias.ringwald #define HCI_EVENT_BTSTACK_WORKING 0x80 638adf0ddaSmatthias.ringwald #define HCI_EVENT_BTSTACK_STATE 0x81 648adf0ddaSmatthias.ringwald 653429f56bSmatthias.ringwald #define COMMAND_COMPLETE_EVENT(event,cmd) ( event[0] == HCI_EVENT_COMMAND_COMPLETE && READ_BT_16(event,3) == cmd.opcode) 663429f56bSmatthias.ringwald 6797addcc5Smatthias.ringwald /** 6897addcc5Smatthias.ringwald * Default INQ Mode 6997addcc5Smatthias.ringwald */ 7097addcc5Smatthias.ringwald #define HCI_INQUIRY_LAP 0x9E8B33L // 0x9E8B33: General/Unlimited Inquiry Access Code (GIAC) 7197addcc5Smatthias.ringwald 72ea6387dfSmatthias.ringwald /** 73ea6387dfSmatthias.ringwald * @brief Length of a bluetooth device address. 74ea6387dfSmatthias.ringwald */ 75ea6387dfSmatthias.ringwald #define BD_ADDR_LEN 6 76ea6387dfSmatthias.ringwald typedef uint8_t bd_addr_t[BD_ADDR_LEN]; 77ea6387dfSmatthias.ringwald 78ea6387dfSmatthias.ringwald /** 79ea6387dfSmatthias.ringwald * @brief The link key type 80ea6387dfSmatthias.ringwald */ 81ea6387dfSmatthias.ringwald #define LINK_KEY_LEN 16 82ea6387dfSmatthias.ringwald typedef uint8_t link_key_t[LINK_KEY_LEN]; 83ea6387dfSmatthias.ringwald 8443625864Smatthias.ringwald /** 8543625864Smatthias.ringwald * @brief hci connection handle type 8643625864Smatthias.ringwald */ 8743625864Smatthias.ringwald typedef uint16_t hci_con_handle_t; 88ea6387dfSmatthias.ringwald 89475c8125Smatthias.ringwald typedef enum { 90475c8125Smatthias.ringwald HCI_POWER_OFF = 0, 91475c8125Smatthias.ringwald HCI_POWER_ON 92475c8125Smatthias.ringwald } HCI_POWER_MODE; 93475c8125Smatthias.ringwald 943429f56bSmatthias.ringwald typedef enum { 953429f56bSmatthias.ringwald HCI_STATE_OFF = 0, 963429f56bSmatthias.ringwald HCI_STATE_INITIALIZING, 973429f56bSmatthias.ringwald HCI_STATE_WORKING, 983429f56bSmatthias.ringwald HCI_STATE_HALTING 993429f56bSmatthias.ringwald } HCI_STATE; 1003429f56bSmatthias.ringwald 101c01e9cbdSmatthias.ringwald typedef enum { 102c01e9cbdSmatthias.ringwald SEND_NEGATIVE_LINK_KEY_REQUEST = 1 << 0, 103c01e9cbdSmatthias.ringwald SEND_PIN_CODE_RESPONSE = 1 << 1 104c01e9cbdSmatthias.ringwald } hci_connection_flags_t; 105c01e9cbdSmatthias.ringwald 10616833f0aSmatthias.ringwald typedef struct hci_connection { 107c01e9cbdSmatthias.ringwald // linked list 10816833f0aSmatthias.ringwald struct hci_connection * next; 109c01e9cbdSmatthias.ringwald 110c01e9cbdSmatthias.ringwald // remote side 11116833f0aSmatthias.ringwald bd_addr_t address; 11216833f0aSmatthias.ringwald hci_con_handle_t con_handle; 113c01e9cbdSmatthias.ringwald 114c01e9cbdSmatthias.ringwald // hci state machine 115c01e9cbdSmatthias.ringwald hci_connection_flags_t flags; 116c01e9cbdSmatthias.ringwald 11716833f0aSmatthias.ringwald } hci_connection_t; 11816833f0aSmatthias.ringwald 119c01e9cbdSmatthias.ringwald 12016833f0aSmatthias.ringwald typedef struct { 12116833f0aSmatthias.ringwald 12216833f0aSmatthias.ringwald hci_transport_t * hci_transport; 12311e23e5fSmatthias.ringwald bt_control_t * control; 12411e23e5fSmatthias.ringwald void * config; 12511e23e5fSmatthias.ringwald 12616833f0aSmatthias.ringwald uint8_t * hci_cmd_buffer; 12716833f0aSmatthias.ringwald hci_connection_t * connections; 12816833f0aSmatthias.ringwald 12916833f0aSmatthias.ringwald /* host to controller flow control */ 13016833f0aSmatthias.ringwald uint8_t num_cmd_packets; 13116833f0aSmatthias.ringwald uint8_t num_acl_packets; 13216833f0aSmatthias.ringwald 13316833f0aSmatthias.ringwald /* callback to L2CAP layer */ 1341cd208adSmatthias.ringwald void (*event_packet_handler)(uint8_t *packet, uint16_t size); 1351cd208adSmatthias.ringwald void (*acl_packet_handler) (uint8_t *packet, uint16_t size); 13616833f0aSmatthias.ringwald 1373429f56bSmatthias.ringwald /* hci state machine */ 1383429f56bSmatthias.ringwald HCI_STATE state; 1393429f56bSmatthias.ringwald uint8_t substate; 1403429f56bSmatthias.ringwald uint8_t cmds_ready; 1413429f56bSmatthias.ringwald 14216833f0aSmatthias.ringwald } hci_stack_t; 14316833f0aSmatthias.ringwald 14402ea9861Smatthias.ringwald 145475c8125Smatthias.ringwald // set up HCI 14611e23e5fSmatthias.ringwald void hci_init(hci_transport_t *transport, void *config, bt_control_t *control); 147475c8125Smatthias.ringwald 1481cd208adSmatthias.ringwald void hci_register_event_packet_handler(void (*handler)(uint8_t *packet, uint16_t size)); 14916833f0aSmatthias.ringwald 1501cd208adSmatthias.ringwald void hci_register_acl_packet_handler (void (*handler)(uint8_t *packet, uint16_t size)); 15116833f0aSmatthias.ringwald 152475c8125Smatthias.ringwald // power control 153475c8125Smatthias.ringwald int hci_power_control(HCI_POWER_MODE mode); 154475c8125Smatthias.ringwald 1553429f56bSmatthias.ringwald /** 1563429f56bSmatthias.ringwald * run the hci daemon loop once 1573429f56bSmatthias.ringwald * 1583429f56bSmatthias.ringwald * @return 0 or next timeout 1593429f56bSmatthias.ringwald */ 1603429f56bSmatthias.ringwald uint32_t hci_run(); 1611f504dbdSmatthias.ringwald 162554588a5Smatthias.ringwald // 16302582713Smatthias.ringwald void hexdump(void *data, int size); 164554588a5Smatthias.ringwald 1651cd208adSmatthias.ringwald // create and send hci command packets based on a template and a list of parameters 16602ea9861Smatthias.ringwald int hci_send_cmd(hci_cmd_t *cmd, ...); 167*d8905019Smatthias.ringwald 168*d8905019Smatthias.ringwald // send complete CMD packet 16949857181Smatthias.ringwald int hci_send_cmd_packet(uint8_t *packet, int size); 170554588a5Smatthias.ringwald 17143625864Smatthias.ringwald // send ACL packet 17243625864Smatthias.ringwald int hci_send_acl_packet(uint8_t *packet, int size); 17343625864Smatthias.ringwald 17443625864Smatthias.ringwald // helper 17543625864Smatthias.ringwald extern void bt_store_16(uint8_t *buffer, uint16_t pos, uint16_t value); 1768b658ebcSmatthias.ringwald extern void bt_store_32(uint8_t *buffer, uint16_t pos, uint32_t value); 17768d92d03Smatthias.ringwald extern void bt_flip_addr(bd_addr_t dest, bd_addr_t src); 17843625864Smatthias.ringwald 179