/**************************************************************************** * drivers/wireless/ieee80211/bcm43xxx/bcmf_bdc.c * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. The * ASF licenses this file to you 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. * ****************************************************************************/ /**************************************************************************** * Included Files ****************************************************************************/ #include #include #include #include #include #include #include #include "bcmf_driver.h" #include "bcmf_ioctl.h" #include "bcmf_cdc.h" #include "bcmf_bdc.h" #include "bcmf_utils.h" /**************************************************************************** * Pre-processor Definitions ****************************************************************************/ #define BCMF_EVENT_ETHER_TYPE 0x6C88 /* Ether type of event frames */ /**************************************************************************** * Private Types ****************************************************************************/ begin_packed_struct struct bcmf_bdc_header { uint8_t flags; /* bdc frame flags */ uint8_t priority; /* bdc frame priority */ uint8_t flags2; /* bdc frame additional flags */ uint8_t data_offset; /* Offset from end of header to payload data, in 4-bytes count */ } end_packed_struct; begin_packed_struct struct bcmf_eth_header { uint16_t type; /* Vendor specific type */ uint16_t len; /* Event data length */ uint8_t version; /* Protocol version */ uint8_t oui[3]; /* Organizationally unique identifier */ uint16_t usr_type; /* User specific type */ } end_packed_struct; begin_packed_struct struct bcmf_event_msg { struct ether_header eth; struct bcmf_eth_header bcm_eth; struct bcmf_event_s event; uint8_t data[0]; } end_packed_struct; /**************************************************************************** * Private Data ****************************************************************************/ static const uint8_t bcmf_broadcom_oui[] = { 0x00, 0x10, 0x18 }; /**************************************************************************** * Public Functions ****************************************************************************/ struct bcmf_frame_s *bcmf_bdc_allocate_frame(FAR struct bcmf_dev_s *priv, uint32_t len, bool block) { struct bcmf_frame_s *frame; /* Allocate data frame */ /* TODO check for integer overflow */ frame = priv->bus->allocate_frame(priv, sizeof(struct bcmf_bdc_header) + len, block, false); if (!frame) { return NULL; } frame->data += sizeof(struct bcmf_bdc_header); return frame; } int bcmf_bdc_process_event_frame(FAR struct bcmf_dev_s *priv, struct bcmf_frame_s *frame) { int data_size; struct bcmf_bdc_header *header; struct bcmf_event_msg *event_msg; uint32_t event_id; event_handler_t handler; /* Check frame header */ data_size = frame->len - (int)(frame->data - frame->base); if (data_size < sizeof(struct bcmf_bdc_header)) { wlerr("invalid event frame -- way too small\n"); goto exit_invalid_frame; } header = (struct bcmf_bdc_header *)frame->data; data_size -= sizeof(struct bcmf_bdc_header) + header->data_offset * 4; if (data_size < sizeof(struct bcmf_event_msg)) { wlerr("invalid event frame -- too small\n"); goto exit_invalid_frame; } data_size -= sizeof(struct ether_header) + sizeof(struct bcmf_eth_header); /* Check ethernet header */ event_msg = (struct bcmf_event_msg *)(frame->data + sizeof(struct bcmf_bdc_header) + header->data_offset * 4); if (event_msg->eth.ether_type != BCMF_EVENT_ETHER_TYPE || memcmp(event_msg->bcm_eth.oui, bcmf_broadcom_oui, 3)) { wlerr("invalid event frame -- bad data\n"); goto exit_invalid_frame; } event_id = bcmf_getle32(&event_msg->event.type); if (event_id >= BCMF_EVENT_COUNT) { wlinfo("Invalid event id %" PRId32 "\n", event_id); return -EINVAL; } /* Dispatch event to registered handler */ handler = priv->event_handlers[event_id]; if (handler != NULL) { handler(priv, &event_msg->event, data_size); } return OK; exit_invalid_frame: bcmf_hexdump(frame->base, frame->len, (unsigned long)frame->base); return -EINVAL; } int bcmf_event_register(FAR struct bcmf_dev_s *priv, event_handler_t handler, unsigned int event_id) { if (event_id >= BCMF_EVENT_COUNT) { /* Invalid event id */ return -EINVAL; } priv->event_handlers[event_id] = handler; return OK; } int bcmf_event_unregister(FAR struct bcmf_dev_s *priv, unsigned int event_id) { return bcmf_event_register(priv, NULL, event_id); } int bcmf_event_push_config(FAR struct bcmf_dev_s *priv) { int i; uint32_t out_len; uint8_t event_mask[(BCMF_EVENT_COUNT + 7) >> 3]; memset(event_mask, 0, sizeof(event_mask)); for (i = 0; i < BCMF_EVENT_COUNT; i++) { if (priv->event_handlers[i] != NULL) { event_mask[i >> 3] |= 1 << (i & 0x7); } } /* Send event mask to chip */ out_len = sizeof(event_mask); if (bcmf_cdc_iovar_request(priv, CHIP_STA_INTERFACE, true, IOVAR_STR_EVENT_MSGS, event_mask, &out_len)) { return -EIO; } return OK; } int bcmf_bdc_transmit_frame(FAR struct bcmf_dev_s *priv, struct bcmf_frame_s *frame) { struct bcmf_bdc_header *header; /* Set frame data for lower layer */ frame->data -= sizeof(struct bcmf_bdc_header); header = (struct bcmf_bdc_header *)frame->data; /* Setup data frame header */ header->flags = 0x20; /* Set bdc protocol version */ header->priority = 0; /* TODO handle priority */ header->flags2 = CHIP_STA_INTERFACE; header->data_offset = 0; /* Send frame */ return priv->bus->txframe(priv, frame, false); } struct bcmf_frame_s *bcmf_bdc_rx_frame(FAR struct bcmf_dev_s *priv) { unsigned int frame_len; struct bcmf_frame_s *frame = priv->bus->rxframe(priv); struct bcmf_bdc_header *header; /* Very that there is an Rx frame */ if (frame == NULL) { return NULL; } /* Process bdc header */ frame_len = frame->len - (unsigned int)(frame->data - frame->base); if (frame_len < sizeof(struct bcmf_bdc_header)) { wlerr("Data frame too small\n"); priv->bus->free_frame(priv, frame); return NULL; } /* Transmit frame to upper layer */ header = (struct bcmf_bdc_header *)frame->data; frame->data += sizeof(struct bcmf_bdc_header) + header->data_offset * 4; return frame; }