/* * Copyright (C) 2014 by Ole Reinhardt * * 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 THE COPYRIGHT HOLDERS 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 THE * COPYRIGHT OWNER 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 btstack@ringwald.ch * */ /* * bnep.c */ #include #include #include // memcpy #include #include #include #include #include #include "btstack_memory.h" #include "hci.h" #include "hci_dump.h" #include "debug.h" #include "bnep.h" #include "l2cap.h" #define BNEP_CONNECTION_TIMEOUT_MS 10000 #define BNEP_CONNECTION_MAX_RETRIES 1 /*static*/ linked_list_t bnep_services = NULL; /*static*/ linked_list_t bnep_channels = NULL; /*static*/ gap_security_level_t bnep_security_level; static void (*app_packet_handler)(void * connection, uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size); static bnep_channel_t * bnep_channel_for_l2cap_cid(uint16_t l2cap_cid); static void bnep_channel_finalize(bnep_channel_t *channel); static void bnep_run(void); static void bnep_channel_start_timer(bnep_channel_t *channel, int timeout); /*inline*/ static void bnep_channel_state_add(bnep_channel_t *channel, BNEP_CHANNEL_STATE_VAR event); static void bnep_emit_open_channel_complete(bnep_channel_t *channel, uint8_t status) { uint8_t event[3 + sizeof(bd_addr_t) + 3 * sizeof(uint16_t)]; log_info("BNEP_EVENT_OPEN_CHANNEL_COMPLETE status 0x%02x bd_addr: %s, handler %p", status, bd_addr_to_str(channel->remote_addr), channel->packet_handler); if (!app_packet_handler) return; event[0] = BNEP_EVENT_OPEN_CHANNEL_COMPLETE; event[1] = sizeof(event) - 2; event[2] = status; little_endian_store_16(event, 3, channel->l2cap_cid); little_endian_store_16(event, 5, channel->uuid_source); little_endian_store_16(event, 7, channel->uuid_dest); little_endian_store_16(event, 9, channel->max_frame_size); BD_ADDR_COPY(&event[11], channel->remote_addr); hci_dump_packet( HCI_EVENT_PACKET, 0, event, sizeof(event)); (*app_packet_handler)(channel->connection, HCI_EVENT_PACKET, 0, (uint8_t *) event, sizeof(event)); } static void bnep_emit_channel_timeout(bnep_channel_t *channel) { uint8_t event[2 + sizeof(bd_addr_t) + 2 * sizeof(uint16_t) + sizeof(uint8_t)]; log_info("BNEP_EVENT_CHANNEL_TIMEOUT bd_addr: %s, handler %p", bd_addr_to_str(channel->remote_addr), channel->packet_handler); if (!app_packet_handler) return; event[0] = BNEP_EVENT_CHANNEL_TIMEOUT; event[1] = sizeof(event) - 2; little_endian_store_16(event, 2, channel->l2cap_cid); little_endian_store_16(event, 4, channel->uuid_source); little_endian_store_16(event, 6, channel->uuid_dest); BD_ADDR_COPY(&event[8], channel->remote_addr); event[14] = channel->state; hci_dump_packet( HCI_EVENT_PACKET, 0, event, sizeof(event)); (*app_packet_handler)(channel->connection, HCI_EVENT_PACKET, 0, (uint8_t *) event, sizeof(event)); } static void bnep_emit_channel_closed(bnep_channel_t *channel) { uint8_t event[2 + sizeof(bd_addr_t) + 3 * sizeof(uint16_t)]; log_info("BNEP_EVENT_CHANNEL_CLOSED bd_addr: %s, handler %p", bd_addr_to_str(channel->remote_addr)); if (!app_packet_handler) return; event[0] = BNEP_EVENT_CHANNEL_CLOSED; event[1] = sizeof(event) - 2; little_endian_store_16(event, 2, channel->l2cap_cid); little_endian_store_16(event, 4, channel->uuid_source); little_endian_store_16(event, 6, channel->uuid_dest); BD_ADDR_COPY(&event[8], channel->remote_addr); hci_dump_packet( HCI_EVENT_PACKET, 0, event, sizeof(event)); (*app_packet_handler)(channel->connection, HCI_EVENT_PACKET, channel->l2cap_cid, (uint8_t *) event, sizeof(event)); } static void bnep_emit_ready_to_send(bnep_channel_t *channel) { if (!app_packet_handler) return; uint8_t event[4]; event[0] = BNEP_EVENT_READY_TO_SEND; event[1] = sizeof(event) - 2; little_endian_store_16(event, 2, channel->l2cap_cid); hci_dump_packet( HCI_EVENT_PACKET, 0, event, sizeof(event)); (*app_packet_handler)(channel->connection, HCI_EVENT_PACKET, 0, (uint8_t *) event, sizeof(event)); } /* Send BNEP connection request */ static int bnep_send_command_not_understood(bnep_channel_t *channel, uint8_t control_type) { uint8_t *bnep_out_buffer = NULL; uint16_t pos = 0; int err = 0; if (channel->state == BNEP_CHANNEL_STATE_CLOSED) { return -1; // TODO } l2cap_reserve_packet_buffer(); bnep_out_buffer = l2cap_get_outgoing_buffer(); /* Setup control packet type */ bnep_out_buffer[pos++] = BNEP_PKT_TYPE_CONTROL; bnep_out_buffer[pos++] = BNEP_CONTROL_TYPE_COMMAND_NOT_UNDERSTOOD; /* Add not understood control type */ bnep_out_buffer[pos++] = control_type; err = l2cap_send_prepared(channel->l2cap_cid, pos); if (err) { // TODO: Log error } return err; } /* Send BNEP connection request */ static int bnep_send_connection_request(bnep_channel_t *channel, uint16_t uuid_source, uint16_t uuid_dest) { uint8_t *bnep_out_buffer = NULL; uint16_t pos = 0; int err = 0; if (channel->state == BNEP_CHANNEL_STATE_CLOSED) { return -1; // TODO } l2cap_reserve_packet_buffer(); bnep_out_buffer = l2cap_get_outgoing_buffer(); /* Setup control packet type */ bnep_out_buffer[pos++] = BNEP_PKT_TYPE_CONTROL; bnep_out_buffer[pos++] = BNEP_CONTROL_TYPE_SETUP_CONNECTION_REQUEST; /* Add UUID Size */ bnep_out_buffer[pos++] = 2; /* Add dest and source UUID */ net_store_16(bnep_out_buffer, pos, uuid_dest); pos += 2; net_store_16(bnep_out_buffer, pos, uuid_source); pos += 2; err = l2cap_send_prepared(channel->l2cap_cid, pos); if (err) { // TODO: Log error } return err; } /* Send BNEP connection response */ static int bnep_send_connection_response(bnep_channel_t *channel, uint16_t response_code) { uint8_t *bnep_out_buffer = NULL; uint16_t pos = 0; int err = 0; if (channel->state == BNEP_CHANNEL_STATE_CLOSED) { return -1; // TODO } l2cap_reserve_packet_buffer(); bnep_out_buffer = l2cap_get_outgoing_buffer(); /* Setup control packet type */ bnep_out_buffer[pos++] = BNEP_PKT_TYPE_CONTROL; bnep_out_buffer[pos++] = BNEP_CONTROL_TYPE_SETUP_CONNECTION_RESPONSE; /* Add response code */ net_store_16(bnep_out_buffer, pos, response_code); pos += 2; err = l2cap_send_prepared(channel->l2cap_cid, pos); if (err) { // TODO: Log error } return err; } /* Send BNEP filter net type set message */ static int bnep_send_filter_net_type_set(bnep_channel_t *channel, bnep_net_filter_t *filter, uint16_t len) { uint8_t *bnep_out_buffer = NULL; uint16_t pos = 0; int err = 0; int i; if (channel->state == BNEP_CHANNEL_STATE_CLOSED) { return -1; } l2cap_reserve_packet_buffer(); bnep_out_buffer = l2cap_get_outgoing_buffer(); /* Setup control packet type */ bnep_out_buffer[pos++] = BNEP_PKT_TYPE_CONTROL; bnep_out_buffer[pos++] = BNEP_CONTROL_TYPE_FILTER_NET_TYPE_SET; net_store_16(bnep_out_buffer, pos, len * 2 * 2); pos += 2; for (i = 0; i < len; i ++) { net_store_16(bnep_out_buffer, pos, filter[i].range_start); pos += 2; net_store_16(bnep_out_buffer, pos, filter[i].range_end); pos += 2; } err = l2cap_send_prepared(channel->l2cap_cid, pos); if (err) { // TODO: Log error } return err; } /* Send BNEP filter net type response message */ static int bnep_send_filter_net_type_response(bnep_channel_t *channel, uint16_t response_code) { uint8_t *bnep_out_buffer = NULL; uint16_t pos = 0; int err = 0; if (channel->state == BNEP_CHANNEL_STATE_CLOSED) { return -1; } l2cap_reserve_packet_buffer(); bnep_out_buffer = l2cap_get_outgoing_buffer(); /* Setup control packet type */ bnep_out_buffer[pos++] = BNEP_PKT_TYPE_CONTROL; bnep_out_buffer[pos++] = BNEP_CONTROL_TYPE_FILTER_NET_TYPE_RESPONSE; /* Add response code */ net_store_16(bnep_out_buffer, pos, response_code); pos += 2; err = l2cap_send_prepared(channel->l2cap_cid, pos); if (err) { // TODO: Log error } return err; } /* Send BNEP filter multicast address set message */ static int bnep_send_filter_multi_addr_set(bnep_channel_t *channel, bnep_multi_filter_t *filter, uint16_t len) { uint8_t *bnep_out_buffer = NULL; uint16_t pos = 0; int err = 0; int i; if (channel->state == BNEP_CHANNEL_STATE_CLOSED) { return -1; } l2cap_reserve_packet_buffer(); bnep_out_buffer = l2cap_get_outgoing_buffer(); /* Setup control packet type */ bnep_out_buffer[pos++] = BNEP_PKT_TYPE_CONTROL; bnep_out_buffer[pos++] = BNEP_CONTROL_TYPE_FILTER_MULTI_ADDR_SET; net_store_16(bnep_out_buffer, pos, len * 2 * ETHER_ADDR_LEN); pos += 2; for (i = 0; i < len; i ++) { BD_ADDR_COPY(bnep_out_buffer + pos, filter[i].addr_start); pos += ETHER_ADDR_LEN; BD_ADDR_COPY(bnep_out_buffer + pos, filter[i].addr_end); pos += ETHER_ADDR_LEN; } err = l2cap_send_prepared(channel->l2cap_cid, pos); if (err) { // TODO: Log error } return err; } /* Send BNEP filter multicast address response message */ static int bnep_send_filter_multi_addr_response(bnep_channel_t *channel, uint16_t response_code) { uint8_t *bnep_out_buffer = NULL; uint16_t pos = 0; int err = 0; if (channel->state == BNEP_CHANNEL_STATE_CLOSED) { return -1; } l2cap_reserve_packet_buffer(); bnep_out_buffer = l2cap_get_outgoing_buffer(); /* Setup control packet type */ bnep_out_buffer[pos++] = BNEP_PKT_TYPE_CONTROL; bnep_out_buffer[pos++] = BNEP_CONTROL_TYPE_FILTER_MULTI_ADDR_RESPONSE; /* Add response code */ net_store_16(bnep_out_buffer, pos, response_code); pos += 2; err = l2cap_send_prepared(channel->l2cap_cid, pos); if (err) { // TODO: Log error } return err; } int bnep_can_send_packet_now(uint16_t bnep_cid) { bnep_channel_t *channel = bnep_channel_for_l2cap_cid(bnep_cid); if (!channel){ log_error("bnep_can_send_packet_now cid 0x%02x doesn't exist!", bnep_cid); return 0; } return l2cap_can_send_packet_now(channel->l2cap_cid); } static int bnep_filter_protocol(bnep_channel_t *channel, uint16_t network_protocol_type) { int i; if (channel->net_filter_count == 0) { /* No filter set */ return 1; } for (i = 0; i < channel->net_filter_count; i ++) { if ((network_protocol_type >= channel->net_filter[i].range_start) && (network_protocol_type <= channel->net_filter[i].range_end)) { return 1; } } return 0; } static int bnep_filter_multicast(bnep_channel_t *channel, bd_addr_t addr_dest) { int i; /* Check if the multicast flag is set int the destination address */ if ((addr_dest[0] & 0x01) == 0x00) { /* Not a multicast frame, do not apply filtering and send it in any case */ return 1; } if (channel->multicast_filter_count == 0) { /* No filter set */ return 1; } for (i = 0; i < channel->multicast_filter_count; i ++) { if ((memcmp(addr_dest, channel->multicast_filter[i].addr_start, sizeof(bd_addr_t)) >= 0) && (memcmp(addr_dest, channel->multicast_filter[i].addr_end, sizeof(bd_addr_t)) <= 0)) { return 1; } } return 0; } /* Send BNEP ethernet packet */ int bnep_send(uint16_t bnep_cid, uint8_t *packet, uint16_t len) { bnep_channel_t *channel; uint8_t *bnep_out_buffer = NULL; uint16_t pos = 0; uint16_t pos_out = 0; uint16_t payload_len; int err = 0; int has_source; int has_dest; bd_addr_t addr_dest; bd_addr_t addr_source; uint16_t network_protocol_type; channel = bnep_channel_for_l2cap_cid(bnep_cid); if (channel == NULL) { log_error("bnep_send cid 0x%02x doesn't exist!", bnep_cid); return 1; } if (channel->state != BNEP_CHANNEL_STATE_CONNECTED) { return BNEP_CHANNEL_NOT_CONNECTED; } /* Check for free ACL buffers */ if (!l2cap_can_send_packet_now(channel->l2cap_cid)) { return BTSTACK_ACL_BUFFERS_FULL; } /* Extract destination and source address from the ethernet packet */ pos = 0; BD_ADDR_COPY(addr_dest, &packet[pos]); pos += sizeof(bd_addr_t); BD_ADDR_COPY(addr_source, &packet[pos]); pos += sizeof(bd_addr_t); network_protocol_type = big_endian_read_16(packet, pos); pos += sizeof(uint16_t); payload_len = len - pos; if (network_protocol_type == ETHERTYPE_VLAN) { /* IEEE 802.1Q tag header */ if (payload_len < 4) { /* Omit this packet */ return 0; } /* The "real" network protocol type is 4 bytes ahead in a VLAN packet */ network_protocol_type = big_endian_read_16(packet, pos + 2); } /* Check network protocol and multicast filters before sending */ if (!bnep_filter_protocol(channel, network_protocol_type) || !bnep_filter_multicast(channel, addr_dest)) { /* Packet did not pass filter... */ if ((network_protocol_type == ETHERTYPE_VLAN) && (payload_len >= 4)) { /* The packet has been tagged as a with IEE 802.1Q tag and has been filtered out. According to the spec the IEE802.1Q tag header shall be sended without ethernet payload. So limit the payload_len to 4. */ payload_len = 4; } else { /* Packet is not tagged with IEE802.1Q header and was filtered out. Omit this packet */ return 0; } } /* Reserve l2cap packet buffer */ l2cap_reserve_packet_buffer(); bnep_out_buffer = l2cap_get_outgoing_buffer(); /* Check if source address is the same as our local address and if the destination address is the same as the remote addr. Maybe we can use the compressed data format */ has_source = (memcmp(addr_source, channel->local_addr, ETHER_ADDR_LEN) != 0); has_dest = (memcmp(addr_dest, channel->remote_addr, ETHER_ADDR_LEN) != 0); /* Check for MTU limits */ if (payload_len > channel->max_frame_size) { log_error("bnep_send: Max frame size (%d) exceeded: %d", channel->max_frame_size, payload_len); return BNEP_DATA_LEN_EXCEEDS_MTU; } /* Fill in the package type depending on the given source and destination address */ if (has_source && has_dest) { bnep_out_buffer[pos_out++] = BNEP_PKT_TYPE_GENERAL_ETHERNET; } else if (has_source && !has_dest) { bnep_out_buffer[pos_out++] = BNEP_PKT_TYPE_COMPRESSED_ETHERNET_SOURCE_ONLY; } else if (!has_source && has_dest) { bnep_out_buffer[pos_out++] = BNEP_PKT_TYPE_COMPRESSED_ETHERNET_DEST_ONLY; } else { bnep_out_buffer[pos_out++] = BNEP_PKT_TYPE_COMPRESSED_ETHERNET; } /* Add the destination address if needed */ if (has_dest) { BD_ADDR_COPY(bnep_out_buffer + pos_out, addr_dest); pos_out += sizeof(bd_addr_t); } /* Add the source address if needed */ if (has_source) { BD_ADDR_COPY(bnep_out_buffer + pos_out, addr_source); pos_out += sizeof(bd_addr_t); } /* Add protocol type */ net_store_16(bnep_out_buffer, pos_out, network_protocol_type); pos_out += 2; /* TODO: Add extension headers, if we may support them at a later stage */ /* Add the payload and then send out the package */ memcpy(bnep_out_buffer + pos_out, packet + pos, payload_len); pos_out += payload_len; err = l2cap_send_prepared(channel->l2cap_cid, pos_out); if (err) { log_error("bnep_send: error %d", err); } return err; } /* Set BNEP network protocol type filter */ int bnep_set_net_type_filter(uint16_t bnep_cid, bnep_net_filter_t *filter, uint16_t len) { bnep_channel_t *channel; if (filter == NULL) { return -1; } channel = bnep_channel_for_l2cap_cid(bnep_cid); if (channel == NULL) { log_error("bnep_set_net_type_filter cid 0x%02x doesn't exist!", bnep_cid); return 1; } if (channel->state != BNEP_CHANNEL_STATE_CONNECTED) { return BNEP_CHANNEL_NOT_CONNECTED; } if (len > MAX_BNEP_NETFILTER_OUT) { return BNEP_DATA_LEN_EXCEEDS_MTU; } channel->net_filter_out = filter; channel->net_filter_out_count = len; /* Set flag to send out the network protocol type filter set reqeuest on next statemachine cycle */ bnep_channel_state_add(channel, BNEP_CHANNEL_STATE_VAR_SND_FILTER_NET_TYPE_SET); bnep_run(); return 0; } /* Set BNEP network protocol type filter */ int bnep_set_multicast_filter(uint16_t bnep_cid, bnep_multi_filter_t *filter, uint16_t len) { bnep_channel_t *channel; if (filter == NULL) { return -1; } channel = bnep_channel_for_l2cap_cid(bnep_cid); if (channel == NULL) { log_error("bnep_set_net_type_filter cid 0x%02x doesn't exist!", bnep_cid); return 1; } if (channel->state != BNEP_CHANNEL_STATE_CONNECTED) { return BNEP_CHANNEL_NOT_CONNECTED; } if (len > MAX_BNEP_MULTICAST_FULTER_OUT) { return BNEP_DATA_LEN_EXCEEDS_MTU; } channel->multicast_filter_out = filter; channel->multicast_filter_out_count = len; /* Set flag to send out the multicast filter set reqeuest on next statemachine cycle */ bnep_channel_state_add(channel, BNEP_CHANNEL_STATE_VAR_SND_FILTER_MULTI_ADDR_SET); bnep_run(); return 0; } /* BNEP timeout timer helper function */ static void bnep_channel_timer_handler(timer_source_t *timer) { bnep_channel_t *channel = (bnep_channel_t *)linked_item_get_user((linked_item_t *) timer); // retry send setup connection at least one time if (channel->state == BNEP_CHANNEL_STATE_WAIT_FOR_CONNECTION_RESPONSE){ if (channel->retry_count < BNEP_CONNECTION_MAX_RETRIES){ channel->retry_count++; bnep_channel_start_timer(channel, BNEP_CONNECTION_TIMEOUT_MS); bnep_channel_state_add(channel, BNEP_CHANNEL_STATE_VAR_SND_CONNECTION_REQUEST); bnep_run(); return; } } log_info( "bnep_channel_timeout_handler callback: shutting down connection!"); bnep_emit_channel_timeout(channel); bnep_channel_finalize(channel); } static void bnep_channel_stop_timer(bnep_channel_t *channel) { if (channel->timer_active) { run_loop_remove_timer(&channel->timer); channel->timer_active = 0; } } static void bnep_channel_start_timer(bnep_channel_t *channel, int timeout) { /* Stop any eventually running timeout timer */ bnep_channel_stop_timer(channel); /* Start bnep channel timeout check timer */ run_loop_set_timer(&channel->timer, timeout); channel->timer.process = bnep_channel_timer_handler; linked_item_set_user((linked_item_t*) &channel->timer, channel); run_loop_add_timer(&channel->timer); channel->timer_active = 1; } /* BNEP statemachine functions */ /*inline*/ static void bnep_channel_state_add(bnep_channel_t *channel, BNEP_CHANNEL_STATE_VAR event){ channel->state_var = (BNEP_CHANNEL_STATE_VAR) (channel->state_var | event); } /*inline*/ static void bnep_channel_state_remove(bnep_channel_t *channel, BNEP_CHANNEL_STATE_VAR event){ channel->state_var = (BNEP_CHANNEL_STATE_VAR) (channel->state_var & ~event); } static uint16_t bnep_max_frame_size_for_l2cap_mtu(uint16_t l2cap_mtu){ /* Assume a standard BNEP header, containing BNEP Type (1 Byte), dest and source address (6 bytes each) and networking protocol type (2 bytes) */ uint16_t max_frame_size = l2cap_mtu - 15; // 15 bytes BNEP header log_info("bnep_max_frame_size_for_l2cap_mtu: %u -> %u", l2cap_mtu, max_frame_size); return max_frame_size; } static bnep_channel_t * bnep_channel_create_for_addr(bd_addr_t *addr) { /* Allocate new channel structure */ bnep_channel_t *channel = btstack_memory_bnep_channel_get(); if (!channel) { return NULL; } /* Initialize the channel struct */ memset(channel, 0, sizeof(bnep_channel_t)); channel->state = BNEP_CHANNEL_STATE_CLOSED; channel->max_frame_size = bnep_max_frame_size_for_l2cap_mtu(l2cap_max_mtu()); BD_ADDR_COPY(&channel->remote_addr, addr); BD_ADDR_COPY(&channel->local_addr, hci_local_bd_addr()); channel->net_filter_count = 0; channel->multicast_filter_count = 0; channel->retry_count = 0; /* Finally add it to the channel list */ linked_list_add(&bnep_channels, (linked_item_t *) channel); return channel; } static bnep_channel_t* bnep_channel_for_addr(bd_addr_t *addr) { linked_item_t *it; for (it = (linked_item_t *) bnep_channels; it ; it = it->next){ bnep_channel_t *channel = ((bnep_channel_t *) it); if (BD_ADDR_CMP(addr, channel->remote_addr) == 0) { return channel; } } return NULL; } static bnep_channel_t * bnep_channel_for_l2cap_cid(uint16_t l2cap_cid) { linked_item_t *it; for (it = (linked_item_t *) bnep_channels; it ; it = it->next){ bnep_channel_t *channel = ((bnep_channel_t *) it); if (channel->l2cap_cid == l2cap_cid) { return channel; } } return NULL; } static bnep_service_t * bnep_service_for_uuid(uint16_t uuid) { linked_item_t *it; for (it = (linked_item_t *) bnep_services; it ; it = it->next){ bnep_service_t * service = ((bnep_service_t *) it); if ( service->service_uuid == uuid){ return service; } } return NULL; } static void bnep_channel_free(bnep_channel_t *channel) { linked_list_remove( &bnep_channels, (linked_item_t *) channel); btstack_memory_bnep_channel_free(channel); } static void bnep_channel_finalize(bnep_channel_t *channel) { uint16_t l2cap_cid; /* Inform application about closed channel */ if (channel->state == BNEP_CHANNEL_STATE_CONNECTED) { bnep_emit_channel_closed(channel); } l2cap_cid = channel->l2cap_cid; /* Stop any eventually running timer */ bnep_channel_stop_timer(channel); /* Free ressources and then close the l2cap channel */ bnep_channel_free(channel); l2cap_disconnect_internal(l2cap_cid, 0x13); } static int bnep_handle_connection_request(bnep_channel_t *channel, uint8_t *packet, uint16_t size) { uint16_t uuid_size; uint16_t uuid_offset; uint16_t response_code; bnep_service_t * service; uuid_size = packet[1]; response_code = BNEP_RESP_SETUP_SUCCESS; /* Sanity check packet size */ if (size < 1 + 1 + 2 * uuid_size) { return 0; } if ((channel->state != BNEP_CHANNEL_STATE_WAIT_FOR_CONNECTION_REQUEST) && (channel->state != BNEP_CHANNEL_STATE_CONNECTED)) { /* Ignore a connection request if not waiting for or still connected */ log_error("BNEP_CONNECTION_REQUEST: ignored in state %d, l2cap_cid: %d!", channel->state, channel->l2cap_cid); return 0; } /* Extract source and destination UUID and convert them to UUID16 format */ switch (uuid_size) { case 2: /* UUID16 */ uuid_offset = 0; break; case 4: /* UUID32 */ case 16: /* UUID128 */ uuid_offset = 2; break; default: log_error("BNEP_CONNECTION_REQUEST: Invalid UUID size %d, l2cap_cid: %d!", channel->state, channel->l2cap_cid); response_code = BNEP_RESP_SETUP_INVALID_SERVICE_UUID_SIZE; break; } /* Check source and destination UUIDs for valid combinations */ if (response_code == BNEP_RESP_SETUP_SUCCESS) { channel->uuid_dest = big_endian_read_16(packet, 2 + uuid_offset); channel->uuid_source = big_endian_read_16(packet, 2 + uuid_offset + uuid_size); if ((channel->uuid_dest != BLUETOOTH_SERVICE_CLASS_PANU) && (channel->uuid_dest != BLUETOOTH_SERVICE_CLASS_NAP) && (channel->uuid_dest != BLUETOOTH_SERVICE_CLASS_GN)) { log_error("BNEP_CONNECTION_REQUEST: Invalid destination service UUID: %04x", channel->uuid_dest); channel->uuid_dest = 0; } if ((channel->uuid_source != BLUETOOTH_SERVICE_CLASS_PANU) && (channel->uuid_source != BLUETOOTH_SERVICE_CLASS_NAP) && (channel->uuid_source != BLUETOOTH_SERVICE_CLASS_GN)) { log_error("BNEP_CONNECTION_REQUEST: Invalid source service UUID: %04x", channel->uuid_source); channel->uuid_source = 0; } /* Check if we have registered a service for the requested destination UUID */ service = bnep_service_for_uuid(channel->uuid_dest); if (service == NULL) { response_code = BNEP_RESP_SETUP_INVALID_DEST_UUID; } else if ((channel->uuid_source != BLUETOOTH_SERVICE_CLASS_PANU) && (channel->uuid_dest != BLUETOOTH_SERVICE_CLASS_PANU)) { response_code = BNEP_RESP_SETUP_INVALID_SOURCE_UUID; } } /* Set flag to send out the connection response on next statemachine cycle */ bnep_channel_state_add(channel, BNEP_CHANNEL_STATE_VAR_SND_CONNECTION_RESPONSE); channel->response_code = response_code; /* Return the number of processed package bytes = BNEP Type, BNEP Control Type, UUID-Size + 2 * UUID */ return 1 + 1 + 2 * uuid_size; } static int bnep_handle_connection_response(bnep_channel_t *channel, uint8_t *packet, uint16_t size) { uint16_t response_code; /* Sanity check packet size */ if (size < 1 + 2) { return 0; } if (channel->state != BNEP_CHANNEL_STATE_WAIT_FOR_CONNECTION_RESPONSE) { /* Ignore a connection response in any state but WAIT_FOR_CONNECTION_RESPONSE */ log_error("BNEP_CONNECTION_RESPONSE: Ignored in channel state %d", channel->state); return 1 + 2; } response_code = big_endian_read_16(packet, 1); if (response_code == BNEP_RESP_SETUP_SUCCESS) { log_info("BNEP_CONNECTION_RESPONSE: Channel established to %s", bd_addr_to_str(channel->remote_addr)); channel->state = BNEP_CHANNEL_STATE_CONNECTED; /* Stop timeout timer! */ bnep_channel_stop_timer(channel); bnep_emit_open_channel_complete(channel, 0); } else { log_error("BNEP_CONNECTION_RESPONSE: Connection to %s failed. Err: %d", bd_addr_to_str(channel->remote_addr), response_code); bnep_channel_finalize(channel); } return 1 + 2; } static int bnep_handle_filter_net_type_set(bnep_channel_t *channel, uint8_t *packet, uint16_t size) { uint16_t list_length; uint16_t response_code = BNEP_RESP_FILTER_SUCCESS; /* Sanity check packet size */ if (size < 3) { return 0; } list_length = big_endian_read_16(packet, 1); /* Sanity check packet size again with known package size */ if (size < 3 + list_length) { return 0; } if (channel->state != BNEP_CHANNEL_STATE_CONNECTED) { /* Ignore filter net type set in any state but CONNECTED */ log_error("BNEP_FILTER_NET_TYPE_SET: Ignored in channel state %d", channel->state); return 3 + list_length; } /* Check if we have enough space for more filters */ if ((list_length / (2*2)) > MAX_BNEP_NETFILTER) { log_info("BNEP_FILTER_NET_TYPE_SET: Too many filter"); response_code = BNEP_RESP_FILTER_ERR_TOO_MANY_FILTERS; } else { int i; channel->net_filter_count = 0; /* There is still enough space, copy the filters to our filter list */ for (i = 0; i < list_length / (2 * 2); i ++) { channel->net_filter[channel->net_filter_count].range_start = big_endian_read_16(packet, 1 + 2 + i * 4); channel->net_filter[channel->net_filter_count].range_end = big_endian_read_16(packet, 1 + 2 + i * 4 + 2); if (channel->net_filter[channel->net_filter_count].range_start > channel->net_filter[channel->net_filter_count].range_end) { /* Invalid filter range, ignore this filter rule */ log_error("BNEP_FILTER_NET_TYPE_SET: Invalid filter: start: %d, end: %d", channel->net_filter[channel->net_filter_count].range_start, channel->net_filter[channel->net_filter_count].range_end); response_code = BNEP_RESP_FILTER_ERR_INVALID_RANGE; } else { /* Valid filter, increase the filter count */ log_info("BNEP_FILTER_NET_TYPE_SET: Add filter: start: %d, end: %d", channel->net_filter[channel->net_filter_count].range_start, channel->net_filter[channel->net_filter_count].range_end); channel->net_filter_count ++; } } } /* Set flag to send out the set net filter response on next statemachine cycle */ bnep_channel_state_add(channel, BNEP_CHANNEL_STATE_VAR_SND_FILTER_NET_TYPE_RESPONSE); channel->response_code = response_code; return 3 + list_length; } static int bnep_handle_filter_net_type_response(bnep_channel_t *channel, uint8_t *packet, uint16_t size) { uint16_t response_code; // TODO: Currently we do not support setting a network filter. /* Sanity check packet size */ if (size < 1 + 2) { return 0; } if (channel->state != BNEP_CHANNEL_STATE_CONNECTED) { /* Ignore a filter net type response in any state but CONNECTED */ log_error("BNEP_FILTER_NET_TYPE_RESPONSE: Ignored in channel state %d", channel->state); return 1 + 2; } response_code = big_endian_read_16(packet, 1); if (response_code == BNEP_RESP_FILTER_SUCCESS) { log_info("BNEP_FILTER_NET_TYPE_RESPONSE: Net filter set successfully for %s", bd_addr_to_str(channel->remote_addr)); } else { log_error("BNEP_FILTER_NET_TYPE_RESPONSE: Net filter setting for %s failed. Err: %d", bd_addr_to_str(channel->remote_addr), response_code); } return 1 + 2; } static int bnep_handle_multi_addr_set(bnep_channel_t *channel, uint8_t *packet, uint16_t size) { uint16_t list_length; uint16_t response_code = BNEP_RESP_FILTER_SUCCESS; /* Sanity check packet size */ if (size < 3) { return 0; } list_length = big_endian_read_16(packet, 1); /* Sanity check packet size again with known package size */ if (size < 3 + list_length) { return 0; } if (channel->state != BNEP_CHANNEL_STATE_CONNECTED) { /* Ignore multicast filter address set in any state but CONNECTED */ log_error("BNEP_MULTI_ADDR_SET: Ignored in channel state %d", channel->state); return 3 + list_length; } /* Check if we have enough space for more filters */ if ((list_length / (2 * ETHER_ADDR_LEN)) > MAX_BNEP_MULTICAST_FILTER) { log_info("BNEP_MULTI_ADDR_SET: Too many filter"); response_code = BNEP_RESP_FILTER_ERR_TOO_MANY_FILTERS; } else { unsigned int i; channel->multicast_filter_count = 0; /* There is enough space, copy the filters to our filter list */ for (i = 0; i < list_length / (2 * ETHER_ADDR_LEN); i ++) { BD_ADDR_COPY(channel->multicast_filter[channel->multicast_filter_count].addr_start, packet + 1 + 2 + i * ETHER_ADDR_LEN * 2); BD_ADDR_COPY(channel->multicast_filter[channel->multicast_filter_count].addr_end, packet + 1 + 2 + i * ETHER_ADDR_LEN * 2 + ETHER_ADDR_LEN); if (memcmp(channel->multicast_filter[channel->multicast_filter_count].addr_start, channel->multicast_filter[channel->multicast_filter_count].addr_end, ETHER_ADDR_LEN) > 0) { /* Invalid filter range, ignore this filter rule */ log_error("BNEP_MULTI_ADDR_SET: Invalid filter: start: %s", bd_addr_to_str(channel->multicast_filter[channel->multicast_filter_count].addr_start)); log_error("BNEP_MULTI_ADDR_SET: Invalid filter: end: %s", bd_addr_to_str(channel->multicast_filter[channel->multicast_filter_count].addr_end)); response_code = BNEP_RESP_FILTER_ERR_INVALID_RANGE; } else { /* Valid filter, increase the filter count */ log_info("BNEP_MULTI_ADDR_SET: Add filter: start: %s", bd_addr_to_str(channel->multicast_filter[channel->multicast_filter_count].addr_start)); log_info("BNEP_MULTI_ADDR_SET: Add filter: end: %s", bd_addr_to_str(channel->multicast_filter[channel->multicast_filter_count].addr_end)); channel->multicast_filter_count ++; } } } /* Set flag to send out the set multi addr response on next statemachine cycle */ bnep_channel_state_add(channel, BNEP_CHANNEL_STATE_VAR_SND_FILTER_MULTI_ADDR_RESPONSE); channel->response_code = response_code; return 3 + list_length; } static int bnep_handle_multi_addr_response(bnep_channel_t *channel, uint8_t *packet, uint16_t size) { uint16_t response_code; // TODO: Currently we do not support setting multicast address filter. /* Sanity check packet size */ if (size < 1 + 2) { return 0; } if (channel->state != BNEP_CHANNEL_STATE_CONNECTED) { /* Ignore multicast filter set response in any state but CONNECTED */ log_error("BNEP_MULTI_ADDR_RESPONSE: Ignored in channel state %d", channel->state); return 1 + 2; } response_code = big_endian_read_16(packet, 1); if (response_code == BNEP_RESP_FILTER_SUCCESS) { log_info("BNEP_MULTI_ADDR_RESPONSE: Multicast address filter set successfully for %s", bd_addr_to_str(channel->remote_addr)); } else { log_error("BNEP_MULTI_ADDR_RESPONSE: Multicast address filter setting for %s failed. Err: %d", bd_addr_to_str(channel->remote_addr), response_code); } return 1 + 2; } static int bnep_handle_ethernet_packet(bnep_channel_t *channel, bd_addr_t addr_dest, bd_addr_t addr_source, uint16_t network_protocol_type, uint8_t *payload, uint16_t size) { uint16_t pos = 0; #if (HCI_INCOMING_PRE_BUFFER_SIZE) && (HCI_INCOMING_PRE_BUFFER_SIZE >= 14 - 8) // 2 * sizeof(bd_addr_t) + sizeof(uint16_t) - L2CAP Header (4) - ACL Header (4) /* In-place modify the package and add the ethernet header in front of the payload. * WARNING: This modifies the data in front of the payload and may overwrite 14 bytes there! */ uint8_t *ethernet_packet = payload - 2 * sizeof(bd_addr_t) - sizeof(uint16_t); /* Restore the ethernet packet header */ BD_ADDR_COPY(ethernet_packet + pos, addr_dest); pos += sizeof(bd_addr_t); BD_ADDR_COPY(ethernet_packet + pos, addr_source); pos += sizeof(bd_addr_t); net_store_16(ethernet_packet, pos, network_protocol_type); /* Payload is just in place... */ #else /* Copy ethernet frame to statically allocated buffer. This solution is more * save, but needs an extra copy and more stack! */ uint8_t ethernet_packet[BNEP_MTU_MIN]; /* Restore the ethernet packet header */ BD_ADDR_COPY(ethernet_packet + pos, addr_dest); pos += sizeof(bd_addr_t); BD_ADDR_COPY(ethernet_packet + pos, addr_source); pos += sizeof(bd_addr_t); net_store_16(ethernet_packet, pos, network_protocol_type); pos += 2; memcpy(ethernet_packet + pos, payload, size); #endif /* Notify application layer and deliver the ethernet packet */ if (app_packet_handler){ (*app_packet_handler)(channel->connection, BNEP_DATA_PACKET, channel->l2cap_cid, ethernet_packet, size + sizeof(uint16_t) + 2 * sizeof(bd_addr_t)); } return size; } static int bnep_handle_control_packet(bnep_channel_t *channel, uint8_t *packet, uint16_t size, int is_extension) { uint16_t len = 0; uint8_t bnep_control_type; bnep_control_type = packet[0]; /* Save last control type. Needed by statemachin in case of unknown control code */ channel->last_control_type = bnep_control_type; log_info("BNEP_CONTROL: Type: %d, size: %d, is_extension: %d", bnep_control_type, size, is_extension); switch (bnep_control_type) { case BNEP_CONTROL_TYPE_COMMAND_NOT_UNDERSTOOD: /* The last command we send was not understood. We should close the connection */ log_error("BNEP_CONTROL: Received COMMAND_NOT_UNDERSTOOD: l2cap_cid: %d, cmd: %d", channel->l2cap_cid, packet[3]); bnep_channel_finalize(channel); len = 2; // Length of command not understood packet - bnep-type field break; case BNEP_CONTROL_TYPE_SETUP_CONNECTION_REQUEST: if (is_extension) { /* Connection requests are not allowed to be send in an extension header * ignore, do not set "COMMAND_NOT_UNDERSTOOD" */ log_error("BNEP_CONTROL: Received SETUP_CONNECTION_REQUEST in extension header: l2cap_cid: %d", channel->l2cap_cid); return 0; } else { len = bnep_handle_connection_request(channel, packet, size); } break; case BNEP_CONTROL_TYPE_SETUP_CONNECTION_RESPONSE: if (is_extension) { /* Connection requests are not allowed to be send in an * extension header, ignore, do not set "COMMAND_NOT_UNDERSTOOD" */ log_error("BNEP_CONTROL: Received SETUP_CONNECTION_RESPONSE in extension header: l2cap_cid: %d", channel->l2cap_cid); return 0; } else { len = bnep_handle_connection_response(channel, packet, size); } break; case BNEP_CONTROL_TYPE_FILTER_NET_TYPE_SET: len = bnep_handle_filter_net_type_set(channel, packet, size); break; case BNEP_CONTROL_TYPE_FILTER_NET_TYPE_RESPONSE: len = bnep_handle_filter_net_type_response(channel, packet, size); break; case BNEP_CONTROL_TYPE_FILTER_MULTI_ADDR_SET: len = bnep_handle_multi_addr_set(channel, packet, size); break; case BNEP_CONTROL_TYPE_FILTER_MULTI_ADDR_RESPONSE: len = bnep_handle_multi_addr_response(channel, packet, size); break; default: log_error("BNEP_CONTROL: Invalid bnep control type: l2cap_cid: %d, cmd: %d", channel->l2cap_cid, bnep_control_type); len = 0; break; } if (len == 0) { /* In case the command could not be handled, send a COMMAND_NOT_UNDERSTOOD message. Set flag to process the request in the next statemachine loop */ bnep_channel_state_add(channel, BNEP_CHANNEL_STATE_VAR_SND_NOT_UNDERSTOOD); } return len; } /** * @return handled packet */ static int bnep_hci_event_handler(uint8_t *packet, uint16_t size) { bd_addr_t event_addr; uint16_t psm; uint16_t l2cap_cid; hci_con_handle_t con_handle; bnep_channel_t *channel = NULL; uint8_t status; switch (packet[0]) { /* Accept an incoming L2CAP connection on PSM_BNEP */ case L2CAP_EVENT_INCOMING_CONNECTION: /* L2CAP event data: event(8), len(8), address(48), handle (16), psm (16), source cid(16) dest cid(16) */ reverse_bd_addr(&packet[2], event_addr); con_handle = little_endian_read_16(packet, 8); psm = little_endian_read_16(packet, 10); l2cap_cid = little_endian_read_16(packet, 12); if (psm != PSM_BNEP) break; channel = bnep_channel_for_addr(&event_addr); if (channel) { log_error("INCOMING_CONNECTION (l2cap_cid 0x%02x) for PSM_BNEP => decline - channel already exists", l2cap_cid); l2cap_decline_connection_internal(l2cap_cid, 0x04); // no resources available return 1; } /* Create a new BNEP channel instance (incoming) */ channel = bnep_channel_create_for_addr(&event_addr); if (!channel) { log_error("INCOMING_CONNECTION (l2cap_cid 0x%02x) for PSM_BNEP => decline - no memory left", l2cap_cid); l2cap_decline_connection_internal(l2cap_cid, 0x04); // no resources available return 1; } /* Assign connection handle and l2cap cid */ channel->con_handle = con_handle; channel->l2cap_cid = l2cap_cid; /* Set channel into accept state */ channel->state = BNEP_CHANNEL_STATE_WAIT_FOR_CONNECTION_REQUEST; /* Start connection timeout timer */ bnep_channel_start_timer(channel, BNEP_CONNECTION_TIMEOUT_MS); log_info("L2CAP_EVENT_INCOMING_CONNECTION (l2cap_cid 0x%02x) for PSM_BNEP => accept", l2cap_cid); l2cap_accept_connection_internal(l2cap_cid); return 1; break; /* Outgoing L2CAP connection has been opened -> store l2cap_cid, remote_addr */ case L2CAP_EVENT_CHANNEL_OPENED: /* Check if the l2cap channel has been opened for PSM_BNEP */ if (little_endian_read_16(packet, 11) != PSM_BNEP) { break; } status = packet[2]; log_info("L2CAP_EVENT_CHANNEL_OPENED for PSM_BNEP, status %u", status); /* Get the bnep channel fpr remote address */ con_handle = little_endian_read_16(packet, 9); l2cap_cid = little_endian_read_16(packet, 13); reverse_bd_addr(&packet[3], event_addr); channel = bnep_channel_for_addr(&event_addr); if (!channel) { log_error("L2CAP_EVENT_CHANNEL_OPENED but no BNEP channel prepared"); return 1; } /* On L2CAP open error discard everything */ if (status) { /* Emit bnep_open_channel_complete with status and free channel */ bnep_emit_open_channel_complete(channel, status); /* Free BNEP channel mempory */ bnep_channel_free(channel); return 1; } switch (channel->state){ case BNEP_CHANNEL_STATE_CLOSED: log_info("L2CAP_EVENT_CHANNEL_OPENED: outgoing connection"); bnep_channel_start_timer(channel, BNEP_CONNECTION_TIMEOUT_MS); /* Assign connection handle and l2cap cid */ channel->l2cap_cid = l2cap_cid; channel->con_handle = con_handle; /* Initiate the connection request */ channel->state = BNEP_CHANNEL_STATE_WAIT_FOR_CONNECTION_RESPONSE; bnep_channel_state_add(channel, BNEP_CHANNEL_STATE_VAR_SND_CONNECTION_REQUEST); channel->max_frame_size = bnep_max_frame_size_for_l2cap_mtu(little_endian_read_16(packet, 17)); bnep_run(); break; case BNEP_CHANNEL_STATE_WAIT_FOR_CONNECTION_REQUEST: /* New information: channel mtu */ channel->max_frame_size = bnep_max_frame_size_for_l2cap_mtu(little_endian_read_16(packet, 17)); break; default: log_error("L2CAP_EVENT_CHANNEL_OPENED: Invalid state: %d", channel->state); break; } return 1; break; case DAEMON_EVENT_HCI_PACKET_SENT: bnep_run(); break; case L2CAP_EVENT_CHANNEL_CLOSED: // data: event (8), len(8), channel (16) l2cap_cid = little_endian_read_16(packet, 2); channel = bnep_channel_for_l2cap_cid(l2cap_cid); log_info("L2CAP_EVENT_CHANNEL_CLOSED cid 0x%0x, channel %p", l2cap_cid, channel); if (!channel) { break; } log_info("L2CAP_EVENT_CHANNEL_CLOSED state %u", channel->state); switch (channel->state) { case BNEP_CHANNEL_STATE_WAIT_FOR_CONNECTION_REQUEST: case BNEP_CHANNEL_STATE_WAIT_FOR_CONNECTION_RESPONSE: case BNEP_CHANNEL_STATE_CONNECTED: bnep_channel_finalize(channel); return 1; default: break; } break; default: bnep_run(); break; } return 0; } static int bnep_l2cap_packet_handler(uint16_t l2cap_cid, uint8_t *packet, uint16_t size) { int rc = 0; uint8_t bnep_type; uint8_t bnep_header_has_ext; uint8_t extension_type; uint16_t pos = 0; bd_addr_t addr_source; bd_addr_t addr_dest; uint16_t network_protocol_type = 0xffff; bnep_channel_t *channel = NULL; /* Get the bnep channel for this package */ channel = bnep_channel_for_l2cap_cid(l2cap_cid); if (!channel) { return rc; } /* Sort out short packages */ if (size < 2) { return rc; } bnep_type = BNEP_TYPE(packet[pos]); bnep_header_has_ext = BNEP_HEADER_HAS_EXT(packet[pos]); pos ++; switch(bnep_type) { case BNEP_PKT_TYPE_GENERAL_ETHERNET: BD_ADDR_COPY(addr_dest, &packet[pos]); pos += sizeof(bd_addr_t); BD_ADDR_COPY(addr_source, &packet[pos]); pos += sizeof(bd_addr_t); network_protocol_type = big_endian_read_16(packet, pos); pos += 2; break; case BNEP_PKT_TYPE_COMPRESSED_ETHERNET: BD_ADDR_COPY(addr_dest, channel->local_addr); BD_ADDR_COPY(addr_source, channel->remote_addr); network_protocol_type = big_endian_read_16(packet, pos); pos += 2; break; case BNEP_PKT_TYPE_COMPRESSED_ETHERNET_SOURCE_ONLY: BD_ADDR_COPY(addr_dest, channel->local_addr); BD_ADDR_COPY(addr_source, &packet[pos]); pos += sizeof(bd_addr_t); network_protocol_type = big_endian_read_16(packet, pos); pos += 2; break; case BNEP_PKT_TYPE_COMPRESSED_ETHERNET_DEST_ONLY: BD_ADDR_COPY(addr_dest, &packet[pos]); pos += sizeof(bd_addr_t); BD_ADDR_COPY(addr_source, channel->remote_addr); network_protocol_type = big_endian_read_16(packet, pos); pos += 2; break; case BNEP_PKT_TYPE_CONTROL: rc = bnep_handle_control_packet(channel, packet + pos, size - pos, 0); pos += rc; break; default: break; } if (bnep_header_has_ext) { do { uint8_t ext_len; /* Read extension type and check for further extensions */ extension_type = BNEP_TYPE(packet[pos]); bnep_header_has_ext = BNEP_HEADER_HAS_EXT(packet[pos]); pos ++; /* Read extension header length */ ext_len = packet[pos]; pos ++; if (size - pos < ext_len) { log_error("BNEP pkt handler: Invalid extension length! Packet ignored"); /* Invalid packet size! */ return 0; } switch (extension_type) { case BNEP_EXT_HEADER_TYPE_EXTENSION_CONTROL: if (ext_len != bnep_handle_control_packet(channel, packet + pos, ext_len, 1)) { log_error("BNEP pkt handler: Ignore invalid control packet in extension header"); } pos += ext_len; break; default: /* Extension header type unknown. Unknown extension SHALL be * SHALL be forwarded in any way. But who shall handle these * extension packets? * For now: We ignore them and just drop them! */ log_error("BNEP pkt handler: Unknown extension type ignored, data dropped!"); pos += ext_len; break; } } while (bnep_header_has_ext); } if (bnep_type != BNEP_PKT_TYPE_CONTROL) { if (channel->state == BNEP_CHANNEL_STATE_CONNECTED) { rc = bnep_handle_ethernet_packet(channel, addr_dest, addr_source, network_protocol_type, packet + pos, size - pos); } else { rc = 0; } } return rc; } void bnep_packet_handler(uint8_t packet_type, uint16_t l2cap_cid, uint8_t *packet, uint16_t size) { int handled = 0; switch (packet_type) { case HCI_EVENT_PACKET: handled = bnep_hci_event_handler(packet, size); break; case L2CAP_DATA_PACKET: handled = bnep_l2cap_packet_handler(l2cap_cid, packet, size); break; default: break; } if (handled) { bnep_run(); return; } /* Forward non l2cap packages to application handler */ if (packet_type != L2CAP_DATA_PACKET) { (*app_packet_handler)(NULL, packet_type, l2cap_cid, packet, size); return; } bnep_run(); } static void bnep_channel_state_machine(bnep_channel_t* channel, bnep_channel_event_t *event) { log_info("bnep_state_machine: state %u, state var: %02x, event %u", channel->state, channel->state_var, event->type); if (event->type == BNEP_CH_EVT_READY_TO_SEND) { /* Send outstanding packets. */ if (channel->state_var & BNEP_CHANNEL_STATE_VAR_SND_NOT_UNDERSTOOD) { bnep_channel_state_remove(channel, BNEP_CHANNEL_STATE_VAR_SND_NOT_UNDERSTOOD); bnep_send_command_not_understood(channel, channel->last_control_type); return; } if (channel->state_var & BNEP_CHANNEL_STATE_VAR_SND_CONNECTION_REQUEST) { bnep_channel_state_remove(channel, BNEP_CHANNEL_STATE_VAR_SND_CONNECTION_REQUEST); channel->state = BNEP_CHANNEL_STATE_WAIT_FOR_CONNECTION_RESPONSE; bnep_send_connection_request(channel, channel->uuid_source, channel->uuid_dest); } if (channel->state_var & BNEP_CHANNEL_STATE_VAR_SND_CONNECTION_RESPONSE) { int emit_connected = 0; if ((channel->state == BNEP_CHANNEL_STATE_CLOSED) || (channel->state == BNEP_CHANNEL_STATE_WAIT_FOR_CONNECTION_REQUEST)) { /* Set channel state to STATE_CONNECTED */ channel->state = BNEP_CHANNEL_STATE_CONNECTED; /* Stop timeout timer! */ bnep_channel_stop_timer(channel); emit_connected = 1; } bnep_channel_state_remove(channel, BNEP_CHANNEL_STATE_VAR_SND_CONNECTION_RESPONSE); bnep_send_connection_response(channel, channel->response_code); if (emit_connected){ bnep_emit_open_channel_complete(channel, 0); } return; } if (channel->state_var & BNEP_CHANNEL_STATE_VAR_SND_FILTER_NET_TYPE_SET) { bnep_channel_state_remove(channel, BNEP_CHANNEL_STATE_VAR_SND_FILTER_NET_TYPE_SET); if ((channel->net_filter_out_count > 0) && (channel->net_filter_out != NULL)) { bnep_send_filter_net_type_set(channel, channel->net_filter_out, channel->net_filter_out_count); channel->net_filter_out_count = 0; channel->net_filter_out = NULL; } return; } if (channel->state_var & BNEP_CHANNEL_STATE_VAR_SND_FILTER_NET_TYPE_RESPONSE) { bnep_channel_state_remove(channel, BNEP_CHANNEL_STATE_VAR_SND_FILTER_NET_TYPE_RESPONSE); bnep_send_filter_net_type_response(channel, channel->response_code); return; } if (channel->state_var & BNEP_CHANNEL_STATE_VAR_SND_FILTER_MULTI_ADDR_SET) { bnep_channel_state_remove(channel, BNEP_CHANNEL_STATE_VAR_SND_FILTER_MULTI_ADDR_SET); if ((channel->multicast_filter_out_count > 0) && (channel->multicast_filter_out != NULL)) { bnep_send_filter_multi_addr_set(channel, channel->multicast_filter_out, channel->multicast_filter_out_count); channel->multicast_filter_out_count = 0; channel->multicast_filter_out = NULL; } return; } if (channel->state_var & BNEP_CHANNEL_STATE_VAR_SND_FILTER_MULTI_ADDR_RESPONSE) { bnep_channel_state_remove(channel, BNEP_CHANNEL_STATE_VAR_SND_FILTER_MULTI_ADDR_RESPONSE); bnep_send_filter_multi_addr_response(channel, channel->response_code); return; } /* If the event was not yet handled, notify the application layer */ bnep_emit_ready_to_send(channel); } } /* Process oustanding signaling tasks */ static void bnep_run(void) { linked_item_t *it; linked_item_t *next; for (it = (linked_item_t *) bnep_channels; it ; it = next){ bnep_channel_t * channel; bnep_channel_event_t channel_event = { BNEP_CH_EVT_READY_TO_SEND }; next = it->next; // be prepared for removal of channel in state machine channel = ((bnep_channel_t *) it); if (!l2cap_can_send_packet_now(channel->l2cap_cid)) { continue; } bnep_channel_state_machine(channel, &channel_event); } } /* BNEP BTStack API */ void bnep_init(void) { bnep_security_level = LEVEL_0; } void bnep_set_required_security_level(gap_security_level_t security_level) { bnep_security_level = security_level; } /* Register application packet handler */ void bnep_register_packet_handler(void (*handler)(void * connection, uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size)){ app_packet_handler = handler; } int bnep_connect(void * connection, bd_addr_t *addr, uint16_t l2cap_psm, uint16_t uuid_dest) { bnep_channel_t *channel; log_info("BNEP_CONNECT addr %s", bd_addr_to_str(*addr)); channel = bnep_channel_create_for_addr(addr); if (channel == NULL) { return -1; } channel->uuid_source = BLUETOOTH_SERVICE_CLASS_PANU; channel->uuid_dest = uuid_dest; l2cap_create_channel_internal(connection, bnep_packet_handler, *addr, l2cap_psm, l2cap_max_mtu()); return 0; } void bnep_disconnect(bd_addr_t *addr) { bnep_channel_t *channel; log_info("BNEP_DISCONNECT"); channel = bnep_channel_for_addr(addr); bnep_channel_finalize(channel); bnep_run(); } int bnep_register_service(void * connection, uint16_t service_uuid, uint16_t max_frame_size) { bnep_service_t * service; log_info("BNEP_REGISTER_SERVICE mtu %d", max_frame_size); /* Check if we already registered a service */ service = bnep_service_for_uuid(service_uuid); if (service) { return BNEP_SERVICE_ALREADY_REGISTERED; } /* Only alow one the three service types: PANU, NAP, GN */ if ((service_uuid != BLUETOOTH_SERVICE_CLASS_PANU) && (service_uuid != BLUETOOTH_SERVICE_CLASS_NAP) && (service_uuid != BLUETOOTH_SERVICE_CLASS_GN)) { log_info("BNEP_REGISTER_SERVICE: Invalid service UUID: %04x", service_uuid); return BNEP_CHANNEL_NOT_CONNECTED; } /* Allocate service memory */ service = (bnep_service_t*) btstack_memory_bnep_service_get(); if (!service) { return BNEP_SERVICE_ALREADY_REGISTERED; } memset(service, 0, sizeof(bnep_service_t)); /* register with l2cap if not registered before, max MTU */ l2cap_register_service_internal(NULL, bnep_packet_handler, PSM_BNEP, 0xffff, bnep_security_level); /* Setup the service struct */ service->connection = connection; service->max_frame_size = max_frame_size; service->service_uuid = service_uuid; /* Add to services list */ linked_list_add(&bnep_services, (linked_item_t *) service); return 0; } void bnep_unregister_service(uint16_t service_uuid) { bnep_service_t *service; log_info("BNEP_UNREGISTER_SERVICE #%04x", service_uuid); service = bnep_service_for_uuid(service_uuid); if (!service) { return; } linked_list_remove(&bnep_services, (linked_item_t *) service); btstack_memory_bnep_service_free(service); service = NULL; l2cap_unregister_service_internal(NULL, PSM_BNEP); }