// SPDX-License-Identifier: Apache-2.0 /* * Copyright (C) 2014 Intel Corporation * */ #define _GNU_SOURCE #include #include "emulator/bthost.h" #include "src/shared/util.h" #include "src/shared/tester.h" #include "src/shared/queue.h" #include "lib/bluetooth.h" #include "android/utils.h" #include "tester-main.h" static struct queue *list; #define req_dsc 0x00, 0x01 #define rsp_dsc 0x02, 0x01, 0x04, 0x08 #define req_get 0x10, 0x02, 0x04 #define rsp_get 0x12, 0x02, 0x01, 0x00, 0x07, 0x06, 0x00, \ 0x00, 0xff, 0xff, 0x02, 0x40 #define req_cfg 0x20, 0x03, 0x04, 0x04, 0x01, 0x00, 0x07, \ 0x06, 0x00, 0x00, 0x21, 0x15, 0x02, \ 0x40 #define rsp_cfg 0x22, 0x03 #define req_open 0x30, 0x06, 0x04 #define rsp_open 0x32, 0x06 #define req_close 0x40, 0x08, 0x04 #define rsp_close 0x42, 0x08 #define req_start 0x40, 0x07, 0x04 #define rsp_start 0x42, 0x07 #define req_suspend 0x50, 0x09, 0x04 #define rsp_suspend 0x52, 0x09 static const struct pdu_set pdus[] = { { raw_pdu(req_dsc), raw_pdu(rsp_dsc) }, { raw_pdu(req_get), raw_pdu(rsp_get) }, { raw_pdu(req_cfg), raw_pdu(rsp_cfg) }, { raw_pdu(req_open), raw_pdu(rsp_open) }, { raw_pdu(req_close), raw_pdu(rsp_close) }, { raw_pdu(req_start), raw_pdu(rsp_start) }, { raw_pdu(req_suspend), raw_pdu(rsp_suspend) }, { end_pdu, end_pdu }, }; static struct emu_l2cap_cid_data cid_data = { .pdu = pdus, }; static void a2dp_connect_request_cb(uint16_t handle, uint16_t cid, void *user_data) { struct emu_l2cap_cid_data *cid_data = user_data; if (cid_data->handle) return; cid_data->handle = handle; cid_data->cid = cid; tester_handle_l2cap_data_exchange(cid_data); } static struct emu_set_l2cap_data l2cap_setup_data = { .psm = 25, .func = a2dp_connect_request_cb, .user_data = &cid_data, }; static void a2dp_connect_action(void) { struct test_data *data = tester_get_data(); const uint8_t *addr = hciemu_get_client_bdaddr(data->hciemu); struct step *step = g_new0(struct step, 1); bt_bdaddr_t bdaddr; cid_data.handle = 0; cid_data.cid = 0; bdaddr2android((const bdaddr_t *) addr, &bdaddr); step->action_status = data->if_a2dp->connect(&bdaddr); schedule_action_verification(step); } static void a2dp_disconnect_action(void) { struct test_data *data = tester_get_data(); const uint8_t *addr = hciemu_get_client_bdaddr(data->hciemu); struct step *step = g_new0(struct step, 1); bt_bdaddr_t bdaddr; bdaddr2android((const bdaddr_t *) addr, &bdaddr); step->action_status = data->if_a2dp->disconnect(&bdaddr); schedule_action_verification(step); } static void audio_resume_action(void) { struct test_data *data = tester_get_data(); struct step *step = g_new0(struct step, 1); int err; err = data->audio->open_output_stream(data->audio, 0, AUDIO_DEVICE_OUT_ALL_A2DP, AUDIO_OUTPUT_FLAG_NONE, NULL, &data->if_stream, NULL); if (err < 0) { step->action_status = BT_STATUS_FAIL; goto done; } /* Write something to force resume */ data->if_stream->write(data->if_stream, &err, sizeof(err)); done: schedule_action_verification(step); } static void audio_suspend_action(void) { struct test_data *data = tester_get_data(); struct step *step = g_new0(struct step, 1); data->if_stream->common.standby(&data->if_stream->common); schedule_action_verification(step); } static struct test_case test_cases[] = { TEST_CASE_BREDRLE("A2DP Init", ACTION_SUCCESS(dummy_action, NULL), ), TEST_CASE_BREDRLE("A2DP Connect - Success", ACTION_SUCCESS(set_default_ssp_request_handler, NULL), ACTION_SUCCESS(bluetooth_enable_action, NULL), CALLBACK_STATE(CB_BT_ADAPTER_STATE_CHANGED, BT_STATE_ON), ACTION_SUCCESS(emu_setup_powered_remote_action, NULL), ACTION_SUCCESS(emu_set_ssp_mode_action, NULL), ACTION_SUCCESS(emu_add_l2cap_server_action, &l2cap_setup_data), ACTION_SUCCESS(a2dp_connect_action, NULL), CALLBACK_AV_CONN_STATE(CB_A2DP_CONN_STATE, BTAV_CONNECTION_STATE_CONNECTING), CALLBACK_AV_CONN_STATE(CB_A2DP_CONN_STATE, BTAV_CONNECTION_STATE_CONNECTED), ACTION_SUCCESS(bluetooth_disable_action, NULL), CALLBACK_AV_CONN_STATE(CB_A2DP_CONN_STATE, BTAV_CONNECTION_STATE_DISCONNECTED), CALLBACK_STATE(CB_BT_ADAPTER_STATE_CHANGED, BT_STATE_OFF), ), TEST_CASE_BREDRLE("A2DP Disconnect - Success", ACTION_SUCCESS(set_default_ssp_request_handler, NULL), ACTION_SUCCESS(bluetooth_enable_action, NULL), CALLBACK_STATE(CB_BT_ADAPTER_STATE_CHANGED, BT_STATE_ON), ACTION_SUCCESS(emu_setup_powered_remote_action, NULL), ACTION_SUCCESS(emu_set_ssp_mode_action, NULL), ACTION_SUCCESS(emu_add_l2cap_server_action, &l2cap_setup_data), ACTION_SUCCESS(a2dp_connect_action, NULL), CALLBACK_AV_CONN_STATE(CB_A2DP_CONN_STATE, BTAV_CONNECTION_STATE_CONNECTING), CALLBACK_AV_CONN_STATE(CB_A2DP_CONN_STATE, BTAV_CONNECTION_STATE_CONNECTED), ACTION_SUCCESS(a2dp_disconnect_action, NULL), CALLBACK_AV_CONN_STATE(CB_A2DP_CONN_STATE, BTAV_CONNECTION_STATE_DISCONNECTING), CALLBACK_AV_CONN_STATE(CB_A2DP_CONN_STATE, BTAV_CONNECTION_STATE_DISCONNECTED), ACTION_SUCCESS(bluetooth_disable_action, NULL), CALLBACK_STATE(CB_BT_ADAPTER_STATE_CHANGED, BT_STATE_OFF), ), TEST_CASE_BREDRLE("A2DP Resume - Success", ACTION_SUCCESS(set_default_ssp_request_handler, NULL), ACTION_SUCCESS(bluetooth_enable_action, NULL), CALLBACK_STATE(CB_BT_ADAPTER_STATE_CHANGED, BT_STATE_ON), ACTION_SUCCESS(emu_setup_powered_remote_action, NULL), ACTION_SUCCESS(emu_set_ssp_mode_action, NULL), ACTION_SUCCESS(emu_add_l2cap_server_action, &l2cap_setup_data), ACTION_SUCCESS(a2dp_connect_action, NULL), CALLBACK_AV_CONN_STATE(CB_A2DP_CONN_STATE, BTAV_CONNECTION_STATE_CONNECTING), CALLBACK_AV_CONN_STATE(CB_A2DP_CONN_STATE, BTAV_CONNECTION_STATE_CONNECTED), ACTION_SUCCESS(audio_resume_action, NULL), CALLBACK_AV_AUDIO_STATE(CB_A2DP_AUDIO_STATE, BTAV_AUDIO_STATE_STARTED), ACTION_SUCCESS(bluetooth_disable_action, NULL), CALLBACK_AV_CONN_STATE(CB_A2DP_CONN_STATE, BTAV_CONNECTION_STATE_DISCONNECTED), CALLBACK_STATE(CB_BT_ADAPTER_STATE_CHANGED, BT_STATE_OFF), ), TEST_CASE_BREDRLE("A2DP Suspend - Success", ACTION_SUCCESS(set_default_ssp_request_handler, NULL), ACTION_SUCCESS(bluetooth_enable_action, NULL), CALLBACK_STATE(CB_BT_ADAPTER_STATE_CHANGED, BT_STATE_ON), ACTION_SUCCESS(emu_setup_powered_remote_action, NULL), ACTION_SUCCESS(emu_set_ssp_mode_action, NULL), ACTION_SUCCESS(emu_add_l2cap_server_action, &l2cap_setup_data), ACTION_SUCCESS(a2dp_connect_action, NULL), CALLBACK_AV_CONN_STATE(CB_A2DP_CONN_STATE, BTAV_CONNECTION_STATE_CONNECTING), CALLBACK_AV_CONN_STATE(CB_A2DP_CONN_STATE, BTAV_CONNECTION_STATE_CONNECTED), ACTION_SUCCESS(audio_resume_action, NULL), CALLBACK_AV_AUDIO_STATE(CB_A2DP_AUDIO_STATE, BTAV_AUDIO_STATE_STARTED), ACTION_SUCCESS(audio_suspend_action, NULL), CALLBACK_AV_AUDIO_STATE(CB_A2DP_AUDIO_STATE, BTAV_AUDIO_STATE_STOPPED), ACTION_SUCCESS(bluetooth_disable_action, NULL), CALLBACK_AV_CONN_STATE(CB_A2DP_CONN_STATE, BTAV_CONNECTION_STATE_DISCONNECTED), CALLBACK_STATE(CB_BT_ADAPTER_STATE_CHANGED, BT_STATE_OFF), ), }; struct queue *get_a2dp_tests(void) { uint16_t i = 0; list = queue_new(); for (; i < sizeof(test_cases) / sizeof(test_cases[0]); ++i) queue_push_tail(list, &test_cases[i]); return list; } void remove_a2dp_tests(void) { queue_destroy(list, NULL); }