2009-07-01 21:55:08 +00:00
|
|
|
/*
|
|
|
|
* btstack.c
|
|
|
|
*
|
|
|
|
* Created by Matthias Ringwald on 7/1/09.
|
|
|
|
*
|
|
|
|
* BTstack client API
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "btstack.h"
|
|
|
|
|
2009-07-20 21:39:43 +00:00
|
|
|
#include "l2cap.h"
|
2009-07-15 22:12:54 +00:00
|
|
|
#include "run_loop.h"
|
|
|
|
#include <netdb.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <stdarg.h>
|
|
|
|
|
2009-07-16 22:04:11 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
|
2009-07-15 22:12:54 +00:00
|
|
|
typedef struct packet_header {
|
|
|
|
uint16_t length;
|
|
|
|
uint8_t type;
|
|
|
|
uint8_t data[0];
|
|
|
|
} packet_header_t;
|
|
|
|
|
|
|
|
typedef enum {
|
|
|
|
SOCKET_W4_HEADER,
|
|
|
|
SOCKET_W4_DATA,
|
|
|
|
} SOCKET_STATE;
|
|
|
|
|
|
|
|
typedef struct connection {
|
|
|
|
data_source_t ds; // used for run loop
|
|
|
|
linked_item_t item; // used for connection list
|
|
|
|
SOCKET_STATE state;
|
|
|
|
uint16_t bytes_read;
|
|
|
|
uint16_t bytes_to_read;
|
|
|
|
uint8_t buffer[3+3+255]; // max HCI CMD + packet_header
|
|
|
|
} connection_t;
|
|
|
|
|
|
|
|
uint8_t hci_cmd_buffer[3+255];
|
|
|
|
|
2009-07-20 21:39:43 +00:00
|
|
|
uint8_t l2cap_sig_buffer[48];
|
2009-07-15 22:12:54 +00:00
|
|
|
|
|
|
|
|
|
|
|
static void dummy_handler(uint8_t *packet, int size);
|
|
|
|
static int btstack_socket_process(struct data_source *ds, int ready);
|
|
|
|
|
|
|
|
static int btstack_socket = -1;
|
|
|
|
static connection_t *btstack_connection = NULL;
|
|
|
|
static const int btstack_port = 1919;
|
|
|
|
|
|
|
|
/* callback to L2CAP layer */
|
|
|
|
static void (*event_packet_handler)(uint8_t *packet, int size) = dummy_handler;
|
|
|
|
static void (*acl_packet_handler) (uint8_t *packet, int size) = dummy_handler;
|
|
|
|
|
|
|
|
|
2009-07-14 18:29:29 +00:00
|
|
|
// init BTstack library
|
2009-07-14 20:41:14 +00:00
|
|
|
int bt_open(){
|
2009-07-15 22:12:54 +00:00
|
|
|
// TCP
|
|
|
|
struct protoent* tcp = getprotobyname("tcp");
|
2009-07-16 22:04:11 +00:00
|
|
|
|
2009-07-15 22:12:54 +00:00
|
|
|
btstack_socket = socket(PF_INET, SOCK_STREAM, tcp->p_proto);
|
|
|
|
if(btstack_socket == -1){
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
// localhost
|
|
|
|
struct sockaddr_in btdaemon_address;
|
|
|
|
btdaemon_address.sin_family = AF_INET;
|
|
|
|
btdaemon_address.sin_port = htons(btstack_port);
|
|
|
|
struct hostent* localhost = gethostbyname("localhost");
|
|
|
|
if(!localhost){
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
// connect
|
|
|
|
char* addr = localhost->h_addr_list[0];
|
|
|
|
memcpy(&btdaemon_address.sin_addr.s_addr, addr, sizeof addr);
|
|
|
|
if(connect(btstack_socket, (struct sockaddr*)&btdaemon_address, sizeof btdaemon_address) == -1){
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// register with run loop
|
|
|
|
btstack_connection = malloc( sizeof(connection_t));
|
|
|
|
if (btstack_connection == NULL) return -1;
|
|
|
|
btstack_connection->ds.fd = btstack_socket;
|
|
|
|
btstack_connection->ds.process = btstack_socket_process;
|
|
|
|
run_loop_add(&btstack_connection->ds);
|
2009-07-16 19:09:17 +00:00
|
|
|
|
|
|
|
// init state machine
|
|
|
|
btstack_connection->state = SOCKET_W4_HEADER;
|
|
|
|
btstack_connection->bytes_read = 0;
|
2009-07-20 21:39:43 +00:00
|
|
|
btstack_connection->bytes_to_read = sizeof(packet_header_t);
|
2009-07-15 22:12:54 +00:00
|
|
|
|
|
|
|
return 0;
|
2009-07-14 20:41:14 +00:00
|
|
|
}
|
2009-07-14 18:29:29 +00:00
|
|
|
|
2009-07-15 22:12:54 +00:00
|
|
|
|
2009-07-14 18:29:29 +00:00
|
|
|
// stop using BTstack library
|
2009-07-14 20:41:14 +00:00
|
|
|
int bt_close(){
|
2009-07-15 22:12:54 +00:00
|
|
|
if (btstack_socket < 0) return 0;
|
|
|
|
shutdown(btstack_socket, SHUT_RDWR);
|
|
|
|
if (btstack_connection) {
|
|
|
|
run_loop_remove(&btstack_connection->ds);
|
|
|
|
free( btstack_connection );
|
|
|
|
btstack_connection = NULL;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// run_loop based data handling
|
|
|
|
int btstack_socket_process(struct data_source *ds, int ready) {
|
|
|
|
connection_t *conn = (connection_t *) ds;
|
2009-07-20 21:39:43 +00:00
|
|
|
int bytes_read = read(ds->fd, &conn->buffer[conn->bytes_read], conn->bytes_to_read);
|
2009-07-15 22:12:54 +00:00
|
|
|
if (bytes_read <= 0){
|
|
|
|
// free
|
|
|
|
run_loop_remove(&conn->ds);
|
|
|
|
free(conn);
|
|
|
|
// TODO notify client of cailed daemon connection!
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
conn->bytes_read += bytes_read;
|
|
|
|
conn->bytes_to_read -= bytes_read;
|
2009-07-20 21:39:43 +00:00
|
|
|
hexdump( conn->buffer, conn->bytes_read);
|
2009-07-15 22:12:54 +00:00
|
|
|
if (conn->bytes_to_read > 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
switch (conn->state){
|
|
|
|
case SOCKET_W4_HEADER:
|
|
|
|
conn->state = SOCKET_W4_DATA;
|
2009-07-20 21:39:43 +00:00
|
|
|
conn->bytes_to_read = READ_BT_16( conn->buffer, 0);
|
2009-07-15 22:12:54 +00:00
|
|
|
break;
|
|
|
|
case SOCKET_W4_DATA:
|
|
|
|
// process packet !!!
|
|
|
|
switch (conn->buffer[2]){
|
|
|
|
case HCI_EVENT_PACKET:
|
2009-07-20 21:39:43 +00:00
|
|
|
(*event_packet_handler)(&conn->buffer[sizeof(packet_header_t)], READ_BT_16( conn->buffer, 0));
|
2009-07-15 22:12:54 +00:00
|
|
|
break;
|
|
|
|
case HCI_ACL_DATA_PACKET:
|
2009-07-20 21:39:43 +00:00
|
|
|
(*acl_packet_handler)(&conn->buffer[sizeof(packet_header_t)], READ_BT_16( conn->buffer, 0));
|
2009-07-15 22:12:54 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// wait for next packet
|
|
|
|
conn->state = SOCKET_W4_HEADER;
|
|
|
|
conn->bytes_read = 0;
|
|
|
|
conn->bytes_to_read = sizeof(packet_header_t);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
2009-07-14 20:41:14 +00:00
|
|
|
}
|
2009-07-14 18:29:29 +00:00
|
|
|
|
2009-07-15 22:12:54 +00:00
|
|
|
|
2009-07-14 18:29:29 +00:00
|
|
|
// send hci cmd packet
|
2009-07-14 20:41:14 +00:00
|
|
|
int bt_send_cmd(hci_cmd_t *cmd, ...){
|
2009-07-15 22:12:54 +00:00
|
|
|
va_list argptr;
|
|
|
|
va_start(argptr, cmd);
|
|
|
|
uint16_t len = hci_create_cmd_internal(hci_cmd_buffer, cmd, argptr);
|
|
|
|
va_end(argptr);
|
2009-07-16 22:04:11 +00:00
|
|
|
packet_header_t header;
|
|
|
|
bt_store_16( (uint8_t*) &header.length, 0, len);
|
|
|
|
header.type = HCI_COMMAND_DATA_PACKET;
|
2009-07-20 21:39:43 +00:00
|
|
|
hexdump( (uint8_t*)&header, sizeof(header));
|
2009-07-16 22:04:11 +00:00
|
|
|
write(btstack_socket, (uint8_t*)&header, sizeof(header));
|
|
|
|
hexdump( hci_cmd_buffer, len);
|
|
|
|
write(btstack_socket, hci_cmd_buffer, len);
|
2009-07-15 22:12:54 +00:00
|
|
|
return len;
|
2009-07-14 20:41:14 +00:00
|
|
|
}
|
2009-07-14 18:29:29 +00:00
|
|
|
|
|
|
|
// send hci acl packet
|
2009-07-14 20:41:14 +00:00
|
|
|
int bt_send_acl_packet(uint8_t *packet, int size){
|
2009-07-20 21:39:43 +00:00
|
|
|
packet_header_t header;
|
|
|
|
bt_store_16( (uint8_t*) &header.length, 0, size);
|
|
|
|
header.type = HCI_ACL_DATA_PACKET;
|
|
|
|
hexdump( (uint8_t*)&header, sizeof(header));
|
|
|
|
write(btstack_socket, (uint8_t*)&header, sizeof(header));
|
|
|
|
hexdump( hci_cmd_buffer, size);
|
|
|
|
write(btstack_socket, packet, size);
|
2009-07-15 22:12:54 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-07-20 21:39:43 +00:00
|
|
|
int bt_send_l2cap_signaling_packet(hci_con_handle_t handle, L2CAP_SIGNALING_COMMANDS cmd, uint8_t identifier, ...){
|
|
|
|
va_list argptr;
|
|
|
|
va_start(argptr, identifier);
|
2009-07-21 19:14:04 +00:00
|
|
|
uint16_t len = l2cap_create_signaling_internal(l2cap_sig_buffer, handle, cmd, identifier, argptr);
|
2009-07-20 21:39:43 +00:00
|
|
|
va_end(argptr);
|
|
|
|
return bt_send_acl_packet(l2cap_sig_buffer, len);
|
|
|
|
}
|
|
|
|
|
2009-07-15 22:12:54 +00:00
|
|
|
static void dummy_handler(uint8_t *packet, int size){
|
2009-07-14 20:41:14 +00:00
|
|
|
}
|
2009-07-14 18:29:29 +00:00
|
|
|
|
|
|
|
// register packet and event handler
|
2009-07-14 20:41:14 +00:00
|
|
|
void bt_register_event_packet_handler(void (*handler)(uint8_t *packet, int size)){
|
2009-07-15 22:12:54 +00:00
|
|
|
event_packet_handler = handler;
|
2009-07-14 20:41:14 +00:00
|
|
|
}
|
2009-07-15 22:12:54 +00:00
|
|
|
|
2009-07-14 20:41:14 +00:00
|
|
|
void bt_register_acl_packet_handler (void (*handler)(uint8_t *packet, int size)){
|
2009-07-15 22:12:54 +00:00
|
|
|
acl_packet_handler = handler;
|
2009-07-14 20:41:14 +00:00
|
|
|
}
|
2009-07-14 18:29:29 +00:00
|
|
|
|