xref: /btstack/port/posix-h4-bcm/main.c (revision 36e2ff5e995e605fb249f227da24ebb53b7c1c78)
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 
38893dbb94SMatthias 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"
72893dbb94SMatthias Ringwald 
73893dbb94SMatthias Ringwald 
74893dbb94SMatthias Ringwald int btstack_main(int argc, const char * argv[]);
75893dbb94SMatthias Ringwald 
76893dbb94SMatthias Ringwald #define TLV_DB_PATH_PREFIX "/tmp/btstack_"
77893dbb94SMatthias Ringwald #define TLV_DB_PATH_POSTFIX ".tlv"
78893dbb94SMatthias Ringwald static char tlv_db_path[100];
79893dbb94SMatthias Ringwald static const btstack_tlv_t * tlv_impl;
80893dbb94SMatthias Ringwald static btstack_tlv_posix_t   tlv_context;
81893dbb94SMatthias Ringwald 
82*36e2ff5eSMatthias Ringwald static const uint32_t baudrate_firmware_download = 921600;
83*36e2ff5eSMatthias Ringwald 
84893dbb94SMatthias Ringwald static hci_transport_config_uart_t transport_config = {
85893dbb94SMatthias Ringwald     HCI_TRANSPORT_CONFIG_UART,
86893dbb94SMatthias Ringwald     115200,
87893dbb94SMatthias Ringwald     921600,  // main baudrate
88893dbb94SMatthias Ringwald     1,       // flow control
89893dbb94SMatthias Ringwald     NULL,
90893dbb94SMatthias Ringwald     BTSTACK_UART_PARITY_OFF, // parity
91893dbb94SMatthias Ringwald };
92893dbb94SMatthias Ringwald static btstack_uart_config_t uart_config;
93893dbb94SMatthias Ringwald 
94893dbb94SMatthias Ringwald static int main_argc;
95893dbb94SMatthias Ringwald static const char ** main_argv;
96893dbb94SMatthias Ringwald 
97893dbb94SMatthias Ringwald static btstack_packet_callback_registration_t hci_event_callback_registration;
98893dbb94SMatthias Ringwald 
99893dbb94SMatthias Ringwald static void sigint_handler(int param){
100893dbb94SMatthias Ringwald     UNUSED(param);
101893dbb94SMatthias Ringwald 
102893dbb94SMatthias Ringwald     printf("CTRL-C - SIGINT received, shutting down..\n");
103893dbb94SMatthias Ringwald     log_info("sigint_handler: shutting down");
104893dbb94SMatthias Ringwald 
105893dbb94SMatthias Ringwald     // reset anyway
106893dbb94SMatthias Ringwald     btstack_stdin_reset();
107893dbb94SMatthias Ringwald 
108893dbb94SMatthias Ringwald     // power down
109893dbb94SMatthias Ringwald     hci_power_control(HCI_POWER_OFF);
110893dbb94SMatthias Ringwald     hci_close();
111893dbb94SMatthias Ringwald     log_info("Good bye, see you.\n");
112893dbb94SMatthias Ringwald     exit(0);
113893dbb94SMatthias Ringwald }
114893dbb94SMatthias Ringwald 
115893dbb94SMatthias Ringwald static int led_state = 0;
116893dbb94SMatthias Ringwald void hal_led_toggle(void){
117893dbb94SMatthias Ringwald     led_state = 1 - led_state;
118893dbb94SMatthias Ringwald     printf("LED State %u\n", led_state);
119893dbb94SMatthias Ringwald }
120893dbb94SMatthias Ringwald 
121893dbb94SMatthias Ringwald static void packet_handler (uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){
122893dbb94SMatthias Ringwald     bd_addr_t addr;
123893dbb94SMatthias Ringwald     if (packet_type != HCI_EVENT_PACKET) return;
124893dbb94SMatthias Ringwald     switch (hci_event_packet_get_type(packet)){
125893dbb94SMatthias Ringwald         case BTSTACK_EVENT_STATE:
126893dbb94SMatthias Ringwald             if (btstack_event_state_get_state(packet) != HCI_STATE_WORKING) break;
127893dbb94SMatthias Ringwald             gap_local_bd_addr(addr);
128893dbb94SMatthias Ringwald             printf("BTstack up and running at %s\n",  bd_addr_to_str(addr));
129893dbb94SMatthias Ringwald             // setup TLV
130893dbb94SMatthias Ringwald             btstack_strcpy(tlv_db_path, sizeof(tlv_db_path), TLV_DB_PATH_PREFIX);
131893dbb94SMatthias Ringwald             btstack_strcat(tlv_db_path, sizeof(tlv_db_path), bd_addr_to_str(addr));
132893dbb94SMatthias Ringwald             btstack_strcat(tlv_db_path, sizeof(tlv_db_path), TLV_DB_PATH_POSTFIX);
133893dbb94SMatthias Ringwald             tlv_impl = btstack_tlv_posix_init_instance(&tlv_context, tlv_db_path);
134893dbb94SMatthias Ringwald             btstack_tlv_set_instance(tlv_impl, &tlv_context);
135893dbb94SMatthias Ringwald #ifdef ENABLE_CLASSIC
136893dbb94SMatthias Ringwald             hci_set_link_key_db(btstack_link_key_db_tlv_get_instance(tlv_impl, &tlv_context));
137893dbb94SMatthias Ringwald #endif
138893dbb94SMatthias Ringwald #ifdef ENABLE_BLE
139893dbb94SMatthias Ringwald             le_device_db_tlv_configure(tlv_impl, &tlv_context);
140893dbb94SMatthias Ringwald #endif
141893dbb94SMatthias Ringwald             break;
142893dbb94SMatthias Ringwald         default:
143893dbb94SMatthias Ringwald             break;
144893dbb94SMatthias Ringwald     }
145893dbb94SMatthias Ringwald }
146893dbb94SMatthias Ringwald 
147893dbb94SMatthias Ringwald static void phase2(int status);
148893dbb94SMatthias Ringwald int main(int argc, const char * argv[]){
149893dbb94SMatthias Ringwald 
150893dbb94SMatthias Ringwald     /// GET STARTED with BTstack ///
151893dbb94SMatthias Ringwald     btstack_memory_init();
152893dbb94SMatthias Ringwald 
153893dbb94SMatthias Ringwald #if 1
154893dbb94SMatthias Ringwald     // log into file using HCI_DUMP_PACKETLOGGER format
155893dbb94SMatthias Ringwald     const char * pklg_path = "/tmp/hci_dump.pklg";
156893dbb94SMatthias Ringwald     hci_dump_posix_fs_open(pklg_path, HCI_DUMP_PACKETLOGGER);
157893dbb94SMatthias Ringwald     const hci_dump_t * hci_dump_impl = hci_dump_posix_fs_get_instance();
158893dbb94SMatthias Ringwald     printf("Packet Log: %s\n", pklg_path);
159893dbb94SMatthias Ringwald #else
160893dbb94SMatthias Ringwald     // log to stdout for debugging/development
161893dbb94SMatthias Ringwald     const hci_dump_t * hci_dump_impl = hci_dump_posix_stdout_get_instance();
162893dbb94SMatthias Ringwald #endif
163893dbb94SMatthias Ringwald     hci_dump_init(hci_dump_impl);
164893dbb94SMatthias Ringwald 
165893dbb94SMatthias Ringwald     // setup run loop
166893dbb94SMatthias Ringwald     btstack_run_loop_init(btstack_run_loop_posix_get_instance());
167893dbb94SMatthias Ringwald 
168893dbb94SMatthias Ringwald     // pick serial port and configure uart driver
169893dbb94SMatthias Ringwald     transport_config.device_name = "/dev/tty.usbserial-FT1XBGIM"; // murata m.2 adapter
170893dbb94SMatthias Ringwald 
171893dbb94SMatthias Ringwald     // get BCM chipset driver
172893dbb94SMatthias Ringwald     const btstack_chipset_t * chipset = btstack_chipset_bcm_instance();
173893dbb94SMatthias Ringwald     chipset->init(&transport_config);
174893dbb94SMatthias Ringwald 
175893dbb94SMatthias Ringwald     // set chipset name
176893dbb94SMatthias Ringwald     btstack_chipset_bcm_set_device_name("CYW55560A1");
177893dbb94SMatthias Ringwald 
178893dbb94SMatthias Ringwald     // setup UART driver
179893dbb94SMatthias Ringwald     const btstack_uart_t * uart_driver = (const btstack_uart_t *) btstack_uart_posix_instance();
180893dbb94SMatthias Ringwald 
181893dbb94SMatthias Ringwald     // extract UART config from transport config
182*36e2ff5eSMatthias Ringwald     uart_config.baudrate    = baudrate_firmware_download;
183893dbb94SMatthias Ringwald     uart_config.flowcontrol = transport_config.flowcontrol;
184893dbb94SMatthias Ringwald     uart_config.device_name = transport_config.device_name;
185893dbb94SMatthias Ringwald     uart_driver->init(&uart_config);
186893dbb94SMatthias Ringwald 
187893dbb94SMatthias Ringwald 
188893dbb94SMatthias Ringwald     // setup HCI (to be able to use bcm chipset driver)
189893dbb94SMatthias Ringwald     // init HCI
190893dbb94SMatthias Ringwald     const hci_transport_t * transport = hci_transport_h4_instance(uart_driver);
191893dbb94SMatthias Ringwald     hci_init(transport, (void*) &transport_config);
192893dbb94SMatthias Ringwald     hci_set_chipset(btstack_chipset_bcm_instance());
193893dbb94SMatthias Ringwald 
194893dbb94SMatthias Ringwald     // inform about BTstack state
195893dbb94SMatthias Ringwald     hci_event_callback_registration.callback = &packet_handler;
196893dbb94SMatthias Ringwald     hci_add_event_handler(&hci_event_callback_registration);
197893dbb94SMatthias Ringwald 
198893dbb94SMatthias Ringwald     // handle CTRL-c
199893dbb94SMatthias Ringwald     signal(SIGINT, sigint_handler);
200893dbb94SMatthias Ringwald 
201893dbb94SMatthias Ringwald     main_argc = argc;
202893dbb94SMatthias Ringwald     main_argv = argv;
203893dbb94SMatthias Ringwald 
204893dbb94SMatthias Ringwald     // phase #1 download firmware
205893dbb94SMatthias Ringwald     printf("Phase 1: Download firmware\n");
206893dbb94SMatthias Ringwald 
207893dbb94SMatthias Ringwald     // phase #2 start main app
208893dbb94SMatthias Ringwald     btstack_chipset_bcm_download_firmware_with_uart(uart_driver, 0, &phase2);
209893dbb94SMatthias Ringwald 
210893dbb94SMatthias Ringwald     // go
211893dbb94SMatthias Ringwald     btstack_run_loop_execute();
212893dbb94SMatthias Ringwald     return 0;
213893dbb94SMatthias Ringwald }
214893dbb94SMatthias Ringwald 
215893dbb94SMatthias Ringwald static void phase2(int status){
216893dbb94SMatthias Ringwald 
217893dbb94SMatthias Ringwald     if (status){
218893dbb94SMatthias Ringwald         printf("Download firmware failed\n");
219893dbb94SMatthias Ringwald         return;
220893dbb94SMatthias Ringwald     }
221893dbb94SMatthias Ringwald 
222893dbb94SMatthias Ringwald     printf("Phase 2: Main app\n");
223893dbb94SMatthias Ringwald 
224893dbb94SMatthias Ringwald     // setup app
225893dbb94SMatthias Ringwald     btstack_main(main_argc, main_argv);
226893dbb94SMatthias Ringwald }
227893dbb94SMatthias Ringwald 
228