| /* |
| * Copyright (C) 2014 Intel Corporation |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| * |
| */ |
| |
| #include <stdbool.h> |
| |
| #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); |
| } |