2009-10-29 20:25:42 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2009 by Matthias Ringwald
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
*
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
* 3. Neither the name of the copyright holders nor the names of
|
|
|
|
* contributors may be used to endorse or promote products derived
|
|
|
|
* from this software without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY MATTHIAS RINGWALD AND CONTRIBUTORS
|
|
|
|
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
|
|
|
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MATTHIAS
|
|
|
|
* RINGWALD OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
|
|
|
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
|
|
|
|
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
|
|
|
|
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
|
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
|
|
|
|
* THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
|
* SUCH DAMAGE.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
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"
|
|
|
|
|
2011-06-03 21:34:00 +00:00
|
|
|
#include <pthread.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
|
|
|
|
2010-02-28 22:02:28 +00:00
|
|
|
#include <getopt.h>
|
|
|
|
|
2010-02-28 21:37:59 +00:00
|
|
|
#include <btstack/btstack.h>
|
2009-09-28 21:19:05 +00:00
|
|
|
#include <btstack/linked_list.h>
|
|
|
|
#include <btstack/run_loop.h>
|
2010-06-13 08:26:47 +00:00
|
|
|
|
2011-05-05 18:45:28 +00:00
|
|
|
#include "debug.h"
|
2010-06-13 08:26:47 +00:00
|
|
|
#include "hci.h"
|
|
|
|
#include "hci_dump.h"
|
2010-09-24 21:01:07 +00:00
|
|
|
#include "hci_transport.h"
|
2010-06-13 08:26:47 +00:00
|
|
|
#include "l2cap.h"
|
2011-05-01 11:36:12 +00:00
|
|
|
#include "rfcomm.h"
|
2010-06-13 08:26:47 +00:00
|
|
|
#include "sdp.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
|
2011-06-03 21:34:00 +00:00
|
|
|
#include <CoreFoundation/CoreFoundation.h>
|
2009-07-11 10:37:48 +00:00
|
|
|
#include "bt_control_iphone.h"
|
2011-04-30 20:11:20 +00:00
|
|
|
#include <notify.h>
|
2009-07-11 10:37:48 +00:00
|
|
|
#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_USB
|
2009-07-09 18:49:43 +00:00
|
|
|
#include <libusb-1.0/libusb.h>
|
2009-07-11 10:37:48 +00:00
|
|
|
#endif
|
2009-07-01 21:55:08 +00:00
|
|
|
|
2011-01-08 22:01:06 +00:00
|
|
|
#define DAEMON_NO_ACTIVE_CLIENT_TIMEOUT 10000
|
2009-08-09 20:18:34 +00:00
|
|
|
|
2011-01-07 21:34:50 +00:00
|
|
|
|
2011-01-09 14:48:03 +00:00
|
|
|
typedef struct {
|
|
|
|
// linked list - assert: first field
|
|
|
|
linked_item_t item;
|
|
|
|
|
|
|
|
// connection
|
|
|
|
connection_t * connection;
|
|
|
|
|
|
|
|
// power mode
|
|
|
|
HCI_POWER_MODE power_mode;
|
|
|
|
|
|
|
|
// discoverable
|
|
|
|
uint8_t discoverable;
|
|
|
|
|
|
|
|
} client_state_t;
|
2011-01-07 21:34:50 +00:00
|
|
|
|
2011-01-10 21:11:58 +00:00
|
|
|
#pragma mark prototypes
|
|
|
|
static void dummy_bluetooth_status_handler(BLUETOOTH_STATE state);
|
|
|
|
static client_state_t * client_for_connection(connection_t *connection);
|
2011-06-06 14:12:38 +00:00
|
|
|
static int clients_require_power_on(void);
|
|
|
|
static int clients_require_discoverable(void);
|
|
|
|
static void clients_clear_power_request(void);
|
|
|
|
static void start_power_off_timer(void);
|
|
|
|
static void stop_power_off_timer(void);
|
2011-01-10 21:11:58 +00:00
|
|
|
|
|
|
|
#pragma mark globals
|
2011-01-07 21:34:50 +00:00
|
|
|
static hci_transport_t * transport;
|
|
|
|
static hci_uart_config_t config;
|
|
|
|
static timer_source_t timeout;
|
|
|
|
static uint8_t timeout_active = 0;
|
2011-01-08 22:01:06 +00:00
|
|
|
static int power_management_sleep = 0;
|
2011-01-09 14:48:03 +00:00
|
|
|
static linked_list_t clients = NULL; // list of connected clients
|
2011-01-10 21:11:58 +00:00
|
|
|
static void (*bluetooth_status_handler)(BLUETOOTH_STATE state) = dummy_bluetooth_status_handler;
|
2011-01-09 14:48:03 +00:00
|
|
|
|
2011-04-30 20:11:20 +00:00
|
|
|
static int global_enable = 0;
|
2010-12-24 00:22:04 +00:00
|
|
|
|
2011-05-03 21:14:41 +00:00
|
|
|
static remote_device_db_t * remote_device_db = NULL;
|
2011-05-27 20:47:31 +00:00
|
|
|
static int rfcomm_channel_generator = 1;
|
2011-05-03 21:14:41 +00:00
|
|
|
|
2009-08-15 21:31:23 +00:00
|
|
|
static void dummy_bluetooth_status_handler(BLUETOOTH_STATE state){
|
2011-05-05 18:45:28 +00:00
|
|
|
log_dbg("Bluetooth status: %u\n", state);
|
2009-08-15 21:31:23 +00:00
|
|
|
};
|
|
|
|
|
2011-06-06 14:12:38 +00:00
|
|
|
static void daemon_no_connections_timeout(void){
|
2011-01-10 21:11:58 +00:00
|
|
|
if (clients_require_power_on()) return; // false alarm :)
|
2011-05-05 18:45:28 +00:00
|
|
|
log_dbg("No active client connection for %u seconds -> POWER OFF\n", DAEMON_NO_ACTIVE_CLIENT_TIMEOUT/1000);
|
2011-01-07 22:21:25 +00:00
|
|
|
hci_power_control(HCI_POWER_OFF);
|
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){
|
2011-01-09 14:48:03 +00:00
|
|
|
|
2009-07-31 21:41:15 +00:00
|
|
|
hci_dump_packet( HCI_COMMAND_DATA_PACKET, 1, packet, size);
|
2011-01-09 14:48:03 +00:00
|
|
|
|
2009-07-31 21:41:15 +00:00
|
|
|
bd_addr_t addr;
|
|
|
|
uint16_t cid;
|
|
|
|
uint16_t psm;
|
2011-04-30 20:30:05 +00:00
|
|
|
uint16_t service_channel;
|
2010-01-25 20:25:20 +00:00
|
|
|
uint16_t mtu;
|
|
|
|
uint8_t reason;
|
2011-04-30 20:30:05 +00:00
|
|
|
uint8_t rfcomm_channel;
|
2010-06-12 21:00:22 +00:00
|
|
|
uint32_t service_record_handle;
|
2011-01-09 14:48:03 +00:00
|
|
|
client_state_t *client;
|
2010-06-12 21:00:22 +00:00
|
|
|
|
2009-07-31 21:41:15 +00:00
|
|
|
// BTstack internal commands - 16 Bit OpCode, 8 Bit ParamLen, Params...
|
|
|
|
switch (READ_CMD_OCF(packet)){
|
2009-09-29 19:40:55 +00:00
|
|
|
case BTSTACK_GET_STATE:
|
2009-07-31 21:41:15 +00:00
|
|
|
hci_emit_state();
|
|
|
|
break;
|
2009-09-29 19:40:55 +00:00
|
|
|
case BTSTACK_SET_POWER_MODE:
|
2011-01-08 22:01:06 +00:00
|
|
|
// track client power requests
|
2011-01-09 14:48:03 +00:00
|
|
|
client = client_for_connection(connection);
|
|
|
|
if (!client) break;
|
|
|
|
client->power_mode = packet[3];
|
2011-01-08 22:01:06 +00:00
|
|
|
// handle merged state
|
2011-01-10 21:11:58 +00:00
|
|
|
if (!clients_require_power_on()){
|
2011-01-09 14:48:03 +00:00
|
|
|
start_power_off_timer();
|
2011-01-08 22:01:06 +00:00
|
|
|
} else if (!power_management_sleep) {
|
2011-01-09 14:48:03 +00:00
|
|
|
stop_power_off_timer();
|
2011-01-08 22:01:06 +00:00
|
|
|
hci_power_control(HCI_POWER_ON);
|
|
|
|
}
|
2009-07-31 21:41:15 +00:00
|
|
|
break;
|
2010-01-09 11:01:23 +00:00
|
|
|
case BTSTACK_GET_VERSION:
|
|
|
|
hci_emit_btstack_version();
|
|
|
|
break;
|
2010-01-09 18:25:30 +00:00
|
|
|
#ifdef USE_BLUETOOL
|
|
|
|
case BTSTACK_SET_SYSTEM_BLUETOOTH_ENABLED:
|
|
|
|
iphone_system_bt_set_enabled(packet[3]);
|
2011-01-10 21:11:58 +00:00
|
|
|
hci_emit_system_bluetooth_enabled(iphone_system_bt_enabled());
|
|
|
|
break;
|
|
|
|
|
2010-01-09 18:25:30 +00:00
|
|
|
case BTSTACK_GET_SYSTEM_BLUETOOTH_ENABLED:
|
|
|
|
hci_emit_system_bluetooth_enabled(iphone_system_bt_enabled());
|
|
|
|
break;
|
|
|
|
#else
|
|
|
|
case BTSTACK_SET_SYSTEM_BLUETOOTH_ENABLED:
|
|
|
|
case BTSTACK_GET_SYSTEM_BLUETOOTH_ENABLED:
|
|
|
|
hci_emit_system_bluetooth_enabled(0);
|
|
|
|
break;
|
|
|
|
#endif
|
2011-01-10 21:11:58 +00:00
|
|
|
case BTSTACK_SET_DISCOVERABLE:
|
|
|
|
// track client discoverable requests
|
|
|
|
client = client_for_connection(connection);
|
|
|
|
if (!client) break;
|
|
|
|
client->discoverable = packet[3];
|
|
|
|
// merge state
|
|
|
|
hci_discoverable_control(clients_require_discoverable());
|
|
|
|
break;
|
2011-04-30 20:11:20 +00:00
|
|
|
case BTSTACK_SET_BLUETOOTH_ENABLED:
|
2011-05-05 18:45:28 +00:00
|
|
|
log_dbg("BTSTACK_SET_BLUETOOTH_ENABLED: %u\n", packet[3]);
|
2011-04-30 20:11:20 +00:00
|
|
|
|
|
|
|
if (packet[3]) {
|
|
|
|
// global enable
|
|
|
|
global_enable = 1;
|
|
|
|
hci_power_control(HCI_POWER_ON);
|
|
|
|
} else {
|
|
|
|
global_enable = 0;
|
|
|
|
clients_clear_power_request();
|
|
|
|
hci_power_control(HCI_POWER_OFF);
|
|
|
|
}
|
|
|
|
break;
|
2010-08-10 20:43:55 +00:00
|
|
|
case L2CAP_CREATE_CHANNEL_MTU:
|
|
|
|
bt_flip_addr(addr, &packet[3]);
|
|
|
|
psm = READ_BT_16(packet, 9);
|
|
|
|
mtu = READ_BT_16(packet, 11);
|
|
|
|
l2cap_create_channel_internal( connection, NULL, addr, psm, mtu);
|
|
|
|
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);
|
2010-08-10 20:43:55 +00:00
|
|
|
l2cap_create_channel_internal( connection, NULL, addr, psm, 150); // until r865
|
2009-07-31 21:41:15 +00:00
|
|
|
break;
|
|
|
|
case L2CAP_DISCONNECT:
|
|
|
|
cid = READ_BT_16(packet, 3);
|
|
|
|
reason = packet[5];
|
|
|
|
l2cap_disconnect_internal(cid, reason);
|
|
|
|
break;
|
2010-01-25 20:25:20 +00:00
|
|
|
case L2CAP_REGISTER_SERVICE:
|
|
|
|
psm = READ_BT_16(packet, 3);
|
|
|
|
mtu = READ_BT_16(packet, 5);
|
2010-06-09 17:34:52 +00:00
|
|
|
l2cap_register_service_internal(connection, NULL, psm, mtu);
|
2010-01-25 20:25:20 +00:00
|
|
|
break;
|
|
|
|
case L2CAP_UNREGISTER_SERVICE:
|
|
|
|
psm = READ_BT_16(packet, 3);
|
|
|
|
l2cap_unregister_service_internal(connection, psm);
|
|
|
|
break;
|
|
|
|
case L2CAP_ACCEPT_CONNECTION:
|
2010-01-26 20:49:02 +00:00
|
|
|
cid = READ_BT_16(packet, 3);
|
|
|
|
l2cap_accept_connection_internal(cid);
|
2010-01-25 20:25:20 +00:00
|
|
|
break;
|
|
|
|
case L2CAP_DECLINE_CONNECTION:
|
2010-01-26 20:49:02 +00:00
|
|
|
cid = READ_BT_16(packet, 3);
|
2010-01-25 20:25:20 +00:00
|
|
|
reason = packet[7];
|
2010-01-26 20:49:02 +00:00
|
|
|
l2cap_decline_connection_internal(cid, reason);
|
2010-01-25 20:25:20 +00:00
|
|
|
break;
|
2011-04-30 20:30:05 +00:00
|
|
|
|
|
|
|
case RFCOMM_CREATE_CHANNEL:
|
|
|
|
bt_flip_addr(addr, &packet[3]);
|
|
|
|
rfcomm_channel = packet[9];
|
|
|
|
rfcomm_create_channel_internal( connection, &addr, rfcomm_channel );
|
|
|
|
break;
|
|
|
|
case RFCOMM_DISCONNECT:
|
|
|
|
cid = READ_BT_16(packet, 3);
|
|
|
|
reason = packet[5];
|
|
|
|
rfcomm_disconnect_internal(cid);
|
|
|
|
break;
|
|
|
|
case RFCOMM_REGISTER_SERVICE:
|
2011-05-03 21:14:41 +00:00
|
|
|
rfcomm_channel = packet[3];
|
|
|
|
mtu = READ_BT_16(packet, 4);
|
|
|
|
rfcomm_register_service_internal(connection, rfcomm_channel, mtu);
|
2011-04-30 20:30:05 +00:00
|
|
|
break;
|
|
|
|
case RFCOMM_UNREGISTER_SERVICE:
|
|
|
|
service_channel = READ_BT_16(packet, 3);
|
|
|
|
rfcomm_unregister_service_internal(service_channel);
|
|
|
|
break;
|
|
|
|
case RFCOMM_ACCEPT_CONNECTION:
|
|
|
|
cid = READ_BT_16(packet, 3);
|
|
|
|
rfcomm_accept_connection_internal(cid);
|
|
|
|
break;
|
|
|
|
case RFCOMM_DECLINE_CONNECTION:
|
|
|
|
cid = READ_BT_16(packet, 3);
|
|
|
|
reason = packet[7];
|
|
|
|
rfcomm_decline_connection_internal(cid);
|
|
|
|
break;
|
2011-05-03 21:14:41 +00:00
|
|
|
case RFCOMM_PERSISTENT_CHANNEL: {
|
|
|
|
if (remote_device_db) {
|
|
|
|
// enforce \0
|
|
|
|
packet[3+248] = 0;
|
2011-05-27 20:47:31 +00:00
|
|
|
rfcomm_channel = remote_device_db->persistent_rfcomm_channel((char*)&packet[3]);
|
2011-05-03 21:14:41 +00:00
|
|
|
} else {
|
|
|
|
// NOTE: hack for non-iOS platforms
|
|
|
|
rfcomm_channel = rfcomm_channel_generator++;
|
|
|
|
}
|
|
|
|
uint8_t event[4];
|
|
|
|
event[0] = RFCOMM_EVENT_PERSISTENT_CHANNEL;
|
|
|
|
event[1] = sizeof(event) - 2;
|
|
|
|
event[2] = 0;
|
|
|
|
event[3] = rfcomm_channel;
|
|
|
|
hci_dump_packet(HCI_EVENT_PACKET, 0, event, sizeof(event));
|
|
|
|
socket_connection_send_packet(connection, HCI_EVENT_PACKET, 0, (uint8_t *) event, sizeof(event));
|
|
|
|
break;
|
|
|
|
}
|
2011-04-30 20:30:05 +00:00
|
|
|
|
2010-06-12 21:00:22 +00:00
|
|
|
case SDP_REGISTER_SERVICE_RECORD:
|
2011-05-05 18:45:28 +00:00
|
|
|
log_dbg("SDP_REGISTER_SERVICE_RECORD size %u\n", size);
|
2010-06-13 08:26:47 +00:00
|
|
|
sdp_register_service_internal(connection, &packet[3]);
|
2010-06-12 21:00:22 +00:00
|
|
|
break;
|
|
|
|
case SDP_UNREGISTER_SERVICE_RECORD:
|
|
|
|
service_record_handle = READ_BT_32(packet, 3);
|
2010-06-13 08:26:47 +00:00
|
|
|
sdp_unregister_service_internal(connection, service_record_handle);
|
2010-06-12 21:00:22 +00:00
|
|
|
break;
|
2009-07-31 21:41:15 +00:00
|
|
|
default:
|
|
|
|
//@TODO: log into hci dump as vendor specific "event"
|
2011-05-05 18:45:28 +00:00
|
|
|
log_err("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){
|
2010-07-29 19:35:34 +00:00
|
|
|
|
|
|
|
int err = 0;
|
2011-01-09 14:48:03 +00:00
|
|
|
client_state_t * client;
|
2010-07-29 19:35:34 +00:00
|
|
|
|
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:
|
2010-07-29 19:35:34 +00:00
|
|
|
err = hci_send_acl_packet(data, length);
|
2009-08-24 21:56:12 +00:00
|
|
|
break;
|
2009-08-02 13:32:55 +00:00
|
|
|
case L2CAP_DATA_PACKET:
|
2009-07-31 21:41:15 +00:00
|
|
|
// process l2cap packet...
|
2010-07-29 19:35:34 +00:00
|
|
|
err = l2cap_send_internal(channel, data, length);
|
2009-07-22 20:21:22 +00:00
|
|
|
break;
|
2011-04-30 20:30:05 +00:00
|
|
|
case RFCOMM_DATA_PACKET:
|
|
|
|
// process l2cap packet...
|
|
|
|
err = rfcomm_send_internal(channel, data, length);
|
|
|
|
break;
|
2009-08-09 17:17:00 +00:00
|
|
|
case DAEMON_EVENT_PACKET:
|
2009-08-09 20:18:34 +00:00
|
|
|
switch (data[0]) {
|
2011-01-09 14:48:03 +00:00
|
|
|
case DAEMON_EVENT_CONNECTION_OPENED:
|
|
|
|
client = malloc(sizeof(client_state_t));
|
|
|
|
if (!client) break; // fail
|
|
|
|
client->connection = connection;
|
|
|
|
client->power_mode = HCI_POWER_OFF;
|
|
|
|
client->discoverable = 0;
|
|
|
|
linked_list_add(&clients, (linked_item_t *) client);
|
|
|
|
break;
|
2009-09-29 20:10:24 +00:00
|
|
|
case DAEMON_EVENT_CONNECTION_CLOSED:
|
2010-06-13 08:26:47 +00:00
|
|
|
sdp_unregister_services_for_connection(connection);
|
2011-05-05 22:18:39 +00:00
|
|
|
rfcomm_close_connection(connection);
|
2010-01-25 18:29:06 +00:00
|
|
|
l2cap_close_connection(connection);
|
2011-01-09 14:48:03 +00:00
|
|
|
client = client_for_connection(connection);
|
|
|
|
if (!client) break;
|
|
|
|
linked_list_remove(&clients, (linked_item_t *) client);
|
|
|
|
free(client);
|
2011-01-10 21:11:58 +00:00
|
|
|
// update discoverable mode
|
|
|
|
hci_discoverable_control(clients_require_discoverable());
|
|
|
|
// start power off, if last active client
|
|
|
|
if (!clients_require_power_on()){
|
2011-01-09 14:48:03 +00:00
|
|
|
start_power_off_timer();
|
2011-01-08 22:01:06 +00:00
|
|
|
}
|
2009-08-09 20:18:34 +00:00
|
|
|
break;
|
|
|
|
case DAEMON_NR_CONNECTIONS_CHANGED:
|
2011-05-05 18:45:28 +00:00
|
|
|
log_dbg("Nr Connections changed, new %u\n",data[1]);
|
2011-01-08 22:01:06 +00:00
|
|
|
break;
|
2009-08-09 20:18:34 +00:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2009-08-09 17:17:00 +00:00
|
|
|
break;
|
2009-07-22 20:21:22 +00:00
|
|
|
}
|
2010-07-31 09:07:48 +00:00
|
|
|
if (err) {
|
2011-05-05 18:45:28 +00:00
|
|
|
log_dbg("Daemon Handler: err %d\n", err);
|
2010-07-31 09:07:48 +00:00
|
|
|
}
|
2010-07-29 19:35:34 +00:00
|
|
|
return err;
|
2009-07-22 20:21:22 +00:00
|
|
|
}
|
|
|
|
|
2011-02-02 21:37:40 +00:00
|
|
|
// local cache used to manage UI status
|
|
|
|
static HCI_STATE hci_state = HCI_STATE_OFF;
|
|
|
|
static int num_connections = 0;
|
2011-06-06 14:12:38 +00:00
|
|
|
static void update_ui_status(void){
|
2011-02-02 21:37:40 +00:00
|
|
|
if (hci_state != HCI_STATE_WORKING) {
|
|
|
|
bluetooth_status_handler(BLUETOOTH_OFF);
|
|
|
|
} else {
|
|
|
|
if (num_connections) {
|
|
|
|
bluetooth_status_handler(BLUETOOTH_ACTIVE);
|
|
|
|
} else {
|
|
|
|
bluetooth_status_handler(BLUETOOTH_ON);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-07-18 17:14:11 +00:00
|
|
|
static void deamon_status_event_handler(uint8_t *packet, uint16_t size){
|
|
|
|
|
2009-09-30 20:37:28 +00:00
|
|
|
|
|
|
|
uint8_t update_status = 0;
|
|
|
|
|
2009-08-15 21:31:23 +00:00
|
|
|
// handle state event
|
2009-09-30 20:37:28 +00:00
|
|
|
switch (packet[0]) {
|
|
|
|
case BTSTACK_EVENT_STATE:
|
|
|
|
hci_state = packet[2];
|
2011-05-05 18:45:28 +00:00
|
|
|
log_dbg("New state: %u\n", hci_state);
|
2009-09-30 20:37:28 +00:00
|
|
|
update_status = 1;
|
|
|
|
break;
|
|
|
|
case BTSTACK_EVENT_NR_CONNECTIONS_CHANGED:
|
|
|
|
num_connections = packet[2];
|
2011-05-05 18:45:28 +00:00
|
|
|
log_dbg("New nr connections: %u\n", num_connections);
|
2009-09-30 20:37:28 +00:00
|
|
|
update_status = 1;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2009-08-15 21:31:23 +00:00
|
|
|
}
|
2009-09-30 20:37:28 +00:00
|
|
|
|
|
|
|
// choose full bluetooth state
|
|
|
|
if (update_status) {
|
2011-02-02 21:37:40 +00:00
|
|
|
update_ui_status();
|
2009-08-15 21:31:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-07-19 17:50:59 +00:00
|
|
|
static void daemon_packet_handler(void * connection, uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){
|
2010-07-31 09:07:48 +00:00
|
|
|
if (packet_type == HCI_EVENT_PACKET) {
|
|
|
|
deamon_status_event_handler(packet, size);
|
|
|
|
if (packet[0] == HCI_EVENT_NUMBER_OF_COMPLETED_PACKETS) {
|
|
|
|
socket_connection_retry_parked();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (connection) {
|
|
|
|
socket_connection_send_packet(connection, packet_type, channel, packet, size);
|
|
|
|
} else {
|
|
|
|
socket_connection_send_packet_all(packet_type, channel, packet, size);
|
2010-07-18 17:05:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-02-02 21:37:40 +00:00
|
|
|
|
2011-01-08 22:45:27 +00:00
|
|
|
static void power_notification_callback(POWER_NOTIFICATION_t notification){
|
|
|
|
switch (notification) {
|
|
|
|
case POWER_WILL_SLEEP:
|
|
|
|
// let's sleep
|
|
|
|
power_management_sleep = 1;
|
|
|
|
hci_power_control(HCI_POWER_SLEEP);
|
|
|
|
break;
|
|
|
|
case POWER_WILL_WAKE_UP:
|
|
|
|
// assume that all clients use Bluetooth -> if connection, start Bluetooth
|
|
|
|
power_management_sleep = 0;
|
2011-01-10 21:11:58 +00:00
|
|
|
if (clients_require_power_on()) {
|
2011-01-08 22:45:27 +00:00
|
|
|
hci_power_control(HCI_POWER_ON);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-08-15 21:31:23 +00:00
|
|
|
static void daemon_sigint_handler(int param){
|
2011-06-03 21:34:00 +00:00
|
|
|
|
2011-04-30 20:11:20 +00:00
|
|
|
#ifdef USE_BLUETOOL
|
2011-06-03 21:34:00 +00:00
|
|
|
// notify daemons
|
2011-04-30 20:11:20 +00:00
|
|
|
notify_post("ch.ringwald.btstack.stopped");
|
|
|
|
#endif
|
|
|
|
|
2011-05-05 18:45:28 +00:00
|
|
|
log_dbg(" <= SIGINT received, shutting down..\n");
|
2011-06-03 21:34:00 +00:00
|
|
|
|
2009-08-15 19:52:46 +00:00
|
|
|
hci_power_control( HCI_POWER_OFF);
|
2010-08-30 20:10:51 +00:00
|
|
|
hci_close();
|
2011-06-03 21:34:00 +00:00
|
|
|
|
2011-05-05 18:45:28 +00:00
|
|
|
log_dbg("Good bye, see you.\n");
|
2011-06-03 21:34:00 +00:00
|
|
|
|
2009-08-15 19:52:46 +00:00
|
|
|
exit(0);
|
|
|
|
}
|
|
|
|
|
2010-02-28 22:02:28 +00:00
|
|
|
static void usage(const char * name) {
|
2011-05-05 18:45:28 +00:00
|
|
|
log_dbg("%s, BTstack background daemon\n", name);
|
|
|
|
log_dbg("usage: %s [-h|--help] [--tcp]\n", name);
|
|
|
|
log_dbg(" -h|--help display this usage\n");
|
|
|
|
log_dbg(" --tcp use TCP server socket instead of local unix socket\n");
|
2010-02-28 22:02:28 +00:00
|
|
|
}
|
|
|
|
|
2011-06-03 21:34:00 +00:00
|
|
|
static void * run_loop_thread(void *context){
|
|
|
|
run_loop_execute();
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2010-02-28 22:02:28 +00:00
|
|
|
int main (int argc, char * const * argv){
|
|
|
|
|
|
|
|
static int tcp_flag = 0;
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
static struct option long_options[] = {
|
|
|
|
{ "tcp", no_argument, &tcp_flag, 1 },
|
|
|
|
{ "help", no_argument, 0, 0 },
|
|
|
|
{ 0,0,0,0 } // This is a filler for -1
|
|
|
|
};
|
|
|
|
|
|
|
|
int c;
|
|
|
|
int option_index = -1;
|
|
|
|
|
|
|
|
c = getopt_long(argc, argv, "h", long_options, &option_index);
|
|
|
|
|
|
|
|
if (c == -1) break; // no more option
|
|
|
|
|
|
|
|
// treat long parameter first
|
|
|
|
if (option_index == -1) {
|
|
|
|
switch (c) {
|
|
|
|
case '?':
|
|
|
|
case 'h':
|
|
|
|
usage(argv[0]);
|
|
|
|
return 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
switch (option_index) {
|
|
|
|
case 1:
|
|
|
|
usage(argv[0]);
|
|
|
|
return 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-07-01 21:55:08 +00:00
|
|
|
|
2010-12-23 20:21:21 +00:00
|
|
|
// make stderr/stdout unbuffered
|
|
|
|
setbuf(stderr, NULL);
|
|
|
|
setbuf(stdout, NULL);
|
2011-05-05 18:45:28 +00:00
|
|
|
log_dbg("BTdaemon started - stdout\n");
|
|
|
|
log_err("BTdaemon started - stderr\n");
|
2010-12-23 20:21:21 +00:00
|
|
|
|
|
|
|
// handle CTRL-c
|
|
|
|
signal(SIGINT, daemon_sigint_handler);
|
|
|
|
// handle SIGTERM - suggested for launchd
|
|
|
|
signal(SIGTERM, daemon_sigint_handler);
|
|
|
|
// handle SIGPIPE
|
|
|
|
struct sigaction act;
|
|
|
|
act.sa_handler = SIG_IGN;
|
|
|
|
sigemptyset (&act.sa_mask);
|
|
|
|
act.sa_flags = 0;
|
|
|
|
sigaction (SIGPIPE, &act, NULL);
|
|
|
|
|
|
|
|
|
2009-07-01 21:55:08 +00:00
|
|
|
bt_control_t * control = NULL;
|
2010-08-30 20:10:51 +00:00
|
|
|
|
2009-07-11 10:37:48 +00:00
|
|
|
#ifdef HAVE_TRANSPORT_H4
|
|
|
|
transport = hci_transport_h4_instance();
|
2011-06-05 09:55:32 +00:00
|
|
|
config.device_name = UART_DEVICE;
|
|
|
|
config.baudrate_init = UART_SPEED;
|
|
|
|
config.baudrate_main = 0;
|
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;
|
2011-02-02 21:37:40 +00:00
|
|
|
platform_iphone_register_window_manager_restart(update_ui_status);
|
2009-08-15 21:31:23 +00:00
|
|
|
#endif
|
|
|
|
|
2010-08-30 20:10:51 +00:00
|
|
|
#ifdef REMOTE_DEVICE_DB
|
|
|
|
remote_device_db = &REMOTE_DEVICE_DB;
|
|
|
|
#endif
|
|
|
|
|
2011-01-07 22:21:25 +00:00
|
|
|
run_loop_init(RUN_LOOP_POSIX);
|
2009-07-14 18:29:29 +00:00
|
|
|
|
2011-01-08 22:45:27 +00:00
|
|
|
// init power management notifications
|
2011-01-12 19:36:27 +00:00
|
|
|
if (control && control->register_for_power_notifications){
|
2011-01-08 22:45:27 +00:00
|
|
|
control->register_for_power_notifications(power_notification_callback);
|
|
|
|
}
|
|
|
|
|
2009-07-16 20:12:42 +00:00
|
|
|
// use logger: format HCI_DUMP_PACKETLOGGER, HCI_DUMP_BLUEZ or HCI_DUMP_STDOUT
|
2009-10-11 23:09:31 +00:00
|
|
|
hci_dump_open("/tmp/hci_dump.pklg", HCI_DUMP_PACKETLOGGER);
|
2010-12-05 22:22:46 +00:00
|
|
|
hci_dump_set_max_packets(1000);
|
2011-01-08 22:45:27 +00:00
|
|
|
|
2009-07-01 21:55:08 +00:00
|
|
|
// init HCI
|
2010-08-30 20:10:51 +00:00
|
|
|
hci_init(transport, &config, control, remote_device_db);
|
2009-07-01 21:55:08 +00:00
|
|
|
|
|
|
|
// init L2CAP
|
|
|
|
l2cap_init();
|
2010-07-18 17:05:56 +00:00
|
|
|
l2cap_register_packet_handler(daemon_packet_handler);
|
2009-08-09 20:18:34 +00:00
|
|
|
timeout.process = daemon_no_connections_timeout;
|
2010-03-02 20:58:27 +00:00
|
|
|
|
2011-04-30 20:30:05 +00:00
|
|
|
#ifdef HAVE_RFCOMM
|
2011-05-05 18:45:28 +00:00
|
|
|
log_dbg("config.h: HAVE_RFCOMM\n");
|
2011-04-30 20:30:05 +00:00
|
|
|
rfcomm_init();
|
|
|
|
rfcomm_register_packet_handler(daemon_packet_handler);
|
|
|
|
#endif
|
|
|
|
|
2010-10-02 15:40:31 +00:00
|
|
|
#ifdef HAVE_SDP
|
2010-06-09 17:34:52 +00:00
|
|
|
sdp_init();
|
2011-05-12 19:15:43 +00:00
|
|
|
sdp_register_packet_handler(daemon_packet_handler);
|
2010-10-02 15:40:31 +00:00
|
|
|
#endif
|
|
|
|
|
2010-03-02 20:58:27 +00:00
|
|
|
#ifdef USE_LAUNCHD
|
|
|
|
socket_connection_create_launchd();
|
|
|
|
#else
|
2009-07-01 21:55:08 +00:00
|
|
|
// create server
|
2010-02-28 22:02:28 +00:00
|
|
|
if (tcp_flag) {
|
|
|
|
socket_connection_create_tcp(BTSTACK_PORT);
|
|
|
|
} else {
|
|
|
|
socket_connection_create_unix(BTSTACK_UNIX);
|
|
|
|
}
|
2010-03-02 20:58:27 +00:00
|
|
|
#endif
|
2009-07-31 21:41:15 +00:00
|
|
|
socket_connection_register_packet_callback(daemon_client_handler);
|
2011-01-08 22:45:27 +00:00
|
|
|
|
2011-04-30 20:11:20 +00:00
|
|
|
#ifdef USE_BLUETOOL
|
2011-06-03 21:34:00 +00:00
|
|
|
// notify daemons
|
2011-04-30 20:11:20 +00:00
|
|
|
notify_post("ch.ringwald.btstack.started");
|
2011-06-03 21:34:00 +00:00
|
|
|
|
|
|
|
// spawn thread to have BTstack run loop on new thread, while main thread is used to keep CFRunLoop
|
|
|
|
pthread_t run_loop;
|
|
|
|
pthread_create(&run_loop, NULL, &run_loop_thread, NULL);
|
|
|
|
|
|
|
|
// needed to receive notifications
|
|
|
|
CFRunLoopRun();
|
2011-04-30 20:11:20 +00:00
|
|
|
#endif
|
|
|
|
|
2009-07-01 21:55:08 +00:00
|
|
|
// go!
|
|
|
|
run_loop_execute();
|
|
|
|
return 0;
|
2011-01-08 22:01:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-01-09 14:48:03 +00:00
|
|
|
#pragma mark manage power off timer
|
2011-01-08 22:01:06 +00:00
|
|
|
|
2011-01-09 22:22:12 +00:00
|
|
|
#define USE_POWER_OFF_TIMER
|
|
|
|
|
2011-06-06 14:12:38 +00:00
|
|
|
static void stop_power_off_timer(void){
|
2011-01-09 22:22:12 +00:00
|
|
|
#ifdef USE_POWER_OFF_TIMER
|
2011-01-08 22:01:06 +00:00
|
|
|
if (timeout_active) {
|
|
|
|
run_loop_remove_timer(&timeout);
|
|
|
|
timeout_active = 0;
|
|
|
|
}
|
2011-01-09 22:22:12 +00:00
|
|
|
#endif
|
2011-01-08 22:01:06 +00:00
|
|
|
}
|
|
|
|
|
2011-06-06 14:12:38 +00:00
|
|
|
static void start_power_off_timer(void){
|
2011-01-09 22:22:12 +00:00
|
|
|
#ifdef USE_POWER_OFF_TIMER
|
2011-01-09 14:48:03 +00:00
|
|
|
stop_power_off_timer();
|
2011-01-08 22:01:06 +00:00
|
|
|
run_loop_set_timer(&timeout, DAEMON_NO_ACTIVE_CLIENT_TIMEOUT);
|
|
|
|
run_loop_add_timer(&timeout);
|
|
|
|
timeout_active = 1;
|
2011-01-09 22:22:12 +00:00
|
|
|
#else
|
|
|
|
hci_power_control(HCI_POWER_OFF);
|
|
|
|
#endif
|
2011-01-08 22:01:06 +00:00
|
|
|
}
|
2011-01-09 14:48:03 +00:00
|
|
|
|
|
|
|
#pragma mark manage list of clients
|
|
|
|
|
|
|
|
|
|
|
|
static client_state_t * client_for_connection(connection_t *connection) {
|
|
|
|
linked_item_t *it;
|
|
|
|
for (it = (linked_item_t *) clients; it ; it = it->next){
|
|
|
|
client_state_t * client_state = (client_state_t *) it;
|
|
|
|
if (client_state->connection == connection) {
|
|
|
|
return client_state;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2011-06-06 14:12:38 +00:00
|
|
|
static void clients_clear_power_request(void){
|
2011-04-30 20:11:20 +00:00
|
|
|
linked_item_t *it;
|
|
|
|
for (it = (linked_item_t *) clients; it ; it = it->next){
|
|
|
|
client_state_t * client_state = (client_state_t *) it;
|
|
|
|
client_state->power_mode = HCI_POWER_OFF;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-06-06 14:12:38 +00:00
|
|
|
static int clients_require_power_on(void){
|
2011-04-30 20:11:20 +00:00
|
|
|
|
|
|
|
if (global_enable) return 1;
|
|
|
|
|
2011-01-09 14:48:03 +00:00
|
|
|
linked_item_t *it;
|
|
|
|
for (it = (linked_item_t *) clients; it ; it = it->next){
|
|
|
|
client_state_t * client_state = (client_state_t *) it;
|
|
|
|
if (client_state->power_mode == HCI_POWER_ON) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
2011-01-10 21:11:58 +00:00
|
|
|
|
2011-06-06 14:12:38 +00:00
|
|
|
static int clients_require_discoverable(void){
|
2011-01-10 21:11:58 +00:00
|
|
|
linked_item_t *it;
|
|
|
|
for (it = (linked_item_t *) clients; it ; it = it->next){
|
|
|
|
client_state_t * client_state = (client_state_t *) it;
|
|
|
|
if (client_state->discoverable) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|