18ef7100fSMilanka Ringwald /* 28ef7100fSMilanka Ringwald * Copyright (C) 2016 BlueKitchen GmbH 38ef7100fSMilanka Ringwald * 48ef7100fSMilanka Ringwald * Redistribution and use in source and binary forms, with or without 58ef7100fSMilanka Ringwald * modification, are permitted provided that the following conditions 68ef7100fSMilanka Ringwald * are met: 78ef7100fSMilanka Ringwald * 88ef7100fSMilanka Ringwald * 1. Redistributions of source code must retain the above copyright 98ef7100fSMilanka Ringwald * notice, this list of conditions and the following disclaimer. 108ef7100fSMilanka Ringwald * 2. Redistributions in binary form must reproduce the above copyright 118ef7100fSMilanka Ringwald * notice, this list of conditions and the following disclaimer in the 128ef7100fSMilanka Ringwald * documentation and/or other materials provided with the distribution. 138ef7100fSMilanka Ringwald * 3. Neither the name of the copyright holders nor the names of 148ef7100fSMilanka Ringwald * contributors may be used to endorse or promote products derived 158ef7100fSMilanka Ringwald * from this software without specific prior written permission. 168ef7100fSMilanka Ringwald * 4. Any redistribution, use, or modification is done solely for 178ef7100fSMilanka Ringwald * personal benefit and not for any commercial purpose or for 188ef7100fSMilanka Ringwald * monetary gain. 198ef7100fSMilanka Ringwald * 208ef7100fSMilanka Ringwald * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS 218ef7100fSMilanka Ringwald * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 228ef7100fSMilanka Ringwald * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 232fca4dadSMilanka Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL BLUEKITCHEN 242fca4dadSMilanka Ringwald * GMBH OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, 258ef7100fSMilanka Ringwald * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 268ef7100fSMilanka Ringwald * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS 278ef7100fSMilanka Ringwald * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 288ef7100fSMilanka Ringwald * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 298ef7100fSMilanka Ringwald * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF 308ef7100fSMilanka Ringwald * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 318ef7100fSMilanka Ringwald * SUCH DAMAGE. 328ef7100fSMilanka Ringwald * 338ef7100fSMilanka Ringwald * Please inquire about commercial licensing options at 348ef7100fSMilanka Ringwald * [email protected] 358ef7100fSMilanka Ringwald * 368ef7100fSMilanka Ringwald */ 37e501bae0SMatthias Ringwald #define BTSTACK_FILE__ "avdtp_acceptor.c" 388ef7100fSMilanka Ringwald 398ef7100fSMilanka Ringwald #include <stdint.h> 408ef7100fSMilanka Ringwald #include <string.h> 418ef7100fSMilanka Ringwald 420da182f5SMilanka Ringwald #include "classic/avdtp.h" 430da182f5SMilanka Ringwald #include "classic/avdtp_util.h" 440da182f5SMilanka Ringwald #include "classic/avdtp_acceptor.h" 458ef7100fSMilanka Ringwald 467c76cd61SMatthias Ringwald #include "btstack_debug.h" 477c76cd61SMatthias Ringwald #include "btstack_util.h" 487c76cd61SMatthias Ringwald #include "l2cap.h" 497c76cd61SMatthias Ringwald 508ef7100fSMilanka Ringwald 518ef7100fSMilanka Ringwald static int avdtp_acceptor_send_accept_response(uint16_t cid, uint8_t transaction_label, avdtp_signal_identifier_t identifier){ 528ef7100fSMilanka Ringwald uint8_t command[2]; 538ef7100fSMilanka Ringwald command[0] = avdtp_header(transaction_label, AVDTP_SINGLE_PACKET, AVDTP_RESPONSE_ACCEPT_MSG); 548ef7100fSMilanka Ringwald command[1] = (uint8_t)identifier; 558ef7100fSMilanka Ringwald return l2cap_send(cid, command, sizeof(command)); 568ef7100fSMilanka Ringwald } 578ef7100fSMilanka Ringwald 5830c0b5e7SMatthias Ringwald // returns true if command complete 5930c0b5e7SMatthias Ringwald static bool avdtp_acceptor_process_chunk(avdtp_signaling_packet_t * signaling_packet, uint8_t * packet, uint16_t size){ 6017796dfbSMatthias Ringwald if ((signaling_packet->size + size) >= sizeof(signaling_packet->command)) { 6130c0b5e7SMatthias Ringwald log_info("Dropping incoming data, doesn't fit into command buffer"); 6230c0b5e7SMatthias Ringwald signaling_packet->size = 0; 6330c0b5e7SMatthias Ringwald return false; 6430c0b5e7SMatthias Ringwald } 6530c0b5e7SMatthias Ringwald 6630c0b5e7SMatthias Ringwald (void)memcpy(signaling_packet->command + signaling_packet->size, packet, size); 678ef7100fSMilanka Ringwald signaling_packet->size += size; 680e588213SMatthias Ringwald return (signaling_packet->packet_type == AVDTP_SINGLE_PACKET) || (signaling_packet->packet_type == AVDTP_END_PACKET); 698ef7100fSMilanka Ringwald } 708ef7100fSMilanka Ringwald 71*fa4f109bSMatthias Ringwald static int avdtp_acceptor_validate_msg_length(avdtp_signal_identifier_t signal_identifier, uint16_t header_size, uint16_t msg_size){ 72*fa4f109bSMatthias Ringwald // payload length is num bytes after header (incl. signaling identifier), e.g. 1 for Get Capabilities 73*fa4f109bSMatthias Ringwald int minimal_payload_length; 748ef7100fSMilanka Ringwald switch (signal_identifier){ 758ef7100fSMilanka Ringwald case AVDTP_SI_GET_CAPABILITIES: 768ef7100fSMilanka Ringwald case AVDTP_SI_GET_ALL_CAPABILITIES: 778ef7100fSMilanka Ringwald case AVDTP_SI_GET_CONFIGURATION: 78*fa4f109bSMatthias Ringwald case AVDTP_SI_RECONFIGURE: 79*fa4f109bSMatthias Ringwald case AVDTP_SI_OPEN: 808ef7100fSMilanka Ringwald case AVDTP_SI_START: 818ef7100fSMilanka Ringwald case AVDTP_SI_CLOSE: 828ef7100fSMilanka Ringwald case AVDTP_SI_ABORT: 83*fa4f109bSMatthias Ringwald case AVDTP_SI_SECURITY_CONTROL: 84*fa4f109bSMatthias Ringwald minimal_payload_length = 1; 858ef7100fSMilanka Ringwald break; 86*fa4f109bSMatthias Ringwald case AVDTP_SI_SET_CONFIGURATION: 87*fa4f109bSMatthias Ringwald minimal_payload_length = 2; 88*fa4f109bSMatthias Ringwald break; 89*fa4f109bSMatthias Ringwald case AVDTP_SI_DELAYREPORT: 90*fa4f109bSMatthias Ringwald minimal_payload_length = 3; 91*fa4f109bSMatthias Ringwald break; 92*fa4f109bSMatthias Ringwald case AVDTP_SI_DISCOVER: 938ef7100fSMilanka Ringwald default: 94*fa4f109bSMatthias Ringwald minimal_payload_length = 0; 958ef7100fSMilanka Ringwald break; 968ef7100fSMilanka Ringwald } 97*fa4f109bSMatthias Ringwald return msg_size >= (header_size + minimal_payload_length); 988ef7100fSMilanka Ringwald } 998ef7100fSMilanka Ringwald 10077092f3eSMatthias Ringwald static void 10177092f3eSMatthias Ringwald avdtp_acceptor_handle_configuration_command(avdtp_connection_t *connection, int offset, uint16_t packet_size, avdtp_stream_endpoint_t *stream_endpoint) { 1027344f9afSMilanka Ringwald log_info("W2_ANSWER_SET_CONFIGURATION cid 0x%02x", connection->avdtp_cid); 103a3ce0109SMatthias Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_CONFIGURATION_SUBSTATEMACHINE; 104a3ce0109SMatthias Ringwald stream_endpoint->connection = connection; 1051ef2d533SMatthias Ringwald 1061ef2d533SMatthias Ringwald // process capabilities, first rejected service category is stored in connection 1071ef2d533SMatthias Ringwald connection->reject_service_category = 0; 108a3ce0109SMatthias Ringwald avdtp_sep_t sep; 10950453b92SMatthias Ringwald sep.seid = connection->acceptor_signaling_packet.command[offset++] >> 2; 11050453b92SMatthias Ringwald sep.configured_service_categories = avdtp_unpack_service_capabilities(connection, connection->acceptor_signaling_packet.signal_identifier, &sep.configuration, connection->acceptor_signaling_packet.command+offset, packet_size-offset); 111a3ce0109SMatthias Ringwald sep.in_use = 1; 112a3ce0109SMatthias Ringwald 1131114e158SMatthias Ringwald // test if sep already in use 1141114e158SMatthias Ringwald if (stream_endpoint->sep.in_use != 0){ 1151114e158SMatthias Ringwald log_info("stream endpoint already in use"); 1161114e158SMatthias Ringwald connection->error_code = AVDTP_ERROR_CODE_SEP_IN_USE; 1171114e158SMatthias Ringwald connection->reject_service_category = 0; 1181114e158SMatthias Ringwald } 1191114e158SMatthias Ringwald 1201ef2d533SMatthias Ringwald // let application validate media configuration as well 1211ef2d533SMatthias Ringwald if (connection->error_code == 0){ 1221ef2d533SMatthias Ringwald if ((sep.configured_service_categories & (1 << AVDTP_MEDIA_CODEC)) != 0){ 123c70720c6SMatthias Ringwald const adtvp_media_codec_capabilities_t * media = &sep.configuration.media_codec; 124c70720c6SMatthias Ringwald uint8_t error_code = avdtp_validate_media_configuration(stream_endpoint, connection->avdtp_cid, 0, media); 1251ef2d533SMatthias Ringwald if (error_code != 0){ 1261ef2d533SMatthias Ringwald log_info("media codec rejected by validator, error 0x%02x", error_code); 1271ef2d533SMatthias Ringwald connection->reject_service_category = AVDTP_MEDIA_CODEC; 1281ef2d533SMatthias Ringwald connection->error_code = error_code; 1291ef2d533SMatthias Ringwald } 1301ef2d533SMatthias Ringwald } 1311ef2d533SMatthias Ringwald } 1321ef2d533SMatthias Ringwald 133a3ce0109SMatthias Ringwald if (connection->error_code){ 13450453b92SMatthias Ringwald connection->reject_signal_identifier = connection->acceptor_signaling_packet.signal_identifier; 135a3ce0109SMatthias Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_REJECT_CATEGORY_WITH_ERROR_CODE; 136a3ce0109SMatthias Ringwald return; 137a3ce0109SMatthias Ringwald } 1381ef2d533SMatthias Ringwald 139cdf930c3SMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_ACCEPT_SET_CONFIGURATION; 140a3ce0109SMatthias Ringwald // find or add sep 141a3ce0109SMatthias Ringwald 1427344f9afSMilanka Ringwald log_info("local seid %d, remote seid %d", connection->acceptor_local_seid, sep.seid); 143a3ce0109SMatthias Ringwald 144a3ce0109SMatthias Ringwald if (is_avdtp_remote_seid_registered(stream_endpoint)){ 145a3ce0109SMatthias Ringwald if (stream_endpoint->remote_sep.in_use){ 1467344f9afSMilanka Ringwald log_info("remote seid already in use"); 1475b391ed7SMatthias Ringwald connection->error_code = AVDTP_ERROR_CODE_SEP_IN_USE; 148a3ce0109SMatthias Ringwald // find first registered category and fire the error 149a3ce0109SMatthias Ringwald connection->reject_service_category = 0; 150a3ce0109SMatthias Ringwald int i; 151a3ce0109SMatthias Ringwald for (i = 1; i < 9; i++){ 152a3ce0109SMatthias Ringwald if (get_bit16(sep.configured_service_categories, i)){ 153a3ce0109SMatthias Ringwald connection->reject_service_category = i; 154a3ce0109SMatthias Ringwald break; 155a3ce0109SMatthias Ringwald } 156a3ce0109SMatthias Ringwald } 15750453b92SMatthias Ringwald connection->reject_signal_identifier = connection->acceptor_signaling_packet.signal_identifier; 158a3ce0109SMatthias Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_REJECT_CATEGORY_WITH_ERROR_CODE; 1590dbf0854SMatthias Ringwald return; 160a3ce0109SMatthias Ringwald } else { 161a3ce0109SMatthias Ringwald stream_endpoint->remote_sep = sep; 1627344f9afSMilanka Ringwald log_info("update remote seid %d", stream_endpoint->remote_sep.seid); 163a3ce0109SMatthias Ringwald } 164a3ce0109SMatthias Ringwald } else { 165a3ce0109SMatthias Ringwald // add new 166a3ce0109SMatthias Ringwald stream_endpoint->remote_sep = sep; 1677344f9afSMilanka Ringwald log_info("add remote seid %d", stream_endpoint->remote_sep.seid); 168a3ce0109SMatthias Ringwald } 169a3ce0109SMatthias Ringwald 1701114e158SMatthias Ringwald // mark as in_use 1711114e158SMatthias Ringwald stream_endpoint->sep.in_use = 1; 1721114e158SMatthias Ringwald 17382767773SMatthias Ringwald // if media codec configuration set, copy configuration and emit event 17482767773SMatthias Ringwald if ((sep.configured_service_categories & (1 << AVDTP_MEDIA_CODEC)) != 0){ 17582767773SMatthias Ringwald if (stream_endpoint->media_codec_configuration_len == sep.configuration.media_codec.media_codec_information_len){ 17682767773SMatthias Ringwald (void) memcpy(stream_endpoint->media_codec_configuration_info, sep.configuration.media_codec.media_codec_information, stream_endpoint->media_codec_configuration_len); 177096e6469SMatthias Ringwald // update media codec info to point to user configuration 178096e6469SMatthias Ringwald stream_endpoint->remote_sep.configuration.media_codec.media_codec_information = stream_endpoint->media_codec_configuration_info; 179096e6469SMatthias Ringwald // emit event 1800d5e852dSMatthias Ringwald avdtp_signaling_emit_configuration(stream_endpoint, connection->avdtp_cid, 0, &sep.configuration, sep.configured_service_categories); 18182767773SMatthias Ringwald } 18282767773SMatthias Ringwald } 18382767773SMatthias Ringwald 184c69f4ba5SMatthias Ringwald avdtp_signaling_emit_accept(connection->avdtp_cid, avdtp_local_seid(stream_endpoint), 185c69f4ba5SMatthias Ringwald connection->acceptor_signaling_packet.signal_identifier, false); 186a3ce0109SMatthias Ringwald } 187a3ce0109SMatthias Ringwald 18877092f3eSMatthias Ringwald void avdtp_acceptor_stream_config_subsm(avdtp_connection_t *connection, uint8_t *packet, uint16_t size, int offset) { 1892d9d000fSMatthias Ringwald avdtp_stream_endpoint_t * stream_endpoint = NULL; 19050453b92SMatthias Ringwald connection->acceptor_transaction_label = connection->acceptor_signaling_packet.transaction_label; 191*fa4f109bSMatthias Ringwald if (!avdtp_acceptor_validate_msg_length(connection->acceptor_signaling_packet.signal_identifier, offset, size)) { 1925b391ed7SMatthias Ringwald connection->error_code = AVDTP_ERROR_CODE_BAD_LENGTH; 1938ef7100fSMilanka Ringwald connection->acceptor_connection_state = AVDTP_SIGNALING_CONNECTION_ACCEPTOR_W2_REJECT_WITH_ERROR_CODE; 19450453b92SMatthias Ringwald connection->reject_signal_identifier = connection->acceptor_signaling_packet.signal_identifier; 195d80ccd43SMatthias Ringwald avdtp_request_can_send_now_acceptor(connection); 1968ef7100fSMilanka Ringwald return; 1978ef7100fSMilanka Ringwald } 1988ef7100fSMilanka Ringwald 19932c78405SMilanka Ringwald // handle error cases 20050453b92SMatthias Ringwald switch (connection->acceptor_signaling_packet.signal_identifier){ 2018ef7100fSMilanka Ringwald case AVDTP_SI_DISCOVER: 2028ef7100fSMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) return; 2037344f9afSMilanka Ringwald log_info("W2_ANSWER_DISCOVER_SEPS"); 2048ef7100fSMilanka Ringwald connection->acceptor_connection_state = AVDTP_SIGNALING_CONNECTION_ACCEPTOR_W2_ANSWER_DISCOVER_SEPS; 205d80ccd43SMatthias Ringwald avdtp_request_can_send_now_acceptor(connection); 2068ef7100fSMilanka Ringwald return; 2078ef7100fSMilanka Ringwald case AVDTP_SI_GET_CAPABILITIES: 2088ef7100fSMilanka Ringwald case AVDTP_SI_GET_ALL_CAPABILITIES: 2098ef7100fSMilanka Ringwald case AVDTP_SI_SET_CONFIGURATION: 2108ef7100fSMilanka Ringwald case AVDTP_SI_GET_CONFIGURATION: 2118ef7100fSMilanka Ringwald case AVDTP_SI_START: 2128ef7100fSMilanka Ringwald case AVDTP_SI_CLOSE: 2138ef7100fSMilanka Ringwald case AVDTP_SI_ABORT: 2148ef7100fSMilanka Ringwald case AVDTP_SI_OPEN: 2158ef7100fSMilanka Ringwald case AVDTP_SI_RECONFIGURE: 21632c78405SMilanka Ringwald case AVDTP_SI_DELAYREPORT: 2175bd73fa2SMatthias Ringwald connection->acceptor_local_seid = packet[offset++] >> 2; 2183338afc0SMatthias Ringwald stream_endpoint = avdtp_get_stream_endpoint_for_seid(connection->acceptor_local_seid); 2198ef7100fSMilanka Ringwald if (!stream_endpoint){ 2207344f9afSMilanka Ringwald log_info("cmd %d - REJECT", connection->acceptor_signaling_packet.signal_identifier); 2215b391ed7SMatthias Ringwald connection->error_code = AVDTP_ERROR_CODE_BAD_ACP_SEID; 22250453b92SMatthias Ringwald if (connection->acceptor_signaling_packet.signal_identifier == AVDTP_SI_OPEN){ 2235b391ed7SMatthias Ringwald connection->error_code = AVDTP_ERROR_CODE_BAD_STATE; 2248ef7100fSMilanka Ringwald } 2258ef7100fSMilanka Ringwald 2268ef7100fSMilanka Ringwald connection->acceptor_connection_state = AVDTP_SIGNALING_CONNECTION_ACCEPTOR_W2_REJECT_WITH_ERROR_CODE; 22750453b92SMatthias Ringwald if (connection->acceptor_signaling_packet.signal_identifier == AVDTP_SI_RECONFIGURE){ 2285bd73fa2SMatthias Ringwald connection->reject_service_category = connection->acceptor_local_seid; 2298ef7100fSMilanka Ringwald connection->acceptor_connection_state = AVDTP_SIGNALING_CONNECTION_ACCEPTOR_W2_REJECT_CATEGORY_WITH_ERROR_CODE; 2308ef7100fSMilanka Ringwald } 23150453b92SMatthias Ringwald connection->reject_signal_identifier = connection->acceptor_signaling_packet.signal_identifier; 232d80ccd43SMatthias Ringwald avdtp_request_can_send_now_acceptor(connection); 2338ef7100fSMilanka Ringwald return; 2348ef7100fSMilanka Ringwald } 2358ef7100fSMilanka Ringwald break; 2368ef7100fSMilanka Ringwald 2378ef7100fSMilanka Ringwald case AVDTP_SI_SUSPEND:{ 2388ef7100fSMilanka Ringwald int i; 2397344f9afSMilanka Ringwald log_info("AVDTP_SI_SUSPEND"); 2408ef7100fSMilanka Ringwald connection->num_suspended_seids = 0; 2418ef7100fSMilanka Ringwald 2428ef7100fSMilanka Ringwald for (i = offset; i < size; i++){ 2438ef7100fSMilanka Ringwald connection->suspended_seids[connection->num_suspended_seids] = packet[i] >> 2; 2448ef7100fSMilanka Ringwald offset++; 2458587e32cSMilanka Ringwald log_info("%d, ", connection->suspended_seids[connection->num_suspended_seids]); 2468ef7100fSMilanka Ringwald connection->num_suspended_seids++; 2478ef7100fSMilanka Ringwald } 2488ef7100fSMilanka Ringwald 2498ef7100fSMilanka Ringwald if (connection->num_suspended_seids == 0) { 2507344f9afSMilanka Ringwald log_info("no susspended seids, BAD_ACP_SEID"); 2515b391ed7SMatthias Ringwald connection->error_code = AVDTP_ERROR_CODE_BAD_ACP_SEID; 2525bd73fa2SMatthias Ringwald connection->reject_service_category = connection->acceptor_local_seid; 2538ef7100fSMilanka Ringwald connection->acceptor_connection_state = AVDTP_SIGNALING_CONNECTION_ACCEPTOR_W2_REJECT_CATEGORY_WITH_ERROR_CODE; 25450453b92SMatthias Ringwald connection->reject_signal_identifier = connection->acceptor_signaling_packet.signal_identifier; 255d80ccd43SMatthias Ringwald avdtp_request_can_send_now_acceptor(connection); 2568ef7100fSMilanka Ringwald return; 2578ef7100fSMilanka Ringwald } 2588ef7100fSMilanka Ringwald // deal with first susspended seid 2595bd73fa2SMatthias Ringwald connection->acceptor_local_seid = connection->suspended_seids[0]; 2603338afc0SMatthias Ringwald stream_endpoint = avdtp_get_stream_endpoint_for_seid(connection->acceptor_local_seid); 2618ef7100fSMilanka Ringwald if (!stream_endpoint){ 2627344f9afSMilanka Ringwald log_info("stream_endpoint not found, BAD_ACP_SEID"); 2635b391ed7SMatthias Ringwald connection->error_code = AVDTP_ERROR_CODE_BAD_ACP_SEID; 2645bd73fa2SMatthias Ringwald connection->reject_service_category = connection->acceptor_local_seid; 2658ef7100fSMilanka Ringwald connection->acceptor_connection_state = AVDTP_SIGNALING_CONNECTION_ACCEPTOR_W2_REJECT_CATEGORY_WITH_ERROR_CODE; 26650453b92SMatthias Ringwald connection->reject_signal_identifier = connection->acceptor_signaling_packet.signal_identifier; 2678ef7100fSMilanka Ringwald connection->num_suspended_seids = 0; 268d80ccd43SMatthias Ringwald avdtp_request_can_send_now_acceptor(connection); 2698ef7100fSMilanka Ringwald return; 2708ef7100fSMilanka Ringwald } 2718ef7100fSMilanka Ringwald break; 2728ef7100fSMilanka Ringwald } 2738ef7100fSMilanka Ringwald default: 2748ef7100fSMilanka Ringwald connection->acceptor_connection_state = AVDTP_SIGNALING_CONNECTION_ACCEPTOR_W2_GENERAL_REJECT_WITH_ERROR_CODE; 27550453b92SMatthias Ringwald connection->reject_signal_identifier = connection->acceptor_signaling_packet.signal_identifier; 27650453b92SMatthias Ringwald log_info("AVDTP_CMD_MSG signal %d not implemented, general reject", connection->acceptor_signaling_packet.signal_identifier); 277d80ccd43SMatthias Ringwald avdtp_request_can_send_now_acceptor(connection); 2788ef7100fSMilanka Ringwald return; 2798ef7100fSMilanka Ringwald } 2808ef7100fSMilanka Ringwald 2812d9d000fSMatthias Ringwald btstack_assert(stream_endpoint != NULL); 2828ef7100fSMilanka Ringwald 28330c0b5e7SMatthias Ringwald bool command_complete = avdtp_acceptor_process_chunk(&connection->acceptor_signaling_packet, packet, size); 28430c0b5e7SMatthias Ringwald if (!command_complete) return; 2858ef7100fSMilanka Ringwald 28650453b92SMatthias Ringwald uint16_t packet_size = connection->acceptor_signaling_packet.size; 28750453b92SMatthias Ringwald connection->acceptor_signaling_packet.size = 0; 2888ef7100fSMilanka Ringwald 2898ef7100fSMilanka Ringwald int request_to_send = 1; 2908ef7100fSMilanka Ringwald switch (stream_endpoint->acceptor_config_state){ 2918ef7100fSMilanka Ringwald case AVDTP_ACCEPTOR_STREAM_CONFIG_IDLE: 29250453b92SMatthias Ringwald switch (connection->acceptor_signaling_packet.signal_identifier){ 29332c78405SMilanka Ringwald case AVDTP_SI_DELAYREPORT: 2947344f9afSMilanka Ringwald log_info("W2_ANSWER_DELAY_REPORT, local seid %d", connection->acceptor_local_seid); 295cdf930c3SMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_ACCEPT_DELAY_REPORT; 2962d9d000fSMatthias Ringwald avdtp_signaling_emit_delay(connection->avdtp_cid, connection->acceptor_local_seid, 2972d9d000fSMatthias Ringwald big_endian_read_16(packet, offset)); 29832c78405SMilanka Ringwald break; 29932c78405SMilanka Ringwald 3008ef7100fSMilanka Ringwald case AVDTP_SI_GET_ALL_CAPABILITIES: 3017344f9afSMilanka Ringwald log_info("AVDTP_SI_GET_ALL_CAPABILITIES, local seid %d", connection->acceptor_local_seid); 302cdf930c3SMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_ACCEPT_GET_ALL_CAPABILITIES; 3038ef7100fSMilanka Ringwald break; 3048ef7100fSMilanka Ringwald case AVDTP_SI_GET_CAPABILITIES: 3057344f9afSMilanka Ringwald log_info("W2_ANSWER_GET_CAPABILITIES, local seid %d", connection->acceptor_local_seid); 306cdf930c3SMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_ACCEPT_GET_CAPABILITIES; 3078ef7100fSMilanka Ringwald break; 3088ef7100fSMilanka Ringwald case AVDTP_SI_SET_CONFIGURATION:{ 3099da6c473SMilanka Ringwald log_info("Received SET_CONFIGURATION cmd: config state %d", connection->configuration_state); 310a3ce0109SMatthias Ringwald switch (connection->configuration_state){ 311a3ce0109SMatthias Ringwald case AVDTP_CONFIGURATION_STATE_IDLE: 31277092f3eSMatthias Ringwald avdtp_acceptor_handle_configuration_command(connection, offset, packet_size, 31377092f3eSMatthias Ringwald stream_endpoint); 314a3ce0109SMatthias Ringwald connection->configuration_state = AVDTP_CONFIGURATION_STATE_REMOTE_INITIATED; 315a3ce0109SMatthias Ringwald break; 316a3ce0109SMatthias Ringwald case AVDTP_CONFIGURATION_STATE_LOCAL_INITIATED: 317b6370429SMilanka Ringwald case AVDTP_CONFIGURATION_STATE_REMOTE_INITIATED: 3187344f9afSMilanka Ringwald log_info("Reject SET_CONFIGURATION BAD_STATE %d", connection->configuration_state); 31950453b92SMatthias Ringwald connection->reject_signal_identifier = connection->acceptor_signaling_packet.signal_identifier; 3203b74a3bbSMatthias Ringwald connection->reject_service_category = 0; 3215b391ed7SMatthias Ringwald connection->error_code = AVDTP_ERROR_CODE_BAD_STATE; 3223b74a3bbSMatthias Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_REJECT_CATEGORY_WITH_ERROR_CODE; 323a3ce0109SMatthias Ringwald break; 324b6370429SMilanka Ringwald case AVDTP_CONFIGURATION_STATE_LOCAL_CONFIGURED: 325b6370429SMilanka Ringwald case AVDTP_CONFIGURATION_STATE_REMOTE_CONFIGURED: 326b6370429SMilanka Ringwald log_info("Reject SET_CONFIGURATION SEP_IN_USE"); 327b6370429SMilanka Ringwald connection->reject_signal_identifier = connection->acceptor_signaling_packet.signal_identifier; 328b6370429SMilanka Ringwald connection->reject_service_category = 0; 3295b391ed7SMatthias Ringwald connection->error_code = AVDTP_ERROR_CODE_SEP_IN_USE; 330b6370429SMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_REJECT_CATEGORY_WITH_ERROR_CODE; 331b6370429SMilanka Ringwald break; 332a3ce0109SMatthias Ringwald default: 33363331bf4SMilanka Ringwald break; 33463331bf4SMilanka Ringwald } 3358ef7100fSMilanka Ringwald break; 3368ef7100fSMilanka Ringwald } 3378ef7100fSMilanka Ringwald case AVDTP_SI_RECONFIGURE:{ 338cdf930c3SMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_ACCEPT_RECONFIGURE; 3398ef7100fSMilanka Ringwald connection->reject_service_category = 0; 3408ef7100fSMilanka Ringwald 3418ef7100fSMilanka Ringwald avdtp_sep_t sep; 3427344f9afSMilanka Ringwald log_info("W2_ANSWER_RECONFIGURE, local seid %d, remote seid %d", connection->acceptor_local_seid, stream_endpoint->remote_sep.seid); 34350453b92SMatthias Ringwald sep.configured_service_categories = avdtp_unpack_service_capabilities(connection, connection->acceptor_signaling_packet.signal_identifier, &sep.configuration, connection->acceptor_signaling_packet.command+offset, packet_size-offset); 3448ef7100fSMilanka Ringwald if (connection->error_code){ 3458ef7100fSMilanka Ringwald // fire configuration parsing errors 34650453b92SMatthias Ringwald connection->reject_signal_identifier = connection->acceptor_signaling_packet.signal_identifier; 3478ef7100fSMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_REJECT_CATEGORY_WITH_ERROR_CODE; 3488ef7100fSMilanka Ringwald break; 3498ef7100fSMilanka Ringwald } 3508ef7100fSMilanka Ringwald 3518ef7100fSMilanka Ringwald // find sep or raise error 352485c0a4cSMilanka Ringwald if (!is_avdtp_remote_seid_registered(stream_endpoint)){ 3537344f9afSMilanka Ringwald log_info("REJECT AVDTP_SI_RECONFIGURE, BAD_ACP_SEID"); 3548ef7100fSMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_REJECT_CATEGORY_WITH_ERROR_CODE; 3555b391ed7SMatthias Ringwald connection->error_code = AVDTP_ERROR_CODE_BAD_ACP_SEID; 35650453b92SMatthias Ringwald connection->reject_signal_identifier = connection->acceptor_signaling_packet.signal_identifier; 3578ef7100fSMilanka Ringwald break; 3588ef7100fSMilanka Ringwald } 359b12cd5a6SMilanka Ringwald stream_endpoint->remote_sep.configured_service_categories = sep.configured_service_categories; 360b12cd5a6SMilanka Ringwald stream_endpoint->remote_sep.configuration = sep.configuration; 361b12cd5a6SMilanka Ringwald 3627344f9afSMilanka Ringwald log_info("update active remote seid %d", stream_endpoint->remote_sep.seid); 3638ef7100fSMilanka Ringwald 36482767773SMatthias Ringwald // if media codec configuration updated, copy configuration and emit event 36582767773SMatthias Ringwald if ((sep.configured_service_categories & (1 << AVDTP_MEDIA_CODEC)) != 0){ 36682767773SMatthias Ringwald if (stream_endpoint->media_codec_configuration_len == sep.configuration.media_codec.media_codec_information_len){ 36782767773SMatthias Ringwald (void) memcpy(stream_endpoint->media_codec_configuration_info, sep.configuration.media_codec.media_codec_information, stream_endpoint->media_codec_configuration_len); 3680d5e852dSMatthias Ringwald stream_endpoint->sep.configuration.media_codec = stream_endpoint->remote_configuration.media_codec; 3696c5b303cSMatthias Ringwald avdtp_signaling_emit_configuration(stream_endpoint, connection->avdtp_cid, 1, &sep.configuration, sep.configured_service_categories); 37082767773SMatthias Ringwald } 3710d5e852dSMatthias Ringwald } 3728ef7100fSMilanka Ringwald break; 3738ef7100fSMilanka Ringwald } 3748ef7100fSMilanka Ringwald 3758ef7100fSMilanka Ringwald case AVDTP_SI_GET_CONFIGURATION: 3767344f9afSMilanka Ringwald log_info("W2_ANSWER_GET_CONFIGURATION"); 377cdf930c3SMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_ACCEPT_GET_CONFIGURATION; 3788ef7100fSMilanka Ringwald break; 3798ef7100fSMilanka Ringwald case AVDTP_SI_OPEN: 3808ef7100fSMilanka Ringwald if (stream_endpoint->state != AVDTP_STREAM_ENDPOINT_CONFIGURED){ 3817344f9afSMilanka Ringwald log_info("REJECT AVDTP_SI_OPEN, BAD_STATE"); 3828ef7100fSMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_REJECT_WITH_ERROR_CODE; 3835b391ed7SMatthias Ringwald connection->error_code = AVDTP_ERROR_CODE_BAD_STATE; 38450453b92SMatthias Ringwald connection->reject_signal_identifier = connection->acceptor_signaling_packet.signal_identifier; 3858ef7100fSMilanka Ringwald break; 3868ef7100fSMilanka Ringwald } 3877344f9afSMilanka Ringwald log_info("AVDTP_STREAM_ENDPOINT_W2_ANSWER_OPEN_STREAM"); 388cdf930c3SMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_ACCEPT_OPEN_STREAM; 3898ef7100fSMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_W4_L2CAP_FOR_MEDIA_CONNECTED; 3905bd73fa2SMatthias Ringwald connection->acceptor_local_seid = stream_endpoint->sep.seid; 3918ef7100fSMilanka Ringwald break; 3928ef7100fSMilanka Ringwald case AVDTP_SI_START: 3938ef7100fSMilanka Ringwald if (stream_endpoint->state != AVDTP_STREAM_ENDPOINT_OPENED){ 3947344f9afSMilanka Ringwald log_info("REJECT AVDTP_SI_START, BAD_STATE, state %d", stream_endpoint->state); 3958ef7100fSMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_REJECT_CATEGORY_WITH_ERROR_CODE; 3965b391ed7SMatthias Ringwald connection->error_code = AVDTP_ERROR_CODE_BAD_STATE; 39750453b92SMatthias Ringwald connection->reject_signal_identifier = connection->acceptor_signaling_packet.signal_identifier; 3988ef7100fSMilanka Ringwald break; 3998ef7100fSMilanka Ringwald } 40048ce193cSMilanka Ringwald #ifdef ENABLE_AVDTP_ACCEPTOR_EXPLICIT_START_STREAM_CONFIRMATION 40148ce193cSMilanka Ringwald log_info("W2_ACCEPT_START_STREAM"); 40248ce193cSMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W4_USER_CONFIRM_START_STREAM; 40348ce193cSMilanka Ringwald #else 4047344f9afSMilanka Ringwald log_info("W2_ANSWER_START_STREAM"); 40548ce193cSMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_ACCEPT_START_STREAM; 40648ce193cSMilanka Ringwald #endif 4078ef7100fSMilanka Ringwald break; 4088ef7100fSMilanka Ringwald case AVDTP_SI_CLOSE: 4098ef7100fSMilanka Ringwald switch (stream_endpoint->state){ 4108ef7100fSMilanka Ringwald case AVDTP_STREAM_ENDPOINT_OPENED: 4118ef7100fSMilanka Ringwald case AVDTP_STREAM_ENDPOINT_STREAMING: 4127344f9afSMilanka Ringwald log_info("W2_ANSWER_CLOSE_STREAM"); 4138ef7100fSMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_CLOSING; 414cdf930c3SMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_ACCEPT_CLOSE_STREAM; 4158ef7100fSMilanka Ringwald break; 4168ef7100fSMilanka Ringwald default: 4177344f9afSMilanka Ringwald log_info("AVDTP_SI_CLOSE, bad state %d ", stream_endpoint->state); 4188ef7100fSMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_REJECT_WITH_ERROR_CODE; 4195b391ed7SMatthias Ringwald connection->error_code = AVDTP_ERROR_CODE_BAD_STATE; 42050453b92SMatthias Ringwald connection->reject_signal_identifier = connection->acceptor_signaling_packet.signal_identifier; 4218ef7100fSMilanka Ringwald break; 4228ef7100fSMilanka Ringwald } 4238ef7100fSMilanka Ringwald break; 4248ef7100fSMilanka Ringwald case AVDTP_SI_ABORT: 4258ef7100fSMilanka Ringwald switch (stream_endpoint->state){ 4268ef7100fSMilanka Ringwald case AVDTP_STREAM_ENDPOINT_CONFIGURED: 4278ef7100fSMilanka Ringwald case AVDTP_STREAM_ENDPOINT_CLOSING: 4288ef7100fSMilanka Ringwald case AVDTP_STREAM_ENDPOINT_OPENED: 4298ef7100fSMilanka Ringwald case AVDTP_STREAM_ENDPOINT_STREAMING: 4307344f9afSMilanka Ringwald log_info("W2_ANSWER_ABORT_STREAM"); 4318ef7100fSMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_ABORTING; 432cdf930c3SMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_ACCEPT_ABORT_STREAM; 4338ef7100fSMilanka Ringwald break; 4348ef7100fSMilanka Ringwald default: 4357344f9afSMilanka Ringwald log_info("AVDTP_SI_ABORT, bad state %d ", stream_endpoint->state); 4368ef7100fSMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_REJECT_WITH_ERROR_CODE; 4375b391ed7SMatthias Ringwald connection->error_code = AVDTP_ERROR_CODE_BAD_STATE; 43850453b92SMatthias Ringwald connection->reject_signal_identifier = connection->acceptor_signaling_packet.signal_identifier; 4398ef7100fSMilanka Ringwald break; 4408ef7100fSMilanka Ringwald } 4418ef7100fSMilanka Ringwald break; 4428ef7100fSMilanka Ringwald case AVDTP_SI_SUSPEND: 4438ef7100fSMilanka Ringwald switch (stream_endpoint->state){ 4448ef7100fSMilanka Ringwald case AVDTP_STREAM_ENDPOINT_STREAMING: 4458ef7100fSMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_OPENED; 4468ef7100fSMilanka Ringwald connection->num_suspended_seids--; 4478ef7100fSMilanka Ringwald if (connection->num_suspended_seids <= 0){ 4487344f9afSMilanka Ringwald log_info("W2_ANSWER_SUSPEND_STREAM"); 449cdf930c3SMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_ACCEPT_SUSPEND_STREAM; 4508ef7100fSMilanka Ringwald } 4518ef7100fSMilanka Ringwald break; 4528ef7100fSMilanka Ringwald default: 4537344f9afSMilanka Ringwald log_info("AVDTP_SI_SUSPEND, bad state %d", stream_endpoint->state); 4548ef7100fSMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_REJECT_CATEGORY_WITH_ERROR_CODE; 4555b391ed7SMatthias Ringwald connection->error_code = AVDTP_ERROR_CODE_BAD_STATE; 45650453b92SMatthias Ringwald connection->reject_signal_identifier = connection->acceptor_signaling_packet.signal_identifier; 4578ef7100fSMilanka Ringwald break; 4588ef7100fSMilanka Ringwald } 4598ef7100fSMilanka Ringwald break; 4608ef7100fSMilanka Ringwald default: 4617344f9afSMilanka Ringwald log_info("NOT IMPLEMENTED, Reject signal_identifier %02x", connection->acceptor_signaling_packet.signal_identifier); 4628ef7100fSMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W2_REJECT_UNKNOWN_CMD; 46350453b92SMatthias Ringwald connection->reject_signal_identifier = connection->acceptor_signaling_packet.signal_identifier; 4648ef7100fSMilanka Ringwald break; 4658ef7100fSMilanka Ringwald } 4668ef7100fSMilanka Ringwald break; 4678ef7100fSMilanka Ringwald default: 4688ef7100fSMilanka Ringwald return; 4698ef7100fSMilanka Ringwald } 4708ef7100fSMilanka Ringwald 4718ef7100fSMilanka Ringwald if (!request_to_send){ 4727344f9afSMilanka Ringwald log_info("NOT IMPLEMENTED"); 4738ef7100fSMilanka Ringwald } 474d80ccd43SMatthias Ringwald avdtp_request_can_send_now_acceptor(connection); 4758ef7100fSMilanka Ringwald } 4768ef7100fSMilanka Ringwald 4778ef7100fSMilanka Ringwald static int avdtp_acceptor_send_seps_response(uint16_t cid, uint8_t transaction_label, avdtp_stream_endpoint_t * endpoints){ 47867ae582dSMilanka Ringwald uint8_t command[2+2*AVDTP_MAX_NUM_SEPS]; 4798ef7100fSMilanka Ringwald int pos = 0; 4808ef7100fSMilanka Ringwald command[pos++] = avdtp_header(transaction_label, AVDTP_SINGLE_PACKET, AVDTP_RESPONSE_ACCEPT_MSG); 4818ef7100fSMilanka Ringwald command[pos++] = (uint8_t)AVDTP_SI_DISCOVER; 4828ef7100fSMilanka Ringwald 4838ef7100fSMilanka Ringwald btstack_linked_list_iterator_t it; 4848ef7100fSMilanka Ringwald btstack_linked_list_iterator_init(&it, (btstack_linked_list_t *) endpoints); 4858ef7100fSMilanka Ringwald while (btstack_linked_list_iterator_has_next(&it)){ 4868ef7100fSMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = (avdtp_stream_endpoint_t *)btstack_linked_list_iterator_next(&it); 4878ef7100fSMilanka Ringwald command[pos++] = (stream_endpoint->sep.seid << 2) | (stream_endpoint->sep.in_use<<1); 4888ef7100fSMilanka Ringwald command[pos++] = (stream_endpoint->sep.media_type << 4) | (stream_endpoint->sep.type << 3); 4898ef7100fSMilanka Ringwald } 4908ef7100fSMilanka Ringwald return l2cap_send(cid, command, pos); 4918ef7100fSMilanka Ringwald } 4928ef7100fSMilanka Ringwald 4938ef7100fSMilanka Ringwald static int avdtp_acceptor_send_response_reject_service_category(uint16_t cid, avdtp_signal_identifier_t identifier, uint8_t category, uint8_t error_code, uint8_t transaction_label){ 4948ef7100fSMilanka Ringwald uint8_t command[4]; 4958ef7100fSMilanka Ringwald command[0] = avdtp_header(transaction_label, AVDTP_SINGLE_PACKET, AVDTP_RESPONSE_REJECT_MSG); 4968ef7100fSMilanka Ringwald command[1] = (uint8_t)identifier; 4978ef7100fSMilanka Ringwald command[2] = category; 4988ef7100fSMilanka Ringwald command[3] = error_code; 4998ef7100fSMilanka Ringwald return l2cap_send(cid, command, sizeof(command)); 5008ef7100fSMilanka Ringwald } 5018ef7100fSMilanka Ringwald 5028ef7100fSMilanka Ringwald static int avdtp_acceptor_send_response_general_reject(uint16_t cid, avdtp_signal_identifier_t identifier, uint8_t transaction_label){ 5038ef7100fSMilanka Ringwald uint8_t command[2]; 5048ef7100fSMilanka Ringwald command[0] = avdtp_header(transaction_label, AVDTP_SINGLE_PACKET, AVDTP_GENERAL_REJECT_MSG); 5058ef7100fSMilanka Ringwald command[1] = (uint8_t)identifier; 5068ef7100fSMilanka Ringwald return l2cap_send(cid, command, sizeof(command)); 5078ef7100fSMilanka Ringwald } 5088ef7100fSMilanka Ringwald 5098ef7100fSMilanka Ringwald static int avdtp_acceptor_send_response_reject(uint16_t cid, avdtp_signal_identifier_t identifier, uint8_t transaction_label){ 5108ef7100fSMilanka Ringwald uint8_t command[2]; 5118ef7100fSMilanka Ringwald command[0] = avdtp_header(transaction_label, AVDTP_SINGLE_PACKET, AVDTP_RESPONSE_REJECT_MSG); 5128ef7100fSMilanka Ringwald command[1] = (uint8_t)identifier; 5138ef7100fSMilanka Ringwald return l2cap_send(cid, command, sizeof(command)); 5148ef7100fSMilanka Ringwald } 5158ef7100fSMilanka Ringwald 5168ef7100fSMilanka Ringwald static int avdtp_acceptor_send_response_reject_with_error_code(uint16_t cid, avdtp_signal_identifier_t identifier, uint8_t error_code, uint8_t transaction_label){ 5178ef7100fSMilanka Ringwald uint8_t command[3]; 5188ef7100fSMilanka Ringwald command[0] = avdtp_header(transaction_label, AVDTP_SINGLE_PACKET, AVDTP_RESPONSE_REJECT_MSG); 5198ef7100fSMilanka Ringwald command[1] = (uint8_t)identifier; 5208ef7100fSMilanka Ringwald command[2] = error_code; 5218ef7100fSMilanka Ringwald return l2cap_send(cid, command, sizeof(command)); 5228ef7100fSMilanka Ringwald } 5238ef7100fSMilanka Ringwald 52477092f3eSMatthias Ringwald void avdtp_acceptor_stream_config_subsm_run(avdtp_connection_t *connection) { 5258ef7100fSMilanka Ringwald int sent = 1; 526d8e15394SMilanka Ringwald btstack_linked_list_t * stream_endpoints = avdtp_get_stream_endpoints(); 5278ef7100fSMilanka Ringwald 5288ef7100fSMilanka Ringwald switch (connection->acceptor_connection_state){ 5298ef7100fSMilanka Ringwald case AVDTP_SIGNALING_CONNECTION_ACCEPTOR_W2_ANSWER_DISCOVER_SEPS: 5308ef7100fSMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_OPENED; 5318ef7100fSMilanka Ringwald connection->acceptor_connection_state = AVDTP_SIGNALING_CONNECTION_ACCEPTOR_IDLE; 5329413b167SMilanka Ringwald avdtp_acceptor_send_seps_response(connection->l2cap_signaling_cid, connection->acceptor_transaction_label, (avdtp_stream_endpoint_t *) stream_endpoints); 53302fa22bcSMatthias Ringwald avdtp_signaling_emit_accept(connection->avdtp_cid, 0, connection->acceptor_signaling_packet.signal_identifier, false); 5348ef7100fSMilanka Ringwald break; 5358ef7100fSMilanka Ringwald case AVDTP_SIGNALING_CONNECTION_ACCEPTOR_W2_REJECT_WITH_ERROR_CODE: 5368ef7100fSMilanka Ringwald connection->acceptor_connection_state = AVDTP_SIGNALING_CONNECTION_ACCEPTOR_IDLE; 5379413b167SMilanka Ringwald avdtp_acceptor_send_response_reject_with_error_code(connection->l2cap_signaling_cid, connection->reject_signal_identifier, connection->error_code, connection->acceptor_transaction_label); 5388ef7100fSMilanka Ringwald break; 5398ef7100fSMilanka Ringwald case AVDTP_SIGNALING_CONNECTION_ACCEPTOR_W2_REJECT_CATEGORY_WITH_ERROR_CODE: 5408ef7100fSMilanka Ringwald connection->acceptor_connection_state = AVDTP_SIGNALING_CONNECTION_ACCEPTOR_IDLE; 5419413b167SMilanka Ringwald avdtp_acceptor_send_response_reject_service_category(connection->l2cap_signaling_cid, connection->reject_signal_identifier, connection->reject_service_category, connection->error_code, connection->acceptor_transaction_label); 5428ef7100fSMilanka Ringwald break; 5438ef7100fSMilanka Ringwald case AVDTP_SIGNALING_CONNECTION_ACCEPTOR_W2_GENERAL_REJECT_WITH_ERROR_CODE: 5448ef7100fSMilanka Ringwald connection->acceptor_connection_state = AVDTP_SIGNALING_CONNECTION_ACCEPTOR_IDLE; 5459413b167SMilanka Ringwald avdtp_acceptor_send_response_general_reject(connection->l2cap_signaling_cid, connection->reject_signal_identifier, connection->acceptor_transaction_label); 546b5281692SMatthias Ringwald break; 5478ef7100fSMilanka Ringwald default: 5488ef7100fSMilanka Ringwald sent = 0; 5498ef7100fSMilanka Ringwald break; 5508ef7100fSMilanka Ringwald } 551747ec646SMilanka Ringwald if (sent){ 5527344f9afSMilanka Ringwald log_info("DONE"); 553747ec646SMilanka Ringwald return; 554747ec646SMilanka Ringwald } 5558ef7100fSMilanka Ringwald 556d8e15394SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_get_stream_endpoint_for_seid(connection->acceptor_local_seid); 5578ef7100fSMilanka Ringwald if (!stream_endpoint) return; 5588ef7100fSMilanka Ringwald 5598ef7100fSMilanka Ringwald uint8_t reject_service_category = connection->reject_service_category; 5608ef7100fSMilanka Ringwald avdtp_signal_identifier_t reject_signal_identifier = connection->reject_signal_identifier; 5618ef7100fSMilanka Ringwald uint8_t error_code = connection->error_code; 5629413b167SMilanka Ringwald uint16_t cid = stream_endpoint->connection ? stream_endpoint->connection->l2cap_signaling_cid : connection->l2cap_signaling_cid; 5638ef7100fSMilanka Ringwald uint8_t trid = stream_endpoint->connection ? stream_endpoint->connection->acceptor_transaction_label : connection->acceptor_transaction_label; 5648ef7100fSMilanka Ringwald 5658ef7100fSMilanka Ringwald avdtp_acceptor_stream_endpoint_state_t acceptor_config_state = stream_endpoint->acceptor_config_state; 5668ef7100fSMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_STREAM_CONFIG_IDLE; 5678ef7100fSMilanka Ringwald uint8_t * out_buffer; 5688ef7100fSMilanka Ringwald uint16_t pos; 569d7af755fSMatthias Ringwald avdtp_sep_t * remote_sep; 5708ef7100fSMilanka Ringwald 571b88e4826SMilanka Ringwald bool emit_accept = false; 572b88e4826SMilanka Ringwald bool emit_reject = false; 573b88e4826SMilanka Ringwald 5748ef7100fSMilanka Ringwald switch (acceptor_config_state){ 5758ef7100fSMilanka Ringwald case AVDTP_ACCEPTOR_STREAM_CONFIG_IDLE: 5768ef7100fSMilanka Ringwald break; 577cdf930c3SMilanka Ringwald case AVDTP_ACCEPTOR_W2_ACCEPT_GET_CAPABILITIES: 57850453b92SMatthias Ringwald avdtp_prepare_capabilities(&connection->acceptor_signaling_packet, trid, stream_endpoint->sep.registered_service_categories, stream_endpoint->sep.capabilities, AVDTP_SI_GET_CAPABILITIES); 5798ef7100fSMilanka Ringwald l2cap_reserve_packet_buffer(); 5808ef7100fSMilanka Ringwald out_buffer = l2cap_get_outgoing_buffer(); 58150453b92SMatthias Ringwald pos = avdtp_signaling_create_fragment(cid, &connection->acceptor_signaling_packet, out_buffer); 58250453b92SMatthias Ringwald if ((connection->acceptor_signaling_packet.packet_type != AVDTP_SINGLE_PACKET) && (connection->acceptor_signaling_packet.packet_type != AVDTP_END_PACKET)){ 5838ef7100fSMilanka Ringwald stream_endpoint->acceptor_config_state = acceptor_config_state; 5847344f9afSMilanka Ringwald log_info("fragmented"); 585747ec646SMilanka Ringwald } else { 5868587e32cSMilanka Ringwald log_info("ACP:DONE"); 587b88e4826SMilanka Ringwald emit_accept = true; 5888ef7100fSMilanka Ringwald } 5898ef7100fSMilanka Ringwald l2cap_send_prepared(cid, pos); 5908ef7100fSMilanka Ringwald break; 591cdf930c3SMilanka Ringwald case AVDTP_ACCEPTOR_W2_ACCEPT_DELAY_REPORT: 5927344f9afSMilanka Ringwald log_info("DONE "); 59332c78405SMilanka Ringwald avdtp_acceptor_send_accept_response(cid, trid, AVDTP_SI_DELAYREPORT); 594b88e4826SMilanka Ringwald emit_accept = true; 59532c78405SMilanka Ringwald break; 596cdf930c3SMilanka Ringwald case AVDTP_ACCEPTOR_W2_ACCEPT_GET_ALL_CAPABILITIES: 59750453b92SMatthias Ringwald avdtp_prepare_capabilities(&connection->acceptor_signaling_packet, trid, stream_endpoint->sep.registered_service_categories, stream_endpoint->sep.capabilities, AVDTP_SI_GET_ALL_CAPABILITIES); 5988ef7100fSMilanka Ringwald l2cap_reserve_packet_buffer(); 5998ef7100fSMilanka Ringwald out_buffer = l2cap_get_outgoing_buffer(); 60050453b92SMatthias Ringwald pos = avdtp_signaling_create_fragment(cid, &connection->acceptor_signaling_packet, out_buffer); 60150453b92SMatthias Ringwald if ((connection->acceptor_signaling_packet.packet_type != AVDTP_SINGLE_PACKET) && (connection->acceptor_signaling_packet.packet_type != AVDTP_END_PACKET)){ 6028ef7100fSMilanka Ringwald stream_endpoint->acceptor_config_state = acceptor_config_state; 6037344f9afSMilanka Ringwald log_info("fragmented"); 604747ec646SMilanka Ringwald } else { 6058587e32cSMilanka Ringwald log_info("ACP:DONE"); 606b88e4826SMilanka Ringwald emit_accept = true; 6078ef7100fSMilanka Ringwald } 6088ef7100fSMilanka Ringwald l2cap_send_prepared(cid, pos); 6098ef7100fSMilanka Ringwald break; 610cdf930c3SMilanka Ringwald case AVDTP_ACCEPTOR_W2_ACCEPT_SET_CONFIGURATION: 6117344f9afSMilanka Ringwald log_info("DONE"); 6128587e32cSMilanka Ringwald log_info(" -> AVDTP_STREAM_ENDPOINT_CONFIGURED"); 6138ef7100fSMilanka Ringwald stream_endpoint->connection = connection; 6148ef7100fSMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_CONFIGURED; 615a3ce0109SMatthias Ringwald connection->configuration_state = AVDTP_CONFIGURATION_STATE_REMOTE_CONFIGURED; 616747ec646SMilanka Ringwald // TODO: consider reconfiguration 6178ef7100fSMilanka Ringwald avdtp_acceptor_send_accept_response(cid, trid, AVDTP_SI_SET_CONFIGURATION); 618b88e4826SMilanka Ringwald emit_accept = true; 6198ef7100fSMilanka Ringwald break; 620cdf930c3SMilanka Ringwald case AVDTP_ACCEPTOR_W2_ACCEPT_RECONFIGURE: 6217344f9afSMilanka Ringwald log_info("DONE "); 6228ef7100fSMilanka Ringwald avdtp_acceptor_send_accept_response(cid, trid, AVDTP_SI_RECONFIGURE); 623b88e4826SMilanka Ringwald emit_accept = true; 6248ef7100fSMilanka Ringwald break; 625d7af755fSMatthias Ringwald case AVDTP_ACCEPTOR_W2_ACCEPT_GET_CONFIGURATION: 626d7af755fSMatthias Ringwald remote_sep = &stream_endpoint->remote_sep; 627d7af755fSMatthias Ringwald avdtp_prepare_capabilities(&connection->acceptor_signaling_packet, trid, remote_sep->configured_service_categories, remote_sep->configuration, AVDTP_SI_GET_CONFIGURATION); 6288ef7100fSMilanka Ringwald l2cap_reserve_packet_buffer(); 6298ef7100fSMilanka Ringwald out_buffer = l2cap_get_outgoing_buffer(); 63050453b92SMatthias Ringwald pos = avdtp_signaling_create_fragment(cid, &connection->acceptor_signaling_packet, out_buffer); 63150453b92SMatthias Ringwald if ((connection->acceptor_signaling_packet.packet_type != AVDTP_SINGLE_PACKET) && (connection->acceptor_signaling_packet.packet_type != AVDTP_END_PACKET)){ 6328ef7100fSMilanka Ringwald stream_endpoint->acceptor_config_state = acceptor_config_state; 6337344f9afSMilanka Ringwald log_info("fragmented"); 634747ec646SMilanka Ringwald } else { 6358587e32cSMilanka Ringwald log_info("ACP:DONE"); 636b88e4826SMilanka Ringwald emit_accept = true; 6378ef7100fSMilanka Ringwald } 6388ef7100fSMilanka Ringwald l2cap_send_prepared(cid, pos); 6398ef7100fSMilanka Ringwald break; 640cdf930c3SMilanka Ringwald case AVDTP_ACCEPTOR_W2_ACCEPT_OPEN_STREAM: 6417344f9afSMilanka Ringwald log_info("DONE"); 6428ef7100fSMilanka Ringwald avdtp_acceptor_send_accept_response(cid, trid, AVDTP_SI_OPEN); 643b88e4826SMilanka Ringwald emit_accept = true; 6448ef7100fSMilanka Ringwald break; 64548ce193cSMilanka Ringwald 64648ce193cSMilanka Ringwald #ifdef ENABLE_AVDTP_ACCEPTOR_EXPLICIT_START_STREAM_CONFIRMATION 64748ce193cSMilanka Ringwald case AVDTP_ACCEPTOR_W4_USER_CONFIRM_START_STREAM: 64848ce193cSMilanka Ringwald // keep state until user calls API to confirm or reject starting the stream 64948ce193cSMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_W4_USER_CONFIRM_START_STREAM; 65048ce193cSMilanka Ringwald avdtp_signaling_emit_accept(connection->avdtp_cid, avdtp_local_seid(stream_endpoint), AVDTP_SI_ACCEPT_START, false); 65148ce193cSMilanka Ringwald break; 65248ce193cSMilanka Ringwald case AVDTP_ACCEPTOR_W2_REJECT_START_STREAM: 65348ce193cSMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_OPENED; 65448ce193cSMilanka Ringwald connection->acceptor_signaling_packet.signal_identifier = AVDTP_SI_START; 65548ce193cSMilanka Ringwald emit_reject = true; 65648ce193cSMilanka Ringwald avdtp_acceptor_send_response_reject(cid, AVDTP_SI_START, trid); 65748ce193cSMilanka Ringwald break; 65848ce193cSMilanka Ringwald #endif 65948ce193cSMilanka Ringwald case AVDTP_ACCEPTOR_W2_ACCEPT_START_STREAM: 6607344f9afSMilanka Ringwald log_info("DONE "); 6618587e32cSMilanka Ringwald log_info(" -> AVDTP_STREAM_ENDPOINT_STREAMING "); 6628ef7100fSMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_STREAMING; 6638ef7100fSMilanka Ringwald avdtp_acceptor_send_accept_response(cid, trid, AVDTP_SI_START); 664b88e4826SMilanka Ringwald emit_accept = true; 6658ef7100fSMilanka Ringwald break; 666cdf930c3SMilanka Ringwald case AVDTP_ACCEPTOR_W2_ACCEPT_CLOSE_STREAM: 6677344f9afSMilanka Ringwald log_info("DONE"); 6688ef7100fSMilanka Ringwald avdtp_acceptor_send_accept_response(cid, trid, AVDTP_SI_CLOSE); 6699da6c473SMilanka Ringwald connection->configuration_state = AVDTP_CONFIGURATION_STATE_IDLE; 670b88e4826SMilanka Ringwald emit_accept = true; 6718ef7100fSMilanka Ringwald break; 672cdf930c3SMilanka Ringwald case AVDTP_ACCEPTOR_W2_ACCEPT_ABORT_STREAM: 6737344f9afSMilanka Ringwald log_info("DONE"); 6748ef7100fSMilanka Ringwald avdtp_acceptor_send_accept_response(cid, trid, AVDTP_SI_ABORT); 675b88e4826SMilanka Ringwald emit_accept = true; 6768ef7100fSMilanka Ringwald break; 677cdf930c3SMilanka Ringwald case AVDTP_ACCEPTOR_W2_ACCEPT_SUSPEND_STREAM: 6787344f9afSMilanka Ringwald log_info("DONE"); 6798ef7100fSMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_OPENED; 6808ef7100fSMilanka Ringwald avdtp_acceptor_send_accept_response(cid, trid, AVDTP_SI_SUSPEND); 681b88e4826SMilanka Ringwald emit_accept = true; 6828ef7100fSMilanka Ringwald break; 6838ef7100fSMilanka Ringwald case AVDTP_ACCEPTOR_W2_REJECT_UNKNOWN_CMD: 6847344f9afSMilanka Ringwald log_info("DONE REJECT"); 685b0920f25SMilanka Ringwald connection->reject_signal_identifier = AVDTP_SI_NONE; 6868ef7100fSMilanka Ringwald avdtp_acceptor_send_response_reject(cid, reject_signal_identifier, trid); 687b88e4826SMilanka Ringwald emit_reject = true; 6888ef7100fSMilanka Ringwald break; 6898ef7100fSMilanka Ringwald case AVDTP_ACCEPTOR_W2_REJECT_CATEGORY_WITH_ERROR_CODE: 6907344f9afSMilanka Ringwald log_info("DONE REJECT CATEGORY"); 6918ef7100fSMilanka Ringwald connection->reject_service_category = 0; 6928ef7100fSMilanka Ringwald avdtp_acceptor_send_response_reject_service_category(cid, reject_signal_identifier, reject_service_category, error_code, trid); 693b88e4826SMilanka Ringwald emit_reject = true; 6948ef7100fSMilanka Ringwald break; 6958ef7100fSMilanka Ringwald case AVDTP_ACCEPTOR_W2_REJECT_WITH_ERROR_CODE: 6967344f9afSMilanka Ringwald log_info("DONE REJECT"); 697b0920f25SMilanka Ringwald connection->reject_signal_identifier = AVDTP_SI_NONE; 6988ef7100fSMilanka Ringwald connection->error_code = 0; 6998ef7100fSMilanka Ringwald avdtp_acceptor_send_response_reject_with_error_code(cid, reject_signal_identifier, error_code, trid); 700b88e4826SMilanka Ringwald emit_reject = true; 7018ef7100fSMilanka Ringwald break; 7028ef7100fSMilanka Ringwald default: 7037344f9afSMilanka Ringwald log_info("NOT IMPLEMENTED"); 7048ef7100fSMilanka Ringwald sent = 0; 705319b19ccSMilanka Ringwald break; 7068ef7100fSMilanka Ringwald } 70732d799b8SMatthias Ringwald 708b88e4826SMilanka Ringwald if (emit_accept == true){ 709c69f4ba5SMatthias Ringwald avdtp_signaling_emit_accept(connection->avdtp_cid, avdtp_local_seid(stream_endpoint), 710c69f4ba5SMatthias Ringwald connection->acceptor_signaling_packet.signal_identifier, false); 711b88e4826SMilanka Ringwald } else if (emit_reject == true){ 712c69f4ba5SMatthias Ringwald avdtp_signaling_emit_reject(connection->avdtp_cid, avdtp_local_seid(stream_endpoint), 713c69f4ba5SMatthias Ringwald connection->acceptor_signaling_packet.signal_identifier, false); 714b88e4826SMilanka Ringwald } 7158ef7100fSMilanka Ringwald // check fragmentation 71650453b92SMatthias Ringwald if ((connection->acceptor_signaling_packet.packet_type != AVDTP_SINGLE_PACKET) && (connection->acceptor_signaling_packet.packet_type != AVDTP_END_PACKET)){ 717d80ccd43SMatthias Ringwald avdtp_request_can_send_now_acceptor(connection); 7188ef7100fSMilanka Ringwald } 7198ef7100fSMilanka Ringwald } 720