2009-07-01 21:55:08 +00:00
|
|
|
/*
|
|
|
|
* daemon.c
|
|
|
|
*
|
|
|
|
* Created by Matthias Ringwald on 7/1/09.
|
|
|
|
*
|
|
|
|
* BTstack background daemon
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2009-07-14 18:29:29 +00:00
|
|
|
#include "../config.h"
|
|
|
|
|
2009-08-15 19:52:46 +00:00
|
|
|
#include <signal.h>
|
2009-07-01 21:55:08 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <strings.h>
|
2009-08-15 19:52:46 +00:00
|
|
|
#include <unistd.h>
|
2009-07-01 21:55:08 +00:00
|
|
|
|
|
|
|
#include "hci.h"
|
|
|
|
#include "hci_dump.h"
|
|
|
|
#include "l2cap.h"
|
2009-07-13 21:51:42 +00:00
|
|
|
#include "linked_list.h"
|
2009-07-01 21:55:08 +00:00
|
|
|
#include "run_loop.h"
|
2009-07-22 20:27:00 +00:00
|
|
|
#include "socket_connection.h"
|
2009-07-01 21:55:08 +00:00
|
|
|
|
2009-07-11 10:37:48 +00:00
|
|
|
#ifdef USE_BLUETOOL
|
|
|
|
#include "bt_control_iphone.h"
|
|
|
|
#endif
|
|
|
|
|
2009-08-16 22:00:58 +00:00
|
|
|
#ifdef USE_SPRINGBOARD
|
|
|
|
#include "platform_iphone.h"
|
|
|
|
#endif
|
|
|
|
|
2009-07-11 10:37:48 +00:00
|
|
|
#ifdef HAVE_TRANSPORT_H4
|
|
|
|
#include "hci_transport_h4.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef HAVE_TRANSPORT_USB
|
2009-07-09 18:49:43 +00:00
|
|
|
#include <libusb-1.0/libusb.h>
|
2009-07-11 10:37:48 +00:00
|
|
|
#include "hci_transport_usb.h"
|
|
|
|
#endif
|
2009-07-01 21:55:08 +00:00
|
|
|
|
2009-08-09 20:18:34 +00:00
|
|
|
#define DAEMON_NO_CONNECTION_TIMEOUT 60
|
|
|
|
|
2009-07-01 21:55:08 +00:00
|
|
|
static hci_transport_t * transport;
|
|
|
|
static hci_uart_config_t config;
|
|
|
|
|
2009-08-09 20:18:34 +00:00
|
|
|
static timer_t timeout;
|
|
|
|
|
2009-08-15 21:31:23 +00:00
|
|
|
static void dummy_bluetooth_status_handler(BLUETOOTH_STATE state){
|
|
|
|
printf("Bluetooth status: %u\n", state);
|
|
|
|
};
|
|
|
|
static void (*bluetooth_status_handler)(BLUETOOTH_STATE state) = dummy_bluetooth_status_handler;
|
|
|
|
|
2009-08-09 20:18:34 +00:00
|
|
|
static void daemon_no_connections_timeout(){
|
2009-08-30 18:17:16 +00:00
|
|
|
#ifdef USE_LAUNCHD
|
2009-08-19 21:31:40 +00:00
|
|
|
printf("No connection for %u seconds -> POWER OFF and quit\n", DAEMON_NO_CONNECTION_TIMEOUT);
|
2009-08-09 20:18:34 +00:00
|
|
|
hci_power_control( HCI_POWER_OFF);
|
2009-08-19 21:31:40 +00:00
|
|
|
exit(0);
|
2009-08-30 18:17:16 +00:00
|
|
|
#else
|
|
|
|
printf("No connection for %u seconds -> POWER OFF\n", DAEMON_NO_CONNECTION_TIMEOUT);
|
|
|
|
hci_power_control( HCI_POWER_OFF);
|
|
|
|
#endif
|
2009-08-09 20:18:34 +00:00
|
|
|
}
|
|
|
|
|
2009-07-31 21:41:15 +00:00
|
|
|
static int btstack_command_handler(connection_t *connection, uint8_t *packet, uint16_t size){
|
|
|
|
// BTstack Commands
|
|
|
|
hci_dump_packet( HCI_COMMAND_DATA_PACKET, 1, packet, size);
|
|
|
|
bd_addr_t addr;
|
|
|
|
uint16_t cid;
|
|
|
|
uint16_t psm;
|
|
|
|
uint8_t reason;
|
|
|
|
// BTstack internal commands - 16 Bit OpCode, 8 Bit ParamLen, Params...
|
|
|
|
switch (READ_CMD_OCF(packet)){
|
|
|
|
case HCI_BTSTACK_GET_STATE:
|
|
|
|
hci_emit_state();
|
|
|
|
break;
|
|
|
|
case HCI_BTSTACK_SET_POWER_MODE:
|
|
|
|
hci_power_control(packet[3]);
|
|
|
|
break;
|
2009-08-24 21:56:12 +00:00
|
|
|
case HCI_BTSTACK_SET_ACL_CAPTURE_MODE:
|
|
|
|
if (packet[3]) {
|
|
|
|
l2cap_set_capture_connection(connection);
|
|
|
|
} else {
|
|
|
|
l2cap_set_capture_connection(NULL);
|
|
|
|
}
|
|
|
|
break;
|
2009-07-31 21:41:15 +00:00
|
|
|
case L2CAP_CREATE_CHANNEL:
|
|
|
|
bt_flip_addr(addr, &packet[3]);
|
|
|
|
psm = READ_BT_16(packet, 9);
|
|
|
|
l2cap_create_channel_internal( connection, addr, psm );
|
|
|
|
break;
|
|
|
|
case L2CAP_DISCONNECT:
|
|
|
|
cid = READ_BT_16(packet, 3);
|
|
|
|
reason = packet[5];
|
|
|
|
l2cap_disconnect_internal(cid, reason);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
//@TODO: log into hci dump as vendor specific "event"
|
2009-09-22 20:03:48 +00:00
|
|
|
fprintf(stderr, "Error: command %u not implemented\n:", READ_CMD_OCF(packet));
|
2009-07-31 21:41:15 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int daemon_client_handler(connection_t *connection, uint16_t packet_type, uint16_t channel, uint8_t *data, uint16_t length){
|
2009-07-22 20:21:22 +00:00
|
|
|
switch (packet_type){
|
|
|
|
case HCI_COMMAND_DATA_PACKET:
|
2009-07-31 21:41:15 +00:00
|
|
|
if (READ_CMD_OGF(data) != OGF_BTSTACK) {
|
|
|
|
// HCI Command
|
|
|
|
hci_send_cmd_packet(data, length);
|
|
|
|
} else {
|
|
|
|
// BTstack command
|
|
|
|
btstack_command_handler(connection, data, length);
|
|
|
|
}
|
2009-07-22 20:21:22 +00:00
|
|
|
break;
|
2009-08-24 21:56:12 +00:00
|
|
|
case HCI_ACL_DATA_PACKET:
|
|
|
|
hci_send_acl_packet(data, length);
|
|
|
|
break;
|
2009-08-02 13:32:55 +00:00
|
|
|
case L2CAP_DATA_PACKET:
|
2009-07-31 21:41:15 +00:00
|
|
|
// process l2cap packet...
|
|
|
|
l2cap_send_internal(channel, data, length);
|
2009-07-22 20:21:22 +00:00
|
|
|
break;
|
2009-08-09 17:17:00 +00:00
|
|
|
case DAEMON_EVENT_PACKET:
|
2009-08-09 20:18:34 +00:00
|
|
|
switch (data[0]) {
|
|
|
|
case DAEMON_CONNECTION_CLOSED:
|
|
|
|
l2cap_close_channels_for_connection(connection);
|
|
|
|
break;
|
|
|
|
case DAEMON_NR_CONNECTIONS_CHANGED:
|
2009-09-22 20:03:48 +00:00
|
|
|
printf("Nr Connections changed, new %u\n", data[1]);
|
2009-08-09 20:18:34 +00:00
|
|
|
if (data[1]) {
|
|
|
|
run_loop_remove_timer(&timeout);
|
|
|
|
} else {
|
|
|
|
run_loop_set_timer(&timeout, DAEMON_NO_CONNECTION_TIMEOUT);
|
|
|
|
run_loop_add_timer(&timeout);
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2009-08-09 17:17:00 +00:00
|
|
|
break;
|
2009-07-22 20:21:22 +00:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-08-15 21:31:23 +00:00
|
|
|
static void daemon_event_handler(uint8_t *packet, uint16_t size){
|
|
|
|
// handle state event
|
|
|
|
if (packet[0] == HCI_EVENT_BTSTACK_WORKING){
|
|
|
|
bluetooth_status_handler(BLUETOOTH_ON);
|
|
|
|
}
|
|
|
|
if (packet[0] == HCI_EVENT_BTSTACK_STATE){
|
|
|
|
if (packet[2] == HCI_STATE_WORKING) {
|
|
|
|
bluetooth_status_handler(BLUETOOTH_ON);
|
|
|
|
}
|
|
|
|
if (packet[2] == HCI_STATE_OFF) {
|
|
|
|
bluetooth_status_handler(BLUETOOTH_OFF);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (packet[0] == HCI_EVENT_NR_CONNECTIONS_CHANGED){
|
|
|
|
if (packet[2]) {
|
|
|
|
bluetooth_status_handler(BLUETOOTH_ACTIVE);
|
|
|
|
} else {
|
|
|
|
bluetooth_status_handler(BLUETOOTH_ON);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// forward event to clients
|
|
|
|
socket_connection_send_packet_all(HCI_EVENT_PACKET, 0, packet, size);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void daemon_sigint_handler(int param){
|
2009-08-15 19:52:46 +00:00
|
|
|
printf(" <= SIGINT received, shutting down..\n");
|
|
|
|
hci_power_control( HCI_POWER_OFF);
|
|
|
|
printf("Good bye, see you.\n");
|
|
|
|
exit(0);
|
|
|
|
}
|
|
|
|
|
2009-07-11 17:42:03 +00:00
|
|
|
int main (int argc, const char * argv[]){
|
2009-07-01 21:55:08 +00:00
|
|
|
|
|
|
|
bt_control_t * control = NULL;
|
|
|
|
|
2009-07-11 10:37:48 +00:00
|
|
|
#ifdef HAVE_TRANSPORT_H4
|
|
|
|
transport = hci_transport_h4_instance();
|
|
|
|
config.device_name = UART_DEVICE;
|
|
|
|
config.baudrate = UART_SPEED;
|
2009-07-01 21:55:08 +00:00
|
|
|
config.flowcontrol = 1;
|
2009-07-11 10:37:48 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef HAVE_TRANSPORT_USB
|
|
|
|
transport = hci_transport_usb_instance();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef USE_BLUETOOL
|
2009-07-01 21:55:08 +00:00
|
|
|
control = &bt_control_iphone;
|
|
|
|
#endif
|
2009-08-15 21:31:23 +00:00
|
|
|
|
|
|
|
#ifdef USE_SPRINGBOARD
|
2009-08-16 22:04:31 +00:00
|
|
|
bluetooth_status_handler = platform_iphone_status_handler;
|
2009-08-15 21:31:23 +00:00
|
|
|
#endif
|
|
|
|
|
2009-07-31 21:41:15 +00:00
|
|
|
// @TODO: allow configuration per HCI CMD
|
2009-07-14 18:29:29 +00:00
|
|
|
|
2009-07-16 20:12:42 +00:00
|
|
|
// use logger: format HCI_DUMP_PACKETLOGGER, HCI_DUMP_BLUEZ or HCI_DUMP_STDOUT
|
2009-09-02 09:19:52 +00:00
|
|
|
// hci_dump_open("/tmp/hci_dump.pklg", HCI_DUMP_PACKETLOGGER);
|
|
|
|
hci_dump_open(NULL, HCI_DUMP_STDOUT);
|
2009-07-16 20:12:42 +00:00
|
|
|
|
2009-07-01 21:55:08 +00:00
|
|
|
// init HCI
|
|
|
|
hci_init(transport, &config, control);
|
|
|
|
|
|
|
|
// init L2CAP
|
|
|
|
l2cap_init();
|
2009-08-15 21:31:23 +00:00
|
|
|
l2cap_register_event_packet_handler(daemon_event_handler);
|
2009-08-09 20:18:34 +00:00
|
|
|
timeout.process = daemon_no_connections_timeout;
|
|
|
|
|
2009-07-31 21:41:15 +00:00
|
|
|
// @TODO: make choice of socket server configurable (TCP and/or Unix Domain Socket)
|
2009-07-14 18:29:29 +00:00
|
|
|
|
2009-07-01 21:55:08 +00:00
|
|
|
// create server
|
2009-08-19 20:09:30 +00:00
|
|
|
// socket_connection_create_tcp(BTSTACK_PORT);
|
|
|
|
socket_connection_create_unix(BTSTACK_UNIX);
|
2009-08-20 22:54:29 +00:00
|
|
|
|
2009-07-31 21:41:15 +00:00
|
|
|
socket_connection_register_packet_callback(daemon_client_handler);
|
2009-08-15 19:52:46 +00:00
|
|
|
|
|
|
|
// handle CTRL-c
|
|
|
|
signal(SIGINT, daemon_sigint_handler);
|
2009-08-19 21:06:05 +00:00
|
|
|
// handle SIGTERM - suggested for launchd
|
|
|
|
signal(SIGTERM, daemon_sigint_handler);
|
|
|
|
// make stderr unbuffered
|
|
|
|
setbuf(stderr, NULL);
|
2009-08-20 22:54:29 +00:00
|
|
|
setbuf(stdout, NULL);
|
|
|
|
printf("BTdaemon started - stdout\n");
|
|
|
|
fprintf(stderr,"BTdaemon started - stderr\n");
|
2009-07-01 21:55:08 +00:00
|
|
|
// go!
|
|
|
|
run_loop_execute();
|
2009-07-14 18:29:29 +00:00
|
|
|
|
2009-07-01 21:55:08 +00:00
|
|
|
return 0;
|
|
|
|
}
|