| /****************************************************************************** |
| * |
| * Copyright (C) 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. |
| * |
| ******************************************************************************/ |
| |
| #define LOG_TAG "bt_btu_task" |
| |
| #include <pthread.h> |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <string.h> |
| |
| #include "bt_common.h" |
| #include "bt_target.h" |
| #include "bt_trace.h" |
| #include "bt_types.h" |
| #include "bt_utils.h" |
| #include "btcore/include/module.h" |
| #include "bte.h" |
| #include "btif_common.h" |
| #include "btm_api.h" |
| #include "btm_int.h" |
| #include "btu.h" |
| #include "gap_int.h" |
| #include "hcimsgs.h" |
| #include "l2c_int.h" |
| #include "osi/include/alarm.h" |
| #include "osi/include/fixed_queue.h" |
| #include "osi/include/future.h" |
| #include "osi/include/log.h" |
| #include "osi/include/osi.h" |
| #include "osi/include/thread.h" |
| #include "port_api.h" |
| #include "port_ext.h" |
| #include "sdpint.h" |
| |
| #if (BNEP_INCLUDED == TRUE) |
| #include "bnep_int.h" |
| #endif |
| |
| #if (PAN_INCLUDED == TRUE) |
| #include "pan_int.h" |
| #endif |
| |
| #if (HID_HOST_INCLUDED == TRUE) |
| #include "hidh_int.h" |
| #endif |
| |
| #if (AVDT_INCLUDED == TRUE) |
| #include "avdt_int.h" |
| #else |
| extern void avdt_rcv_sync_info(BT_HDR* p_buf); /* this is for hci_test */ |
| #endif |
| |
| #if (MCA_INCLUDED == TRUE) |
| #include "mca_api.h" |
| #include "mca_defs.h" |
| #include "mca_int.h" |
| #endif |
| |
| #include "bta_sys.h" |
| #include "btm_ble_int.h" |
| #include "gatt_int.h" |
| #include "smp_int.h" |
| |
| /* Define BTU storage area |
| */ |
| uint8_t btu_trace_level = HCI_INITIAL_TRACE_LEVEL; |
| |
| // Communication queue between btu_task and bta. |
| extern fixed_queue_t* btu_bta_msg_queue; |
| |
| // Communication queue between btu_task and hci. |
| extern fixed_queue_t* btu_hci_msg_queue; |
| |
| // General timer queue. |
| extern fixed_queue_t* btu_general_alarm_queue; |
| |
| extern fixed_queue_t* event_queue; |
| extern fixed_queue_t* btif_msg_queue; |
| |
| extern thread_t* bt_workqueue_thread; |
| |
| static void btu_hci_msg_process(BT_HDR* p_msg); |
| |
| void btu_hci_msg_ready(fixed_queue_t* queue, UNUSED_ATTR void* context) { |
| BT_HDR* p_msg = (BT_HDR*)fixed_queue_dequeue(queue); |
| btu_hci_msg_process(p_msg); |
| } |
| |
| void btu_bta_msg_ready(fixed_queue_t* queue, UNUSED_ATTR void* context) { |
| BT_HDR* p_msg = (BT_HDR*)fixed_queue_dequeue(queue); |
| bta_sys_event(p_msg); |
| } |
| |
| static void btu_hci_msg_process(BT_HDR* p_msg) { |
| /* Determine the input message type. */ |
| switch (p_msg->event & BT_EVT_MASK) { |
| case BTU_POST_TO_TASK_NO_GOOD_HORRIBLE_HACK: // TODO(zachoverflow): remove |
| // this |
| ((post_to_task_hack_t*)(&p_msg->data[0]))->callback(p_msg); |
| break; |
| case BT_EVT_TO_BTU_HCI_ACL: |
| /* All Acl Data goes to L2CAP */ |
| l2c_rcv_acl_data(p_msg); |
| break; |
| |
| case BT_EVT_TO_BTU_L2C_SEG_XMIT: |
| /* L2CAP segment transmit complete */ |
| l2c_link_segments_xmitted(p_msg); |
| break; |
| |
| case BT_EVT_TO_BTU_HCI_SCO: |
| #if (BTM_SCO_INCLUDED == TRUE) |
| btm_route_sco_data(p_msg); |
| break; |
| #endif |
| |
| case BT_EVT_TO_BTU_HCI_EVT: |
| btu_hcif_process_event((uint8_t)(p_msg->event & BT_SUB_EVT_MASK), p_msg); |
| osi_free(p_msg); |
| break; |
| |
| case BT_EVT_TO_BTU_HCI_CMD: |
| btu_hcif_send_cmd((uint8_t)(p_msg->event & BT_SUB_EVT_MASK), p_msg); |
| break; |
| |
| default: |
| osi_free(p_msg); |
| break; |
| } |
| } |
| |
| void btu_task_start_up(UNUSED_ATTR void* context) { |
| BT_TRACE(TRACE_LAYER_BTU, TRACE_TYPE_API, |
| "btu_task pending for preload complete event"); |
| |
| LOG_INFO(LOG_TAG, "Bluetooth chip preload is complete"); |
| |
| BT_TRACE(TRACE_LAYER_BTU, TRACE_TYPE_API, |
| "btu_task received preload complete event"); |
| |
| /* Initialize the mandatory core stack control blocks |
| (BTU, BTM, L2CAP, and SDP) |
| */ |
| btu_init_core(); |
| |
| /* Initialize any optional stack components */ |
| BTE_InitStack(); |
| |
| bta_sys_init(); |
| |
| /* Initialise platform trace levels at this point as BTE_InitStack() and |
| * bta_sys_init() |
| * reset the control blocks and preset the trace level with |
| * XXX_INITIAL_TRACE_LEVEL |
| */ |
| module_init(get_module(BTE_LOGMSG_MODULE)); |
| |
| // Inform the bt jni thread initialization is ok. |
| btif_transfer_context(btif_init_ok, 0, NULL, 0, NULL); |
| |
| fixed_queue_register_dequeue(btu_bta_msg_queue, |
| thread_get_reactor(bt_workqueue_thread), |
| btu_bta_msg_ready, NULL); |
| |
| fixed_queue_register_dequeue(btu_hci_msg_queue, |
| thread_get_reactor(bt_workqueue_thread), |
| btu_hci_msg_ready, NULL); |
| |
| alarm_register_processing_queue(btu_general_alarm_queue, bt_workqueue_thread); |
| } |
| |
| void btu_task_shut_down(UNUSED_ATTR void* context) { |
| fixed_queue_unregister_dequeue(btu_bta_msg_queue); |
| fixed_queue_unregister_dequeue(btu_hci_msg_queue); |
| alarm_unregister_processing_queue(btu_general_alarm_queue); |
| |
| module_clean_up(get_module(BTE_LOGMSG_MODULE)); |
| |
| bta_sys_free(); |
| btu_free_core(); |
| } |