1893dbb94SMatthias Ringwald /*
2893dbb94SMatthias Ringwald * Copyright (C) 2014 BlueKitchen GmbH
3893dbb94SMatthias Ringwald *
4893dbb94SMatthias Ringwald * Redistribution and use in source and binary forms, with or without
5893dbb94SMatthias Ringwald * modification, are permitted provided that the following conditions
6893dbb94SMatthias Ringwald * are met:
7893dbb94SMatthias Ringwald *
8893dbb94SMatthias Ringwald * 1. Redistributions of source code must retain the above copyright
9893dbb94SMatthias Ringwald * notice, this list of conditions and the following disclaimer.
10893dbb94SMatthias Ringwald * 2. Redistributions in binary form must reproduce the above copyright
11893dbb94SMatthias Ringwald * notice, this list of conditions and the following disclaimer in the
12893dbb94SMatthias Ringwald * documentation and/or other materials provided with the distribution.
13893dbb94SMatthias Ringwald * 3. Neither the name of the copyright holders nor the names of
14893dbb94SMatthias Ringwald * contributors may be used to endorse or promote products derived
15893dbb94SMatthias Ringwald * from this software without specific prior written permission.
16893dbb94SMatthias Ringwald * 4. Any redistribution, use, or modification is done solely for
17893dbb94SMatthias Ringwald * personal benefit and not for any commercial purpose or for
18893dbb94SMatthias Ringwald * monetary gain.
19893dbb94SMatthias Ringwald *
20893dbb94SMatthias Ringwald * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS
21893dbb94SMatthias Ringwald * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
22893dbb94SMatthias Ringwald * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
23893dbb94SMatthias Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL BLUEKITCHEN
24893dbb94SMatthias Ringwald * GMBH OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
25893dbb94SMatthias Ringwald * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
26893dbb94SMatthias Ringwald * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
27893dbb94SMatthias Ringwald * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
28893dbb94SMatthias Ringwald * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
29893dbb94SMatthias Ringwald * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
30893dbb94SMatthias Ringwald * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
31893dbb94SMatthias Ringwald * SUCH DAMAGE.
32893dbb94SMatthias Ringwald *
33893dbb94SMatthias Ringwald * Please inquire about commercial licensing options at
34893dbb94SMatthias Ringwald * [email protected]
35893dbb94SMatthias Ringwald *
36893dbb94SMatthias Ringwald */
37893dbb94SMatthias Ringwald
38bc6a318fSMatthias Ringwald #define BTSTACK_FILE__ "main.c"
39893dbb94SMatthias Ringwald
40893dbb94SMatthias Ringwald // *****************************************************************************
41893dbb94SMatthias Ringwald //
42893dbb94SMatthias Ringwald // minimal setup for HCI code
43893dbb94SMatthias Ringwald //
44893dbb94SMatthias Ringwald // *****************************************************************************
45893dbb94SMatthias Ringwald
46893dbb94SMatthias Ringwald #include <stdint.h>
47893dbb94SMatthias Ringwald #include <stdio.h>
48893dbb94SMatthias Ringwald #include <stdlib.h>
49893dbb94SMatthias Ringwald #include <string.h>
50893dbb94SMatthias Ringwald #include <signal.h>
51893dbb94SMatthias Ringwald
52893dbb94SMatthias Ringwald #include "btstack_config.h"
53893dbb94SMatthias Ringwald
54893dbb94SMatthias Ringwald #include "bluetooth_company_id.h"
55893dbb94SMatthias Ringwald #include "ble/le_device_db_tlv.h"
56893dbb94SMatthias Ringwald #include "btstack_chipset_bcm.h"
57893dbb94SMatthias Ringwald #include "btstack_chipset_bcm_download_firmware.h"
58893dbb94SMatthias Ringwald #include "btstack_debug.h"
59893dbb94SMatthias Ringwald #include "btstack_event.h"
60893dbb94SMatthias Ringwald #include "btstack_memory.h"
61893dbb94SMatthias Ringwald #include "btstack_run_loop.h"
62893dbb94SMatthias Ringwald #include "btstack_run_loop_posix.h"
63893dbb94SMatthias Ringwald #include "btstack_stdin.h"
64893dbb94SMatthias Ringwald #include "btstack_uart.h"
65893dbb94SMatthias Ringwald #include "btstack_tlv_posix.h"
66893dbb94SMatthias Ringwald #include "classic/btstack_link_key_db_tlv.h"
67893dbb94SMatthias Ringwald #include "hci.h"
68893dbb94SMatthias Ringwald #include "hci_dump.h"
69893dbb94SMatthias Ringwald #include "hci_transport_h4.h"
70893dbb94SMatthias Ringwald #include "hci_dump_posix_fs.h"
71893dbb94SMatthias Ringwald #include "hci_dump_posix_stdout.h"
72*c8249f33SMatthias Ringwald #include "btstack_audio.h"
73893dbb94SMatthias Ringwald
74893dbb94SMatthias Ringwald
75893dbb94SMatthias Ringwald int btstack_main(int argc, const char * argv[]);
76893dbb94SMatthias Ringwald
77893dbb94SMatthias Ringwald #define TLV_DB_PATH_PREFIX "/tmp/btstack_"
78893dbb94SMatthias Ringwald #define TLV_DB_PATH_POSTFIX ".tlv"
79893dbb94SMatthias Ringwald static char tlv_db_path[100];
80893dbb94SMatthias Ringwald static const btstack_tlv_t * tlv_impl;
81893dbb94SMatthias Ringwald static btstack_tlv_posix_t tlv_context;
82893dbb94SMatthias Ringwald
8336e2ff5eSMatthias Ringwald static const uint32_t baudrate_firmware_download = 921600;
8436e2ff5eSMatthias Ringwald
85893dbb94SMatthias Ringwald static hci_transport_config_uart_t transport_config = {
86893dbb94SMatthias Ringwald HCI_TRANSPORT_CONFIG_UART,
87893dbb94SMatthias Ringwald 115200,
88893dbb94SMatthias Ringwald 921600, // main baudrate
89893dbb94SMatthias Ringwald 1, // flow control
90893dbb94SMatthias Ringwald NULL,
91893dbb94SMatthias Ringwald BTSTACK_UART_PARITY_OFF, // parity
92893dbb94SMatthias Ringwald };
93893dbb94SMatthias Ringwald static btstack_uart_config_t uart_config;
94893dbb94SMatthias Ringwald
95893dbb94SMatthias Ringwald static int main_argc;
96893dbb94SMatthias Ringwald static const char ** main_argv;
97893dbb94SMatthias Ringwald
98893dbb94SMatthias Ringwald static btstack_packet_callback_registration_t hci_event_callback_registration;
99893dbb94SMatthias Ringwald
sigint_handler(int param)100893dbb94SMatthias Ringwald static void sigint_handler(int param){
101893dbb94SMatthias Ringwald UNUSED(param);
102893dbb94SMatthias Ringwald
103893dbb94SMatthias Ringwald printf("CTRL-C - SIGINT received, shutting down..\n");
104893dbb94SMatthias Ringwald log_info("sigint_handler: shutting down");
105893dbb94SMatthias Ringwald
106893dbb94SMatthias Ringwald // reset anyway
107893dbb94SMatthias Ringwald btstack_stdin_reset();
108893dbb94SMatthias Ringwald
109893dbb94SMatthias Ringwald // power down
110893dbb94SMatthias Ringwald hci_power_control(HCI_POWER_OFF);
111893dbb94SMatthias Ringwald hci_close();
112893dbb94SMatthias Ringwald log_info("Good bye, see you.\n");
113893dbb94SMatthias Ringwald exit(0);
114893dbb94SMatthias Ringwald }
115893dbb94SMatthias Ringwald
116893dbb94SMatthias Ringwald static int led_state = 0;
hal_led_toggle(void)117893dbb94SMatthias Ringwald void hal_led_toggle(void){
118893dbb94SMatthias Ringwald led_state = 1 - led_state;
119893dbb94SMatthias Ringwald printf("LED State %u\n", led_state);
120893dbb94SMatthias Ringwald }
121893dbb94SMatthias Ringwald
packet_handler(uint8_t packet_type,uint16_t channel,uint8_t * packet,uint16_t size)122893dbb94SMatthias Ringwald static void packet_handler (uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){
123893dbb94SMatthias Ringwald bd_addr_t addr;
124893dbb94SMatthias Ringwald if (packet_type != HCI_EVENT_PACKET) return;
125893dbb94SMatthias Ringwald switch (hci_event_packet_get_type(packet)){
126893dbb94SMatthias Ringwald case BTSTACK_EVENT_STATE:
127893dbb94SMatthias Ringwald if (btstack_event_state_get_state(packet) != HCI_STATE_WORKING) break;
128893dbb94SMatthias Ringwald gap_local_bd_addr(addr);
129893dbb94SMatthias Ringwald printf("BTstack up and running at %s\n", bd_addr_to_str(addr));
130893dbb94SMatthias Ringwald // setup TLV
131893dbb94SMatthias Ringwald btstack_strcpy(tlv_db_path, sizeof(tlv_db_path), TLV_DB_PATH_PREFIX);
132893dbb94SMatthias Ringwald btstack_strcat(tlv_db_path, sizeof(tlv_db_path), bd_addr_to_str(addr));
133893dbb94SMatthias Ringwald btstack_strcat(tlv_db_path, sizeof(tlv_db_path), TLV_DB_PATH_POSTFIX);
134893dbb94SMatthias Ringwald tlv_impl = btstack_tlv_posix_init_instance(&tlv_context, tlv_db_path);
135893dbb94SMatthias Ringwald btstack_tlv_set_instance(tlv_impl, &tlv_context);
136893dbb94SMatthias Ringwald #ifdef ENABLE_CLASSIC
137893dbb94SMatthias Ringwald hci_set_link_key_db(btstack_link_key_db_tlv_get_instance(tlv_impl, &tlv_context));
138893dbb94SMatthias Ringwald #endif
139893dbb94SMatthias Ringwald #ifdef ENABLE_BLE
140893dbb94SMatthias Ringwald le_device_db_tlv_configure(tlv_impl, &tlv_context);
141893dbb94SMatthias Ringwald #endif
142893dbb94SMatthias Ringwald break;
143893dbb94SMatthias Ringwald default:
144893dbb94SMatthias Ringwald break;
145893dbb94SMatthias Ringwald }
146893dbb94SMatthias Ringwald }
147893dbb94SMatthias Ringwald
148893dbb94SMatthias Ringwald static void phase2(int status);
main(int argc,const char * argv[])149893dbb94SMatthias Ringwald int main(int argc, const char * argv[]){
150893dbb94SMatthias Ringwald
151893dbb94SMatthias Ringwald /// GET STARTED with BTstack ///
152893dbb94SMatthias Ringwald btstack_memory_init();
153893dbb94SMatthias Ringwald
154893dbb94SMatthias Ringwald #if 1
155893dbb94SMatthias Ringwald // log into file using HCI_DUMP_PACKETLOGGER format
156893dbb94SMatthias Ringwald const char * pklg_path = "/tmp/hci_dump.pklg";
157893dbb94SMatthias Ringwald hci_dump_posix_fs_open(pklg_path, HCI_DUMP_PACKETLOGGER);
158893dbb94SMatthias Ringwald const hci_dump_t * hci_dump_impl = hci_dump_posix_fs_get_instance();
159893dbb94SMatthias Ringwald printf("Packet Log: %s\n", pklg_path);
160893dbb94SMatthias Ringwald #else
161893dbb94SMatthias Ringwald // log to stdout for debugging/development
162893dbb94SMatthias Ringwald const hci_dump_t * hci_dump_impl = hci_dump_posix_stdout_get_instance();
163893dbb94SMatthias Ringwald #endif
164893dbb94SMatthias Ringwald hci_dump_init(hci_dump_impl);
165893dbb94SMatthias Ringwald
166893dbb94SMatthias Ringwald // setup run loop
167893dbb94SMatthias Ringwald btstack_run_loop_init(btstack_run_loop_posix_get_instance());
168893dbb94SMatthias Ringwald
169893dbb94SMatthias Ringwald // pick serial port and configure uart driver
170893dbb94SMatthias Ringwald transport_config.device_name = "/dev/tty.usbserial-FT1XBGIM"; // murata m.2 adapter
1717ae42287SMatthias Ringwald printf("tty: %s", transport_config.device_name);
172893dbb94SMatthias Ringwald // get BCM chipset driver
173893dbb94SMatthias Ringwald const btstack_chipset_t * chipset = btstack_chipset_bcm_instance();
174893dbb94SMatthias Ringwald chipset->init(&transport_config);
175893dbb94SMatthias Ringwald
176893dbb94SMatthias Ringwald // setup UART driver
177893dbb94SMatthias Ringwald const btstack_uart_t * uart_driver = (const btstack_uart_t *) btstack_uart_posix_instance();
178893dbb94SMatthias Ringwald
179893dbb94SMatthias Ringwald // extract UART config from transport config
18036e2ff5eSMatthias Ringwald uart_config.baudrate = baudrate_firmware_download;
181893dbb94SMatthias Ringwald uart_config.flowcontrol = transport_config.flowcontrol;
182893dbb94SMatthias Ringwald uart_config.device_name = transport_config.device_name;
183893dbb94SMatthias Ringwald uart_driver->init(&uart_config);
184893dbb94SMatthias Ringwald
185893dbb94SMatthias Ringwald
186893dbb94SMatthias Ringwald // setup HCI (to be able to use bcm chipset driver)
187893dbb94SMatthias Ringwald // init HCI
188893dbb94SMatthias Ringwald const hci_transport_t * transport = hci_transport_h4_instance(uart_driver);
189893dbb94SMatthias Ringwald hci_init(transport, (void*) &transport_config);
190893dbb94SMatthias Ringwald hci_set_chipset(btstack_chipset_bcm_instance());
191893dbb94SMatthias Ringwald
192893dbb94SMatthias Ringwald // inform about BTstack state
193893dbb94SMatthias Ringwald hci_event_callback_registration.callback = &packet_handler;
194893dbb94SMatthias Ringwald hci_add_event_handler(&hci_event_callback_registration);
195893dbb94SMatthias Ringwald
196893dbb94SMatthias Ringwald // handle CTRL-c
197893dbb94SMatthias Ringwald signal(SIGINT, sigint_handler);
198893dbb94SMatthias Ringwald
199893dbb94SMatthias Ringwald main_argc = argc;
200893dbb94SMatthias Ringwald main_argv = argv;
201893dbb94SMatthias Ringwald
202893dbb94SMatthias Ringwald // phase #1 download firmware
203893dbb94SMatthias Ringwald printf("Phase 1: Download firmware\n");
204893dbb94SMatthias Ringwald
205893dbb94SMatthias Ringwald // phase #2 start main app
206893dbb94SMatthias Ringwald btstack_chipset_bcm_download_firmware_with_uart(uart_driver, 0, &phase2);
207893dbb94SMatthias Ringwald
208893dbb94SMatthias Ringwald // go
209893dbb94SMatthias Ringwald btstack_run_loop_execute();
210893dbb94SMatthias Ringwald return 0;
211893dbb94SMatthias Ringwald }
212893dbb94SMatthias Ringwald
phase2(int status)213893dbb94SMatthias Ringwald static void phase2(int status){
214893dbb94SMatthias Ringwald
215893dbb94SMatthias Ringwald if (status){
216893dbb94SMatthias Ringwald printf("Download firmware failed\n");
217893dbb94SMatthias Ringwald return;
218893dbb94SMatthias Ringwald }
219893dbb94SMatthias Ringwald
220893dbb94SMatthias Ringwald printf("Phase 2: Main app\n");
221893dbb94SMatthias Ringwald
222*c8249f33SMatthias Ringwald #ifdef HAVE_PORTAUDIO
223*c8249f33SMatthias Ringwald btstack_audio_sink_set_instance(btstack_audio_portaudio_sink_get_instance());
224*c8249f33SMatthias Ringwald btstack_audio_source_set_instance(btstack_audio_portaudio_source_get_instance());
225*c8249f33SMatthias Ringwald #endif
226*c8249f33SMatthias Ringwald
227893dbb94SMatthias Ringwald // setup app
228893dbb94SMatthias Ringwald btstack_main(main_argc, main_argv);
229893dbb94SMatthias Ringwald }
230893dbb94SMatthias Ringwald
231