1dc5611fdSMilanka Ringwald /* 2dc5611fdSMilanka Ringwald * Copyright (C) 2014 BlueKitchen GmbH 3dc5611fdSMilanka Ringwald * 4dc5611fdSMilanka Ringwald * Redistribution and use in source and binary forms, with or without 5dc5611fdSMilanka Ringwald * modification, are permitted provided that the following conditions 6dc5611fdSMilanka Ringwald * are met: 7dc5611fdSMilanka Ringwald * 8dc5611fdSMilanka Ringwald * 1. Redistributions of source code must retain the above copyright 9dc5611fdSMilanka Ringwald * notice, this list of conditions and the following disclaimer. 10dc5611fdSMilanka Ringwald * 2. Redistributions in binary form must reproduce the above copyright 11dc5611fdSMilanka Ringwald * notice, this list of conditions and the following disclaimer in the 12dc5611fdSMilanka Ringwald * documentation and/or other materials provided with the distribution. 13dc5611fdSMilanka Ringwald * 3. Neither the name of the copyright holders nor the names of 14dc5611fdSMilanka Ringwald * contributors may be used to endorse or promote products derived 15dc5611fdSMilanka Ringwald * from this software without specific prior written permission. 16dc5611fdSMilanka Ringwald * 4. Any redistribution, use, or modification is done solely for 17dc5611fdSMilanka Ringwald * personal benefit and not for any commercial purpose or for 18dc5611fdSMilanka Ringwald * monetary gain. 19dc5611fdSMilanka Ringwald * 20dc5611fdSMilanka Ringwald * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS 21dc5611fdSMilanka Ringwald * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 22dc5611fdSMilanka Ringwald * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 23dc5611fdSMilanka Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MATTHIAS 24dc5611fdSMilanka Ringwald * RINGWALD OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, 25dc5611fdSMilanka Ringwald * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 26dc5611fdSMilanka Ringwald * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS 27dc5611fdSMilanka Ringwald * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 28dc5611fdSMilanka Ringwald * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 29dc5611fdSMilanka Ringwald * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF 30dc5611fdSMilanka Ringwald * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 31dc5611fdSMilanka Ringwald * SUCH DAMAGE. 32dc5611fdSMilanka Ringwald * 33dc5611fdSMilanka Ringwald * Please inquire about commercial licensing options at 34dc5611fdSMilanka Ringwald * [email protected] 35dc5611fdSMilanka Ringwald * 36dc5611fdSMilanka Ringwald */ 37dc5611fdSMilanka Ringwald 38dc5611fdSMilanka Ringwald // ***************************************************************************** 39dc5611fdSMilanka Ringwald // 40dc5611fdSMilanka Ringwald // HFP BTstack Mocks 41dc5611fdSMilanka Ringwald // 42dc5611fdSMilanka Ringwald // ***************************************************************************** 43dc5611fdSMilanka Ringwald 4412381011SMilanka Ringwald #include <stdint.h> 4512381011SMilanka Ringwald #include <stdio.h> 4612381011SMilanka Ringwald #include <stdlib.h> 4712381011SMilanka Ringwald #include <string.h> 4812381011SMilanka Ringwald 4912381011SMilanka Ringwald #include <btstack/btstack.h> 5012381011SMilanka Ringwald #include "hci.h" 5112381011SMilanka Ringwald #include "hci_dump.h" 5212381011SMilanka Ringwald #include "sdp_query_rfcomm.h" 5312381011SMilanka Ringwald #include "rfcomm.h" 54dc5611fdSMilanka Ringwald #include "hfp_hf.h" 55dc5611fdSMilanka Ringwald 56dc5611fdSMilanka Ringwald #include "mock.h" 5712381011SMilanka Ringwald 5812381011SMilanka Ringwald static void *registered_sdp_app_context; 5912381011SMilanka Ringwald static uint8_t sdp_rfcomm_channel_nr = 1; 6012381011SMilanka Ringwald const char sdp_rfcomm_service_name[] = "BTstackMock"; 6112381011SMilanka Ringwald static uint16_t rfcomm_cid = 1; 626f1de21cSMilanka Ringwald static bd_addr_t dev_addr; 636f1de21cSMilanka Ringwald static uint16_t sco_handle = 10; 6412381011SMilanka Ringwald static uint8_t rfcomm_payload[200]; 6512381011SMilanka Ringwald static uint16_t rfcomm_payload_len; 6612381011SMilanka Ringwald void * active_connection; 6712381011SMilanka Ringwald 6812381011SMilanka Ringwald void (*registered_rfcomm_packet_handler)(void * connection, uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size); 6912381011SMilanka Ringwald void (*registered_sdp_app_callback)(sdp_query_event_t * event, void * context); 7012381011SMilanka Ringwald 7112381011SMilanka Ringwald uint8_t * get_rfcomm_payload(){ 7212381011SMilanka Ringwald return &rfcomm_payload[0]; 7312381011SMilanka Ringwald } 7412381011SMilanka Ringwald 7512381011SMilanka Ringwald uint16_t get_rfcomm_payload_len(){ 7612381011SMilanka Ringwald return rfcomm_payload_len; 7712381011SMilanka Ringwald } 7812381011SMilanka Ringwald 79fc6b77dcSMilanka Ringwald static void prepare_rfcomm_buffer(uint8_t * data, int len){ 80656f7353SMilanka Ringwald if (len <= 0) return; 8112381011SMilanka Ringwald memset(&rfcomm_payload, 0, 200); 82fc6b77dcSMilanka Ringwald int pos = 0; 83656f7353SMilanka Ringwald 84656f7353SMilanka Ringwald if (strncmp((char*)data, "AT", 2) == 0){ 85656f7353SMilanka Ringwald strncpy((char*)&rfcomm_payload[pos], (char*)data, len); 86656f7353SMilanka Ringwald pos += len; 87656f7353SMilanka Ringwald } else { 88fc6b77dcSMilanka Ringwald rfcomm_payload[pos++] = '\r'; 89fc6b77dcSMilanka Ringwald rfcomm_payload[pos++] = '\n'; 90656f7353SMilanka Ringwald strncpy((char*)&rfcomm_payload[pos], (char*)data, len); 91fc6b77dcSMilanka Ringwald pos += len; 92656f7353SMilanka Ringwald 93656f7353SMilanka Ringwald if (memcmp((char*)data, "+BAC", 4) != 0 && 94656f7353SMilanka Ringwald memcmp((char*)data, "+BCS", 4) != 0){ 95fc6b77dcSMilanka Ringwald rfcomm_payload[pos++] = '\r'; 96dc5611fdSMilanka Ringwald rfcomm_payload[pos++] = '\n'; 97dc5611fdSMilanka Ringwald rfcomm_payload[pos++] = 'O'; 98dc5611fdSMilanka Ringwald rfcomm_payload[pos++] = 'K'; 99fc6b77dcSMilanka Ringwald } 100656f7353SMilanka Ringwald 101656f7353SMilanka Ringwald } 102dc5611fdSMilanka Ringwald rfcomm_payload[pos++] = '\r'; 103656f7353SMilanka Ringwald rfcomm_payload[pos++] = '\n'; 104656f7353SMilanka Ringwald rfcomm_payload[pos] = 0; 105fc6b77dcSMilanka Ringwald rfcomm_payload_len = pos; 106fc6b77dcSMilanka Ringwald } 107fc6b77dcSMilanka Ringwald 10862656e33SMilanka Ringwald static void print_without_newlines(uint8_t *data, uint16_t len){ 10962656e33SMilanka Ringwald int found_newline = 0; 11062656e33SMilanka Ringwald int found_item = 0; 111c26caa21SMilanka Ringwald 11262656e33SMilanka Ringwald for (int i=0; i<len; i++){ 11362656e33SMilanka Ringwald if (data[i] == '\r' || data[i] == '\n'){ 11462656e33SMilanka Ringwald if (!found_newline && found_item) printf("\n"); 11562656e33SMilanka Ringwald found_newline = 1; 11662656e33SMilanka Ringwald } else { 11762656e33SMilanka Ringwald printf("%c", data[i]); 11862656e33SMilanka Ringwald found_newline = 0; 11962656e33SMilanka Ringwald found_item = 1; 12062656e33SMilanka Ringwald } 12162656e33SMilanka Ringwald } 12262656e33SMilanka Ringwald printf("\n"); 12362656e33SMilanka Ringwald } 124fc6b77dcSMilanka Ringwald 12512381011SMilanka Ringwald int rfcomm_send_internal(uint16_t rfcomm_cid, uint8_t *data, uint16_t len){ 126656f7353SMilanka Ringwald if (strncmp((char*)data, "AT", 2) == 0){ 12762656e33SMilanka Ringwald printf("Verify HF state machine response: "); 12862656e33SMilanka Ringwald print_without_newlines(data,len); 129fc6b77dcSMilanka Ringwald } else { 13062656e33SMilanka Ringwald printf("Verify AG state machine response: "); 13162656e33SMilanka Ringwald print_without_newlines(data,len); 132fc6b77dcSMilanka Ringwald } 133656f7353SMilanka Ringwald strncpy((char*)&rfcomm_payload[0], (char*)data, len); 134656f7353SMilanka Ringwald rfcomm_payload_len = len; 13512381011SMilanka Ringwald return 0; 13612381011SMilanka Ringwald } 13712381011SMilanka Ringwald 1386f1de21cSMilanka Ringwald static void hci_event_sco_complete(){ 1396f1de21cSMilanka Ringwald uint8_t event[20]; 1406f1de21cSMilanka Ringwald uint8_t pos = 0; 1416f1de21cSMilanka Ringwald event[pos++] = HCI_EVENT_SYNCHRONOUS_CONNECTION_COMPLETE; 1426f1de21cSMilanka Ringwald event[pos++] = sizeof(event) - 2; 1436f1de21cSMilanka Ringwald 1446f1de21cSMilanka Ringwald event[pos++] = 0; //status 1456f1de21cSMilanka Ringwald bt_store_16(event, pos, sco_handle); pos += 2; // sco handle 1466f1de21cSMilanka Ringwald bt_flip_addr(&event[pos], dev_addr); pos += 6; 1476f1de21cSMilanka Ringwald printf("hci_event_sco_complete sco_handle 0x%02x, address %s\n", sco_handle, bd_addr_to_str(&event[pos-6])); 1486f1de21cSMilanka Ringwald 1496f1de21cSMilanka Ringwald event[pos++] = 0; // link_type 1506f1de21cSMilanka Ringwald event[pos++] = 0; // transmission_interval 1516f1de21cSMilanka Ringwald event[pos++] = 0; // retransmission_interval 1526f1de21cSMilanka Ringwald 1536f1de21cSMilanka Ringwald bt_store_16(event, pos, 0); pos += 2; // rx_packet_length 1546f1de21cSMilanka Ringwald bt_store_16(event, pos, 0); pos += 2; // tx_packet_length 1556f1de21cSMilanka Ringwald 1566f1de21cSMilanka Ringwald event[pos++] = 0; // air_mode 1576f1de21cSMilanka Ringwald (*registered_rfcomm_packet_handler)(0, HCI_EVENT_PACKET, 0, event, sizeof(event)); 1586f1de21cSMilanka Ringwald } 1596f1de21cSMilanka Ringwald 16012381011SMilanka Ringwald int hci_send_cmd(const hci_cmd_t *cmd, ...){ 1616f1de21cSMilanka Ringwald // printf("hci_send_cmd opcode 0x%02x\n", cmd->opcode); 1626f1de21cSMilanka Ringwald if (cmd->opcode == 0x428){ 1636f1de21cSMilanka Ringwald hci_event_sco_complete(); 1646f1de21cSMilanka Ringwald } 16512381011SMilanka Ringwald return 0; 16612381011SMilanka Ringwald } 16712381011SMilanka Ringwald 16812381011SMilanka Ringwald 16912381011SMilanka Ringwald void sdp_query_rfcomm_register_callback(void(*sdp_app_callback)(sdp_query_event_t * event, void * context), void * context){ 17012381011SMilanka Ringwald registered_sdp_app_callback = sdp_app_callback; 17112381011SMilanka Ringwald registered_sdp_app_context = context; 17212381011SMilanka Ringwald } 17312381011SMilanka Ringwald 17412381011SMilanka Ringwald static void sdp_query_complete_response(uint8_t status){ 17512381011SMilanka Ringwald sdp_query_complete_event_t complete_event = { 17612381011SMilanka Ringwald SDP_QUERY_COMPLETE, 17712381011SMilanka Ringwald status 17812381011SMilanka Ringwald }; 17912381011SMilanka Ringwald (*registered_sdp_app_callback)((sdp_query_event_t*)&complete_event, registered_sdp_app_context); 18012381011SMilanka Ringwald } 18112381011SMilanka Ringwald 18212381011SMilanka Ringwald static void sdp_query_rfcomm_service_response(uint8_t status){ 18312381011SMilanka Ringwald sdp_query_rfcomm_service_event_t service_event = { 18412381011SMilanka Ringwald SDP_QUERY_RFCOMM_SERVICE, 18512381011SMilanka Ringwald sdp_rfcomm_channel_nr, 18612381011SMilanka Ringwald (uint8_t *)sdp_rfcomm_service_name 18712381011SMilanka Ringwald }; 18812381011SMilanka Ringwald (*registered_sdp_app_callback)((sdp_query_event_t*)&service_event, registered_sdp_app_context); 18912381011SMilanka Ringwald } 19012381011SMilanka Ringwald 19112381011SMilanka Ringwald void sdp_query_rfcomm_channel_and_name_for_uuid(bd_addr_t remote, uint16_t uuid){ 19212381011SMilanka Ringwald // printf("sdp_query_rfcomm_channel_and_name_for_uuid %p\n", registered_sdp_app_callback); 19312381011SMilanka Ringwald sdp_query_rfcomm_service_response(0); 19412381011SMilanka Ringwald sdp_query_complete_response(0); 19512381011SMilanka Ringwald } 19612381011SMilanka Ringwald 1976f1de21cSMilanka Ringwald 19812381011SMilanka Ringwald void rfcomm_create_channel_internal(void * connection, bd_addr_t addr, uint8_t channel){ 19912381011SMilanka Ringwald // RFCOMM_EVENT_OPEN_CHANNEL_COMPLETE 20012381011SMilanka Ringwald // printf("rfcomm_create_channel_internal\n"); 20112381011SMilanka Ringwald active_connection = connection; 20212381011SMilanka Ringwald uint8_t event[16]; 20312381011SMilanka Ringwald uint8_t pos = 0; 20412381011SMilanka Ringwald event[pos++] = RFCOMM_EVENT_OPEN_CHANNEL_COMPLETE; 20512381011SMilanka Ringwald event[pos++] = sizeof(event) - 2; 20612381011SMilanka Ringwald event[pos++] = 0; 20712381011SMilanka Ringwald 20812381011SMilanka Ringwald bt_flip_addr(&event[pos], addr); pos += 6; 2096f1de21cSMilanka Ringwald bt_flip_addr(dev_addr, addr); 2106f1de21cSMilanka Ringwald 21112381011SMilanka Ringwald bt_store_16(event, pos, 1); pos += 2; 21212381011SMilanka Ringwald event[pos++] = 0; 21312381011SMilanka Ringwald 21412381011SMilanka Ringwald bt_store_16(event, pos, rfcomm_cid); pos += 2; // channel ID 21512381011SMilanka Ringwald bt_store_16(event, pos, 200); pos += 2; // max frame size 21612381011SMilanka Ringwald (*registered_rfcomm_packet_handler)(connection, HCI_EVENT_PACKET, 0, (uint8_t *) event, pos); 21712381011SMilanka Ringwald } 21812381011SMilanka Ringwald 219d86bb743SMilanka Ringwald int rfcomm_can_send_packet_now(uint16_t rfcomm_cid){ 220d86bb743SMilanka Ringwald return 1; 221d86bb743SMilanka Ringwald } 22212381011SMilanka Ringwald 22312381011SMilanka Ringwald void rfcomm_disconnect_internal(uint16_t rfcomm_cid){ 2249280a6ccSMilanka Ringwald uint8_t event[4]; 2259280a6ccSMilanka Ringwald event[0] = RFCOMM_EVENT_CHANNEL_CLOSED; 2269280a6ccSMilanka Ringwald event[1] = sizeof(event) - 2; 2279280a6ccSMilanka Ringwald bt_store_16(event, 2, rfcomm_cid); 2289280a6ccSMilanka Ringwald (*registered_rfcomm_packet_handler)(active_connection, HCI_EVENT_PACKET, 0, (uint8_t *) event, sizeof(event)); 22912381011SMilanka Ringwald } 23012381011SMilanka Ringwald 23112381011SMilanka Ringwald void rfcomm_register_packet_handler(void (*handler)(void * connection, uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size)){ 23212381011SMilanka Ringwald registered_rfcomm_packet_handler = handler; 23312381011SMilanka Ringwald } 23412381011SMilanka Ringwald 23512381011SMilanka Ringwald void rfcomm_register_service_internal(void * connection, uint8_t channel, uint16_t max_frame_size){ 23612381011SMilanka Ringwald printf("rfcomm_register_service_internal\n"); 23712381011SMilanka Ringwald } 23812381011SMilanka Ringwald 23912381011SMilanka Ringwald 24012381011SMilanka Ringwald void sdp_query_rfcomm_channel_and_name_for_search_pattern(bd_addr_t remote, uint8_t * des_serviceSearchPattern){ 24112381011SMilanka Ringwald printf("sdp_query_rfcomm_channel_and_name_for_search_pattern\n"); 24212381011SMilanka Ringwald } 24312381011SMilanka Ringwald 24412381011SMilanka Ringwald 24512381011SMilanka Ringwald void rfcomm_accept_connection_internal(uint16_t rfcomm_cid){ 24612381011SMilanka Ringwald printf("rfcomm_accept_connection_internal \n"); 24712381011SMilanka Ringwald } 24812381011SMilanka Ringwald 2496f1de21cSMilanka Ringwald void hci_emit_disconnection_complete(uint16_t handle, uint8_t reason){ 2506f1de21cSMilanka Ringwald uint8_t event[6]; 2516f1de21cSMilanka Ringwald event[0] = HCI_EVENT_DISCONNECTION_COMPLETE; 2526f1de21cSMilanka Ringwald event[1] = sizeof(event) - 2; 2536f1de21cSMilanka Ringwald event[2] = 0; // status = OK 2546f1de21cSMilanka Ringwald bt_store_16(event, 3, handle); 2556f1de21cSMilanka Ringwald event[5] = reason; 2566f1de21cSMilanka Ringwald (*registered_rfcomm_packet_handler)(0, HCI_EVENT_PACKET, 0, event, sizeof(event)); 2576f1de21cSMilanka Ringwald } 2586f1de21cSMilanka Ringwald 2596f1de21cSMilanka Ringwald le_command_status_t gap_disconnect(hci_con_handle_t handle){ 2606f1de21cSMilanka Ringwald hci_emit_disconnection_complete(handle, 0); 2616f1de21cSMilanka Ringwald return BLE_PERIPHERAL_OK; 2626f1de21cSMilanka Ringwald } 2636f1de21cSMilanka Ringwald 264*dd5554ccSMilanka Ringwald uint16_t hci_get_sco_voice_setting(){ 265*dd5554ccSMilanka Ringwald return 0x40; 266*dd5554ccSMilanka Ringwald } 267dc5611fdSMilanka Ringwald 268656f7353SMilanka Ringwald void inject_rfcomm_command_to_hf(uint8_t * data, int len){ 269656f7353SMilanka Ringwald if (memcmp((char*)data, "AT", 2) == 0) return; 270656f7353SMilanka Ringwald 271656f7353SMilanka Ringwald prepare_rfcomm_buffer(data, len); 272c26caa21SMilanka Ringwald if (data[0] == '+' || (data[0] == 'O' && data[1] == 'K')){ 273c26caa21SMilanka Ringwald printf("Send cmd to HF state machine: %s\n", data); 274656f7353SMilanka Ringwald } else { 275c26caa21SMilanka Ringwald printf("Trigger HF state machine - %s", data); 276656f7353SMilanka Ringwald } 277656f7353SMilanka Ringwald (*registered_rfcomm_packet_handler)(active_connection, RFCOMM_DATA_PACKET, rfcomm_cid, (uint8_t *) &rfcomm_payload[0], rfcomm_payload_len); 278656f7353SMilanka Ringwald } 279656f7353SMilanka Ringwald 280656f7353SMilanka Ringwald void inject_rfcomm_command_to_ag(uint8_t * data, int len){ 281656f7353SMilanka Ringwald if (data[0] == '+') return; 282656f7353SMilanka Ringwald 283656f7353SMilanka Ringwald prepare_rfcomm_buffer(data, len); 284656f7353SMilanka Ringwald if (memcmp((char*)data, "AT", 2) == 0){ 28562656e33SMilanka Ringwald printf("Send cmd to AG state machine: %s\n", data); 286656f7353SMilanka Ringwald } else { 287c26caa21SMilanka Ringwald printf("Trigger AG state machine - %s", data); 288656f7353SMilanka Ringwald } 289656f7353SMilanka Ringwald (*registered_rfcomm_packet_handler)(active_connection, RFCOMM_DATA_PACKET, rfcomm_cid, (uint8_t *) &rfcomm_payload[0], rfcomm_payload_len); 290656f7353SMilanka Ringwald } 291dc5611fdSMilanka Ringwald 292dc5611fdSMilanka Ringwald 293dc5611fdSMilanka Ringwald 294dc5611fdSMilanka Ringwald 295