1f61339eaSMatthias Ringwald /*
2f61339eaSMatthias Ringwald * Copyright (C) 2014 BlueKitchen GmbH
3f61339eaSMatthias Ringwald *
4f61339eaSMatthias Ringwald * Redistribution and use in source and binary forms, with or without
5f61339eaSMatthias Ringwald * modification, are permitted provided that the following conditions
6f61339eaSMatthias Ringwald * are met:
7f61339eaSMatthias Ringwald *
8f61339eaSMatthias Ringwald * 1. Redistributions of source code must retain the above copyright
9f61339eaSMatthias Ringwald * notice, this list of conditions and the following disclaimer.
10f61339eaSMatthias Ringwald * 2. Redistributions in binary form must reproduce the above copyright
11f61339eaSMatthias Ringwald * notice, this list of conditions and the following disclaimer in the
12f61339eaSMatthias Ringwald * documentation and/or other materials provided with the distribution.
13f61339eaSMatthias Ringwald * 3. Neither the name of the copyright holders nor the names of
14f61339eaSMatthias Ringwald * contributors may be used to endorse or promote products derived
15f61339eaSMatthias Ringwald * from this software without specific prior written permission.
16f61339eaSMatthias Ringwald * 4. Any redistribution, use, or modification is done solely for
17f61339eaSMatthias Ringwald * personal benefit and not for any commercial purpose or for
18f61339eaSMatthias Ringwald * monetary gain.
19f61339eaSMatthias Ringwald *
20f61339eaSMatthias Ringwald * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS
21f61339eaSMatthias Ringwald * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
22f61339eaSMatthias Ringwald * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
232fca4dadSMilanka Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL BLUEKITCHEN
242fca4dadSMilanka Ringwald * GMBH OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
25f61339eaSMatthias Ringwald * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
26f61339eaSMatthias Ringwald * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
27f61339eaSMatthias Ringwald * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
28f61339eaSMatthias Ringwald * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
29f61339eaSMatthias Ringwald * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
30f61339eaSMatthias Ringwald * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
31f61339eaSMatthias Ringwald * SUCH DAMAGE.
32f61339eaSMatthias Ringwald *
33f61339eaSMatthias Ringwald * Please inquire about commercial licensing options at
34f61339eaSMatthias Ringwald * [email protected]
35f61339eaSMatthias Ringwald *
36f61339eaSMatthias Ringwald */
37f61339eaSMatthias Ringwald
38d5b02eadSMatthias Ringwald #define BTSTACK_FILE__ "main.c"
39f61339eaSMatthias Ringwald
40f61339eaSMatthias Ringwald // *****************************************************************************
41f61339eaSMatthias Ringwald //
42f61339eaSMatthias Ringwald // minimal setup for HCI code
43f61339eaSMatthias Ringwald //
44f61339eaSMatthias Ringwald // *****************************************************************************
45f61339eaSMatthias Ringwald
46f61339eaSMatthias Ringwald #include <stdint.h>
47f61339eaSMatthias Ringwald #include <stdio.h>
48f61339eaSMatthias Ringwald #include <stdlib.h>
49f61339eaSMatthias Ringwald #include <string.h>
50f61339eaSMatthias Ringwald
51f61339eaSMatthias Ringwald #include "btstack_config.h"
52f61339eaSMatthias Ringwald
53d5b02eadSMatthias Ringwald #include "ble/le_device_db_tlv.h"
5442c5c558SMatthias Ringwald #include "btstack_audio.h"
55d5b02eadSMatthias Ringwald #include "btstack_chipset_intel_firmware.h"
56f61339eaSMatthias Ringwald #include "btstack_debug.h"
57f61339eaSMatthias Ringwald #include "btstack_event.h"
58f61339eaSMatthias Ringwald #include "btstack_memory.h"
59f61339eaSMatthias Ringwald #include "btstack_run_loop.h"
60f61339eaSMatthias Ringwald #include "btstack_run_loop_windows.h"
61d5b02eadSMatthias Ringwald #include "btstack_stdin.h"
62d5b02eadSMatthias Ringwald #include "btstack_stdin_windows.h"
633086bcceSMatthias Ringwald #include "btstack_tlv_windows.h"
646486d278SMatthias Ringwald #include "classic/btstack_link_key_db_tlv.h"
65f61339eaSMatthias Ringwald #include "hal_led.h"
66f61339eaSMatthias Ringwald #include "hci.h"
67f61339eaSMatthias Ringwald #include "hci_dump.h"
683086bcceSMatthias Ringwald #include "hci_dump_windows_fs.h"
69c8dfe071SMatthias Ringwald #include "hci_transport.h"
70d5b02eadSMatthias Ringwald #include "hci_transport_usb.h"
71f61339eaSMatthias Ringwald
72f61339eaSMatthias Ringwald int btstack_main(int argc, const char * argv[]);
73f61339eaSMatthias Ringwald
746486d278SMatthias Ringwald
756486d278SMatthias Ringwald #define TLV_DB_PATH_PREFIX "btstack_"
766486d278SMatthias Ringwald #define TLV_DB_PATH_POSTFIX ".tlv"
776486d278SMatthias Ringwald static char tlv_db_path[100];
786486d278SMatthias Ringwald static const btstack_tlv_t * tlv_impl;
793086bcceSMatthias Ringwald static btstack_tlv_windows_t tlv_context;
806486d278SMatthias Ringwald static bd_addr_t local_addr;
816486d278SMatthias Ringwald
82f61339eaSMatthias Ringwald static btstack_packet_callback_registration_t hci_event_callback_registration;
83f61339eaSMatthias Ringwald static int main_argc;
84f61339eaSMatthias Ringwald static const char ** main_argv;
85f61339eaSMatthias Ringwald static const hci_transport_t * transport;
86948e8bfeSMatthias Ringwald static int intel_firmware_loaded;
87d5b02eadSMatthias Ringwald static bool shutdown_triggered;
88f61339eaSMatthias Ringwald
packet_handler(uint8_t packet_type,uint16_t channel,uint8_t * packet,uint16_t size)89f61339eaSMatthias Ringwald static void packet_handler (uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){
90f61339eaSMatthias Ringwald if (packet_type != HCI_EVENT_PACKET) return;
91f61339eaSMatthias Ringwald if (hci_event_packet_get_type(packet) != BTSTACK_EVENT_STATE) return;
92d5b02eadSMatthias Ringwald switch (btstack_event_state_get_state(packet)){
93d5b02eadSMatthias Ringwald case HCI_STATE_WORKING:
946486d278SMatthias Ringwald gap_local_bd_addr(local_addr);
956486d278SMatthias Ringwald printf("BTstack up and running on %s.\n", bd_addr_to_str(local_addr));
9654736c11SMatthias Ringwald btstack_strcpy(tlv_db_path, sizeof(tlv_db_path), TLV_DB_PATH_PREFIX);
9754736c11SMatthias Ringwald btstack_strcat(tlv_db_path, sizeof(tlv_db_path), bd_addr_to_str_with_delimiter(local_addr, '-'));
9854736c11SMatthias Ringwald btstack_strcat(tlv_db_path, sizeof(tlv_db_path), TLV_DB_PATH_POSTFIX);
993086bcceSMatthias Ringwald tlv_impl = btstack_tlv_windows_init_instance(&tlv_context, tlv_db_path);
1006486d278SMatthias Ringwald btstack_tlv_set_instance(tlv_impl, &tlv_context);
1016486d278SMatthias Ringwald #ifdef ENABLE_CLASSIC
1026486d278SMatthias Ringwald hci_set_link_key_db(btstack_link_key_db_tlv_get_instance(tlv_impl, &tlv_context));
1036486d278SMatthias Ringwald #endif
1046486d278SMatthias Ringwald #ifdef ENABLE_BLE
1056486d278SMatthias Ringwald le_device_db_tlv_configure(tlv_impl, &tlv_context);
1066486d278SMatthias Ringwald #endif
107d5b02eadSMatthias Ringwald break;
108d5b02eadSMatthias Ringwald case HCI_STATE_OFF:
1093086bcceSMatthias Ringwald btstack_tlv_windows_deinit(&tlv_context);
110d5b02eadSMatthias Ringwald if (!shutdown_triggered) break;
111d5b02eadSMatthias Ringwald // reset stdin
112d5b02eadSMatthias Ringwald btstack_stdin_reset();
113d5b02eadSMatthias Ringwald log_info("Good bye, see you.\n");
114d5b02eadSMatthias Ringwald exit(0);
115d5b02eadSMatthias Ringwald break;
116d5b02eadSMatthias Ringwald default:
117d5b02eadSMatthias Ringwald break;
118d5b02eadSMatthias Ringwald }
119f61339eaSMatthias Ringwald }
120f61339eaSMatthias Ringwald
trigger_shutdown(void)121d5b02eadSMatthias Ringwald static void trigger_shutdown(void){
122f61339eaSMatthias Ringwald printf("CTRL-C - SIGINT received, shutting down..\n");
123f61339eaSMatthias Ringwald log_info("sigint_handler: shutting down");
124d5b02eadSMatthias Ringwald shutdown_triggered = true;
125f61339eaSMatthias Ringwald hci_power_control(HCI_POWER_OFF);
126f61339eaSMatthias Ringwald }
127f61339eaSMatthias Ringwald
128f61339eaSMatthias Ringwald static int led_state = 0;
hal_led_toggle(void)129f61339eaSMatthias Ringwald void hal_led_toggle(void){
130f61339eaSMatthias Ringwald led_state = 1 - led_state;
131f61339eaSMatthias Ringwald printf("LED State %u\n", led_state);
132f61339eaSMatthias Ringwald }
133f61339eaSMatthias Ringwald
134f61339eaSMatthias Ringwald
intel_firmware_done(int result)135f61339eaSMatthias Ringwald static void intel_firmware_done(int result){
136f61339eaSMatthias Ringwald
137f61339eaSMatthias Ringwald printf("Done %x\n", result);
138f61339eaSMatthias Ringwald
139*1beba4d4SMatthias Ringwald if (result != 0){
140*1beba4d4SMatthias Ringwald exit(result);
141*1beba4d4SMatthias Ringwald }
142*1beba4d4SMatthias Ringwald
143948e8bfeSMatthias Ringwald intel_firmware_loaded = 1;
144948e8bfeSMatthias Ringwald
145f61339eaSMatthias Ringwald // init HCI
146f61339eaSMatthias Ringwald hci_init(transport, NULL);
147f61339eaSMatthias Ringwald
14842c5c558SMatthias Ringwald #ifdef HAVE_PORTAUDIO
14942c5c558SMatthias Ringwald btstack_audio_sink_set_instance(btstack_audio_portaudio_sink_get_instance());
15042c5c558SMatthias Ringwald btstack_audio_source_set_instance(btstack_audio_portaudio_source_get_instance());
15142c5c558SMatthias Ringwald #endif
15242c5c558SMatthias Ringwald
153f61339eaSMatthias Ringwald // inform about BTstack state
154f61339eaSMatthias Ringwald hci_event_callback_registration.callback = &packet_handler;
155f61339eaSMatthias Ringwald hci_add_event_handler(&hci_event_callback_registration);
156f61339eaSMatthias Ringwald
157f61339eaSMatthias Ringwald // setup app
158f61339eaSMatthias Ringwald btstack_main(main_argc, main_argv);
159f61339eaSMatthias Ringwald }
160f61339eaSMatthias Ringwald
161f61339eaSMatthias Ringwald #define USB_MAX_PATH_LEN 7
main(int argc,const char * argv[])162f61339eaSMatthias Ringwald int main(int argc, const char * argv[]){
163f61339eaSMatthias Ringwald
164f61339eaSMatthias Ringwald // Prevent stdout buffering
165f61339eaSMatthias Ringwald setvbuf(stdout, NULL, _IONBF, 0);
166f61339eaSMatthias Ringwald
167f61339eaSMatthias Ringwald main_argc = argc;
168f61339eaSMatthias Ringwald main_argv = argv;
169f61339eaSMatthias Ringwald
170f61339eaSMatthias Ringwald printf("BTstack/windows-winusb booting up\n");
171f61339eaSMatthias Ringwald
172f61339eaSMatthias Ringwald /// GET STARTED with BTstack ///
173f61339eaSMatthias Ringwald btstack_memory_init();
174f61339eaSMatthias Ringwald btstack_run_loop_init(btstack_run_loop_windows_get_instance());
175f61339eaSMatthias Ringwald
1767435ec7bSMatthias Ringwald // log into file using HCI_DUMP_PACKETLOGGER format
17734c6d5e0SMatthias Ringwald const char * pklg_path = "hci_dump.pklg";
1783086bcceSMatthias Ringwald hci_dump_windows_fs_open(pklg_path, HCI_DUMP_PACKETLOGGER);
1793086bcceSMatthias Ringwald const hci_dump_t * hci_dump_impl = hci_dump_windows_fs_get_instance();
1807435ec7bSMatthias Ringwald hci_dump_init(hci_dump_impl);
181f61339eaSMatthias Ringwald printf("Packet Log: %s\n", pklg_path);
182f61339eaSMatthias Ringwald
183d5b02eadSMatthias Ringwald // setup stdin to handle CTRL-c
184d5b02eadSMatthias Ringwald btstack_stdin_windows_init();
185d5b02eadSMatthias Ringwald btstack_stdin_window_register_ctrl_c_callback(&trigger_shutdown);
186f61339eaSMatthias Ringwald
187f61339eaSMatthias Ringwald // setup USB Transport
188f61339eaSMatthias Ringwald transport = hci_transport_usb_instance();
189f61339eaSMatthias Ringwald btstack_chipset_intel_download_firmware(hci_transport_usb_instance(), &intel_firmware_done);
190f61339eaSMatthias Ringwald
191f61339eaSMatthias Ringwald // go
192f61339eaSMatthias Ringwald btstack_run_loop_execute();
193f61339eaSMatthias Ringwald
194f61339eaSMatthias Ringwald return 0;
195f61339eaSMatthias Ringwald }
196