/******************************************************************************
*
* Copyright 1999-2012 Broadcom Corporation
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at:
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
******************************************************************************/
/******************************************************************************
*
* This file contains main functions to support PAN profile
* commands and events.
*
******************************************************************************/
#define LOG_TAG "pan"
#include
#include
#include // memset
#include
#include "os/log.h"
#include "osi/include/allocator.h"
#include "stack/include/bnep_api.h"
#include "stack/include/bt_hdr.h"
#include "stack/include/bt_uuid16.h"
#include "stack/pan/pan_int.h"
#include "types/bluetooth/uuid.h"
#include "types/raw_address.h"
using namespace bluetooth;
using bluetooth::Uuid;
tPAN_CB pan_cb;
/*******************************************************************************
*
* Function pan_register_with_bnep
*
* Description This function registers PAN profile with BNEP
*
* Parameters: none
*
* Returns none
*
******************************************************************************/
void pan_register_with_bnep(void) {
tBNEP_REGISTER reg_info;
memset(®_info, 0, sizeof(tBNEP_REGISTER));
reg_info.p_conn_ind_cb = pan_conn_ind_cb;
reg_info.p_conn_state_cb = pan_connect_state_cb;
reg_info.p_data_buf_cb = pan_data_buf_ind_cb;
reg_info.p_data_ind_cb = NULL;
reg_info.p_tx_data_flow_cb = pan_tx_data_flow_cb;
reg_info.p_filter_ind_cb = pan_proto_filt_ind_cb;
reg_info.p_mfilter_ind_cb = pan_mcast_filt_ind_cb;
BNEP_Register(®_info);
}
/*******************************************************************************
*
* Function pan_conn_ind_cb
*
* Description This function is registered with BNEP as connection
* indication callback. BNEP will call this when there is
* connection request from the peer. PAN should call
* BNEP_ConnectResp to indicate whether to accept the
* connection or reject
*
* Parameters: handle - handle for the connection
* p_bda - BD Addr of the peer requesting the connection
* remote_uuid - UUID of the source role (peer device role)
* local_uuid - UUID of the destination role (local device
* role)
* is_role_change - Flag to indicate that it is a role change
*
* Returns none
*
******************************************************************************/
void pan_conn_ind_cb(uint16_t handle, const RawAddress& p_bda,
const Uuid& remote_uuid, const Uuid& local_uuid,
bool is_role_change) {
/* If we are in GN or NAP role and have one or more active connections and the
* received connection is for user role reject it. If we are in user role with
* one connection active reject the connection. Allocate PCB and store the
* parameters. Make bridge request to the host system if connection is for NAP
*/
if (!remote_uuid.Is16Bit()) {
log::error("PAN Connection failed because of wrong remote UUID");
BNEP_ConnectResp(handle, BNEP_CONN_FAILED_SRC_UUID);
return;
}
if (!local_uuid.Is16Bit()) {
log::error("PAN Connection failed because of wrong local UUID");
BNEP_ConnectResp(handle, BNEP_CONN_FAILED_DST_UUID);
return;
}
uint16_t remote_uuid16 = remote_uuid.As16Bit();
uint16_t local_uuid16 = local_uuid.As16Bit();
log::verbose(
"handle {}, current role {}, dst uuid 0x{:x}, src uuid 0x{:x}, role "
"change {}",
handle, pan_cb.role, local_uuid16, remote_uuid16,
is_role_change ? "YES" : "NO");
/* Check if the source UUID is a valid one */
if (remote_uuid16 != UUID_SERVCLASS_PANU &&
remote_uuid16 != UUID_SERVCLASS_NAP &&
remote_uuid16 != UUID_SERVCLASS_GN) {
log::error("Src UUID 0x{:x} is not valid", remote_uuid16);
BNEP_ConnectResp(handle, BNEP_CONN_FAILED_SRC_UUID);
return;
}
/* Check if the destination UUID is a valid one */
if (local_uuid16 != UUID_SERVCLASS_PANU &&
local_uuid16 != UUID_SERVCLASS_NAP && local_uuid16 != UUID_SERVCLASS_GN) {
log::error("Dst UUID 0x{:x} is not valid", local_uuid16);
BNEP_ConnectResp(handle, BNEP_CONN_FAILED_DST_UUID);
return;
}
/* Check if currently we support the destination role requested */
if (((!(pan_cb.role & UUID_SERVCLASS_PANU)) &&
local_uuid16 == UUID_SERVCLASS_PANU) ||
((!(pan_cb.role & UUID_SERVCLASS_GN)) &&
local_uuid16 == UUID_SERVCLASS_GN) ||
((!(pan_cb.role & UUID_SERVCLASS_NAP)) &&
local_uuid16 == UUID_SERVCLASS_NAP)) {
log::error(
"PAN Connection failed because of unsupported destination UUID 0x{:x}",
local_uuid16);
BNEP_ConnectResp(handle, BNEP_CONN_FAILED_DST_UUID);
return;
}
/* Check for valid interactions between the three PAN profile roles */
/*
* For reference, see Table 1 in PAN Profile v1.0 spec.
* Note: the remote is the initiator.
*/
bool is_valid_interaction = false;
switch (remote_uuid16) {
case UUID_SERVCLASS_NAP:
case UUID_SERVCLASS_GN:
if (local_uuid16 == UUID_SERVCLASS_PANU) is_valid_interaction = true;
break;
case UUID_SERVCLASS_PANU:
is_valid_interaction = true;
break;
}
/*
* Explicitly disable connections to the local PANU if the remote is
* not PANU.
*/
if ((local_uuid16 == UUID_SERVCLASS_PANU) &&
(remote_uuid16 != UUID_SERVCLASS_PANU)) {
is_valid_interaction = false;
}
if (!is_valid_interaction) {
log::error(
"PAN Connection failed because of invalid PAN profile roles "
"interaction: Remote UUID 0x{:x} Local UUID 0x{:x}",
remote_uuid16, local_uuid16);
BNEP_ConnectResp(handle, BNEP_CONN_FAILED_SRC_UUID);
return;
}
uint8_t req_role;
/* Requested destination role is */
if (local_uuid16 == UUID_SERVCLASS_PANU)
req_role = PAN_ROLE_CLIENT;
else
req_role = PAN_ROLE_NAP_SERVER;
/* If the connection indication is for the existing connection
** Check if the new destination role is acceptable
*/
tPAN_CONN* pcb = pan_get_pcb_by_handle(handle);
if (pcb) {
if (pan_cb.num_conns > 1 && local_uuid16 == UUID_SERVCLASS_PANU) {
/* There are connections other than this one
** so we cann't accept PANU role. Reject
*/
log::error(
"Dst UUID should be either GN or NAP only because there are other "
"connections");
BNEP_ConnectResp(handle, BNEP_CONN_FAILED_DST_UUID);
return;
}
/* If it is already in connected state check for bridging status */
if (pcb->con_state == PAN_STATE_CONNECTED) {
log::verbose("PAN Role changing New Src 0x{:x} Dst 0x{:x}", remote_uuid16,
local_uuid16);
pcb->prv_src_uuid = pcb->src_uuid;
pcb->prv_dst_uuid = pcb->dst_uuid;
if (pcb->src_uuid == UUID_SERVCLASS_NAP &&
local_uuid16 != UUID_SERVCLASS_NAP) {
/* Remove bridging */
if (pan_cb.pan_bridge_req_cb)
(*pan_cb.pan_bridge_req_cb)(pcb->rem_bda, false);
}
}
/* Set the latest active PAN role */
pan_cb.active_role = req_role;
pcb->src_uuid = local_uuid16;
pcb->dst_uuid = remote_uuid16;
BNEP_ConnectResp(handle, BNEP_SUCCESS);
return;
} else {
/* If this a new connection and destination is PANU role and
** we already have a connection then reject the request.
** If we have a connection in PANU role then reject it
*/
if (pan_cb.num_conns && (local_uuid16 == UUID_SERVCLASS_PANU ||
pan_cb.active_role == PAN_ROLE_CLIENT)) {
log::error("PAN already have a connection and can't be user");
BNEP_ConnectResp(handle, BNEP_CONN_FAILED_DST_UUID);
return;
}
}
/* This is a new connection */
log::verbose("New connection indication for handle {}", handle);
pcb = pan_allocate_pcb(p_bda, handle);
if (!pcb) {
log::error("PAN no control block for new connection");
BNEP_ConnectResp(handle, BNEP_CONN_FAILED);
return;
}
log::verbose("PAN connection destination UUID is 0x{:x}", local_uuid16);
/* Set the latest active PAN role */
pan_cb.active_role = req_role;
pcb->src_uuid = local_uuid16;
pcb->dst_uuid = remote_uuid16;
pcb->con_state = PAN_STATE_CONN_START;
pan_cb.num_conns++;
BNEP_ConnectResp(handle, BNEP_SUCCESS);
return;
}
/*******************************************************************************
*
* Function pan_connect_state_cb
*
* Description This function is registered with BNEP as connection state
* change callback. BNEP will call this when the connection
* is established successfully or terminated
*
* Parameters: handle - handle for the connection given in the connection
* indication callback
* rem_bda - remote device bd addr
* result - indicates whether the connection is up or down
* BNEP_SUCCESS if the connection is up all other
* values indicate appropriate errors.
* is_role_change - flag to indicate that it is a role change
*
* Returns none
*
******************************************************************************/
void pan_connect_state_cb(uint16_t handle, const RawAddress& /* rem_bda */,
tBNEP_RESULT result, bool is_role_change) {
tPAN_CONN* pcb;
uint8_t peer_role;
log::verbose("pan_connect_state_cb - for handle {}, result {}", handle,
result);
pcb = pan_get_pcb_by_handle(handle);
if (!pcb) {
log::error("PAN State change indication for wrong handle {}", handle);
return;
}
/* If the connection is getting terminated remove bridging */
if (result != BNEP_SUCCESS) {
/* Inform the application that connection is down */
if (pan_cb.pan_conn_state_cb)
(*pan_cb.pan_conn_state_cb)(pcb->handle, pcb->rem_bda,
(tPAN_RESULT)result, is_role_change,
PAN_ROLE_INACTIVE, PAN_ROLE_INACTIVE);
/* Check if this failure is for role change only */
if (pcb->con_state != PAN_STATE_CONNECTED &&
(pcb->con_flags & PAN_FLAGS_CONN_COMPLETED)) {
/* restore the original values */
log::verbose("restoring the connection state to active");
pcb->con_state = PAN_STATE_CONNECTED;
pcb->con_flags &= (~PAN_FLAGS_CONN_COMPLETED);
pcb->src_uuid = pcb->prv_src_uuid;
pcb->dst_uuid = pcb->prv_dst_uuid;
pan_cb.active_role = pan_cb.prv_active_role;
if ((pcb->src_uuid == UUID_SERVCLASS_NAP) && pan_cb.pan_bridge_req_cb)
(*pan_cb.pan_bridge_req_cb)(pcb->rem_bda, true);
return;
}
if (pcb->con_state == PAN_STATE_CONNECTED) {
/* If the connections destination role is NAP remove bridging */
if ((pcb->src_uuid == UUID_SERVCLASS_NAP) && pan_cb.pan_bridge_req_cb)
(*pan_cb.pan_bridge_req_cb)(pcb->rem_bda, false);
}
pan_cb.num_conns--;
pan_release_pcb(pcb);
return;
}
/* Requested destination role is */
if (pcb->src_uuid == UUID_SERVCLASS_PANU)
pan_cb.active_role = PAN_ROLE_CLIENT;
else
pan_cb.active_role = PAN_ROLE_NAP_SERVER;
if (pcb->dst_uuid == UUID_SERVCLASS_PANU)
peer_role = PAN_ROLE_CLIENT;
else
peer_role = PAN_ROLE_NAP_SERVER;
pcb->con_state = PAN_STATE_CONNECTED;
/* Inform the application that connection is down */
if (pan_cb.pan_conn_state_cb)
(*pan_cb.pan_conn_state_cb)(pcb->handle, pcb->rem_bda, PAN_SUCCESS,
is_role_change, pan_cb.active_role, peer_role);
/* Create bridge if the destination role is NAP */
if (pan_cb.pan_bridge_req_cb && pcb->src_uuid == UUID_SERVCLASS_NAP) {
log::verbose("PAN requesting for bridge");
(*pan_cb.pan_bridge_req_cb)(pcb->rem_bda, true);
}
}
/*******************************************************************************
*
* Function pan_data_buf_ind_cb
*
* Description This function is registered with BNEP as data buffer
* indication callback. BNEP will call this when the peer sends
* any data on this connection. PAN is responsible to release
* the buffer
*
* Parameters: handle - handle for the connection
* src - source BD Addr
* dst - destination BD Addr
* protocol - Network protocol of the Eth packet
* p_buf - pointer to the data buffer
* ext - to indicate whether the data contains any
* extension headers before the payload
*
* Returns none
*
******************************************************************************/
void pan_data_buf_ind_cb(uint16_t handle, const RawAddress& src,
const RawAddress& dst, uint16_t protocol,
BT_HDR* p_buf, bool ext) {
tPAN_CONN *pcb, *dst_pcb;
tBNEP_RESULT result;
uint16_t i, len;
uint8_t* p_data;
bool forward = false;
/* Check if the connection is in right state */
pcb = pan_get_pcb_by_handle(handle);
if (!pcb) {
log::error("PAN Data buffer indication for wrong handle {}", handle);
osi_free(p_buf);
return;
}
if (pcb->con_state != PAN_STATE_CONNECTED) {
log::error("PAN Data indication in wrong state {} for handle {}",
pcb->con_state, handle);
pcb->read.drops++;
osi_free(p_buf);
return;
}
p_data = (uint8_t*)(p_buf + 1) + p_buf->offset;
len = p_buf->len;
pcb->read.octets += len;
pcb->read.packets++;
log::verbose(
"pan_data_buf_ind_cb - for handle {}, protocol 0x{:x}, length {}, ext {}",
handle, protocol, len, ext);
if (pcb->src_uuid == UUID_SERVCLASS_NAP)
forward = true;
else
forward = false;
/* Check if it is broadcast or multicast packet */
if (pcb->src_uuid != UUID_SERVCLASS_PANU) {
if (dst.address[0] & 0x01) {
log::verbose(
"PAN received broadcast packet on handle {}, src uuid 0x{:x}", handle,
pcb->src_uuid);
for (i = 0; i < MAX_PAN_CONNS; i++) {
if (pan_cb.pcb[i].con_state == PAN_STATE_CONNECTED &&
pan_cb.pcb[i].handle != handle &&
pcb->src_uuid == pan_cb.pcb[i].src_uuid) {
BNEP_Write(pan_cb.pcb[i].handle, dst, p_data, len, protocol, src,
ext);
}
}
if (pan_cb.pan_data_buf_ind_cb)
(*pan_cb.pan_data_buf_ind_cb)(pcb->handle, src, dst, protocol, p_buf,
ext, forward);
else if (pan_cb.pan_data_ind_cb)
(*pan_cb.pan_data_ind_cb)(pcb->handle, src, dst, protocol, p_data, len,
ext, forward);
osi_free(p_buf);
return;
}
/* Check if it is for any other PAN connection */
dst_pcb = pan_get_pcb_by_addr(dst);
if (dst_pcb) {
log::verbose(
"destination PANU found on handle {} and sending data, len: {}",
dst_pcb->handle, len);
result =
BNEP_Write(dst_pcb->handle, dst, p_data, len, protocol, src, ext);
if (result != BNEP_SUCCESS && result != BNEP_IGNORE_CMD)
log::error("Failed to write data for PAN connection handle {}",
dst_pcb->handle);
pcb->read.errors++;
osi_free(p_buf);
return;
}
}
/* Send it over the LAN or give it to host software */
if (pan_cb.pan_data_buf_ind_cb)
(*pan_cb.pan_data_buf_ind_cb)(pcb->handle, src, dst, protocol, p_buf, ext,
forward);
else if (pan_cb.pan_data_ind_cb)
(*pan_cb.pan_data_ind_cb)(pcb->handle, src, dst, protocol, p_data, len, ext,
forward);
osi_free(p_buf);
return;
}
/*******************************************************************************
*
* Function pan_proto_filt_ind_cb
*
* Description This function is registered with BNEP to receive tx data
* flow status
*
* Parameters: handle - handle for the connection
* event - flow status
*
* Returns none
*
******************************************************************************/
void pan_tx_data_flow_cb(uint16_t handle, tBNEP_RESULT result) {
if (pan_cb.pan_tx_data_flow_cb)
(*pan_cb.pan_tx_data_flow_cb)(handle, (tPAN_RESULT)result);
return;
}
/*******************************************************************************
*
* Function pan_proto_filt_ind_cb
*
* Description This function is registered with BNEP as proto filter
* indication callback. BNEP will call this when the peer sends
* any protocol filter set for the connection or to indicate
* the result of the protocol filter set by the local device
*
* Parameters: handle - handle for the connection
* indication - true if this is indication
* false if it is called to give the result of
* local device protocol filter set
* result - This gives the result of the filter set
* operation
* num_filters - number of filters set by the peer device
* p_filters - pointer to the filters set by the peer device
*
* Returns none
*
******************************************************************************/
void pan_proto_filt_ind_cb(uint16_t handle, bool indication,
tBNEP_RESULT result, uint16_t num_filters,
uint8_t* p_filters) {
log::verbose(
"pan_proto_filt_ind_cb - called for handle {} with ind {}, result {}, "
"num {}",
handle, indication, result, num_filters);
if (pan_cb.pan_pfilt_ind_cb)
(*pan_cb.pan_pfilt_ind_cb)(handle, indication, (tPAN_RESULT)result,
num_filters, p_filters);
}
/*******************************************************************************
*
* Function pan_mcast_filt_ind_cb
*
* Description This function is registered with BNEP as mcast filter
* indication callback. BNEP will call this when the peer sends
* any multicast filter set for the connection or to indicate
* the result of the multicast filter set by the local device
*
* Parameters: handle - handle for the connection
* indication - true if this is indication
* false if it is called to give the result of
* local device multicast filter set
* result - This gives the result of the filter set
* operation
* num_filters - number of filters set by the peer device
* p_filters - pointer to the filters set by the peer device
*
* Returns none
*
******************************************************************************/
void pan_mcast_filt_ind_cb(uint16_t handle, bool indication,
tBNEP_RESULT result, uint16_t num_filters,
uint8_t* p_filters) {
log::verbose(
"pan_mcast_filt_ind_cb - called for handle {} with ind {}, result {}, "
"num {}",
handle, indication, result, num_filters);
if (pan_cb.pan_mfilt_ind_cb)
(*pan_cb.pan_mfilt_ind_cb)(handle, indication, (tPAN_RESULT)result,
num_filters, p_filters);
}