btstack/example/libusb/ble_peripheral.c

462 lines
14 KiB
C
Raw Normal View History

2013-10-17 12:42:18 +00:00
/*
2013-11-09 23:20:11 +00:00
* Copyright (C) 2011-2013 by Matthias Ringwald
2013-10-17 12:42:18 +00:00
*
* 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.
* 4. This software may not be used in a commercial product
* without an explicit license granted by the copyright holder.
*
* 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.
*
*/
//*****************************************************************************
//
// BLE Peripheral Demo
2013-10-17 12:42:18 +00:00
//
//*****************************************************************************
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
2014-01-24 10:51:28 +00:00
#include <sys/uio.h>
#include <unistd.h>
#include <termios.h>
2013-10-17 12:42:18 +00:00
#include "btstack-config.h"
2013-10-17 12:42:18 +00:00
#include <btstack/run_loop.h>
#include "debug.h"
#include "btstack_memory.h"
#include "hci.h"
#include "hci_dump.h"
#include "l2cap.h"
2014-01-03 19:19:37 +00:00
#include "sm.h"
2013-10-17 12:42:18 +00:00
#include "att.h"
2014-01-05 21:39:26 +00:00
#include "att_server.h"
2014-01-05 20:13:37 +00:00
#include "gap_le.h"
2014-01-03 19:19:37 +00:00
#include "central_device_db.h"
2013-10-17 12:42:18 +00:00
#define HEARTBEAT_PERIOD_MS 1000
2013-10-17 12:42:18 +00:00
// test profile
#include "profile.h"
2014-01-24 21:47:59 +00:00
typedef enum {
DISABLE_ADVERTISEMENTS = 1 << 0,
SET_ADVERTISEMENT_PARAMS = 1 << 1,
SET_ADVERTISEMENT_DATA = 1 << 2,
SET_SCAN_RESPONSE_DATA = 1 << 3,
ENABLE_ADVERTISEMENTS = 1 << 4,
} todo_t;
static todo_t todos = 0;
///------
static int advertisements_enabled = 0;
2014-01-24 15:43:52 +00:00
static int gap_discoverable = 1;
static int gap_connectable = 1;
static int gap_bondable = 1;
static timer_source_t heartbeat;
static uint8_t counter = 0;
static int update_client = 0;
static int client_configuration = 0;
static void app_run();
2014-01-24 10:51:28 +00:00
static void show_usage();
2014-01-24 21:47:59 +00:00
static void update_advertisements();
// some test data
static uint8_t adv_data_0[] = { 2, 01, 05, 03, 02, 0xf0, 0xff };
// AD Manufacturer Specific Data - Ericsson, 1, 2, 3, 4
static uint8_t adv_data_1[] = { 7, 0xff, 0x00, 0x00, 1, 2, 3, 4 };
// AD Local Name - 'BTstack'
static uint8_t adv_data_2[] = { 8, 0x09, 'B', 'T', 's', 't', 'a', 'c', 'k' };
// AD Flags - 2 - General Discoverable mode
static uint8_t adv_data_3[] = { 2, 01, 02 };
// AD Service Data - 0x1812 HID over LE
static uint8_t adv_data_4[] = { 3, 0x16, 0x12, 0x18 };
// AD Service Solicitation - 0x1812 HID over LE
static uint8_t adv_data_5[] = { 3, 0x14, 0x12, 0x18 };
// AD Services
static uint8_t adv_data_6[] = { 3, 0x03, 0x12, 0x18 };
// AD Slave Preferred Connection Interval Range - no min, no max
static uint8_t adv_data_7[] = { 5, 0x12, 0xff, 0xff, 0xff, 0xff };
// AD Tx Power Level - +4 dBm
static uint8_t adv_data_8[] = { 2, 0x0a, 4 };
static uint8_t adv_data_len;
static uint8_t adv_data[32];
typedef struct {
uint16_t len;
uint8_t * data;
} advertisement_t;
#define ADV(a) { sizeof(a), &a[0]}
static advertisement_t advertisements[] = {
ADV(adv_data_0),
ADV(adv_data_1),
ADV(adv_data_2),
ADV(adv_data_3),
ADV(adv_data_4),
ADV(adv_data_5),
ADV(adv_data_6),
ADV(adv_data_7),
ADV(adv_data_8),
};
static int advertisement_index = 0;
static void heartbeat_handler(struct timer *ts){
// restart timer
run_loop_set_timer(ts, HEARTBEAT_PERIOD_MS);
run_loop_add_timer(ts);
counter++;
update_client = 1;
app_run();
}
static void app_run(){
if (!update_client) return;
if (!att_server_can_send()) return;
int result = -1;
switch (client_configuration){
case 0x01:
printf("Notify value %u\n", counter);
result = att_server_notify(0x0f, &counter, 1);
break;
case 0x02:
printf("Indicate value %u\n", counter);
result = att_server_indicate(0x0f, &counter, 1);
break;
default:
return;
}
if (result){
printf("Error 0x%02x\n", result);
return;
}
update_client = 0;
}
2014-01-05 21:39:26 +00:00
// write requests
static int att_write_callback(uint16_t handle, uint16_t transaction_mode, uint16_t offset, uint8_t *buffer, uint16_t buffer_size, signature_t * signature){
2014-01-05 21:39:26 +00:00
printf("WRITE Callback, handle %04x\n", handle);
switch(handle){
case 0x0010:
client_configuration = buffer[0];
printf("Client Configuration set to %u\n", client_configuration);
2014-01-05 21:39:26 +00:00
break;
default:
printf("Value: ");
hexdump(buffer, buffer_size);
2014-01-05 21:39:26 +00:00
break;
}
return 1;
2014-01-05 21:39:26 +00:00
}
2014-01-24 15:43:52 +00:00
static uint8_t gap_adv_type(){
if (gap_connectable){
return 0;
}
return 0x03;
}
2014-01-24 21:47:59 +00:00
static void gap_run(){
if (!hci_can_send_packet_now(HCI_COMMAND_DATA_PACKET)) return;
if (todos & DISABLE_ADVERTISEMENTS){
todos &= ~DISABLE_ADVERTISEMENTS;
advertisements_enabled = 0;
hci_send_cmd(&hci_le_set_advertise_enable, 0);
return;
}
if (todos & SET_ADVERTISEMENT_DATA){
todos &= ~SET_ADVERTISEMENT_DATA;
hci_send_cmd(&hci_le_set_advertising_data, adv_data_len, adv_data);
return;
}
if (todos & SET_ADVERTISEMENT_PARAMS){
todos &= ~SET_ADVERTISEMENT_PARAMS;
bd_addr_t null;
hci_send_cmd(&hci_le_set_advertising_parameters,0x0800, 0x0800, gap_adv_type(), 0, 0, &null, 0x07, 0x00);
return;
}
// if (todos & SET_SCAN_RESPONSE_DATA){
// todos &= ~SET_SCAN_RESPONSE_DATA;
// hci_send_cmd(&hci_le_set_scan_response_data, adv_data_len, adv_data);
// return;
// }
if (todos & ENABLE_ADVERTISEMENTS){
todos &= ~ENABLE_ADVERTISEMENTS;
advertisements_enabled = 1;
hci_send_cmd(&hci_le_set_advertise_enable, 1);
show_usage();
return;
}
}
2014-01-24 15:43:52 +00:00
static void app_packet_handler (uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){
switch (packet_type) {
case HCI_EVENT_PACKET:
switch (packet[0]) {
case BTSTACK_EVENT_STATE:
// bt stack activated, get started
if (packet[2] == HCI_STATE_WORKING) {
printf("SM Init completed\n");
2014-01-24 21:47:59 +00:00
todos = SET_ADVERTISEMENT_PARAMS | SET_ADVERTISEMENT_DATA | SET_SCAN_RESPONSE_DATA | ENABLE_ADVERTISEMENTS;
update_advertisements();
gap_run();
}
break;
case HCI_EVENT_LE_META:
switch (packet[2]) {
case HCI_SUBEVENT_LE_CONNECTION_COMPLETE:
2014-01-05 21:39:26 +00:00
advertisements_enabled = 0;
// request connection parameter update - test parameters
// l2cap_le_request_connection_parameter_update(READ_BT_16(packet, 4), 20, 1000, 100, 100);
break;
default:
break;
}
break;
case HCI_EVENT_DISCONNECTION_COMPLETE:
2014-01-24 21:47:59 +00:00
if (!advertisements_enabled == 0 && gap_discoverable){
todos = ENABLE_ADVERTISEMENTS;
}
break;
case SM_PASSKEY_DISPLAY_NUMBER: {
// display number
sm_event_t * event = (sm_event_t *) packet;
printf("GAP Bonding: Display Passkey '%06u\n", event->passkey);
break;
}
case SM_PASSKEY_DISPLAY_CANCEL:
printf("GAP Bonding: Display cancel\n");
break;
2014-01-06 18:44:26 +00:00
case SM_AUTHORIZATION_REQUEST: {
// auto-authorize connection if requested
sm_event_t * event = (sm_event_t *) packet;
sm_authorization_grant(event->addr_type, event->address);
break;
}
case ATT_HANDLE_VALUE_INDICATION_COMPLETE:
printf("ATT_HANDLE_VALUE_INDICATION_COMPLETE status %u\n", packet[2]);
break;
default:
break;
}
}
2014-01-24 21:47:59 +00:00
gap_run();
}
2013-10-17 12:42:18 +00:00
void setup(void){
/// GET STARTED with BTstack ///
btstack_memory_init();
run_loop_init(RUN_LOOP_POSIX);
// use logger: format HCI_DUMP_PACKETLOGGER, HCI_DUMP_BLUEZ or HCI_DUMP_STDOUT
hci_dump_open("/tmp/hci_dump.pklg", HCI_DUMP_PACKETLOGGER);
// init HCI
hci_transport_t * transport = hci_transport_usb_instance();
2013-10-17 13:35:43 +00:00
hci_uart_config_t * config = NULL;
2013-10-17 12:42:18 +00:00
bt_control_t * control = NULL;
remote_device_db_t * remote_db = (remote_device_db_t *) &remote_device_db_memory;
hci_init(transport, config, control, remote_db);
// set up l2cap_le
l2cap_init();
2014-01-05 21:17:25 +00:00
// setup central device db
central_device_db_init();
// setup SM: Display only
sm_init();
sm_set_io_capabilities(IO_CAPABILITY_DISPLAY_ONLY);
sm_set_authentication_requirements( SM_AUTHREQ_BONDING | SM_AUTHREQ_MITM_PROTECTION);
2014-01-08 08:44:04 +00:00
// sm_set_request_security(1);
// sm_set_encryption_key_size_range(7,15);
2014-01-05 21:39:26 +00:00
// setup ATT server
att_server_init(profile_data, NULL, att_write_callback);
att_server_register_packet_handler(app_packet_handler);
2013-10-17 12:42:18 +00:00
}
2014-01-24 10:51:28 +00:00
void show_usage(){
printf("\n--- CLI for LE Peripheral ---\n");
2014-01-24 15:43:52 +00:00
printf("Status: discoverable %u, connectable %u, bondable %u, advertisements enabled %u \n", gap_discoverable, gap_connectable, gap_bondable, advertisements_enabled);
printf("---\n");
printf("b - bondable off\n");
printf("B - bondable on\n");
printf("c - connectable off\n");
printf("C - connectable on\n");
printf("d - discoverable off\n");
printf("D - discoverable on\n");
2014-01-24 21:47:59 +00:00
printf("---\n");
printf("1 - AD Manufacturer Specific Data\n");
printf("2 - AD Local Name\n");
printf("3 - AD Flags\n");
printf("4 - AD Service Data\n");
printf("5 - AD Service Solicitation\n");
printf("6 - AD Services\n");
printf("7 - AD Slave Preferred Connection Interval Range\n");
printf("8 - AD Tx Power Level\n");
printf("---\n");
2014-01-24 10:51:28 +00:00
printf("Ctrl-c - exit\n");
printf("---\n");
}
2014-01-24 15:43:52 +00:00
void update_advertisements(){
2014-01-24 21:47:59 +00:00
memset(adv_data, 0, 32);
memcpy(adv_data, advertisements[advertisement_index].data, advertisements[advertisement_index].len);
2014-01-24 15:43:52 +00:00
if (!gap_discoverable){
gap_connectable = 0;
}
if (!gap_connectable){
gap_bondable = 0;
}
2014-01-24 21:47:59 +00:00
if (!advertisements_enabled) return;
2014-01-24 15:43:52 +00:00
2014-01-24 21:47:59 +00:00
// update all
todos = DISABLE_ADVERTISEMENTS | SET_ADVERTISEMENT_PARAMS | SET_ADVERTISEMENT_DATA | SET_SCAN_RESPONSE_DATA | ENABLE_ADVERTISEMENTS;
gap_run();
2014-01-24 15:43:52 +00:00
}
2014-01-24 10:51:28 +00:00
int stdin_process(struct data_source *ds){
char buffer;
read(ds->fd, &buffer, 1);
switch (buffer){
2014-01-24 15:43:52 +00:00
case 'b':
gap_bondable = 0;
sm_set_authentication_requirements(SM_AUTHREQ_NO_BONDING);
show_usage();
break;
case 'B':
gap_bondable = 1;
sm_set_authentication_requirements(SM_AUTHREQ_BONDING);
show_usage();
break;
case 'c':
gap_connectable = 0;
update_advertisements();
break;
case 'C':
gap_connectable = 1;
update_advertisements();
break;
case 'd':
gap_discoverable = 0;
update_advertisements();
break;
case 'D':
gap_discoverable = 1;
update_advertisements();
break;
2014-01-24 21:47:59 +00:00
case '1':
case '2':
case '3':
case '4':
case '5':
case '6':
case '7':
case '8':
advertisement_index = buffer - '0';
update_advertisements();
break;
2014-01-24 10:51:28 +00:00
default:
show_usage();
break;
}
return 0;
}
static data_source_t stdin_source;
void setup_cli(){
struct termios term = {0};
if (tcgetattr(0, &term) < 0)
perror("tcsetattr()");
term.c_lflag &= ~ICANON;
term.c_lflag &= ~ECHO;
term.c_cc[VMIN] = 1;
term.c_cc[VTIME] = 0;
if (tcsetattr(0, TCSANOW, &term) < 0)
perror("tcsetattr ICANON");
stdin_source.fd = 0; // stdin
stdin_source.process = &stdin_process;
run_loop_add_data_source(&stdin_source);
}
2013-10-17 12:42:18 +00:00
int main(void)
{
2014-01-24 21:47:59 +00:00
printf("BTstack LE Peripheral starting up...\n");
2013-10-17 12:42:18 +00:00
setup();
2014-01-24 10:51:28 +00:00
setup_cli();
gap_random_address_set_update_period(60000);
gap_random_address_set_mode(GAP_RANDOM_ADDRESS_RESOLVABLE);
2013-10-17 12:42:18 +00:00
// set one-shot timer
heartbeat.process = &heartbeat_handler;
run_loop_set_timer(&heartbeat, HEARTBEAT_PERIOD_MS);
run_loop_add_timer(&heartbeat);
2013-10-17 12:42:18 +00:00
// turn on!
hci_power_control(HCI_POWER_ON);
// go!
run_loop_execute();
// happy compiler!
return 0;
}