test/gap: unit test for gap le scanning

This commit is contained in:
Matthias Ringwald 2020-04-02 16:22:07 +02:00
parent a5a5d24ebb
commit 1470db0ca8
5 changed files with 195 additions and 0 deletions

View File

@ -5405,4 +5405,8 @@ void hci_free_connections_fuzz(void){
btstack_memory_hci_connection_free(con);
}
}
void hci_simulate_working_fuzz(void){
hci_init_done();
hci_stack->num_cmd_packets = 255;
}
#endif

View File

@ -1251,6 +1251,9 @@ void hci_setup_test_connections_fuzz(void);
// free all connections, used for fuzzing
void hci_free_connections_fuzz(void);
// simulate stack bootup
void hci_simulate_working_fuzz(void);
#if defined __cplusplus
}
#endif

View File

@ -14,6 +14,7 @@ SUBDIRS = \
flash_tlv \
gatt_client \
gatt_server \
gap \
hfp \
hid_parser \
linked_list \

48
test/gap/Makefile Normal file
View File

@ -0,0 +1,48 @@
CC = g++
# Requirements: cpputest.github.io
BTSTACK_ROOT = ../..
CFLAGS = -DUNIT_TEST -x c++ -g -Wall -Wnarrowing -Wconversion-null -I. -I../ -I${BTSTACK_ROOT}/src
CFLAGS += -fsanitize=address
CFLAGS += -DFUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION
CFLAGS += -fprofile-arcs -ftest-coverage
LDFLAGS += -lCppUTest -lCppUTestExt
VPATH += ${BTSTACK_ROOT}/src
VPATH += ${BTSTACK_ROOT}/src/ble
VPATH += ${BTSTACK_ROOT}/platform/posix
COMMON = \
ad_parser.c \
btstack_linked_list.c \
btstack_memory.c \
btstack_memory_pool.c \
btstack_util.c \
btstack_run_loop.c \
hci.c \
hci_cmd.c \
hci_dump.c \
le_device_db_memory.c \
COMMON_OBJ = $(COMMON:.c=.o)
all: test_le_scan
# compile .ble description
profile.h: profile.gatt
python ${BTSTACK_ROOT}/tool/compile_gatt.py $< $@
test_le_scan: ${COMMON_OBJ} test_le_scan.o
${CC} ${COMMON_OBJ} test_le_scan.o ${CFLAGS} ${LDFLAGS} -o $@
test: all
./test_le_scan
clean:
rm -f test_le_scan
rm -f *.o
rm -rf *.dSYM
rm -f *.gcno *.gcda

139
test/gap/test_le_scan.c Normal file
View File

