2009-07-23 21:25:39 +00:00
|
|
|
/*
|
|
|
|
* hci_cmds.c
|
|
|
|
*
|
|
|
|
* Created by Matthias Ringwald on 7/23/09.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "hci_cmds.h"
|
|
|
|
|
|
|
|
/**
|
|
|
|
* construct HCI Command based on template
|
|
|
|
*
|
|
|
|
* Format:
|
|
|
|
* 1,2,3,4: one to four byte value
|
|
|
|
* H: HCI connection handle
|
|
|
|
* B: Bluetooth Baseband Address (BD_ADDR)
|
|
|
|
* P: 16 byte Pairing code
|
|
|
|
* N: Name up to 248 chars
|
|
|
|
* E: Extended Inquiry Result
|
|
|
|
*/
|
|
|
|
uint16_t hci_create_cmd_internal(uint8_t *hci_cmd_buffer, hci_cmd_t *cmd, va_list argptr){
|
|
|
|
|
|
|
|
hci_cmd_buffer[0] = cmd->opcode & 0xff;
|
|
|
|
hci_cmd_buffer[1] = cmd->opcode >> 8;
|
|
|
|
int pos = 3;
|
|
|
|
|
|
|
|
const char *format = cmd->format;
|
|
|
|
uint16_t word;
|
|
|
|
uint32_t longword;
|
|
|
|
uint8_t * ptr;
|
|
|
|
while (*format) {
|
|
|
|
switch(*format) {
|
|
|
|
case '1': // 8 bit value
|
|
|
|
case '2': // 16 bit value
|
|
|
|
case 'H': // hci_handle
|
|
|
|
word = va_arg(argptr, int); // minimal va_arg is int: 2 bytes on 8+16 bit CPUs
|
|
|
|
hci_cmd_buffer[pos++] = word & 0xff;
|
|
|
|
if (*format == '2') {
|
|
|
|
hci_cmd_buffer[pos++] = word >> 8;
|
|
|
|
} else if (*format == 'H') {
|
|
|
|
// TODO
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case '3':
|
|
|
|
case '4':
|
|
|
|
longword = va_arg(argptr, uint32_t);
|
|
|
|
// longword = va_arg(argptr, int);
|
|
|
|
hci_cmd_buffer[pos++] = longword;
|
|
|
|
hci_cmd_buffer[pos++] = longword >> 8;
|
|
|
|
hci_cmd_buffer[pos++] = longword >> 16;
|
|
|
|
if (*format == '4'){
|
|
|
|
hci_cmd_buffer[pos++] = longword >> 24;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'B': // bt-addr
|
|
|
|
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;
|
|
|
|
break;
|
|
|
|
case 'N': // UTF-8 string, null terminated
|
|
|
|
ptr = va_arg(argptr, uint8_t *);
|
|
|
|
memcpy(&hci_cmd_buffer[pos], ptr, 248);
|
|
|
|
pos += 248;
|
|
|
|
break;
|
|
|
|
case 'E': // Extended Inquiry Information 240 octets
|
|
|
|
ptr = va_arg(argptr, uint8_t *);
|
|
|
|
memcpy(&hci_cmd_buffer[pos], ptr, 240);
|
|
|
|
pos += 240;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
format++;
|
|
|
|
};
|
|
|
|
hci_cmd_buffer[2] = pos - 3;
|
|
|
|
return pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* construct HCI Command based on template
|
|
|
|
*
|
|
|
|
* mainly calls hci_create_cmd_internal
|
|
|
|
*/
|
|
|
|
uint16_t hci_create_cmd(uint8_t *hci_cmd_buffer, hci_cmd_t *cmd, ...){
|
|
|
|
va_list argptr;
|
|
|
|
va_start(argptr, cmd);
|
|
|
|
uint16_t len = hci_create_cmd_internal(hci_cmd_buffer, cmd, argptr);
|
|
|
|
va_end(argptr);
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Link Control Commands
|
|
|
|
*/
|
|
|
|
hci_cmd_t hci_inquiry = {
|
|
|
|
OPCODE(OGF_LINK_CONTROL, 0x01), "311"
|
|
|
|
// LAP, Inquiry length, Num_responses
|
|
|
|
};
|
|
|
|
hci_cmd_t hci_inquiry_cancel = {
|
|
|
|
OPCODE(OGF_LINK_CONTROL, 0x02), ""
|
|
|
|
// no params
|
|
|
|
};
|
|
|
|
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_accept_connection_request = {
|
|
|
|
OPCODE(OGF_LINK_CONTROL, 0x09), "B1"
|
|
|
|
// BD_ADDR, Role: become master, stay slave
|
|
|
|
};
|
|
|
|
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
|
|
|
|
};
|
|
|
|
hci_cmd_t hci_remote_name_request = {
|
|
|
|
OPCODE(OGF_LINK_CONTROL, 0x19), "B112"
|
|
|
|
// BD_ADDR, Page_Scan_Repetition_Mode, Reserved, Clock_Offset
|
|
|
|
};
|
|
|
|
hci_cmd_t hci_remote_name_request_cancel = {
|
|
|
|
OPCODE(OGF_LINK_CONTROL, 0x1A), "B"
|
|
|
|
// BD_ADDR
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Controller & Baseband Commands
|
|
|
|
*/
|
|
|
|
hci_cmd_t hci_set_event_mask = {
|
|
|
|
OPCODE(OGF_CONTROLLER_BASEBAND, 0x01), "44"
|
|
|
|
// event_mask lower 4 octets, higher 4 bytes
|
|
|
|
};
|
|
|
|
hci_cmd_t hci_reset = {
|
|
|
|
OPCODE(OGF_CONTROLLER_BASEBAND, 0x03), ""
|
|
|
|
// no params
|
|
|
|
};
|
|
|
|
hci_cmd_t hci_delete_stored_link_key = {
|
|
|
|
OPCODE(OGF_CONTROLLER_BASEBAND, 0x12), "B1"
|
|
|
|
// BD_ADDR, Delete_All_Flag
|
|
|
|
};
|
|
|
|
hci_cmd_t hci_write_local_name = {
|
|
|
|
OPCODE(OGF_CONTROLLER_BASEBAND, 0x13), "N"
|
|
|
|
// Local name (UTF-8, Null Terminated, max 248 octets)
|
|
|
|
};
|
|
|
|
hci_cmd_t hci_write_page_timeout = {
|
|
|
|
OPCODE(OGF_CONTROLLER_BASEBAND, 0x18), "2"
|
|
|
|
// Page_Timeout * 0.625 ms
|
|
|
|
};
|
|
|
|
hci_cmd_t hci_write_scan_enable = {
|
|
|
|
OPCODE(OGF_CONTROLLER_BASEBAND, 0x1A), "1"
|
|
|
|
// Scan_enable: no, inq, page, inq+page
|
|
|
|
};
|
|
|
|
hci_cmd_t hci_write_authentication_enable = {
|
|
|
|
OPCODE(OGF_CONTROLLER_BASEBAND, 0x20), "1"
|
|
|
|
// Authentication_Enable
|
|
|
|
};
|
|
|
|
hci_cmd_t hci_write_class_of_device = {
|
|
|
|
OPCODE(OGF_CONTROLLER_BASEBAND, 0x24), "3"
|
|
|
|
// Class of Device
|
|
|
|
};
|
|
|
|
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:
|
|
|
|
};
|
|
|
|
|
|
|
|
hci_cmd_t hci_write_inquiry_mode = {
|
|
|
|
OPCODE(OGF_CONTROLLER_BASEBAND, 0x45), "1"
|
|
|
|
// Inquiry mode: 0x00 = standard, 0x01 = with RSSI, 0x02 = extended
|
|
|
|
};
|
|
|
|
|
|
|
|
hci_cmd_t hci_write_extended_inquiry_response = {
|
|
|
|
OPCODE(OGF_CONTROLLER_BASEBAND, 0x52), "1E"
|
|
|
|
// FEC_Required, Exstended Inquiry Response
|
|
|
|
};
|
|
|
|
|
|
|
|
hci_cmd_t hci_write_simple_pairing_mode = {
|
|
|
|
OPCODE(OGF_CONTROLLER_BASEBAND, 0x56), "1"
|
|
|
|
// mode: 0 = off, 1 = on
|
|
|
|
};
|
|
|
|
|
|
|
|
hci_cmd_t hci_read_bd_addr = {
|
|
|
|
OPCODE(OGF_INFORMATIONAL_PARAMETERS, 0x09), ""
|
|
|
|
// no params
|
|
|
|
};
|
|
|
|
|
2009-07-24 21:37:45 +00:00
|
|
|
// BTstack commands
|
|
|
|
|
2009-07-23 21:25:39 +00:00
|
|
|
hci_cmd_t hci_get_btstack_state = {
|
|
|
|
OPCODE(OGF_BTSTACK, HCI_BTSTACK_GET_STATE), ""
|
|
|
|
// no params ->
|
|
|
|
};
|
|
|
|
|
2009-07-24 21:37:45 +00:00
|
|
|
hci_cmd_t hci_set_power_mode = {
|
|
|
|
OPCODE(OGF_BTSTACK, HCI_BTSTACK_SET_POWER_MODE), "1"
|
|
|
|
// mode: 0 = off, 1 = on
|
|
|
|
};
|
|
|
|
|
|
|
|
|