btstack/test/hfp/mock.c

301 lines
9.8 KiB
C
Raw Normal View History

/*
* Copyright (C) 2014 BlueKitchen GmbH
*
* 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. Any redistribution, use, or modification is done solely for
* personal benefit and not for any commercial purpose or for
* monetary gain.
*
* THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH 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.
*
* Please inquire about commercial licensing options at
* contact@bluekitchen-gmbh.com
*
*/
// *****************************************************************************
//
// HFP BTstack Mocks
//
// *****************************************************************************
2015-08-26 18:31:31 +02:00
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <btstack/btstack.h>
#include "hci.h"
#include "hci_dump.h"
#include "sdp_query_rfcomm.h"
#include "rfcomm.h"
#include "hfp_hf.h"
#include "mock.h"
2015-08-26 18:31:31 +02:00
static void *registered_sdp_app_context;
static uint8_t sdp_rfcomm_channel_nr = 1;
const char sdp_rfcomm_service_name[] = "BTstackMock";
static uint16_t rfcomm_cid = 1;
2015-11-11 15:45:35 +01:00
static bd_addr_t dev_addr;
static uint16_t sco_handle = 10;
2015-08-26 18:31:31 +02:00
static uint8_t rfcomm_payload[200];
static uint16_t rfcomm_payload_len;
void * active_connection;
2015-11-12 14:16:20 +01:00
hfp_connection_t * hfp_context;
2015-08-26 18:31:31 +02:00
void (*registered_rfcomm_packet_handler)(void * connection, uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size);
void (*registered_sdp_app_callback)(sdp_query_event_t * event, void * context);
uint8_t * get_rfcomm_payload(){
return &rfcomm_payload[0];
}
uint16_t get_rfcomm_payload_len(){
return rfcomm_payload_len;
}
2015-08-27 11:27:44 +02:00
static void prepare_rfcomm_buffer(uint8_t * data, int len){
2015-10-15 13:55:28 +02:00
if (len <= 0) return;
memset(&rfcomm_payload, 0, 200);
2015-08-27 11:27:44 +02:00
int pos = 0;
2015-10-15 13:55:28 +02:00
if (strncmp((char*)data, "AT", 2) == 0){
strncpy((char*)&rfcomm_payload[pos], (char*)data, len);
pos += len;
} else {
rfcomm_payload[pos++] = '\r';
rfcomm_payload[pos++] = '\n';
2015-10-15 13:55:28 +02:00
strncpy((char*)&rfcomm_payload[pos], (char*)data, len);
pos += len;
if (memcmp((char*)data, "+BAC", 4) != 0 &&
memcmp((char*)data, "+BCS", 4) != 0){
rfcomm_payload[pos++] = '\r';
rfcomm_payload[pos++] = '\n';
rfcomm_payload[pos++] = 'O';
rfcomm_payload[pos++] = 'K';
}
}
rfcomm_payload[pos++] = '\r';
rfcomm_payload[pos++] = '\n';
rfcomm_payload[pos] = 0;
2015-08-27 11:27:44 +02:00
rfcomm_payload_len = pos;
}
static void print_without_newlines(uint8_t *data, uint16_t len){
int found_newline = 0;
int found_item = 0;
2015-11-11 12:15:35 +01:00
for (int i=0; i<len; i++){
if (data[i] == '\r' || data[i] == '\n'){
if (!found_newline && found_item) printf("\n");
found_newline = 1;
} else {
printf("%c", data[i]);
found_newline = 0;
found_item = 1;
}
}
printf("\n");
}
2015-08-27 11:27:44 +02:00
extern "C" void l2cap_init(void){}
extern "C" void l2cap_register_packet_handler(void (*handler)(void * connection, uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size)){
}
2015-08-26 18:31:31 +02:00
int rfcomm_send_internal(uint16_t rfcomm_cid, uint8_t *data, uint16_t len){
2015-10-15 13:55:28 +02:00
if (strncmp((char*)data, "AT", 2) == 0){
printf("Verify HF state machine response: ");
print_without_newlines(data,len);
2015-08-27 11:27:44 +02:00
} else {
printf("Verify AG state machine response: ");
print_without_newlines(data,len);
2015-08-27 11:27:44 +02:00
}
2015-10-15 13:55:28 +02:00
strncpy((char*)&rfcomm_payload[0], (char*)data, len);
rfcomm_payload_len = len;
2015-08-26 18:31:31 +02:00
return 0;
}
2015-11-11 15:45:35 +01:00
static void hci_event_sco_complete(){
2015-11-12 14:16:20 +01:00
uint8_t event[19];
2015-11-11 15:45:35 +01:00
uint8_t pos = 0;
event[pos++] = HCI_EVENT_SYNCHRONOUS_CONNECTION_COMPLETE;
event[pos++] = sizeof(event) - 2;
event[pos++] = 0; //status
bt_store_16(event, pos, sco_handle); pos += 2; // sco handle
2015-11-12 14:16:20 +01:00
bt_flip_addr(&event[pos], dev_addr); pos += 6;
2015-11-11 15:45:35 +01:00
event[pos++] = 0; // link_type
event[pos++] = 0; // transmission_interval
event[pos++] = 0; // retransmission_interval
bt_store_16(event, pos, 0); pos += 2; // rx_packet_length
bt_store_16(event, pos, 0); pos += 2; // tx_packet_length
event[pos++] = 0; // air_mode
2015-11-12 14:16:20 +01:00
(*registered_rfcomm_packet_handler)(active_connection, HCI_EVENT_PACKET, 0, event, sizeof(event));
2015-11-11 15:45:35 +01:00
}
2015-08-26 18:31:31 +02:00
int hci_send_cmd(const hci_cmd_t *cmd, ...){
2015-11-12 14:16:20 +01:00
printf("hci_send_cmd opcode 0x%02x\n", cmd->opcode);
2015-11-11 15:45:35 +01:00
if (cmd->opcode == 0x428){
hci_event_sco_complete();
}
2015-08-26 18:31:31 +02:00
return 0;
}
void sdp_query_rfcomm_register_callback(void(*sdp_app_callback)(sdp_query_event_t * event, void * context), void * context){
registered_sdp_app_callback = sdp_app_callback;
registered_sdp_app_context = context;
}
static void sdp_query_complete_response(uint8_t status){
sdp_query_complete_event_t complete_event = {
SDP_QUERY_COMPLETE,
status
};
(*registered_sdp_app_callback)((sdp_query_event_t*)&complete_event, registered_sdp_app_context);
}
static void sdp_query_rfcomm_service_response(uint8_t status){
sdp_query_rfcomm_service_event_t service_event = {
SDP_QUERY_RFCOMM_SERVICE,
sdp_rfcomm_channel_nr,
(uint8_t *)sdp_rfcomm_service_name
};
(*registered_sdp_app_callback)((sdp_query_event_t*)&service_event, registered_sdp_app_context);
}
void sdp_query_rfcomm_channel_and_name_for_uuid(bd_addr_t remote, uint16_t uuid){
// printf("sdp_query_rfcomm_channel_and_name_for_uuid %p\n", registered_sdp_app_callback);
sdp_query_rfcomm_service_response(0);
sdp_query_complete_response(0);
}
2015-11-11 15:45:35 +01:00
2015-08-26 18:31:31 +02:00
void rfcomm_create_channel_internal(void * connection, bd_addr_t addr, uint8_t channel){
// RFCOMM_EVENT_OPEN_CHANNEL_COMPLETE
active_connection = connection;
uint8_t event[16];
uint8_t pos = 0;
event[pos++] = RFCOMM_EVENT_OPEN_CHANNEL_COMPLETE;
event[pos++] = sizeof(event) - 2;
event[pos++] = 0;
2015-11-12 14:16:20 +01:00
bt_flip_addr(&event[pos], addr);
memcpy(dev_addr, addr, 6);
pos += 6;
2015-11-11 15:45:35 +01:00
2015-08-26 18:31:31 +02:00
bt_store_16(event, pos, 1); pos += 2;
event[pos++] = 0;
bt_store_16(event, pos, rfcomm_cid); pos += 2; // channel ID
bt_store_16(event, pos, 200); pos += 2; // max frame size
2015-11-12 14:16:20 +01:00
(*registered_rfcomm_packet_handler)(active_connection, HCI_EVENT_PACKET, 0, (uint8_t *) event, pos);
2015-08-26 18:31:31 +02:00
}
int rfcomm_can_send_packet_now(uint16_t rfcomm_cid){
return 1;
}
2015-08-26 18:31:31 +02:00
void rfcomm_disconnect_internal(uint16_t rfcomm_cid){
2015-08-27 00:08:33 +02:00
uint8_t event[4];
event[0] = RFCOMM_EVENT_CHANNEL_CLOSED;
event[1] = sizeof(event) - 2;
bt_store_16(event, 2, rfcomm_cid);
(*registered_rfcomm_packet_handler)(active_connection, HCI_EVENT_PACKET, 0, (uint8_t *) event, sizeof(event));
2015-08-26 18:31:31 +02:00
}
void rfcomm_register_packet_handler(void (*handler)(void * connection, uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size)){
registered_rfcomm_packet_handler = handler;
}
void rfcomm_register_service_internal(void * connection, uint8_t channel, uint16_t max_frame_size){
printf("rfcomm_register_service_internal\n");
}
void sdp_query_rfcomm_channel_and_name_for_search_pattern(bd_addr_t remote, uint8_t * des_serviceSearchPattern){
printf("sdp_query_rfcomm_channel_and_name_for_search_pattern\n");
}
void rfcomm_accept_connection_internal(uint16_t rfcomm_cid){
printf("rfcomm_accept_connection_internal \n");
}
2015-11-11 15:45:35 +01:00
void hci_emit_disconnection_complete(uint16_t handle, uint8_t reason){
uint8_t event[6];
event[0] = HCI_EVENT_DISCONNECTION_COMPLETE;
event[1] = sizeof(event) - 2;
event[2] = 0; // status = OK
bt_store_16(event, 3, handle);
event[5] = reason;
2015-11-12 14:16:20 +01:00
(*registered_rfcomm_packet_handler)(active_connection, HCI_EVENT_PACKET, 0, event, sizeof(event));
2015-11-11 15:45:35 +01:00
}
le_command_status_t gap_disconnect(hci_con_handle_t handle){
hci_emit_disconnection_complete(handle, 0);
return BLE_PERIPHERAL_OK;
}
2015-11-12 12:13:43 +01:00
uint16_t hci_get_sco_voice_setting(){
return 0x40;
}
2015-10-15 13:55:28 +02:00
void inject_rfcomm_command_to_hf(uint8_t * data, int len){
if (memcmp((char*)data, "AT", 2) == 0) return;
2015-10-15 13:55:28 +02:00
prepare_rfcomm_buffer(data, len);
2015-11-11 12:15:35 +01:00
if (data[0] == '+' || (data[0] == 'O' && data[1] == 'K')){
printf("Send cmd to HF state machine: %s\n", data);
2015-10-15 13:55:28 +02:00
} else {
2015-11-11 12:15:35 +01:00
printf("Trigger HF state machine - %s", data);
}
2015-10-15 13:55:28 +02:00
(*registered_rfcomm_packet_handler)(active_connection, RFCOMM_DATA_PACKET, rfcomm_cid, (uint8_t *) &rfcomm_payload[0], rfcomm_payload_len);
}
2015-10-15 13:55:28 +02:00
void inject_rfcomm_command_to_ag(uint8_t * data, int len){
if (data[0] == '+') return;
prepare_rfcomm_buffer(data, len);
2015-10-15 13:55:28 +02:00
if (memcmp((char*)data, "AT", 2) == 0){
printf("Send cmd to AG state machine: %s\n", data);
2015-10-15 13:55:28 +02:00
} else {
2015-11-11 12:15:35 +01:00
printf("Trigger AG state machine - %s", data);
2015-10-15 13:55:28 +02:00
}
(*registered_rfcomm_packet_handler)(active_connection, RFCOMM_DATA_PACKET, rfcomm_cid, (uint8_t *) &rfcomm_payload[0], rfcomm_payload_len);
}