@ -0,0 +1,139 @@
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "CppUTest/TestHarness.h"
#include "CppUTest/CommandLineTestRunner.h"
#include "CppUTestExt/MockSupport.h"
#include "hci_cmd.h"
#include "btstack_memory.h"
#include "hci.h"
#include "ble/gatt_client.h"
#include "btstack_event.h"
#include "hci_dump.h"
#include "btstack_debug.h"
typedef struct {
uint8_t type;
uint16_t size;
uint8_t buffer[258];
} hci_packet_t;
#define MAX_HCI_PACKETS 10
static uint16_t transport_count_packets;
static hci_packet_t transport_packets[MAX_HCI_PACKETS];
static void (*packet_handler)(uint8_t packet_type, uint8_t *packet, uint16_t size);
static const uint8_t packet_sent_event[] = { HCI_EVENT_TRANSPORT_PACKET_SENT, 0};
static int hci_transport_test_set_baudrate(uint32_t baudrate){
return 0;
}
static int hci_transport_test_can_send_now(uint8_t packet_type){
return 1;
}
static int hci_transport_test_send_packet(uint8_t packet_type, uint8_t * packet, int size){
btstack_assert(transport_count_packets < MAX_HCI_PACKETS);
memcpy(transport_packets[transport_count_packets].buffer, packet, size);
transport_packets[transport_count_packets].type = packet_type;
transport_packets[transport_count_packets].size = size;
transport_count_packets++;
// notify upper stack that it can send again
packet_handler(HCI_EVENT_PACKET, (uint8_t *) &packet_sent_event[0], sizeof(packet_sent_event));
return 0;
}
static void hci_transport_test_init(const void * transport_config){
}
static int hci_transport_test_open(void){
return 0;
}
static int hci_transport_test_close(void){
return 0;
}
static void hci_transport_test_register_packet_handler(void (*handler)(uint8_t packet_type, uint8_t *packet, uint16_t size)){
packet_handler = handler;
}
static const hci_transport_t hci_transport_test = {
/* const char * name; */ "TEST",
/* void (*init) (const void *transport_config); */ &hci_transport_test_init,
/* int (*open)(void); */ &hci_transport_test_open,
/* int (*close)(void); */ &hci_transport_test_close,
/* void (*register_packet_handler)(void (*handler)(...); */ &hci_transport_test_register_packet_handler,
/* int (*can_send_packet_now)(uint8_t packet_type); */ &hci_transport_test_can_send_now,
/* int (*send_packet)(...); */ &hci_transport_test_send_packet,
/* int (*set_baudrate)(uint32_t baudrate); */ &hci_transport_test_set_baudrate,
/* void (*reset_link)(void); */ NULL,
/* void (*set_sco_config)(uint16_t voice_setting, int num_connections); */ NULL,
};
static uint16_t next_hci_packet;
void CHECK_EQUAL_ARRAY(const uint8_t * expected, uint8_t * actual, int size){
for (int i=0; i<size; i++){
BYTES_EQUAL(expected[i], actual[i]);
}
}
void CHECK_HCI_COMMAND(const hci_cmd_t * expected_hci_command){
uint16_t actual_opcode = little_endian_read_16(transport_packets[next_hci_packet].buffer, 0);
next_hci_packet++;
CHECK_EQUAL(expected_hci_command->opcode, actual_opcode);
}
TEST_GROUP(GAP_LE){
void setup(void){
transport_count_packets = 0;
next_hci_packet = 0;
hci_init(&hci_transport_test, NULL);
hci_simulate_working_fuzz();
// register for HCI events
mock().expectOneCall("hci_can_send_packet_now_using_packet_buffer").andReturnValue(1);
}
void teardown(void){
mock().clear();
}
};
TEST(GAP_LE, ScanStart){
log_info("TEST(GAP_LE, ScanStart)");
gap_start_scan();
CHECK_EQUAL(1, transport_count_packets);
CHECK_HCI_COMMAND(&hci_le_set_scan_enable);
}
TEST(GAP_LE, ScanStartStop){
log_info("TEST(GAP_LE, ScanStartStop)");
gap_start_scan();
gap_stop_scan();
CHECK_EQUAL(2, transport_count_packets);
CHECK_HCI_COMMAND(&hci_le_set_scan_enable);
CHECK_HCI_COMMAND(&hci_le_set_scan_enable);
}
TEST(GAP_LE, ScanStartParam){
log_info("TEST(GAP_LE, ScanStartParam)");
gap_start_scan();
gap_set_scan_parameters(0, 10, 10);
CHECK_EQUAL(4, transport_count_packets);
CHECK_HCI_COMMAND(&hci_le_set_scan_enable);
CHECK_HCI_COMMAND(&hci_le_set_scan_enable);
CHECK_HCI_COMMAND(&hci_le_set_scan_parameters);
CHECK_HCI_COMMAND(&hci_le_set_scan_enable);
}
int main (int argc, const char * argv[]){
const char * log_path = "/tmp/test_scan.pklg";
printf("Log: %s\n", log_path);
hci_dump_open(log_path, HCI_DUMP_PACKETLOGGER);
return CommandLineTestRunner::RunAllTests(argc, argv);
}