2009-04-29 22:00:24 +00:00
|
|
|
/*
|
|
|
|
* hci.c
|
|
|
|
*
|
|
|
|
* Created by Matthias Ringwald on 4/29/09.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2009-05-03 21:28:40 +00:00
|
|
|
#include <unistd.h>
|
2009-05-09 17:34:17 +00:00
|
|
|
#include <stdarg.h>
|
|
|
|
#include <string.h>
|
2009-05-09 17:43:27 +00:00
|
|
|
#include <stdio.h>
|
2009-04-29 22:00:24 +00:00
|
|
|
#include "hci.h"
|
|
|
|
|
2009-05-10 14:16:03 +00:00
|
|
|
// calculate combined ogf/ocf value
|
|
|
|
#define OPCODE(ogf, ocf) (ocf | ogf << 10)
|
2009-05-10 19:55:02 +00:00
|
|
|
#define OGF_LINK_CONTROL 0x01
|
|
|
|
#define OGF_CONTROLLER_BASEBAND 0x03
|
2009-05-09 17:34:17 +00:00
|
|
|
|
|
|
|
hci_cmd_t hci_inquiry = {
|
2009-05-10 20:57:01 +00:00
|
|
|
OPCODE(OGF_LINK_CONTROL, 0x01), "311"
|
|
|
|
// LAP, Inquiry length, Num_responses
|
|
|
|
};
|
|
|
|
|
|
|
|
hci_cmd_t hci_link_key_request_negative_reply = {
|
|
|
|
OPCODE(OGF_LINK_CONTROL, 0x0c), "B"
|
|
|
|
};
|
|
|
|
|
|
|
|
hci_cmd_t hci_pin_code_request_reply = {
|
|
|
|
OPCODE(OGF_LINK_CONTROL, 0x0d), "B1P"
|
|
|
|
// BD_ADDR, pin length, PIN: c-string
|
2009-05-09 17:34:17 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
hci_cmd_t hci_reset = {
|
2009-05-10 19:55:02 +00:00
|
|
|
OPCODE(OGF_CONTROLLER_BASEBAND, 0x03), ""
|
2009-05-09 17:34:17 +00:00
|
|
|
};
|
|
|
|
|
2009-05-10 19:55:02 +00:00
|
|
|
hci_cmd_t hci_create_connection = {
|
|
|
|
OPCODE(OGF_LINK_CONTROL, 0x05), "B21121"
|
|
|
|
// BD_ADDR, Packet_Type, Page_Scan_Repetition_Mode, Reserved, Clock_Offset, Allow_Role_Switch
|
|
|
|
};
|
|
|
|
|
|
|
|
hci_cmd_t hci_write_page_timeout = {
|
|
|
|
OPCODE(OGF_CONTROLLER_BASEBAND, 0x18), "2"
|
|
|
|
// Page_Timeout * 0.625 ms
|
|
|
|
};
|
|
|
|
|
2009-05-10 20:57:01 +00:00
|
|
|
hci_cmd_t hci_write_authentication_enable = {
|
|
|
|
OPCODE(OGF_CONTROLLER_BASEBAND, 0x20), "1"
|
|
|
|
// Authentication_Enable
|
|
|
|
};
|
|
|
|
|
2009-05-10 19:55:02 +00:00
|
|
|
hci_cmd_t hci_host_buffer_size = {
|
|
|
|
OPCODE(OGF_CONTROLLER_BASEBAND, 0x33), "2122"
|
|
|
|
// Host_ACL_Data_Packet_Length:, Host_Synchronous_Data_Packet_Length:, Host_Total_Num_ACL_Data_Packets:, Host_Total_Num_Synchronous_Data_Packets:
|
|
|
|
};
|
2009-05-09 17:34:17 +00:00
|
|
|
|
2009-05-10 19:55:02 +00:00
|
|
|
|
2009-05-17 20:32:14 +00:00
|
|
|
// the stack is here
|
|
|
|
static hci_stack_t hci_stack;
|
|
|
|
|
2009-05-03 21:28:40 +00:00
|
|
|
|
2009-05-16 21:29:51 +00:00
|
|
|
void bt_store_16(uint8_t *buffer, uint16_t pos, uint16_t value){
|
|
|
|
buffer[pos] = value & 0xff;
|
|
|
|
buffer[pos+1] = value >> 8;
|
|
|
|
}
|
|
|
|
|
2009-05-17 21:06:39 +00:00
|
|
|
void bt_flip_addr(bd_addr_t dest, bd_addr_t src){
|
|
|
|
dest[0] = src[5];
|
|
|
|
dest[1] = src[4];
|
|
|
|
dest[2] = src[3];
|
|
|
|
dest[3] = src[2];
|
|
|
|
dest[4] = src[1];
|
|
|
|
dest[5] = src[0];
|
|
|
|
}
|
|
|
|
|
2009-05-09 17:43:27 +00:00
|
|
|
void hexdump(uint8_t *data, int size){
|
|
|
|
int i;
|
|
|
|
for (i=0; i<size;i++){
|
|
|
|
printf("%02X ", data[i]);
|
|
|
|
}
|
|
|
|
printf("\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
static void *hci_daemon_thread(void *arg){
|
|
|
|
printf("HCI Daemon started\n");
|
|
|
|
hci_run(transport, &config);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2009-05-19 21:49:12 +00:00
|
|
|
/**
|
|
|
|
* Linked link list
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* get link for given address
|
|
|
|
*
|
|
|
|
* @return connection OR NULL, if not found
|
|
|
|
*/
|
2009-05-20 21:02:04 +00:00
|
|
|
#if 0
|
2009-05-19 21:49:12 +00:00
|
|
|
static hci_connection_t *link_for_addr(bd_addr_t addr){
|
|
|
|
return NULL;
|
|
|
|
}
|
2009-05-20 21:02:04 +00:00
|
|
|
#endif
|
2009-05-19 21:49:12 +00:00
|
|
|
|
2009-05-17 20:32:14 +00:00
|
|
|
/**
|
|
|
|
* Handler called by HCI transport
|
|
|
|
*/
|
|
|
|
static void dummy_handler(uint8_t *packet, int size){
|
|
|
|
}
|
|
|
|
|
|
|
|
static void acl_handler(uint8_t *packet, int size){
|
|
|
|
hci_stack.acl_packet_handler(packet, size);
|
|
|
|
}
|
2009-05-17 20:51:48 +00:00
|
|
|
|
2009-05-17 20:32:14 +00:00
|
|
|
static void event_handler(uint8_t *packet, int size){
|
2009-05-17 21:06:39 +00:00
|
|
|
bd_addr_t addr;
|
2009-05-17 20:51:48 +00:00
|
|
|
|
2009-05-18 20:18:42 +00:00
|
|
|
// Get Num_HCI_Command_Packets
|
|
|
|
if (packet[0] == HCI_EVENT_COMMAND_COMPLETE ||
|
|
|
|
packet[0] == HCI_EVENT_COMMAND_STATUS){
|
|
|
|
hci_stack.num_cmd_packets = packet[2];
|
|
|
|
}
|
|
|
|
|
|
|
|
// handle BT initialization
|
|
|
|
if (hci_stack.state == HCI_STATE_INITIALIZING){
|
|
|
|
if (hci_stack.substate % 2){
|
|
|
|
// odd: waiting for event
|
|
|
|
if (packet[0] == HCI_EVENT_COMMAND_COMPLETE){
|
|
|
|
hci_stack.substate++;
|
|
|
|
}
|
|
|
|
}
|
2009-05-17 20:51:48 +00:00
|
|
|
}
|
|
|
|
|
2009-05-17 21:06:39 +00:00
|
|
|
// link key request
|
2009-05-18 20:18:42 +00:00
|
|
|
if (packet[0] == HCI_EVENT_LINK_KEY_REQUEST){
|
2009-05-17 21:06:39 +00:00
|
|
|
bt_flip_addr(addr, &packet[2]);
|
|
|
|
hci_send_cmd(&hci_link_key_request_negative_reply, &addr);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// pin code request
|
2009-05-18 20:18:42 +00:00
|
|
|
if (packet[0] == HCI_EVENT_PIN_CODE_REQUEST){
|
2009-05-17 21:06:39 +00:00
|
|
|
bt_flip_addr(addr, &packet[2]);
|
|
|
|
hci_send_cmd(&hci_pin_code_request_reply, &addr, 4, "1234");
|
|
|
|
}
|
|
|
|
|
2009-05-17 20:32:14 +00:00
|
|
|
hci_stack.event_packet_handler(packet, size);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Register L2CAP handlers */
|
|
|
|
void hci_register_event_packet_handler(void (*handler)(uint8_t *packet, int size)){
|
|
|
|
hci_stack.event_packet_handler = handler;
|
|
|
|
}
|
|
|
|
void hci_register_acl_packet_handler (void (*handler)(uint8_t *packet, int size)){
|
|
|
|
hci_stack.acl_packet_handler = handler;
|
|
|
|
}
|
|
|
|
|
2009-05-20 20:46:35 +00:00
|
|
|
void hci_init(hci_transport_t *transport, void *config, bt_control_t *control){
|
2009-05-03 21:28:40 +00:00
|
|
|
|
|
|
|
// reference to use transport layer implementation
|
2009-05-17 20:32:14 +00:00
|
|
|
hci_stack.hci_transport = transport;
|
2009-05-03 21:28:40 +00:00
|
|
|
|
2009-05-20 20:46:35 +00:00
|
|
|
// references to used control implementation
|
|
|
|
hci_stack.control = control;
|
|
|
|
|
|
|
|
// reference to used config
|
|
|
|
hci_stack.config = config;
|
|
|
|
|
2009-05-10 19:55:02 +00:00
|
|
|
// empty cmd buffer
|
2009-05-17 20:32:14 +00:00
|
|
|
hci_stack.hci_cmd_buffer = malloc(3+255);
|
|
|
|
|
2009-05-18 20:18:42 +00:00
|
|
|
// set up state
|
|
|
|
hci_stack.num_cmd_packets = 1; // assume that one cmd can be sent
|
|
|
|
hci_stack.state = HCI_STATE_INITIALIZING;
|
|
|
|
hci_stack.substate = 0;
|
|
|
|
|
2009-05-17 20:32:14 +00:00
|
|
|
// higher level handler
|
|
|
|
hci_stack.event_packet_handler = dummy_handler;
|
|
|
|
hci_stack.acl_packet_handler = dummy_handler;
|
|
|
|
|
|
|
|
// register packet handlers with transport
|
|
|
|
transport->register_event_packet_handler( event_handler);
|
|
|
|
transport->register_acl_packet_handler( acl_handler);
|
|
|
|
|
|
|
|
// open low-level device
|
2009-05-17 20:51:48 +00:00
|
|
|
transport->open(config);
|
2009-05-03 21:28:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int hci_power_control(HCI_POWER_MODE power_mode){
|
2009-05-20 20:46:35 +00:00
|
|
|
if (hci_stack.control) {
|
|
|
|
if (power_mode == HCI_POWER_ON) {
|
|
|
|
hci_stack.control->on(hci_stack.config);
|
|
|
|
} else if (power_mode == HCI_POWER_OFF){
|
|
|
|
hci_stack.control->off(hci_stack.config);
|
|
|
|
}
|
|
|
|
}
|
2009-05-03 21:28:40 +00:00
|
|
|
return 0;
|
2009-04-29 22:00:24 +00:00
|
|
|
}
|
2009-05-03 21:28:40 +00:00
|
|
|
|
2009-05-18 20:18:42 +00:00
|
|
|
uint32_t hci_run(){
|
|
|
|
uint8_t micro_packet;
|
|
|
|
switch (hci_stack.state){
|
|
|
|
case HCI_STATE_INITIALIZING:
|
|
|
|
if (hci_stack.substate % 2) {
|
|
|
|
// odd: waiting for command completion
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (hci_stack.num_cmd_packets == 0) {
|
|
|
|
// cannot send command yet
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
switch (hci_stack.substate/2){
|
|
|
|
case 0:
|
|
|
|
hci_send_cmd(&hci_reset);
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
// ca. 15 sec
|
|
|
|
hci_send_cmd(&hci_write_page_timeout, 0x6000);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
// done.
|
|
|
|
hci_stack.state = HCI_STATE_WORKING;
|
|
|
|
micro_packet = BTSTACK_EVENT_HCI_WORKING;
|
|
|
|
hci_stack.event_packet_handler(µ_packet, 1);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
hci_stack.substate++;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2009-05-03 21:28:40 +00:00
|
|
|
}
|
2009-05-18 20:18:42 +00:00
|
|
|
|
|
|
|
// don't check for timetous yet
|
|
|
|
return 0;
|
2009-05-09 17:34:17 +00:00
|
|
|
}
|
|
|
|
|
2009-05-17 20:32:14 +00:00
|
|
|
|
2009-05-16 21:29:51 +00:00
|
|
|
int hci_send_acl_packet(uint8_t *packet, int size){
|
2009-05-17 20:32:14 +00:00
|
|
|
return hci_stack.hci_transport->send_acl_packet(packet, size);
|
2009-05-16 21:29:51 +00:00
|
|
|
}
|
|
|
|
|
2009-05-18 20:18:42 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* pre: numcmds >= 0 - it's allowed to send a command to the controller
|
|
|
|
*/
|
2009-05-10 19:55:02 +00:00
|
|
|
int hci_send_cmd(hci_cmd_t *cmd, ...){
|
2009-05-17 20:32:14 +00:00
|
|
|
uint8_t * hci_cmd_buffer = hci_stack.hci_cmd_buffer;
|
2009-05-10 19:55:02 +00:00
|
|
|
hci_cmd_buffer[0] = cmd->opcode & 0xff;
|
|
|
|
hci_cmd_buffer[1] = cmd->opcode >> 8;
|
2009-05-09 17:34:17 +00:00
|
|
|
int pos = 3;
|
|
|
|
|
|
|
|
va_list argptr;
|
|
|
|
va_start(argptr, cmd);
|
|
|
|
const char *format = cmd->format;
|
|
|
|
uint16_t word;
|
|
|
|
uint32_t longword;
|
2009-05-10 20:57:01 +00:00
|
|
|
uint8_t * ptr;
|
2009-05-09 17:34:17 +00:00
|
|
|
while (*format) {
|
|
|
|
switch(*format) {
|
|
|
|
case '1': // 8 bit value
|
|
|
|
case '2': // 16 bit value
|
|
|
|
case 'H': // hci_handle
|
2009-05-09 17:45:45 +00:00
|
|
|
word = va_arg(argptr, int); // minimal va_arg is int: 2 bytes on 8+16 bit CPUs
|
2009-05-10 19:55:02 +00:00
|
|
|
hci_cmd_buffer[pos++] = word & 0xff;
|
2009-05-09 17:34:17 +00:00
|
|
|
if (*format == '2') {
|
2009-05-10 19:55:02 +00:00
|
|
|
hci_cmd_buffer[pos++] = word >> 8;
|
2009-05-09 17:34:17 +00:00
|
|
|
} else if (*format == 'H') {
|
2009-05-09 17:45:45 +00:00
|
|
|
// TODO
|
2009-05-09 17:34:17 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case '3':
|
|
|
|
case '4':
|
|
|
|
longword = va_arg(argptr, uint32_t);
|
|
|
|
// longword = va_arg(argptr, int);
|
2009-05-10 19:55:02 +00:00
|
|
|
hci_cmd_buffer[pos++] = longword;
|
|
|
|
hci_cmd_buffer[pos++] = longword >> 8;
|
|
|
|
hci_cmd_buffer[pos++] = longword >> 16;
|
2009-05-09 17:34:17 +00:00
|
|
|
if (*format == '4'){
|
2009-05-10 19:55:02 +00:00
|
|
|
hci_cmd_buffer[pos++] = longword >> 24;
|
2009-05-09 17:34:17 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'B': // bt-addr
|
2009-05-10 20:57:01 +00:00
|
|
|
ptr = va_arg(argptr, uint8_t *);
|
|
|
|
hci_cmd_buffer[pos++] = ptr[5];
|
|
|
|
hci_cmd_buffer[pos++] = ptr[4];
|
|
|
|
hci_cmd_buffer[pos++] = ptr[3];
|
|
|
|
hci_cmd_buffer[pos++] = ptr[2];
|
|
|
|
hci_cmd_buffer[pos++] = ptr[1];
|
|
|
|
hci_cmd_buffer[pos++] = ptr[0];
|
|
|
|
break;
|
|
|
|
case 'P': // c string passed as pascal string with leading 1-byte len
|
|
|
|
ptr = va_arg(argptr, uint8_t *);
|
|
|
|
memcpy(&hci_cmd_buffer[pos], ptr, 16);
|
|
|
|
pos += 16;
|
2009-05-09 17:34:17 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
format++;
|
|
|
|
};
|
|
|
|
va_end(argptr);
|
2009-05-10 19:55:02 +00:00
|
|
|
hci_cmd_buffer[2] = pos - 3;
|
|
|
|
// send packet
|
2009-05-18 20:18:42 +00:00
|
|
|
hci_stack.num_cmd_packets--;
|
2009-05-17 20:32:14 +00:00
|
|
|
return hci_stack.hci_transport->send_cmd_packet(hci_cmd_buffer, pos);
|
2009-05-03 21:28:40 +00:00
|
|
|
}
|