| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | // Copyright 2015-2016 Espressif Systems (Shanghai) PTE LTD
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // 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.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <stddef.h>
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <stdio.h>
 | 
					
						
							| 
									
										
										
										
											2017-04-05 21:19:15 +08:00
										 |  |  | #include <string.h>
 | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-30 11:39:42 +08:00
										 |  |  | #include "sdkconfig.h"
 | 
					
						
							|  |  |  | #include "esp_heap_caps.h"
 | 
					
						
							| 
									
										
										
										
											2017-09-12 22:36:17 +08:00
										 |  |  | #include "esp_heap_caps_init.h"
 | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | #include "freertos/FreeRTOS.h"
 | 
					
						
							|  |  |  | #include "freertos/task.h"
 | 
					
						
							|  |  |  | #include "freertos/queue.h"
 | 
					
						
							|  |  |  | #include "freertos/semphr.h"
 | 
					
						
							|  |  |  | #include "freertos/xtensa_api.h"
 | 
					
						
							|  |  |  | #include "freertos/portmacro.h"
 | 
					
						
							| 
									
										
										
										
											2017-10-20 17:09:03 +08:00
										 |  |  | #include "xtensa/core-macros.h"
 | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | #include "esp_types.h"
 | 
					
						
							|  |  |  | #include "esp_system.h"
 | 
					
						
							|  |  |  | #include "esp_task.h"
 | 
					
						
							| 
									
										
										
										
											2019-03-26 16:30:43 +08:00
										 |  |  | #include "esp_intr_alloc.h"
 | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | #include "esp_attr.h"
 | 
					
						
							| 
									
										
										
										
											2017-02-16 22:06:02 +08:00
										 |  |  | #include "esp_phy_init.h"
 | 
					
						
							| 
									
										
										
										
											2017-12-07 21:48:27 +08:00
										 |  |  | #include "esp_bt.h"
 | 
					
						
							| 
									
										
										
										
											2017-09-12 22:36:17 +08:00
										 |  |  | #include "esp_err.h"
 | 
					
						
							|  |  |  | #include "esp_log.h"
 | 
					
						
							| 
									
										
										
										
											2017-10-09 15:34:31 +08:00
										 |  |  | #include "esp_pm.h"
 | 
					
						
							| 
									
										
										
										
											2017-11-01 17:05:38 +08:00
										 |  |  | #include "driver/periph_ctrl.h"
 | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  | #include "soc/rtc.h"
 | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | #include "soc/soc_memory_layout.h"
 | 
					
						
							| 
									
										
										
										
											2019-03-18 15:46:15 +08:00
										 |  |  | #include "esp32/clk.h"
 | 
					
						
							| 
									
										
										
										
											2019-02-20 21:01:27 +08:00
										 |  |  | #include "esp_coexist_internal.h"
 | 
					
						
							| 
									
										
										
										
											2019-08-15 12:35:59 +05:30
										 |  |  | #if !CONFIG_FREERTOS_UNICORE
 | 
					
						
							|  |  |  | #include "esp_ipc.h"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-21 13:07:34 +08:00
										 |  |  | #include "esp_rom_sys.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | #if CONFIG_BT_ENABLED
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | /* Macro definition
 | 
					
						
							|  |  |  |  ************************************************************************ | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-12 22:36:17 +08:00
										 |  |  | #define BTDM_LOG_TAG                        "BTDM_INIT"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-13 17:14:50 +08:00
										 |  |  | #define BTDM_INIT_PERIOD                    (5000)    /* ms */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-21 17:46:59 +08:00
										 |  |  | /* Bluetooth system and controller config */ | 
					
						
							| 
									
										
										
										
											2017-09-19 20:10:35 +08:00
										 |  |  | #define BTDM_CFG_BT_DATA_RELEASE            (1<<0)
 | 
					
						
							|  |  |  | #define BTDM_CFG_HCI_UART                   (1<<1)
 | 
					
						
							|  |  |  | #define BTDM_CFG_CONTROLLER_RUN_APP_CPU     (1<<2)
 | 
					
						
							| 
									
										
										
										
											2018-05-03 20:22:08 +08:00
										 |  |  | #define BTDM_CFG_SCAN_DUPLICATE_OPTIONS     (1<<3)
 | 
					
						
							|  |  |  | #define BTDM_CFG_SEND_ADV_RESERVED_SIZE     (1<<4)
 | 
					
						
							| 
									
										
										
										
											2018-11-28 20:00:40 +08:00
										 |  |  | #define BTDM_CFG_BLE_FULL_SCAN_SUPPORTED    (1<<5)
 | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | /* Sleep mode */ | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  | #define BTDM_MODEM_SLEEP_MODE_NONE          (0)
 | 
					
						
							|  |  |  | #define BTDM_MODEM_SLEEP_MODE_ORIG          (1)
 | 
					
						
							| 
									
										
										
										
											2018-12-21 19:30:20 +08:00
										 |  |  | #define BTDM_MODEM_SLEEP_MODE_EVED          (2)  // sleep mode for BLE controller, used only for internal test.
 | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | /* Low Power Clock Selection */ | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  | #define BTDM_LPCLK_SEL_XTAL      (0)
 | 
					
						
							|  |  |  | #define BTDM_LPCLK_SEL_XTAL32K   (1)
 | 
					
						
							|  |  |  | #define BTDM_LPCLK_SEL_RTC_SLOW  (2)
 | 
					
						
							|  |  |  | #define BTDM_LPCLK_SEL_8M        (3)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  | /* Sleep and wakeup interval control */ | 
					
						
							|  |  |  | #define BTDM_MIN_SLEEP_DURATION          (12) // threshold of interval in slots to allow to fall into modem sleep
 | 
					
						
							|  |  |  | #define BTDM_MODEM_WAKE_UP_DELAY         (4)  // delay in slots of modem wake up procedure, including re-enable PHY/RF
 | 
					
						
							| 
									
										
										
										
											2017-09-19 20:10:35 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | #define BT_DEBUG(...)
 | 
					
						
							|  |  |  | #define BT_API_CALL_CHECK(info, api_call, ret) \
 | 
					
						
							|  |  |  | do{\ | 
					
						
							|  |  |  |     esp_err_t __err = (api_call);\ | 
					
						
							|  |  |  |     if ((ret) != __err) {\ | 
					
						
							| 
									
										
										
										
											2017-11-06 18:22:45 +08:00
										 |  |  |         BT_DEBUG("%s %d %s ret=0x%X\n", __FUNCTION__, __LINE__, (info), __err);\ | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  |         return __err;\ | 
					
						
							|  |  |  |     }\ | 
					
						
							|  |  |  | } while(0) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  | #define OSI_FUNCS_TIME_BLOCKING  0xffffffff
 | 
					
						
							| 
									
										
										
										
											2021-09-23 16:48:54 +08:00
										 |  |  | #define OSI_VERSION              0x00010003
 | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | #define OSI_MAGIC_VALUE          0xFADEBEAD
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* SPIRAM Configuration */ | 
					
						
							|  |  |  | #if CONFIG_SPIRAM_USE_MALLOC
 | 
					
						
							|  |  |  | #define BTDM_MAX_QUEUE_NUM       (5)
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Types definition
 | 
					
						
							|  |  |  |  ************************************************************************ | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | /* VHCI function interface */ | 
					
						
							|  |  |  | typedef struct vhci_host_callback { | 
					
						
							|  |  |  |     void (*notify_host_send_available)(void);               /*!< callback used to notify that the host can send packet to controller */ | 
					
						
							|  |  |  |     int (*notify_host_recv)(uint8_t *data, uint16_t len);   /*!< callback used to notify that the controller has a packet to send to the host*/ | 
					
						
							|  |  |  | } vhci_host_callback_t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Dram region */ | 
					
						
							| 
									
										
										
										
											2017-09-19 20:10:35 +08:00
										 |  |  | typedef struct { | 
					
						
							|  |  |  |     esp_bt_mode_t mode; | 
					
						
							|  |  |  |     intptr_t start; | 
					
						
							|  |  |  |     intptr_t end; | 
					
						
							|  |  |  | } btdm_dram_available_region_t; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | /* PSRAM configuration */ | 
					
						
							| 
									
										
										
										
											2018-03-30 11:39:42 +08:00
										 |  |  | #if CONFIG_SPIRAM_USE_MALLOC
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							|  |  |  |     QueueHandle_t handle; | 
					
						
							|  |  |  |     void *storage; | 
					
						
							|  |  |  |     void *buffer; | 
					
						
							|  |  |  | } btdm_queue_item_t; | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2018-03-30 11:39:42 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | /* OSI function */ | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | struct osi_funcs_t { | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  |     uint32_t _version; | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  |     xt_handler (*_set_isr)(int n, xt_handler f, void *arg); | 
					
						
							|  |  |  |     void (*_ints_on)(unsigned int mask); | 
					
						
							|  |  |  |     void (*_interrupt_disable)(void); | 
					
						
							|  |  |  |     void (*_interrupt_restore)(void); | 
					
						
							|  |  |  |     void (*_task_yield)(void); | 
					
						
							| 
									
										
										
										
											2017-06-07 14:58:17 +08:00
										 |  |  |     void (*_task_yield_from_isr)(void); | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  |     void *(*_semphr_create)(uint32_t max, uint32_t init); | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  |     void (*_semphr_delete)(void *semphr); | 
					
						
							|  |  |  |     int32_t (*_semphr_take_from_isr)(void *semphr, void *hptw); | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  |     int32_t (*_semphr_give_from_isr)(void *semphr, void *hptw); | 
					
						
							|  |  |  |     int32_t (*_semphr_take)(void *semphr, uint32_t block_time_ms); | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  |     int32_t (*_semphr_give)(void *semphr); | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  |     void *(*_mutex_create)(void); | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  |     void (*_mutex_delete)(void *mutex); | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  |     int32_t (*_mutex_lock)(void *mutex); | 
					
						
							|  |  |  |     int32_t (*_mutex_unlock)(void *mutex); | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  |     void *(* _queue_create)(uint32_t queue_len, uint32_t item_size); | 
					
						
							|  |  |  |     void (* _queue_delete)(void *queue); | 
					
						
							|  |  |  |     int32_t (* _queue_send)(void *queue, void *item, uint32_t block_time_ms); | 
					
						
							|  |  |  |     int32_t (* _queue_send_from_isr)(void *queue, void *item, void *hptw); | 
					
						
							|  |  |  |     int32_t (* _queue_recv)(void *queue, void *item, uint32_t block_time_ms); | 
					
						
							|  |  |  |     int32_t (* _queue_recv_from_isr)(void *queue, void *item, void *hptw); | 
					
						
							|  |  |  |     int32_t (* _task_create)(void *task_func, const char *name, uint32_t stack_depth, void *param, uint32_t prio, void *task_handle, uint32_t core_id); | 
					
						
							|  |  |  |     void (* _task_delete)(void *task_handle); | 
					
						
							| 
									
										
										
										
											2017-07-25 19:58:39 +08:00
										 |  |  |     bool (* _is_in_isr)(void); | 
					
						
							| 
									
										
										
										
											2017-10-20 17:09:03 +08:00
										 |  |  |     int (* _cause_sw_intr_to_core)(int core_id, int intr_no); | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  |     void *(* _malloc)(uint32_t size); | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  |     void *(* _malloc_internal)(uint32_t size); | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  |     void (* _free)(void *p); | 
					
						
							| 
									
										
										
										
											2017-03-01 21:04:12 +08:00
										 |  |  |     int32_t (* _read_efuse_mac)(uint8_t mac[6]); | 
					
						
							| 
									
										
										
										
											2017-03-24 14:57:07 +08:00
										 |  |  |     void (* _srand)(unsigned int seed); | 
					
						
							|  |  |  |     int (* _rand)(void); | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  |     uint32_t (* _btdm_lpcycles_2_us)(uint32_t cycles); | 
					
						
							|  |  |  |     uint32_t (* _btdm_us_2_lpcycles)(uint32_t us); | 
					
						
							|  |  |  |     bool (* _btdm_sleep_check_duration)(uint32_t *slot_cnt); | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  |     void (* _btdm_sleep_enter_phase1)(uint32_t lpcycles);  /* called when interrupt is disabled */ | 
					
						
							|  |  |  |     void (* _btdm_sleep_enter_phase2)(void); | 
					
						
							|  |  |  |     void (* _btdm_sleep_exit_phase1)(void);  /* called from ISR */ | 
					
						
							|  |  |  |     void (* _btdm_sleep_exit_phase2)(void);  /* called from ISR */ | 
					
						
							|  |  |  |     void (* _btdm_sleep_exit_phase3)(void);  /* called from task */ | 
					
						
							| 
									
										
										
										
											2020-03-20 17:58:05 +08:00
										 |  |  |     bool (* _coex_bt_wakeup_request)(void); | 
					
						
							|  |  |  |     void (* _coex_bt_wakeup_request_end)(void); | 
					
						
							| 
									
										
										
										
											2019-02-20 21:01:27 +08:00
										 |  |  |     int (* _coex_bt_request)(uint32_t event, uint32_t latency, uint32_t duration); | 
					
						
							|  |  |  |     int (* _coex_bt_release)(uint32_t event); | 
					
						
							|  |  |  |     int (* _coex_register_bt_cb)(coex_func_cb_t cb); | 
					
						
							|  |  |  |     uint32_t (* _coex_bb_reset_lock)(void); | 
					
						
							|  |  |  |     void (* _coex_bb_reset_unlock)(uint32_t restore); | 
					
						
							| 
									
										
										
										
											2021-03-23 16:55:30 +08:00
										 |  |  |     int (* _coex_schm_register_btdm_callback)(void *callback); | 
					
						
							|  |  |  |     void (* _coex_schm_status_bit_clear)(uint32_t type, uint32_t status); | 
					
						
							|  |  |  |     void (* _coex_schm_status_bit_set)(uint32_t type, uint32_t status); | 
					
						
							|  |  |  |     uint32_t (* _coex_schm_interval_get)(void); | 
					
						
							|  |  |  |     uint8_t (* _coex_schm_curr_period_get)(void); | 
					
						
							|  |  |  |     void *(* _coex_schm_curr_phase_get)(void); | 
					
						
							|  |  |  |     int (* _coex_wifi_channel_get)(uint8_t *primary, uint8_t *secondary); | 
					
						
							|  |  |  |     int (* _coex_register_wifi_channel_change_callback)(void *cb); | 
					
						
							| 
									
										
										
										
											2021-09-23 16:48:54 +08:00
										 |  |  |     xt_handler (*_set_isr_l3)(int n, xt_handler f, void *arg); | 
					
						
							|  |  |  |     void (*_interrupt_l3_disable)(void); | 
					
						
							|  |  |  |     void (*_interrupt_l3_restore)(void); | 
					
						
							|  |  |  |     void *(* _customer_queue_create)(uint32_t queue_len, uint32_t item_size); | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  |     uint32_t _magic; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  | typedef void (*workitem_handler_t)(void* arg); | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* External functions or values
 | 
					
						
							|  |  |  |  ************************************************************************ | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* not for user call, so don't put to include file */ | 
					
						
							|  |  |  | /* OSI */ | 
					
						
							|  |  |  | extern int btdm_osi_funcs_register(void *osi_funcs); | 
					
						
							|  |  |  | /* Initialise and De-initialise */ | 
					
						
							|  |  |  | extern int btdm_controller_init(uint32_t config_mask, esp_bt_controller_config_t *config_opts); | 
					
						
							|  |  |  | extern void btdm_controller_deinit(void); | 
					
						
							|  |  |  | extern int btdm_controller_enable(esp_bt_mode_t mode); | 
					
						
							|  |  |  | extern void btdm_controller_disable(void); | 
					
						
							|  |  |  | extern uint8_t btdm_controller_get_mode(void); | 
					
						
							|  |  |  | extern const char *btdm_controller_get_compile_version(void); | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  | extern void btdm_rf_bb_init_phase2(void); // shall be called after PHY/RF is enabled
 | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  | extern int btdm_dispatch_work_to_controller(workitem_handler_t callback, void *arg, bool blocking); | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | /* Sleep */ | 
					
						
							|  |  |  | extern void btdm_controller_enable_sleep(bool enable); | 
					
						
							|  |  |  | extern void btdm_controller_set_sleep_mode(uint8_t mode); | 
					
						
							|  |  |  | extern uint8_t btdm_controller_get_sleep_mode(void); | 
					
						
							|  |  |  | extern bool btdm_power_state_active(void); | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  | extern void btdm_wakeup_request(void); | 
					
						
							|  |  |  | extern void btdm_in_wakeup_requesting_set(bool in_wakeup_requesting); | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | /* Low Power Clock */ | 
					
						
							|  |  |  | extern bool btdm_lpclk_select_src(uint32_t sel); | 
					
						
							|  |  |  | extern bool btdm_lpclk_set_div(uint32_t div); | 
					
						
							|  |  |  | /* VHCI */ | 
					
						
							|  |  |  | extern bool API_vhci_host_check_send_available(void); | 
					
						
							|  |  |  | extern void API_vhci_host_send_packet(uint8_t *data, uint16_t len); | 
					
						
							|  |  |  | extern int API_vhci_host_register_callback(const vhci_host_callback_t *callback); | 
					
						
							|  |  |  | /* TX power */ | 
					
						
							|  |  |  | extern int ble_txpwr_set(int power_type, int power_level); | 
					
						
							|  |  |  | extern int ble_txpwr_get(int power_type); | 
					
						
							|  |  |  | extern int bredr_txpwr_set(int min_power_level, int max_power_level); | 
					
						
							|  |  |  | extern int bredr_txpwr_get(int *min_power_level, int *max_power_level); | 
					
						
							|  |  |  | extern void bredr_sco_datapath_set(uint8_t data_path); | 
					
						
							| 
									
										
										
										
											2018-08-20 15:04:37 +08:00
										 |  |  | extern void btdm_controller_scan_duplicate_list_clear(void); | 
					
						
							| 
									
										
										
										
											2019-02-20 21:01:27 +08:00
										 |  |  | /* Coexistence */ | 
					
						
							| 
									
										
										
										
											2020-08-26 17:51:13 +08:00
										 |  |  | extern int coex_bt_request(uint32_t event, uint32_t latency, uint32_t duration); | 
					
						
							|  |  |  | extern int coex_bt_release(uint32_t event); | 
					
						
							|  |  |  | extern int coex_register_bt_cb(coex_func_cb_t cb); | 
					
						
							|  |  |  | extern uint32_t coex_bb_reset_lock(void); | 
					
						
							|  |  |  | extern void coex_bb_reset_unlock(uint32_t restore); | 
					
						
							| 
									
										
										
										
											2021-03-23 16:55:30 +08:00
										 |  |  | extern int coex_schm_register_btdm_callback(void *callback); | 
					
						
							|  |  |  | extern void coex_schm_status_bit_clear(uint32_t type, uint32_t status); | 
					
						
							|  |  |  | extern void coex_schm_status_bit_set(uint32_t type, uint32_t status); | 
					
						
							|  |  |  | extern uint32_t coex_schm_interval_get(void); | 
					
						
							|  |  |  | extern uint8_t coex_schm_curr_period_get(void); | 
					
						
							|  |  |  | extern void * coex_schm_curr_phase_get(void); | 
					
						
							|  |  |  | extern int coex_wifi_channel_get(uint8_t *primary, uint8_t *secondary); | 
					
						
							|  |  |  | extern int coex_register_wifi_channel_change_callback(void *cb); | 
					
						
							| 
									
										
										
										
											2021-11-01 20:39:37 +08:00
										 |  |  | /* Shutdown */ | 
					
						
							|  |  |  | extern void esp_bt_controller_shutdown(void); | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | extern char _bss_start_btdm; | 
					
						
							|  |  |  | extern char _bss_end_btdm; | 
					
						
							|  |  |  | extern char _data_start_btdm; | 
					
						
							|  |  |  | extern char _data_end_btdm; | 
					
						
							|  |  |  | extern uint32_t _data_start_btdm_rom; | 
					
						
							|  |  |  | extern uint32_t _data_end_btdm_rom; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | extern uint32_t _bt_bss_start; | 
					
						
							|  |  |  | extern uint32_t _bt_bss_end; | 
					
						
							| 
									
										
										
										
											2019-06-25 09:03:58 +08:00
										 |  |  | extern uint32_t _nimble_bss_start; | 
					
						
							|  |  |  | extern uint32_t _nimble_bss_end; | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | extern uint32_t _btdm_bss_start; | 
					
						
							|  |  |  | extern uint32_t _btdm_bss_end; | 
					
						
							|  |  |  | extern uint32_t _bt_data_start; | 
					
						
							|  |  |  | extern uint32_t _bt_data_end; | 
					
						
							| 
									
										
										
										
											2019-06-25 09:03:58 +08:00
										 |  |  | extern uint32_t _nimble_data_start; | 
					
						
							|  |  |  | extern uint32_t _nimble_data_end; | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | extern uint32_t _btdm_data_start; | 
					
						
							|  |  |  | extern uint32_t _btdm_data_end; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Local Function Declare
 | 
					
						
							|  |  |  |  ********************************************************************* | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #if CONFIG_SPIRAM_USE_MALLOC
 | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static bool btdm_queue_generic_register(const btdm_queue_item_t *queue); | 
					
						
							|  |  |  | static bool btdm_queue_generic_deregister(btdm_queue_item_t *queue); | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | #endif /* CONFIG_SPIRAM_USE_MALLOC */
 | 
					
						
							|  |  |  | static void IRAM_ATTR interrupt_disable(void); | 
					
						
							|  |  |  | static void IRAM_ATTR interrupt_restore(void); | 
					
						
							| 
									
										
										
										
											2021-09-23 16:48:54 +08:00
										 |  |  | static void IRAM_ATTR task_yield(void); | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | static void IRAM_ATTR task_yield_from_isr(void); | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static void *semphr_create_wrapper(uint32_t max, uint32_t init); | 
					
						
							|  |  |  | static void semphr_delete_wrapper(void *semphr); | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | static int32_t IRAM_ATTR semphr_take_from_isr_wrapper(void *semphr, void *hptw); | 
					
						
							|  |  |  | static int32_t IRAM_ATTR semphr_give_from_isr_wrapper(void *semphr, void *hptw); | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static int32_t  semphr_take_wrapper(void *semphr, uint32_t block_time_ms); | 
					
						
							|  |  |  | static int32_t  semphr_give_wrapper(void *semphr); | 
					
						
							|  |  |  | static void *mutex_create_wrapper(void); | 
					
						
							|  |  |  | static void mutex_delete_wrapper(void *mutex); | 
					
						
							|  |  |  | static int32_t mutex_lock_wrapper(void *mutex); | 
					
						
							|  |  |  | static int32_t mutex_unlock_wrapper(void *mutex); | 
					
						
							|  |  |  | static void *queue_create_wrapper(uint32_t queue_len, uint32_t item_size); | 
					
						
							|  |  |  | static void queue_delete_wrapper(void *queue); | 
					
						
							|  |  |  | static int32_t queue_send_wrapper(void *queue, void *item, uint32_t block_time_ms); | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | static int32_t IRAM_ATTR queue_send_from_isr_wrapper(void *queue, void *item, void *hptw); | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static int32_t queue_recv_wrapper(void *queue, void *item, uint32_t block_time_ms); | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | static int32_t IRAM_ATTR queue_recv_from_isr_wrapper(void *queue, void *item, void *hptw); | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static int32_t task_create_wrapper(void *task_func, const char *name, uint32_t stack_depth, void *param, uint32_t prio, void *task_handle, uint32_t core_id); | 
					
						
							|  |  |  | static void task_delete_wrapper(void *task_handle); | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | static bool IRAM_ATTR is_in_isr_wrapper(void); | 
					
						
							|  |  |  | static void IRAM_ATTR cause_sw_intr(void *arg); | 
					
						
							|  |  |  | static int IRAM_ATTR cause_sw_intr_to_core_wrapper(int core_id, int intr_no); | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static void *malloc_internal_wrapper(size_t size); | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | static int32_t IRAM_ATTR read_mac_wrapper(uint8_t mac[6]); | 
					
						
							|  |  |  | static void IRAM_ATTR srand_wrapper(unsigned int seed); | 
					
						
							|  |  |  | static int IRAM_ATTR rand_wrapper(void); | 
					
						
							|  |  |  | static uint32_t IRAM_ATTR btdm_lpcycles_2_us(uint32_t cycles); | 
					
						
							|  |  |  | static uint32_t IRAM_ATTR btdm_us_2_lpcycles(uint32_t us); | 
					
						
							|  |  |  | static bool IRAM_ATTR btdm_sleep_check_duration(uint32_t *slot_cnt); | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  | static void btdm_sleep_enter_phase1_wrapper(uint32_t lpcycles); | 
					
						
							|  |  |  | static void btdm_sleep_enter_phase2_wrapper(void); | 
					
						
							|  |  |  | static void btdm_sleep_exit_phase3_wrapper(void); | 
					
						
							| 
									
										
										
										
											2020-03-20 17:58:05 +08:00
										 |  |  | static bool coex_bt_wakeup_request(void); | 
					
						
							|  |  |  | static void coex_bt_wakeup_request_end(void); | 
					
						
							| 
									
										
										
										
											2020-08-26 17:51:13 +08:00
										 |  |  | static int coex_bt_request_wrapper(uint32_t event, uint32_t latency, uint32_t duration); | 
					
						
							|  |  |  | static int coex_bt_release_wrapper(uint32_t event); | 
					
						
							|  |  |  | static int coex_register_bt_cb_wrapper(coex_func_cb_t cb); | 
					
						
							|  |  |  | static uint32_t coex_bb_reset_lock_wrapper(void); | 
					
						
							|  |  |  | static void coex_bb_reset_unlock_wrapper(uint32_t restore); | 
					
						
							| 
									
										
										
										
											2021-03-23 16:55:30 +08:00
										 |  |  | static int coex_schm_register_btdm_callback_wrapper(void *callback); | 
					
						
							|  |  |  | static void coex_schm_status_bit_clear_wrapper(uint32_t type, uint32_t status); | 
					
						
							|  |  |  | static void coex_schm_status_bit_set_wrapper(uint32_t type, uint32_t status); | 
					
						
							|  |  |  | static uint32_t coex_schm_interval_get_wrapper(void); | 
					
						
							|  |  |  | static uint8_t coex_schm_curr_period_get_wrapper(void); | 
					
						
							|  |  |  | static void * coex_schm_curr_phase_get_wrapper(void); | 
					
						
							|  |  |  | static int coex_wifi_channel_get_wrapper(uint8_t *primary, uint8_t *secondary); | 
					
						
							|  |  |  | static int coex_register_wifi_channel_change_callback_wrapper(void *cb); | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | /* Local variable definition
 | 
					
						
							|  |  |  |  *************************************************************************** | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | /* OSI funcs */ | 
					
						
							|  |  |  | static const struct osi_funcs_t osi_funcs_ro = { | 
					
						
							|  |  |  |     ._version = OSI_VERSION, | 
					
						
							|  |  |  |     ._set_isr = xt_set_interrupt_handler, | 
					
						
							|  |  |  |     ._ints_on = xt_ints_on, | 
					
						
							|  |  |  |     ._interrupt_disable = interrupt_disable, | 
					
						
							|  |  |  |     ._interrupt_restore = interrupt_restore, | 
					
						
							| 
									
										
										
										
											2021-09-23 16:48:54 +08:00
										 |  |  |     ._task_yield = task_yield, | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  |     ._task_yield_from_isr = task_yield_from_isr, | 
					
						
							|  |  |  |     ._semphr_create = semphr_create_wrapper, | 
					
						
							|  |  |  |     ._semphr_delete = semphr_delete_wrapper, | 
					
						
							|  |  |  |     ._semphr_take_from_isr = semphr_take_from_isr_wrapper, | 
					
						
							|  |  |  |     ._semphr_give_from_isr = semphr_give_from_isr_wrapper, | 
					
						
							|  |  |  |     ._semphr_take = semphr_take_wrapper, | 
					
						
							|  |  |  |     ._semphr_give = semphr_give_wrapper, | 
					
						
							|  |  |  |     ._mutex_create = mutex_create_wrapper, | 
					
						
							|  |  |  |     ._mutex_delete = mutex_delete_wrapper, | 
					
						
							|  |  |  |     ._mutex_lock = mutex_lock_wrapper, | 
					
						
							|  |  |  |     ._mutex_unlock = mutex_unlock_wrapper, | 
					
						
							|  |  |  |     ._queue_create = queue_create_wrapper, | 
					
						
							|  |  |  |     ._queue_delete = queue_delete_wrapper, | 
					
						
							|  |  |  |     ._queue_send = queue_send_wrapper, | 
					
						
							|  |  |  |     ._queue_send_from_isr = queue_send_from_isr_wrapper, | 
					
						
							|  |  |  |     ._queue_recv = queue_recv_wrapper, | 
					
						
							|  |  |  |     ._queue_recv_from_isr = queue_recv_from_isr_wrapper, | 
					
						
							|  |  |  |     ._task_create = task_create_wrapper, | 
					
						
							|  |  |  |     ._task_delete = task_delete_wrapper, | 
					
						
							|  |  |  |     ._is_in_isr = is_in_isr_wrapper, | 
					
						
							|  |  |  |     ._cause_sw_intr_to_core = cause_sw_intr_to_core_wrapper, | 
					
						
							|  |  |  |     ._malloc = malloc, | 
					
						
							|  |  |  |     ._malloc_internal = malloc_internal_wrapper, | 
					
						
							|  |  |  |     ._free = free, | 
					
						
							|  |  |  |     ._read_efuse_mac = read_mac_wrapper, | 
					
						
							|  |  |  |     ._srand = srand_wrapper, | 
					
						
							|  |  |  |     ._rand = rand_wrapper, | 
					
						
							|  |  |  |     ._btdm_lpcycles_2_us = btdm_lpcycles_2_us, | 
					
						
							|  |  |  |     ._btdm_us_2_lpcycles = btdm_us_2_lpcycles, | 
					
						
							|  |  |  |     ._btdm_sleep_check_duration = btdm_sleep_check_duration, | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  |     ._btdm_sleep_enter_phase1 = btdm_sleep_enter_phase1_wrapper, | 
					
						
							|  |  |  |     ._btdm_sleep_enter_phase2 = btdm_sleep_enter_phase2_wrapper, | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  |     ._btdm_sleep_exit_phase1 = NULL, | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  |     ._btdm_sleep_exit_phase2 = NULL, | 
					
						
							|  |  |  |     ._btdm_sleep_exit_phase3 = btdm_sleep_exit_phase3_wrapper, | 
					
						
							| 
									
										
										
										
											2020-03-20 17:58:05 +08:00
										 |  |  |     ._coex_bt_wakeup_request = coex_bt_wakeup_request, | 
					
						
							|  |  |  |     ._coex_bt_wakeup_request_end = coex_bt_wakeup_request_end, | 
					
						
							| 
									
										
										
										
											2019-02-20 21:01:27 +08:00
										 |  |  |     ._coex_bt_request = coex_bt_request_wrapper, | 
					
						
							|  |  |  |     ._coex_bt_release = coex_bt_release_wrapper, | 
					
						
							|  |  |  |     ._coex_register_bt_cb = coex_register_bt_cb_wrapper, | 
					
						
							|  |  |  |     ._coex_bb_reset_lock = coex_bb_reset_lock_wrapper, | 
					
						
							|  |  |  |     ._coex_bb_reset_unlock = coex_bb_reset_unlock_wrapper, | 
					
						
							| 
									
										
										
										
											2021-03-23 16:55:30 +08:00
										 |  |  |     ._coex_schm_register_btdm_callback = coex_schm_register_btdm_callback_wrapper, | 
					
						
							|  |  |  |     ._coex_schm_status_bit_clear = coex_schm_status_bit_clear_wrapper, | 
					
						
							|  |  |  |     ._coex_schm_status_bit_set = coex_schm_status_bit_set_wrapper, | 
					
						
							|  |  |  |     ._coex_schm_interval_get = coex_schm_interval_get_wrapper, | 
					
						
							|  |  |  |     ._coex_schm_curr_period_get = coex_schm_curr_period_get_wrapper, | 
					
						
							|  |  |  |     ._coex_schm_curr_phase_get = coex_schm_curr_phase_get_wrapper, | 
					
						
							|  |  |  |     ._coex_wifi_channel_get = coex_wifi_channel_get_wrapper, | 
					
						
							|  |  |  |     ._coex_register_wifi_channel_change_callback = coex_register_wifi_channel_change_callback_wrapper, | 
					
						
							| 
									
										
										
										
											2021-09-23 16:48:54 +08:00
										 |  |  |     ._set_isr_l3 = xt_set_interrupt_handler, | 
					
						
							|  |  |  |     ._interrupt_l3_disable = interrupt_disable, | 
					
						
							|  |  |  |     ._interrupt_l3_restore = interrupt_restore, | 
					
						
							|  |  |  |     ._customer_queue_create = NULL, | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  |     ._magic = OSI_MAGIC_VALUE, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* the mode column will be modified by release function to indicate the available region */ | 
					
						
							|  |  |  | static btdm_dram_available_region_t btdm_dram_available_region[] = { | 
					
						
							|  |  |  |     //following is .data
 | 
					
						
							|  |  |  |     {ESP_BT_MODE_BTDM,          SOC_MEM_BT_DATA_START,      SOC_MEM_BT_DATA_END         }, | 
					
						
							|  |  |  |     //following is memory which HW will use
 | 
					
						
							|  |  |  |     {ESP_BT_MODE_BTDM,          SOC_MEM_BT_EM_BTDM0_START,  SOC_MEM_BT_EM_BTDM0_END     }, | 
					
						
							|  |  |  |     {ESP_BT_MODE_BLE,           SOC_MEM_BT_EM_BLE_START,    SOC_MEM_BT_EM_BLE_END      }, | 
					
						
							|  |  |  |     {ESP_BT_MODE_BTDM,          SOC_MEM_BT_EM_BTDM1_START,  SOC_MEM_BT_EM_BTDM1_END     }, | 
					
						
							|  |  |  |     {ESP_BT_MODE_CLASSIC_BT,    SOC_MEM_BT_EM_BREDR_START,  SOC_MEM_BT_EM_BREDR_REAL_END}, | 
					
						
							|  |  |  |     //following is .bss
 | 
					
						
							|  |  |  |     {ESP_BT_MODE_BTDM,          SOC_MEM_BT_BSS_START,       SOC_MEM_BT_BSS_END          }, | 
					
						
							|  |  |  |     {ESP_BT_MODE_BTDM,          SOC_MEM_BT_MISC_START,      SOC_MEM_BT_MISC_END         }, | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | /* Reserve the full memory region used by Bluetooth Controller,
 | 
					
						
							|  |  |  |  *    some may be released later at runtime. */ | 
					
						
							|  |  |  | SOC_RESERVE_MEMORY_REGION(SOC_MEM_BT_EM_START,   SOC_MEM_BT_EM_BREDR_REAL_END,  rom_bt_em); | 
					
						
							|  |  |  | SOC_RESERVE_MEMORY_REGION(SOC_MEM_BT_BSS_START,  SOC_MEM_BT_BSS_END,            rom_bt_bss); | 
					
						
							|  |  |  | SOC_RESERVE_MEMORY_REGION(SOC_MEM_BT_MISC_START, SOC_MEM_BT_MISC_END,           rom_bt_misc); | 
					
						
							|  |  |  | SOC_RESERVE_MEMORY_REGION(SOC_MEM_BT_DATA_START, SOC_MEM_BT_DATA_END,           rom_bt_data); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-14 11:29:00 +08:00
										 |  |  | static DRAM_ATTR struct osi_funcs_t *osi_funcs_p; | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if CONFIG_SPIRAM_USE_MALLOC
 | 
					
						
							| 
									
										
										
										
											2018-12-14 11:29:00 +08:00
										 |  |  | static DRAM_ATTR btdm_queue_item_t btdm_queue_table[BTDM_MAX_QUEUE_NUM]; | 
					
						
							|  |  |  | static DRAM_ATTR SemaphoreHandle_t btdm_queue_table_mux = NULL; | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | #endif /* #if CONFIG_SPIRAM_USE_MALLOC */
 | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-17 19:24:58 +08:00
										 |  |  | /* Static variable declare */ | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  | // timestamp when PHY/RF was switched on
 | 
					
						
							| 
									
										
										
										
											2018-12-14 11:29:00 +08:00
										 |  |  | static DRAM_ATTR int64_t s_time_phy_rf_just_enabled = 0; | 
					
						
							|  |  |  | static DRAM_ATTR esp_bt_controller_status_t btdm_controller_status = ESP_BT_CONTROLLER_STATUS_IDLE; | 
					
						
							| 
									
										
										
										
											2017-02-17 19:24:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-14 11:29:00 +08:00
										 |  |  | static DRAM_ATTR portMUX_TYPE global_int_mux = portMUX_INITIALIZER_UNLOCKED; | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  | // measured average low power clock period in micro seconds
 | 
					
						
							| 
									
										
										
										
											2018-12-14 11:29:00 +08:00
										 |  |  | static DRAM_ATTR uint32_t btdm_lpcycle_us = 0; | 
					
						
							|  |  |  | static DRAM_ATTR uint8_t btdm_lpcycle_us_frac = 0; // number of fractional bit for btdm_lpcycle_us
 | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-09 20:58:13 +08:00
										 |  |  | #if CONFIG_BTDM_CTRL_MODEM_SLEEP_MODE_ORIG
 | 
					
						
							| 
									
										
										
										
											2020-02-20 14:34:28 +08:00
										 |  |  | // used low power clock
 | 
					
						
							|  |  |  | static DRAM_ATTR uint8_t btdm_lpclk_sel; | 
					
						
							| 
									
										
										
										
											2020-07-09 20:58:13 +08:00
										 |  |  | #endif /* #ifdef CONFIG_BTDM_CTRL_MODEM_SLEEP_MODE_ORIG */
 | 
					
						
							| 
									
										
										
										
											2020-02-20 14:34:28 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  | static DRAM_ATTR QueueHandle_t s_wakeup_req_sem = NULL; | 
					
						
							| 
									
										
										
										
											2017-10-09 15:34:31 +08:00
										 |  |  | #ifdef CONFIG_PM_ENABLE
 | 
					
						
							| 
									
										
										
										
											2018-12-14 11:29:00 +08:00
										 |  |  | static DRAM_ATTR esp_timer_handle_t s_btdm_slp_tmr; | 
					
						
							|  |  |  | static DRAM_ATTR esp_pm_lock_handle_t s_pm_lock; | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  | static bool s_pm_lock_acquired = true; | 
					
						
							| 
									
										
										
										
											2020-02-20 14:34:28 +08:00
										 |  |  | static DRAM_ATTR bool s_btdm_allow_light_sleep; | 
					
						
							| 
									
										
										
										
											2019-04-09 19:48:13 +08:00
										 |  |  | // pm_lock to prevent light sleep when using main crystal as Bluetooth low power clock
 | 
					
						
							|  |  |  | static DRAM_ATTR esp_pm_lock_handle_t s_light_sleep_pm_lock; | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  | static void btdm_slp_tmr_callback(void *arg); | 
					
						
							| 
									
										
										
										
											2020-02-20 14:34:28 +08:00
										 |  |  | #endif /* #ifdef CONFIG_PM_ENABLE */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-09 15:34:31 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  | static inline void btdm_check_and_init_bb(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* init BT-BB if PHY/RF has been switched off since last BT-BB init */ | 
					
						
							|  |  |  |     int64_t latest_ts = esp_phy_rf_get_on_ts(); | 
					
						
							|  |  |  |     if (latest_ts != s_time_phy_rf_just_enabled || | 
					
						
							|  |  |  |         s_time_phy_rf_just_enabled == 0) { | 
					
						
							|  |  |  |         btdm_rf_bb_init_phase2(); | 
					
						
							|  |  |  |         s_time_phy_rf_just_enabled = latest_ts; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-30 11:39:42 +08:00
										 |  |  | #if CONFIG_SPIRAM_USE_MALLOC
 | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static bool btdm_queue_generic_register(const btdm_queue_item_t *queue) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:39:42 +08:00
										 |  |  | { | 
					
						
							|  |  |  |     if (!btdm_queue_table_mux || !queue) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool ret = false; | 
					
						
							|  |  |  |     btdm_queue_item_t *item; | 
					
						
							|  |  |  |     xSemaphoreTake(btdm_queue_table_mux, portMAX_DELAY); | 
					
						
							|  |  |  |     for (int i = 0; i < BTDM_MAX_QUEUE_NUM; ++i) { | 
					
						
							|  |  |  |         item = &btdm_queue_table[i]; | 
					
						
							|  |  |  |         if (item->handle == NULL) { | 
					
						
							|  |  |  |             memcpy(item, queue, sizeof(btdm_queue_item_t)); | 
					
						
							|  |  |  |             ret = true; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     xSemaphoreGive(btdm_queue_table_mux); | 
					
						
							|  |  |  |     return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static bool btdm_queue_generic_deregister(btdm_queue_item_t *queue) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:39:42 +08:00
										 |  |  | { | 
					
						
							|  |  |  |     if (!btdm_queue_table_mux || !queue) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool ret = false; | 
					
						
							|  |  |  |     btdm_queue_item_t *item; | 
					
						
							|  |  |  |     xSemaphoreTake(btdm_queue_table_mux, portMAX_DELAY); | 
					
						
							|  |  |  |     for (int i = 0; i < BTDM_MAX_QUEUE_NUM; ++i) { | 
					
						
							|  |  |  |         item = &btdm_queue_table[i]; | 
					
						
							|  |  |  |         if (item->handle == queue->handle) { | 
					
						
							|  |  |  |             memcpy(queue, item, sizeof(btdm_queue_item_t)); | 
					
						
							|  |  |  |             memset(item, 0, sizeof(btdm_queue_item_t)); | 
					
						
							|  |  |  |             ret = true; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     xSemaphoreGive(btdm_queue_table_mux); | 
					
						
							|  |  |  |     return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif /* CONFIG_SPIRAM_USE_MALLOC */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | static void IRAM_ATTR interrupt_disable(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-07-01 19:05:00 +05:30
										 |  |  |     if (xPortInIsrContext()) { | 
					
						
							|  |  |  |         portENTER_CRITICAL_ISR(&global_int_mux); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         portENTER_CRITICAL(&global_int_mux); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void IRAM_ATTR interrupt_restore(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-07-01 19:05:00 +05:30
										 |  |  |     if (xPortInIsrContext()) { | 
					
						
							|  |  |  |         portEXIT_CRITICAL_ISR(&global_int_mux); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         portEXIT_CRITICAL(&global_int_mux); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-23 16:48:54 +08:00
										 |  |  | static void IRAM_ATTR task_yield(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     vPortYield(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-07 14:58:17 +08:00
										 |  |  | static void IRAM_ATTR task_yield_from_isr(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     portYIELD_FROM_ISR(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static void *semphr_create_wrapper(uint32_t max, uint32_t init) | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-30 11:39:42 +08:00
										 |  |  | #if !CONFIG_SPIRAM_USE_MALLOC
 | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  |     return (void *)xSemaphoreCreateCounting(max, init); | 
					
						
							| 
									
										
										
										
											2018-03-30 11:39:42 +08:00
										 |  |  | #else
 | 
					
						
							|  |  |  |     StaticQueue_t *queue_buffer = NULL; | 
					
						
							|  |  |  |     QueueHandle_t handle = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     queue_buffer = heap_caps_malloc(sizeof(StaticQueue_t), MALLOC_CAP_INTERNAL|MALLOC_CAP_8BIT); | 
					
						
							|  |  |  |     if (!queue_buffer) { | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     handle = xSemaphoreCreateCountingStatic(max, init, queue_buffer); | 
					
						
							|  |  |  |     if (!handle) { | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     btdm_queue_item_t item = { | 
					
						
							|  |  |  |         .handle = handle, | 
					
						
							|  |  |  |         .storage = NULL, | 
					
						
							|  |  |  |         .buffer = queue_buffer, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!btdm_queue_generic_register(&item)) { | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return handle; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  error: | 
					
						
							|  |  |  |     if (handle) { | 
					
						
							|  |  |  |         vSemaphoreDelete(handle); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (queue_buffer) { | 
					
						
							|  |  |  |         free(queue_buffer); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static void semphr_delete_wrapper(void *semphr) | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-30 11:39:42 +08:00
										 |  |  | #if !CONFIG_SPIRAM_USE_MALLOC
 | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  |     vSemaphoreDelete(semphr); | 
					
						
							| 
									
										
										
										
											2018-03-30 11:39:42 +08:00
										 |  |  | #else
 | 
					
						
							|  |  |  |     btdm_queue_item_t item = { | 
					
						
							|  |  |  |         .handle = semphr, | 
					
						
							|  |  |  |         .storage = NULL, | 
					
						
							|  |  |  |         .buffer = NULL, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (btdm_queue_generic_deregister(&item)) { | 
					
						
							|  |  |  |         vSemaphoreDelete(item.handle); | 
					
						
							|  |  |  |         free(item.buffer); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int32_t IRAM_ATTR semphr_take_from_isr_wrapper(void *semphr, void *hptw) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return (int32_t)xSemaphoreTakeFromISR(semphr, hptw); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | static int32_t IRAM_ATTR semphr_give_from_isr_wrapper(void *semphr, void *hptw) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return (int32_t)xSemaphoreGiveFromISR(semphr, hptw); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static int32_t semphr_take_wrapper(void *semphr, uint32_t block_time_ms) | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  |     if (block_time_ms == OSI_FUNCS_TIME_BLOCKING) { | 
					
						
							|  |  |  |         return (int32_t)xSemaphoreTake(semphr, portMAX_DELAY); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         return (int32_t)xSemaphoreTake(semphr, block_time_ms / portTICK_PERIOD_MS); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static int32_t semphr_give_wrapper(void *semphr) | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  | { | 
					
						
							|  |  |  |     return (int32_t)xSemaphoreGive(semphr); | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static void *mutex_create_wrapper(void) | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-30 11:39:42 +08:00
										 |  |  | #if CONFIG_SPIRAM_USE_MALLOC
 | 
					
						
							|  |  |  |     StaticQueue_t *queue_buffer = NULL; | 
					
						
							|  |  |  |     QueueHandle_t handle = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     queue_buffer = heap_caps_malloc(sizeof(StaticQueue_t), MALLOC_CAP_INTERNAL|MALLOC_CAP_8BIT); | 
					
						
							|  |  |  |     if (!queue_buffer) { | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     handle = xSemaphoreCreateMutexStatic(queue_buffer); | 
					
						
							|  |  |  |     if (!handle) { | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     btdm_queue_item_t item = { | 
					
						
							|  |  |  |         .handle = handle, | 
					
						
							|  |  |  |         .storage = NULL, | 
					
						
							|  |  |  |         .buffer = queue_buffer, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!btdm_queue_generic_register(&item)) { | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return handle; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  error: | 
					
						
							|  |  |  |     if (handle) { | 
					
						
							|  |  |  |         vSemaphoreDelete(handle); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (queue_buffer) { | 
					
						
							|  |  |  |         free(queue_buffer); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  |     return (void *)xSemaphoreCreateMutex(); | 
					
						
							| 
									
										
										
										
											2018-03-30 11:39:42 +08:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static void mutex_delete_wrapper(void *mutex) | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-30 11:39:42 +08:00
										 |  |  | #if !CONFIG_SPIRAM_USE_MALLOC
 | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  |     vSemaphoreDelete(mutex); | 
					
						
							| 
									
										
										
										
											2018-03-30 11:39:42 +08:00
										 |  |  | #else
 | 
					
						
							|  |  |  |     btdm_queue_item_t item = { | 
					
						
							|  |  |  |         .handle = mutex, | 
					
						
							|  |  |  |         .storage = NULL, | 
					
						
							|  |  |  |         .buffer = NULL, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (btdm_queue_generic_deregister(&item)) { | 
					
						
							|  |  |  |         vSemaphoreDelete(item.handle); | 
					
						
							|  |  |  |         free(item.buffer); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static int32_t mutex_lock_wrapper(void *mutex) | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | { | 
					
						
							|  |  |  |     return (int32_t)xSemaphoreTake(mutex, portMAX_DELAY); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static int32_t mutex_unlock_wrapper(void *mutex) | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | { | 
					
						
							|  |  |  |     return (int32_t)xSemaphoreGive(mutex); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static void *queue_create_wrapper(uint32_t queue_len, uint32_t item_size) | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-30 11:39:42 +08:00
										 |  |  | #if CONFIG_SPIRAM_USE_MALLOC
 | 
					
						
							|  |  |  |     StaticQueue_t *queue_buffer = NULL; | 
					
						
							|  |  |  |     uint8_t *queue_storage = NULL; | 
					
						
							|  |  |  |     QueueHandle_t handle = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     queue_buffer = heap_caps_malloc(sizeof(StaticQueue_t), MALLOC_CAP_INTERNAL|MALLOC_CAP_8BIT); | 
					
						
							|  |  |  |     if (!queue_buffer) { | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     queue_storage = heap_caps_malloc((queue_len*item_size), MALLOC_CAP_INTERNAL|MALLOC_CAP_8BIT); | 
					
						
							|  |  |  |     if (!queue_storage ) { | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     handle = xQueueCreateStatic(queue_len, item_size, queue_storage, queue_buffer); | 
					
						
							|  |  |  |     if (!handle) { | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     btdm_queue_item_t item = { | 
					
						
							|  |  |  |         .handle = handle, | 
					
						
							|  |  |  |         .storage = queue_storage, | 
					
						
							|  |  |  |         .buffer = queue_buffer, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!btdm_queue_generic_register(&item)) { | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return handle; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  error: | 
					
						
							|  |  |  |     if (handle) { | 
					
						
							|  |  |  |         vQueueDelete(handle); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (queue_storage) { | 
					
						
							|  |  |  |         free(queue_storage); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (queue_buffer) { | 
					
						
							|  |  |  |         free(queue_buffer); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  |     return (void *)xQueueCreate(queue_len, item_size); | 
					
						
							| 
									
										
										
										
											2018-03-30 11:39:42 +08:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static void queue_delete_wrapper(void *queue) | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-30 11:39:42 +08:00
										 |  |  | #if !CONFIG_SPIRAM_USE_MALLOC
 | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  |     vQueueDelete(queue); | 
					
						
							| 
									
										
										
										
											2018-03-30 11:39:42 +08:00
										 |  |  | #else
 | 
					
						
							|  |  |  |     btdm_queue_item_t item = { | 
					
						
							|  |  |  |         .handle = queue, | 
					
						
							|  |  |  |         .storage = NULL, | 
					
						
							|  |  |  |         .buffer = NULL, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (btdm_queue_generic_deregister(&item)) { | 
					
						
							|  |  |  |         vQueueDelete(item.handle); | 
					
						
							|  |  |  |         free(item.storage); | 
					
						
							|  |  |  |         free(item.buffer); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static int32_t queue_send_wrapper(void *queue, void *item, uint32_t block_time_ms) | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  | { | 
					
						
							|  |  |  |     if (block_time_ms == OSI_FUNCS_TIME_BLOCKING) { | 
					
						
							|  |  |  |         return (int32_t)xQueueSend(queue, item, portMAX_DELAY); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         return (int32_t)xQueueSend(queue, item, block_time_ms / portTICK_PERIOD_MS); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int32_t IRAM_ATTR queue_send_from_isr_wrapper(void *queue, void *item, void *hptw) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return (int32_t)xQueueSendFromISR(queue, item, hptw); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static int32_t queue_recv_wrapper(void *queue, void *item, uint32_t block_time_ms) | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  | { | 
					
						
							|  |  |  |     if (block_time_ms == OSI_FUNCS_TIME_BLOCKING) { | 
					
						
							|  |  |  |         return (int32_t)xQueueReceive(queue, item, portMAX_DELAY); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         return (int32_t)xQueueReceive(queue, item, block_time_ms / portTICK_PERIOD_MS); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int32_t IRAM_ATTR queue_recv_from_isr_wrapper(void *queue, void *item, void *hptw) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return (int32_t)xQueueReceiveFromISR(queue, item, hptw); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static int32_t task_create_wrapper(void *task_func, const char *name, uint32_t stack_depth, void *param, uint32_t prio, void *task_handle, uint32_t core_id) | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  | { | 
					
						
							|  |  |  |     return (uint32_t)xTaskCreatePinnedToCore(task_func, name, stack_depth, param, prio, task_handle, (core_id < portNUM_PROCESSORS ? core_id : tskNO_AFFINITY)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static void task_delete_wrapper(void *task_handle) | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  | { | 
					
						
							|  |  |  |     vTaskDelete(task_handle); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-25 19:58:39 +08:00
										 |  |  | static bool IRAM_ATTR is_in_isr_wrapper(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-10-28 18:43:35 +08:00
										 |  |  |     return !xPortCanYield(); | 
					
						
							| 
									
										
										
										
											2017-07-25 19:58:39 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-20 17:09:03 +08:00
										 |  |  | static void IRAM_ATTR cause_sw_intr(void *arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* just convert void * to int, because the width is the same */ | 
					
						
							|  |  |  |     uint32_t intr_no = (uint32_t)arg; | 
					
						
							|  |  |  |     XTHAL_SET_INTSET((1<<intr_no)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int IRAM_ATTR cause_sw_intr_to_core_wrapper(int core_id, int intr_no) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     esp_err_t err = ESP_OK; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-15 12:35:59 +05:30
										 |  |  | #if CONFIG_FREERTOS_UNICORE
 | 
					
						
							|  |  |  |     cause_sw_intr((void *)intr_no); | 
					
						
							|  |  |  | #else /* CONFIG_FREERTOS_UNICORE */
 | 
					
						
							| 
									
										
										
										
											2017-10-20 17:09:03 +08:00
										 |  |  |     if (xPortGetCoreID() == core_id) { | 
					
						
							|  |  |  |         cause_sw_intr((void *)intr_no); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         err = esp_ipc_call(core_id, cause_sw_intr, (void *)intr_no); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-08-15 12:35:59 +05:30
										 |  |  | #endif /* !CONFIG_FREERTOS_UNICORE */
 | 
					
						
							| 
									
										
										
										
											2017-10-20 17:09:03 +08:00
										 |  |  |     return err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-29 16:54:32 +08:00
										 |  |  | static void *malloc_internal_wrapper(size_t size) | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-10-31 15:19:17 +11:00
										 |  |  |     return heap_caps_malloc(size, MALLOC_CAP_8BIT|MALLOC_CAP_DMA|MALLOC_CAP_INTERNAL); | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-01 21:04:12 +08:00
										 |  |  | static int32_t IRAM_ATTR read_mac_wrapper(uint8_t mac[6]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return esp_read_mac(mac, ESP_MAC_BT); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-24 14:57:07 +08:00
										 |  |  | static void IRAM_ATTR srand_wrapper(unsigned int seed) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* empty function */ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int IRAM_ATTR rand_wrapper(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return (int)esp_random(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  | static uint32_t IRAM_ATTR btdm_lpcycles_2_us(uint32_t cycles) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  |     // The number of lp cycles should not lead to overflow. Thrs: 100s
 | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  |     // clock measurement is conducted
 | 
					
						
							|  |  |  |     uint64_t us = (uint64_t)btdm_lpcycle_us * cycles; | 
					
						
							|  |  |  |     us = (us + (1 << (btdm_lpcycle_us_frac - 1))) >> btdm_lpcycle_us_frac; | 
					
						
							|  |  |  |     return (uint32_t)us; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * @brief Converts a duration in slots into a number of low power clock cycles. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static uint32_t IRAM_ATTR btdm_us_2_lpcycles(uint32_t us) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-06-27 17:23:23 +08:00
										 |  |  |     // The number of sleep duration(us) should not lead to overflow. Thrs: 100s
 | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  |     // Compute the sleep duration in us to low power clock cycles, with calibration result applied
 | 
					
						
							|  |  |  |     // clock measurement is conducted
 | 
					
						
							|  |  |  |     uint64_t cycles = ((uint64_t)(us) << btdm_lpcycle_us_frac) / btdm_lpcycle_us; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return (uint32_t)cycles; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool IRAM_ATTR btdm_sleep_check_duration(uint32_t *slot_cnt) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (*slot_cnt < BTDM_MIN_SLEEP_DURATION) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  |     /* wake up in advance considering the delay in enabling PHY/RF */ | 
					
						
							|  |  |  |     *slot_cnt -= BTDM_MODEM_WAKE_UP_DELAY; | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  | static void btdm_sleep_enter_phase1_wrapper(uint32_t lpcycles) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifdef CONFIG_PM_ENABLE
 | 
					
						
							|  |  |  |     // start a timer to wake up and acquire the pm_lock before modem_sleep awakes
 | 
					
						
							|  |  |  |     uint32_t us_to_sleep = btdm_lpcycles_2_us(lpcycles); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-09 17:06:16 +08:00
										 |  |  | #define BTDM_MIN_TIMER_UNCERTAINTY_US      (500)
 | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  |     assert(us_to_sleep > BTDM_MIN_TIMER_UNCERTAINTY_US); | 
					
						
							|  |  |  |     // allow a maximum time uncertainty to be about 488ppm(1/2048) at least as clock drift
 | 
					
						
							|  |  |  |     // and set the timer in advance
 | 
					
						
							|  |  |  |     uint32_t uncertainty = (us_to_sleep >> 11); | 
					
						
							|  |  |  |     if (uncertainty < BTDM_MIN_TIMER_UNCERTAINTY_US) { | 
					
						
							|  |  |  |         uncertainty = BTDM_MIN_TIMER_UNCERTAINTY_US; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (esp_timer_start_once(s_btdm_slp_tmr, us_to_sleep - uncertainty) != ESP_OK) { | 
					
						
							|  |  |  |         ESP_LOGW(BTDM_LOG_TAG, "timer start failed"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void btdm_sleep_enter_phase2_wrapper(void) | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  | { | 
					
						
							|  |  |  |     if (btdm_controller_get_sleep_mode() == BTDM_MODEM_SLEEP_MODE_ORIG) { | 
					
						
							| 
									
										
										
										
											2020-05-22 17:15:28 +08:00
										 |  |  |         esp_phy_disable(); | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  | #ifdef CONFIG_PM_ENABLE
 | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  |         if (s_pm_lock_acquired) { | 
					
						
							|  |  |  |             esp_pm_lock_release(s_pm_lock); | 
					
						
							|  |  |  |             s_pm_lock_acquired = false; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  |     } else if (btdm_controller_get_sleep_mode() == BTDM_MODEM_SLEEP_MODE_EVED) { | 
					
						
							| 
									
										
										
										
											2020-05-22 17:15:28 +08:00
										 |  |  |         esp_phy_disable(); | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  |         // pause bluetooth baseband
 | 
					
						
							|  |  |  |         periph_module_disable(PERIPH_BT_BASEBAND_MODULE); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  | static void btdm_sleep_exit_phase3_wrapper(void) | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  | { | 
					
						
							|  |  |  | #ifdef CONFIG_PM_ENABLE
 | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  |     if (!s_pm_lock_acquired) { | 
					
						
							|  |  |  |         s_pm_lock_acquired = true; | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  |         esp_pm_lock_acquire(s_pm_lock); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  |     if (btdm_controller_get_sleep_mode() == BTDM_MODEM_SLEEP_MODE_ORIG) { | 
					
						
							| 
									
										
										
										
											2020-05-22 17:15:28 +08:00
										 |  |  |         esp_phy_enable(); | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  |         btdm_check_and_init_bb(); | 
					
						
							|  |  |  | #ifdef CONFIG_PM_ENABLE
 | 
					
						
							|  |  |  |         esp_timer_stop(s_btdm_slp_tmr); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  |     } else if (btdm_controller_get_sleep_mode() == BTDM_MODEM_SLEEP_MODE_EVED) { | 
					
						
							|  |  |  |         // resume bluetooth baseband
 | 
					
						
							|  |  |  |         periph_module_enable(PERIPH_BT_BASEBAND_MODULE); | 
					
						
							| 
									
										
										
										
											2020-05-22 17:15:28 +08:00
										 |  |  |         esp_phy_enable(); | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  | #ifdef CONFIG_PM_ENABLE
 | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  | static void btdm_slp_tmr_customer_callback(void * arg) | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  |     (void)(arg); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!s_pm_lock_acquired) { | 
					
						
							|  |  |  |         s_pm_lock_acquired = true; | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  |         esp_pm_lock_acquire(s_pm_lock); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void IRAM_ATTR btdm_slp_tmr_callback(void *arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     (void)(arg); | 
					
						
							|  |  |  |     btdm_dispatch_work_to_controller(btdm_slp_tmr_customer_callback, NULL, true); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  | #define BTDM_ASYNC_WAKEUP_REQ_HCI       0
 | 
					
						
							|  |  |  | #define BTDM_ASYNC_WAKEUP_REQ_COEX      1
 | 
					
						
							|  |  |  | #define BTDM_ASYNC_WAKEUP_REQ_CTRL_DISA 2
 | 
					
						
							|  |  |  | #define BTDM_ASYNC_WAKEUP_REQMAX        3
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void btdm_wakeup_request_callback(void * arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     (void)(arg); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if CONFIG_PM_ENABLE
 | 
					
						
							|  |  |  |     if (!s_pm_lock_acquired) { | 
					
						
							|  |  |  |         s_pm_lock_acquired = true; | 
					
						
							|  |  |  |         esp_pm_lock_acquire(s_pm_lock); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     esp_timer_stop(s_btdm_slp_tmr); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     btdm_wakeup_request(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     semphr_give_wrapper(s_wakeup_req_sem); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2017-01-03 15:53:06 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-20 17:58:05 +08:00
										 |  |  | static bool async_wakeup_request(int event) | 
					
						
							| 
									
										
										
										
											2017-01-03 15:53:06 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  |     bool do_wakeup_request = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-20 17:58:05 +08:00
										 |  |  |     switch (event) { | 
					
						
							|  |  |  |         case BTDM_ASYNC_WAKEUP_REQ_HCI: | 
					
						
							| 
									
										
										
										
											2020-10-14 20:20:57 +08:00
										 |  |  |             btdm_in_wakeup_requesting_set(true); | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  |             // NO break
 | 
					
						
							|  |  |  |         case BTDM_ASYNC_WAKEUP_REQ_CTRL_DISA: | 
					
						
							|  |  |  |             if (!btdm_power_state_active()) { | 
					
						
							|  |  |  |                 do_wakeup_request = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 btdm_dispatch_work_to_controller(btdm_wakeup_request_callback, NULL, true); | 
					
						
							|  |  |  |                 semphr_take_wrapper(s_wakeup_req_sem, OSI_FUNCS_TIME_BLOCKING); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-03-20 17:58:05 +08:00
										 |  |  |             break; | 
					
						
							|  |  |  |         case BTDM_ASYNC_WAKEUP_REQ_COEX: | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  |             if (!btdm_power_state_active()) { | 
					
						
							|  |  |  |                 do_wakeup_request = true; | 
					
						
							|  |  |  | #if CONFIG_PM_ENABLE
 | 
					
						
							|  |  |  |                 if (!s_pm_lock_acquired) { | 
					
						
							|  |  |  |                     s_pm_lock_acquired = true; | 
					
						
							|  |  |  |                     esp_pm_lock_acquire(s_pm_lock); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 esp_timer_stop(s_btdm_slp_tmr); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |                 btdm_wakeup_request(); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-03-20 17:58:05 +08:00
										 |  |  |             break; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return do_wakeup_request; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void async_wakeup_request_end(int event) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     bool request_lock = false; | 
					
						
							|  |  |  |     switch (event) { | 
					
						
							|  |  |  |         case BTDM_ASYNC_WAKEUP_REQ_HCI: | 
					
						
							|  |  |  |             request_lock = true; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case BTDM_ASYNC_WAKEUP_REQ_COEX: | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  |         case BTDM_ASYNC_WAKEUP_REQ_CTRL_DISA: | 
					
						
							| 
									
										
										
										
											2020-03-20 17:58:05 +08:00
										 |  |  |             request_lock = false; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (request_lock) { | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  |         btdm_in_wakeup_requesting_set(false); | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-07-27 17:47:33 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-20 17:58:05 +08:00
										 |  |  |     return; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool coex_bt_wakeup_request(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return async_wakeup_request(BTDM_ASYNC_WAKEUP_REQ_COEX); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void coex_bt_wakeup_request_end(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     async_wakeup_request_end(BTDM_ASYNC_WAKEUP_REQ_COEX); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-23 16:55:30 +08:00
										 |  |  | static int IRAM_ATTR coex_bt_request_wrapper(uint32_t event, uint32_t latency, uint32_t duration) | 
					
						
							| 
									
										
										
										
											2020-08-26 17:51:13 +08:00
										 |  |  | { | 
					
						
							|  |  |  | #if CONFIG_SW_COEXIST_ENABLE
 | 
					
						
							|  |  |  |     return coex_bt_request(event, latency, duration); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-23 16:55:30 +08:00
										 |  |  | static int IRAM_ATTR coex_bt_release_wrapper(uint32_t event) | 
					
						
							| 
									
										
										
										
											2020-08-26 17:51:13 +08:00
										 |  |  | { | 
					
						
							|  |  |  | #if CONFIG_SW_COEXIST_ENABLE
 | 
					
						
							|  |  |  |     return coex_bt_release(event); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-23 16:55:30 +08:00
										 |  |  | static int coex_register_bt_cb_wrapper(coex_func_cb_t cb) | 
					
						
							| 
									
										
										
										
											2020-08-26 17:51:13 +08:00
										 |  |  | { | 
					
						
							|  |  |  | #if CONFIG_SW_COEXIST_ENABLE
 | 
					
						
							|  |  |  |     return coex_register_bt_cb(cb); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-23 16:55:30 +08:00
										 |  |  | static uint32_t IRAM_ATTR coex_bb_reset_lock_wrapper(void) | 
					
						
							| 
									
										
										
										
											2020-08-26 17:51:13 +08:00
										 |  |  | { | 
					
						
							|  |  |  | #if CONFIG_SW_COEXIST_ENABLE
 | 
					
						
							|  |  |  |     return coex_bb_reset_lock(); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-23 16:55:30 +08:00
										 |  |  | static void IRAM_ATTR coex_bb_reset_unlock_wrapper(uint32_t restore) | 
					
						
							| 
									
										
										
										
											2020-08-26 17:51:13 +08:00
										 |  |  | { | 
					
						
							|  |  |  | #if CONFIG_SW_COEXIST_ENABLE
 | 
					
						
							|  |  |  |     coex_bb_reset_unlock(restore); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-23 16:55:30 +08:00
										 |  |  | static int coex_schm_register_btdm_callback_wrapper(void *callback) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #if CONFIG_SW_COEXIST_ENABLE
 | 
					
						
							|  |  |  |     return coex_schm_register_btdm_callback(callback); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void coex_schm_status_bit_clear_wrapper(uint32_t type, uint32_t status) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #if CONFIG_SW_COEXIST_ENABLE
 | 
					
						
							|  |  |  |     coex_schm_status_bit_clear(type, status); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void coex_schm_status_bit_set_wrapper(uint32_t type, uint32_t status) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #if CONFIG_SW_COEXIST_ENABLE
 | 
					
						
							|  |  |  |     coex_schm_status_bit_set(type, status); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static uint32_t coex_schm_interval_get_wrapper(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #if CONFIG_SW_COEXIST_ENABLE
 | 
					
						
							|  |  |  |     return coex_schm_interval_get(); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static uint8_t coex_schm_curr_period_get_wrapper(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #if CONFIG_SW_COEXIST_ENABLE
 | 
					
						
							|  |  |  |     return coex_schm_curr_period_get(); | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2021-03-23 21:11:12 +08:00
										 |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2021-03-23 16:55:30 +08:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void * coex_schm_curr_phase_get_wrapper(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #if CONFIG_SW_COEXIST_ENABLE
 | 
					
						
							|  |  |  |     return coex_schm_curr_phase_get(); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int coex_wifi_channel_get_wrapper(uint8_t *primary, uint8_t *secondary) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #if CONFIG_SW_COEXIST_ENABLE
 | 
					
						
							|  |  |  |     return coex_wifi_channel_get(primary, secondary); | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2021-03-23 21:11:12 +08:00
										 |  |  |     return -1; | 
					
						
							| 
									
										
										
										
											2021-03-23 16:55:30 +08:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int coex_register_wifi_channel_change_callback_wrapper(void *cb) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #if CONFIG_SW_COEXIST_ENABLE
 | 
					
						
							|  |  |  |     return coex_register_wifi_channel_change_callback(cb); | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2021-03-23 21:11:12 +08:00
										 |  |  |     return -1; | 
					
						
							| 
									
										
										
										
											2021-03-23 16:55:30 +08:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-20 17:58:05 +08:00
										 |  |  | bool esp_vhci_host_check_send_available(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return API_vhci_host_check_send_available(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void esp_vhci_host_send_packet(uint8_t *data, uint16_t len) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-10-14 20:20:57 +08:00
										 |  |  |     async_wakeup_request(BTDM_ASYNC_WAKEUP_REQ_HCI); | 
					
						
							| 
									
										
										
										
											2020-03-20 17:58:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-03 15:53:06 +08:00
										 |  |  |     API_vhci_host_send_packet(data, len); | 
					
						
							| 
									
										
										
										
											2019-07-27 17:47:33 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-14 20:20:57 +08:00
										 |  |  |     async_wakeup_request_end(BTDM_ASYNC_WAKEUP_REQ_HCI); | 
					
						
							| 
									
										
										
										
											2017-01-03 15:53:06 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  | esp_err_t esp_vhci_host_register_callback(const esp_vhci_host_callback_t *callback) | 
					
						
							| 
									
										
										
										
											2017-01-03 15:53:06 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  |     return API_vhci_host_register_callback((const vhci_host_callback_t *)callback) == 0 ? ESP_OK : ESP_FAIL; | 
					
						
							| 
									
										
										
										
											2017-01-03 15:53:06 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-21 17:46:59 +08:00
										 |  |  | static uint32_t btdm_config_mask_load(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint32_t mask = 0x0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-02 15:36:06 +02:00
										 |  |  | #if CONFIG_BTDM_CTRL_HCI_MODE_UART_H4
 | 
					
						
							| 
									
										
										
										
											2017-03-16 17:14:20 +08:00
										 |  |  |     mask |= BTDM_CFG_HCI_UART; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-05-02 15:36:06 +02:00
										 |  |  | #if CONFIG_BTDM_CTRL_PINNED_TO_CORE == 1
 | 
					
						
							| 
									
										
										
										
											2017-03-16 17:14:20 +08:00
										 |  |  |     mask |= BTDM_CFG_CONTROLLER_RUN_APP_CPU; | 
					
						
							| 
									
										
										
										
											2017-02-21 17:46:59 +08:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-05-02 15:36:06 +02:00
										 |  |  | #if CONFIG_BTDM_CTRL_FULL_SCAN_SUPPORTED
 | 
					
						
							| 
									
										
										
										
											2018-11-28 20:00:40 +08:00
										 |  |  |     mask |= BTDM_CFG_BLE_FULL_SCAN_SUPPORTED; | 
					
						
							| 
									
										
										
										
											2019-05-02 15:36:06 +02:00
										 |  |  | #endif /* CONFIG_BTDM_CTRL_FULL_SCAN_SUPPORTED */
 | 
					
						
							| 
									
										
										
										
											2018-05-03 20:22:08 +08:00
										 |  |  |     mask |= BTDM_CFG_SCAN_DUPLICATE_OPTIONS; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     mask |= BTDM_CFG_SEND_ADV_RESERVED_SIZE; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-21 17:46:59 +08:00
										 |  |  |     return mask; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-19 20:10:35 +08:00
										 |  |  | static void btdm_controller_mem_init(void) | 
					
						
							| 
									
										
										
										
											2017-09-12 22:36:17 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  |     /* initialise .data section */ | 
					
						
							| 
									
										
										
										
											2017-09-19 20:10:35 +08:00
										 |  |  |     memcpy(&_data_start_btdm, (void *)_data_start_btdm_rom, &_data_end_btdm - &_data_start_btdm); | 
					
						
							| 
									
										
										
										
											2018-08-28 13:31:07 +05:30
										 |  |  |     ESP_LOGD(BTDM_LOG_TAG, ".data initialise [0x%08x] <== [0x%08x]", (uint32_t)&_data_start_btdm, _data_start_btdm_rom); | 
					
						
							| 
									
										
										
										
											2017-09-19 20:10:35 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  |     //initial em, .bss section
 | 
					
						
							| 
									
										
										
										
											2017-09-19 20:10:35 +08:00
										 |  |  |     for (int i = 1; i < sizeof(btdm_dram_available_region)/sizeof(btdm_dram_available_region_t); i++) { | 
					
						
							|  |  |  |         if (btdm_dram_available_region[i].mode != ESP_BT_MODE_IDLE) { | 
					
						
							|  |  |  |             memset((void *)btdm_dram_available_region[i].start, 0x0, btdm_dram_available_region[i].end - btdm_dram_available_region[i].start); | 
					
						
							| 
									
										
										
										
											2018-08-28 13:31:07 +05:30
										 |  |  |             ESP_LOGD(BTDM_LOG_TAG, ".bss initialise [0x%08x] - [0x%08x]", btdm_dram_available_region[i].start, btdm_dram_available_region[i].end); | 
					
						
							| 
									
										
										
										
											2017-09-19 20:10:35 +08:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-28 13:31:07 +05:30
										 |  |  | static esp_err_t try_heap_caps_add_region(intptr_t start, intptr_t end) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int ret = heap_caps_add_region(start, end); | 
					
						
							|  |  |  |     /* heap_caps_add_region() returns ESP_ERR_INVALID_SIZE if the memory region is
 | 
					
						
							|  |  |  |      * is too small to fit a heap. This cannot be termed as a fatal error and hence | 
					
						
							|  |  |  |      * we replace it by ESP_OK | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     if (ret == ESP_ERR_INVALID_SIZE) { | 
					
						
							|  |  |  |         return ESP_OK; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-19 20:10:35 +08:00
										 |  |  | esp_err_t esp_bt_controller_mem_release(esp_bt_mode_t mode) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     bool update = true; | 
					
						
							| 
									
										
										
										
											2020-03-12 15:07:03 +08:00
										 |  |  |     intptr_t mem_start=(intptr_t) NULL, mem_end=(intptr_t) NULL; | 
					
						
							| 
									
										
										
										
											2017-09-19 20:10:35 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-12 15:24:43 +08:00
										 |  |  |     if (btdm_controller_status != ESP_BT_CONTROLLER_STATUS_IDLE) { | 
					
						
							|  |  |  |         return ESP_ERR_INVALID_STATE; | 
					
						
							| 
									
										
										
										
											2017-09-19 20:10:35 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-13 12:25:18 +05:30
										 |  |  |     //already released
 | 
					
						
							| 
									
										
										
										
											2017-09-19 20:10:35 +08:00
										 |  |  |     if (!(mode & btdm_dram_available_region[0].mode)) { | 
					
						
							|  |  |  |         return ESP_ERR_INVALID_STATE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (int i = 0; i < sizeof(btdm_dram_available_region)/sizeof(btdm_dram_available_region_t); i++) { | 
					
						
							|  |  |  |         //skip the share mode, idle mode and other mode
 | 
					
						
							|  |  |  |         if (btdm_dram_available_region[i].mode == ESP_BT_MODE_IDLE | 
					
						
							|  |  |  |                 || (mode & btdm_dram_available_region[i].mode) != btdm_dram_available_region[i].mode) { | 
					
						
							|  |  |  |             //clear the bit of the mode which will be released
 | 
					
						
							|  |  |  |             btdm_dram_available_region[i].mode &= ~mode; | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             //clear the bit of the mode which will be released
 | 
					
						
							|  |  |  |             btdm_dram_available_region[i].mode &= ~mode; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (update) { | 
					
						
							|  |  |  |             mem_start = btdm_dram_available_region[i].start; | 
					
						
							|  |  |  |             mem_end = btdm_dram_available_region[i].end; | 
					
						
							|  |  |  |             update = false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (i < sizeof(btdm_dram_available_region)/sizeof(btdm_dram_available_region_t) - 1) { | 
					
						
							|  |  |  |             mem_end = btdm_dram_available_region[i].end; | 
					
						
							|  |  |  |             if (btdm_dram_available_region[i+1].mode != ESP_BT_MODE_IDLE | 
					
						
							|  |  |  |                     && (mode & btdm_dram_available_region[i+1].mode) == btdm_dram_available_region[i+1].mode | 
					
						
							|  |  |  |                     && mem_end == btdm_dram_available_region[i+1].start) { | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2018-08-28 13:31:07 +05:30
										 |  |  |                 ESP_LOGD(BTDM_LOG_TAG, "Release DRAM [0x%08x] - [0x%08x]", mem_start, mem_end); | 
					
						
							|  |  |  |                 ESP_ERROR_CHECK(try_heap_caps_add_region(mem_start, mem_end)); | 
					
						
							| 
									
										
										
										
											2017-09-19 20:10:35 +08:00
										 |  |  |                 update = true; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             mem_end = btdm_dram_available_region[i].end; | 
					
						
							| 
									
										
										
										
											2018-08-28 13:31:07 +05:30
										 |  |  |             ESP_LOGD(BTDM_LOG_TAG, "Release DRAM [0x%08x] - [0x%08x]", mem_start, mem_end); | 
					
						
							|  |  |  |             ESP_ERROR_CHECK(try_heap_caps_add_region(mem_start, mem_end)); | 
					
						
							| 
									
										
										
										
											2017-09-19 20:10:35 +08:00
										 |  |  |             update = true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-11 11:52:32 +05:30
										 |  |  |     if (mode == ESP_BT_MODE_BTDM) { | 
					
						
							|  |  |  |         mem_start = (intptr_t)&_btdm_bss_start; | 
					
						
							|  |  |  |         mem_end = (intptr_t)&_btdm_bss_end; | 
					
						
							| 
									
										
										
										
											2018-08-17 08:49:48 +02:00
										 |  |  |         if (mem_start != mem_end) { | 
					
						
							| 
									
										
										
										
											2018-08-28 13:31:07 +05:30
										 |  |  |             ESP_LOGD(BTDM_LOG_TAG, "Release BTDM BSS [0x%08x] - [0x%08x]", mem_start, mem_end); | 
					
						
							|  |  |  |             ESP_ERROR_CHECK(try_heap_caps_add_region(mem_start, mem_end)); | 
					
						
							| 
									
										
										
										
											2018-08-17 08:49:48 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-07-11 11:52:32 +05:30
										 |  |  |         mem_start = (intptr_t)&_btdm_data_start; | 
					
						
							|  |  |  |         mem_end = (intptr_t)&_btdm_data_end; | 
					
						
							| 
									
										
										
										
											2018-08-17 08:49:48 +02:00
										 |  |  |         if (mem_start != mem_end) { | 
					
						
							| 
									
										
										
										
											2018-08-28 13:31:07 +05:30
										 |  |  |             ESP_LOGD(BTDM_LOG_TAG, "Release BTDM Data [0x%08x] - [0x%08x]", mem_start, mem_end); | 
					
						
							|  |  |  |             ESP_ERROR_CHECK(try_heap_caps_add_region(mem_start, mem_end)); | 
					
						
							| 
									
										
										
										
											2018-08-17 08:49:48 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-07-11 11:52:32 +05:30
										 |  |  |     } | 
					
						
							|  |  |  |     return ESP_OK; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | esp_err_t esp_bt_mem_release(esp_bt_mode_t mode) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int ret; | 
					
						
							|  |  |  |     intptr_t mem_start, mem_end; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ret = esp_bt_controller_mem_release(mode); | 
					
						
							|  |  |  |     if (ret != ESP_OK) { | 
					
						
							|  |  |  |         return ret; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (mode == ESP_BT_MODE_BTDM) { | 
					
						
							|  |  |  |         mem_start = (intptr_t)&_bt_bss_start; | 
					
						
							|  |  |  |         mem_end = (intptr_t)&_bt_bss_end; | 
					
						
							| 
									
										
										
										
											2018-08-17 08:49:48 +02:00
										 |  |  |         if (mem_start != mem_end) { | 
					
						
							| 
									
										
										
										
											2018-08-28 13:31:07 +05:30
										 |  |  |             ESP_LOGD(BTDM_LOG_TAG, "Release BT BSS [0x%08x] - [0x%08x]", mem_start, mem_end); | 
					
						
							|  |  |  |             ESP_ERROR_CHECK(try_heap_caps_add_region(mem_start, mem_end)); | 
					
						
							| 
									
										
										
										
											2018-08-17 08:49:48 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-07-11 11:52:32 +05:30
										 |  |  |         mem_start = (intptr_t)&_bt_data_start; | 
					
						
							|  |  |  |         mem_end = (intptr_t)&_bt_data_end; | 
					
						
							| 
									
										
										
										
											2018-08-17 08:49:48 +02:00
										 |  |  |         if (mem_start != mem_end) { | 
					
						
							| 
									
										
										
										
											2018-08-28 13:31:07 +05:30
										 |  |  |             ESP_LOGD(BTDM_LOG_TAG, "Release BT Data [0x%08x] - [0x%08x]", mem_start, mem_end); | 
					
						
							| 
									
										
										
										
											2019-06-25 09:03:58 +08:00
										 |  |  |             ESP_ERROR_CHECK(try_heap_caps_add_region(mem_start, mem_end)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         mem_start = (intptr_t)&_nimble_bss_start; | 
					
						
							|  |  |  |         mem_end = (intptr_t)&_nimble_bss_end; | 
					
						
							|  |  |  |         if (mem_start != mem_end) { | 
					
						
							|  |  |  |             ESP_LOGD(BTDM_LOG_TAG, "Release NimBLE BSS [0x%08x] - [0x%08x]", mem_start, mem_end); | 
					
						
							|  |  |  |             ESP_ERROR_CHECK(try_heap_caps_add_region(mem_start, mem_end)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         mem_start = (intptr_t)&_nimble_data_start; | 
					
						
							|  |  |  |         mem_end = (intptr_t)&_nimble_data_end; | 
					
						
							|  |  |  |         if (mem_start != mem_end) { | 
					
						
							|  |  |  |             ESP_LOGD(BTDM_LOG_TAG, "Release NimBLE Data [0x%08x] - [0x%08x]", mem_start, mem_end); | 
					
						
							| 
									
										
										
										
											2018-08-28 13:31:07 +05:30
										 |  |  |             ESP_ERROR_CHECK(try_heap_caps_add_region(mem_start, mem_end)); | 
					
						
							| 
									
										
										
										
											2018-08-17 08:49:48 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-07-11 11:52:32 +05:30
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-09-19 20:10:35 +08:00
										 |  |  |     return ESP_OK; | 
					
						
							| 
									
										
										
										
											2017-09-12 22:36:17 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-05 21:19:15 +08:00
										 |  |  | esp_err_t esp_bt_controller_init(esp_bt_controller_config_t *cfg) | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  |     esp_err_t err; | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  |     uint32_t btdm_cfg_mask = 0; | 
					
						
							| 
									
										
										
										
											2017-04-05 21:19:15 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-26 15:56:55 +08:00
										 |  |  |     //if all the bt available memory was already released, cannot initialize bluetooth controller
 | 
					
						
							|  |  |  |     if (btdm_dram_available_region[0].mode == ESP_BT_MODE_IDLE) { | 
					
						
							|  |  |  |         return ESP_ERR_INVALID_STATE; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-11-10 18:40:01 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  |     osi_funcs_p = (struct osi_funcs_t *)malloc_internal_wrapper(sizeof(struct osi_funcs_t)); | 
					
						
							|  |  |  |     if (osi_funcs_p == NULL) { | 
					
						
							|  |  |  |         return ESP_ERR_NO_MEM; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     memcpy(osi_funcs_p, &osi_funcs_ro, sizeof(struct osi_funcs_t)); | 
					
						
							|  |  |  |     if (btdm_osi_funcs_register(osi_funcs_p) != 0) { | 
					
						
							|  |  |  |         return ESP_ERR_INVALID_ARG; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-21 01:05:37 +08:00
										 |  |  |     if (btdm_controller_status != ESP_BT_CONTROLLER_STATUS_IDLE) { | 
					
						
							| 
									
										
										
										
											2017-04-05 21:19:15 +08:00
										 |  |  |         return ESP_ERR_INVALID_STATE; | 
					
						
							| 
									
										
										
										
											2017-02-21 01:05:37 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-05 21:19:15 +08:00
										 |  |  |     if (cfg == NULL) { | 
					
						
							|  |  |  |         return ESP_ERR_INVALID_ARG; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  |     if (cfg->controller_task_prio != ESP_TASK_BT_CONTROLLER_PRIO | 
					
						
							|  |  |  |             || cfg->controller_task_stack_size < ESP_TASK_BT_CONTROLLER_STACK) { | 
					
						
							|  |  |  |         return ESP_ERR_INVALID_ARG; | 
					
						
							| 
									
										
										
										
											2017-06-13 17:14:50 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  |     //overwrite some parameters
 | 
					
						
							| 
									
										
										
										
											2019-05-02 15:36:06 +02:00
										 |  |  |     cfg->bt_max_sync_conn = CONFIG_BTDM_CTRL_BR_EDR_MAX_SYNC_CONN_EFF; | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  |     cfg->magic  = ESP_BT_CONTROLLER_CONFIG_MAGIC_VAL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (((cfg->mode & ESP_BT_MODE_BLE) && (cfg->ble_max_conn <= 0 || cfg->ble_max_conn > BTDM_CONTROLLER_BLE_MAX_CONN_LIMIT)) | 
					
						
							|  |  |  |             || ((cfg->mode & ESP_BT_MODE_CLASSIC_BT) && (cfg->bt_max_acl_conn <= 0 || cfg->bt_max_acl_conn > BTDM_CONTROLLER_BR_EDR_MAX_ACL_CONN_LIMIT)) | 
					
						
							|  |  |  |             || ((cfg->mode & ESP_BT_MODE_CLASSIC_BT) && (cfg->bt_max_sync_conn > BTDM_CONTROLLER_BR_EDR_MAX_SYNC_CONN_LIMIT))) { | 
					
						
							|  |  |  |         return ESP_ERR_INVALID_ARG; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-28 13:31:07 +05:30
										 |  |  |     ESP_LOGI(BTDM_LOG_TAG, "BT controller compile version [%s]", btdm_controller_get_compile_version()); | 
					
						
							| 
									
										
										
										
											2018-03-27 16:35:00 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-30 11:39:42 +08:00
										 |  |  | #if CONFIG_SPIRAM_USE_MALLOC
 | 
					
						
							|  |  |  |     btdm_queue_table_mux = xSemaphoreCreateMutex(); | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  |     if (btdm_queue_table_mux == NULL) { | 
					
						
							| 
									
										
										
										
											2018-03-30 11:39:42 +08:00
										 |  |  |         return ESP_ERR_NO_MEM; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     memset(btdm_queue_table, 0, sizeof(btdm_queue_item_t) * BTDM_MAX_QUEUE_NUM); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  |     s_wakeup_req_sem = semphr_create_wrapper(1, 0); | 
					
						
							|  |  |  |     if (s_wakeup_req_sem == NULL) { | 
					
						
							|  |  |  |         err = ESP_ERR_NO_MEM; | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-20 14:34:28 +08:00
										 |  |  |     btdm_controller_mem_init(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     periph_module_enable(PERIPH_BT_MODULE); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  | #ifdef CONFIG_PM_ENABLE
 | 
					
						
							| 
									
										
										
										
											2020-02-20 14:34:28 +08:00
										 |  |  |     s_btdm_allow_light_sleep = false; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-26 15:58:44 +08:00
										 |  |  |     // set default sleep clock cycle and its fractional bits
 | 
					
						
							|  |  |  |     btdm_lpcycle_us_frac = RTC_CLK_CAL_FRACT; | 
					
						
							|  |  |  |     btdm_lpcycle_us = 2 << (btdm_lpcycle_us_frac); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-09 20:58:13 +08:00
										 |  |  | #if CONFIG_BTDM_CTRL_MODEM_SLEEP_MODE_ORIG
 | 
					
						
							| 
									
										
										
										
											2020-02-20 14:34:28 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     btdm_lpclk_sel = BTDM_LPCLK_SEL_XTAL; // set default value
 | 
					
						
							| 
									
										
										
										
											2020-07-09 20:58:13 +08:00
										 |  |  | #if CONFIG_BTDM_CTRL_LPCLK_SEL_EXT_32K_XTAL
 | 
					
						
							| 
									
										
										
										
											2020-02-20 14:34:28 +08:00
										 |  |  |     // check whether or not EXT_CRYS is working
 | 
					
						
							|  |  |  |     if (rtc_clk_slow_freq_get() == RTC_SLOW_FREQ_32K_XTAL) { | 
					
						
							| 
									
										
										
										
											2021-08-10 15:51:09 +08:00
										 |  |  |         btdm_lpclk_sel = BTDM_LPCLK_SEL_XTAL32K; // External 32kHz XTAL
 | 
					
						
							| 
									
										
										
										
											2020-02-20 14:34:28 +08:00
										 |  |  | #ifdef CONFIG_PM_ENABLE
 | 
					
						
							|  |  |  |         s_btdm_allow_light_sleep = true; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         ESP_LOGW(BTDM_LOG_TAG, "32.768kHz XTAL not detected, fall back to main XTAL as Bluetooth sleep clock\n" | 
					
						
							|  |  |  |                  "light sleep mode will not be able to apply when bluetooth is enabled"); | 
					
						
							|  |  |  |         btdm_lpclk_sel = BTDM_LPCLK_SEL_XTAL; // set default value
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     btdm_lpclk_sel = BTDM_LPCLK_SEL_XTAL; // set default value
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool select_src_ret, set_div_ret; | 
					
						
							|  |  |  |     if (btdm_lpclk_sel == BTDM_LPCLK_SEL_XTAL) { | 
					
						
							|  |  |  |         select_src_ret = btdm_lpclk_select_src(BTDM_LPCLK_SEL_XTAL); | 
					
						
							|  |  |  |         set_div_ret = btdm_lpclk_set_div(rtc_clk_xtal_freq_get() * 2 - 1); | 
					
						
							|  |  |  |         assert(select_src_ret && set_div_ret); | 
					
						
							|  |  |  |         btdm_lpcycle_us_frac = RTC_CLK_CAL_FRACT; | 
					
						
							|  |  |  |         btdm_lpcycle_us = 2 << (btdm_lpcycle_us_frac); | 
					
						
							|  |  |  |     } else { // btdm_lpclk_sel == BTDM_LPCLK_SEL_XTAL32K
 | 
					
						
							|  |  |  |         select_src_ret = btdm_lpclk_select_src(BTDM_LPCLK_SEL_XTAL32K); | 
					
						
							|  |  |  |         set_div_ret = btdm_lpclk_set_div(0); | 
					
						
							|  |  |  |         assert(select_src_ret && set_div_ret); | 
					
						
							|  |  |  |         btdm_lpcycle_us_frac = RTC_CLK_CAL_FRACT; | 
					
						
							|  |  |  |         btdm_lpcycle_us = (RTC_CLK_CAL_FRACT > 15) ? (1000000 << (RTC_CLK_CAL_FRACT - 15)) : | 
					
						
							|  |  |  |             (1000000 >> (15 - RTC_CLK_CAL_FRACT)); | 
					
						
							|  |  |  |         assert(btdm_lpcycle_us != 0); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     btdm_controller_set_sleep_mode(BTDM_MODEM_SLEEP_MODE_ORIG); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-09 20:58:13 +08:00
										 |  |  | #elif CONFIG_BTDM_CTRL_MODEM_SLEEP_MODE_EVED
 | 
					
						
							| 
									
										
										
										
											2020-02-20 14:34:28 +08:00
										 |  |  |     btdm_controller_set_sleep_mode(BTDM_MODEM_SLEEP_MODE_EVED); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     btdm_controller_set_sleep_mode(BTDM_MODEM_SLEEP_MODE_NONE); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef CONFIG_PM_ENABLE
 | 
					
						
							|  |  |  |     if (!s_btdm_allow_light_sleep) { | 
					
						
							|  |  |  |         if ((err = esp_pm_lock_create(ESP_PM_NO_LIGHT_SLEEP, 0, "btLS", &s_light_sleep_pm_lock)) != ESP_OK) { | 
					
						
							|  |  |  |             goto error; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-12-17 19:54:57 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  |     if ((err = esp_pm_lock_create(ESP_PM_APB_FREQ_MAX, 0, "bt", &s_pm_lock)) != ESP_OK) { | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     esp_timer_create_args_t create_args = { | 
					
						
							|  |  |  |         .callback = btdm_slp_tmr_callback, | 
					
						
							|  |  |  |         .arg = NULL, | 
					
						
							|  |  |  |         .name = "btSlp" | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     if ((err = esp_timer_create(&create_args, &s_btdm_slp_tmr)) != ESP_OK) { | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  |     s_pm_lock_acquired = true; | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-26 17:51:13 +08:00
										 |  |  | #if CONFIG_SW_COEXIST_ENABLE
 | 
					
						
							|  |  |  |     coex_init(); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  |     btdm_cfg_mask = btdm_config_mask_load(); | 
					
						
							| 
									
										
										
										
											2017-04-05 21:19:15 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  |     if (btdm_controller_init(btdm_cfg_mask, cfg) != 0) { | 
					
						
							|  |  |  |         err = ESP_ERR_NO_MEM; | 
					
						
							|  |  |  |         goto error; | 
					
						
							| 
									
										
										
										
											2017-04-05 21:19:15 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  |     btdm_controller_status = ESP_BT_CONTROLLER_STATUS_INITED; | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-05 21:19:15 +08:00
										 |  |  |     return ESP_OK; | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | error: | 
					
						
							|  |  |  | #ifdef CONFIG_PM_ENABLE
 | 
					
						
							| 
									
										
										
										
											2020-02-20 14:34:28 +08:00
										 |  |  |     if (!s_btdm_allow_light_sleep) { | 
					
						
							|  |  |  |         if (s_light_sleep_pm_lock != NULL) { | 
					
						
							|  |  |  |             esp_pm_lock_delete(s_light_sleep_pm_lock); | 
					
						
							|  |  |  |             s_light_sleep_pm_lock = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-12-17 19:54:57 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  |     if (s_pm_lock != NULL) { | 
					
						
							|  |  |  |         esp_pm_lock_delete(s_pm_lock); | 
					
						
							|  |  |  |         s_pm_lock = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (s_btdm_slp_tmr != NULL) { | 
					
						
							|  |  |  |         esp_timer_delete(s_btdm_slp_tmr); | 
					
						
							|  |  |  |         s_btdm_slp_tmr = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  |     if (s_wakeup_req_sem) { | 
					
						
							|  |  |  |         semphr_delete_wrapper(s_wakeup_req_sem); | 
					
						
							|  |  |  |         s_wakeup_req_sem = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  |     return err; | 
					
						
							| 
									
										
										
										
											2017-02-17 19:24:58 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  | esp_err_t esp_bt_controller_deinit(void) | 
					
						
							| 
									
										
										
										
											2017-02-17 19:24:58 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  |     if (btdm_controller_status != ESP_BT_CONTROLLER_STATUS_INITED) { | 
					
						
							|  |  |  |         return ESP_ERR_INVALID_STATE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-17 15:46:56 +08:00
										 |  |  |     btdm_controller_deinit(); | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-01 17:05:38 +08:00
										 |  |  |     periph_module_disable(PERIPH_BT_MODULE); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  | #ifdef CONFIG_PM_ENABLE
 | 
					
						
							| 
									
										
										
										
											2020-02-20 14:34:28 +08:00
										 |  |  |     if (!s_btdm_allow_light_sleep) { | 
					
						
							|  |  |  |         esp_pm_lock_delete(s_light_sleep_pm_lock); | 
					
						
							|  |  |  |         s_light_sleep_pm_lock = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-11-16 10:43:25 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  |     if (s_pm_lock != NULL) { | 
					
						
							|  |  |  |         esp_pm_lock_delete(s_pm_lock); | 
					
						
							|  |  |  |         s_pm_lock = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (s_btdm_slp_tmr != NULL) { | 
					
						
							|  |  |  |         esp_timer_stop(s_btdm_slp_tmr); | 
					
						
							|  |  |  |         esp_timer_delete(s_btdm_slp_tmr); | 
					
						
							|  |  |  |         s_btdm_slp_tmr = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  |     s_pm_lock_acquired = false; | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  |     semphr_delete_wrapper(s_wakeup_req_sem); | 
					
						
							|  |  |  |     s_wakeup_req_sem = NULL; | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-30 11:39:42 +08:00
										 |  |  | #if CONFIG_SPIRAM_USE_MALLOC
 | 
					
						
							|  |  |  |     vSemaphoreDelete(btdm_queue_table_mux); | 
					
						
							|  |  |  |     btdm_queue_table_mux = NULL; | 
					
						
							|  |  |  |     memset(btdm_queue_table, 0, sizeof(btdm_queue_item_t) * BTDM_MAX_QUEUE_NUM); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  |     free(osi_funcs_p); | 
					
						
							|  |  |  |     osi_funcs_p = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-10 15:35:17 +08:00
										 |  |  |     btdm_controller_status = ESP_BT_CONTROLLER_STATUS_IDLE; | 
					
						
							| 
									
										
										
										
											2017-10-09 15:34:31 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  |     btdm_lpcycle_us = 0; | 
					
						
							|  |  |  |     btdm_controller_set_sleep_mode(BTDM_MODEM_SLEEP_MODE_NONE); | 
					
						
							| 
									
										
										
										
											2017-10-09 15:34:31 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  |     return ESP_OK; | 
					
						
							| 
									
										
										
										
											2017-02-17 19:24:58 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-21 20:08:10 +08:00
										 |  |  | static void bt_shutdown(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-11-01 20:39:37 +08:00
										 |  |  |     if (btdm_controller_status != ESP_BT_CONTROLLER_STATUS_ENABLED) { | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2020-12-21 20:08:10 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-11-01 20:39:37 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     esp_bt_controller_shutdown(); | 
					
						
							|  |  |  |     esp_phy_disable(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-21 20:08:10 +08:00
										 |  |  |     return; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-17 19:24:58 +08:00
										 |  |  | esp_err_t esp_bt_controller_enable(esp_bt_mode_t mode) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-21 01:05:37 +08:00
										 |  |  |     if (btdm_controller_status != ESP_BT_CONTROLLER_STATUS_INITED) { | 
					
						
							|  |  |  |         return ESP_ERR_INVALID_STATE; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-09-19 20:10:35 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-15 21:40:51 +08:00
										 |  |  |     //As the history reason, mode should be equal to the mode which set in esp_bt_controller_init()
 | 
					
						
							|  |  |  |     if (mode != btdm_controller_get_mode()) { | 
					
						
							| 
									
										
										
										
											2017-02-17 19:24:58 +08:00
										 |  |  |         return ESP_ERR_INVALID_ARG; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-09 15:34:31 +08:00
										 |  |  | #ifdef CONFIG_PM_ENABLE
 | 
					
						
							| 
									
										
										
										
											2020-02-20 14:34:28 +08:00
										 |  |  |     if (!s_btdm_allow_light_sleep) { | 
					
						
							|  |  |  |         esp_pm_lock_acquire(s_light_sleep_pm_lock); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-10-09 15:34:31 +08:00
										 |  |  |     esp_pm_lock_acquire(s_pm_lock); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-22 17:15:28 +08:00
										 |  |  |     esp_phy_enable(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-26 17:51:13 +08:00
										 |  |  | #if CONFIG_SW_COEXIST_ENABLE
 | 
					
						
							|  |  |  |     coex_enable(); | 
					
						
							| 
									
										
										
										
											2020-05-22 17:15:28 +08:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2017-11-10 10:54:50 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  |     if (btdm_controller_get_sleep_mode() == BTDM_MODEM_SLEEP_MODE_ORIG) { | 
					
						
							|  |  |  |         btdm_controller_enable_sleep(true); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-02-17 19:24:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-11 16:49:01 +08:00
										 |  |  |     // inititalize bluetooth baseband
 | 
					
						
							|  |  |  |     btdm_check_and_init_bb(); | 
					
						
							| 
									
										
										
										
											2017-02-17 19:24:58 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ret = btdm_controller_enable(mode); | 
					
						
							| 
									
										
										
										
											2020-05-22 17:15:28 +08:00
										 |  |  |     if (ret != 0) { | 
					
						
							| 
									
										
										
										
											2020-08-26 17:51:13 +08:00
										 |  |  | #if CONFIG_SW_COEXIST_ENABLE
 | 
					
						
							|  |  |  |         coex_disable(); | 
					
						
							| 
									
										
										
										
											2020-05-22 17:15:28 +08:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |         esp_phy_disable(); | 
					
						
							| 
									
										
										
										
											2018-07-17 15:46:56 +08:00
										 |  |  | #ifdef CONFIG_PM_ENABLE
 | 
					
						
							| 
									
										
										
										
											2020-02-20 14:34:28 +08:00
										 |  |  |         if (!s_btdm_allow_light_sleep) { | 
					
						
							|  |  |  |             esp_pm_lock_release(s_light_sleep_pm_lock); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-07-17 15:46:56 +08:00
										 |  |  |         esp_pm_lock_release(s_pm_lock); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2017-02-17 19:24:58 +08:00
										 |  |  |         return ESP_ERR_INVALID_STATE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-21 01:05:37 +08:00
										 |  |  |     btdm_controller_status = ESP_BT_CONTROLLER_STATUS_ENABLED; | 
					
						
							| 
									
										
										
										
											2020-12-21 20:08:10 +08:00
										 |  |  |     ret = esp_register_shutdown_handler(bt_shutdown); | 
					
						
							|  |  |  |     if (ret != ESP_OK) { | 
					
						
							|  |  |  |         ESP_LOGW(BTDM_LOG_TAG, "Register shutdown handler failed, ret = 0x%x", ret); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-02-21 01:05:37 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-17 19:24:58 +08:00
										 |  |  |     return ESP_OK; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-19 20:10:35 +08:00
										 |  |  | esp_err_t esp_bt_controller_disable(void) | 
					
						
							| 
									
										
										
										
											2017-02-17 19:24:58 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-02-21 01:05:37 +08:00
										 |  |  |     if (btdm_controller_status != ESP_BT_CONTROLLER_STATUS_ENABLED) { | 
					
						
							|  |  |  |         return ESP_ERR_INVALID_STATE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  |     // disable modem sleep and wake up from sleep mode
 | 
					
						
							|  |  |  |     if (btdm_controller_get_sleep_mode() == BTDM_MODEM_SLEEP_MODE_ORIG) { | 
					
						
							|  |  |  |         btdm_controller_enable_sleep(false); | 
					
						
							| 
									
										
										
										
											2020-09-03 20:51:19 +08:00
										 |  |  |         async_wakeup_request(BTDM_ASYNC_WAKEUP_REQ_CTRL_DISA); | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  |         while (!btdm_power_state_active()) { | 
					
						
							| 
									
										
										
										
											2020-07-21 13:07:34 +08:00
										 |  |  |             esp_rom_delay_us(1000); | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-17 15:46:56 +08:00
										 |  |  |     btdm_controller_disable(); | 
					
						
							| 
									
										
										
										
											2017-02-17 19:24:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-26 17:51:13 +08:00
										 |  |  | #if CONFIG_SW_COEXIST_ENABLE
 | 
					
						
							|  |  |  |     coex_disable(); | 
					
						
							| 
									
										
										
										
											2020-05-22 17:15:28 +08:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     esp_phy_disable(); | 
					
						
							| 
									
										
										
										
											2018-07-17 15:46:56 +08:00
										 |  |  |     btdm_controller_status = ESP_BT_CONTROLLER_STATUS_INITED; | 
					
						
							| 
									
										
										
										
											2020-12-21 20:08:10 +08:00
										 |  |  |     esp_unregister_shutdown_handler(bt_shutdown); | 
					
						
							| 
									
										
										
										
											2017-02-21 01:05:37 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-09 15:34:31 +08:00
										 |  |  | #ifdef CONFIG_PM_ENABLE
 | 
					
						
							| 
									
										
										
										
											2020-02-20 14:34:28 +08:00
										 |  |  |     if (!s_btdm_allow_light_sleep) { | 
					
						
							|  |  |  |         esp_pm_lock_release(s_light_sleep_pm_lock); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-10-09 15:34:31 +08:00
										 |  |  |     esp_pm_lock_release(s_pm_lock); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-17 19:24:58 +08:00
										 |  |  |     return ESP_OK; | 
					
						
							| 
									
										
										
										
											2016-10-18 21:49:00 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-21 01:05:37 +08:00
										 |  |  | esp_bt_controller_status_t esp_bt_controller_get_status(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return btdm_controller_status; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-11 21:46:17 +08:00
										 |  |  | /* extra functions */ | 
					
						
							|  |  |  | esp_err_t esp_ble_tx_power_set(esp_ble_power_type_t power_type, esp_power_level_t power_level) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (ble_txpwr_set(power_type, power_level) != 0) { | 
					
						
							|  |  |  |         return ESP_ERR_INVALID_ARG; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return ESP_OK; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | esp_power_level_t esp_ble_tx_power_get(esp_ble_power_type_t power_type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return (esp_power_level_t)ble_txpwr_get(power_type); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:22:49 +08:00
										 |  |  | esp_err_t esp_bredr_tx_power_set(esp_power_level_t min_power_level, esp_power_level_t max_power_level) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     esp_err_t err; | 
					
						
							|  |  |  |     int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ret = bredr_txpwr_set(min_power_level, max_power_level); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (ret == 0) { | 
					
						
							|  |  |  |         err = ESP_OK; | 
					
						
							|  |  |  |     } else if (ret == -1) { | 
					
						
							|  |  |  |         err = ESP_ERR_INVALID_ARG; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         err = ESP_ERR_INVALID_STATE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | esp_err_t esp_bredr_tx_power_get(esp_power_level_t *min_power_level, esp_power_level_t *max_power_level) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (bredr_txpwr_get((int *)min_power_level, (int *)max_power_level) != 0) { | 
					
						
							|  |  |  |         return ESP_ERR_INVALID_ARG; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return ESP_OK; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  | esp_err_t esp_bt_sleep_enable (void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     esp_err_t status; | 
					
						
							|  |  |  |     if (btdm_controller_status != ESP_BT_CONTROLLER_STATUS_ENABLED) { | 
					
						
							|  |  |  |         return ESP_ERR_INVALID_STATE; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-05-22 17:15:28 +08:00
										 |  |  |     if (btdm_controller_get_sleep_mode() == BTDM_MODEM_SLEEP_MODE_ORIG || | 
					
						
							|  |  |  |             btdm_controller_get_sleep_mode() == BTDM_MODEM_SLEEP_MODE_EVED) { | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  |         btdm_controller_enable_sleep (true); | 
					
						
							|  |  |  |         status = ESP_OK; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         status = ESP_ERR_NOT_SUPPORTED; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return status; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | esp_err_t esp_bt_sleep_disable (void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     esp_err_t status; | 
					
						
							|  |  |  |     if (btdm_controller_status != ESP_BT_CONTROLLER_STATUS_ENABLED) { | 
					
						
							|  |  |  |         return ESP_ERR_INVALID_STATE; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-05-22 17:15:28 +08:00
										 |  |  |     if (btdm_controller_get_sleep_mode() == BTDM_MODEM_SLEEP_MODE_ORIG || | 
					
						
							|  |  |  |             btdm_controller_get_sleep_mode() == BTDM_MODEM_SLEEP_MODE_EVED) { | 
					
						
							| 
									
										
										
										
											2018-04-08 19:19:47 +08:00
										 |  |  |         btdm_controller_enable_sleep (false); | 
					
						
							|  |  |  |         status = ESP_OK; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         status = ESP_ERR_NOT_SUPPORTED; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return status; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-21 11:33:30 +08:00
										 |  |  | esp_err_t esp_bredr_sco_datapath_set(esp_sco_data_path_t data_path) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (btdm_controller_status != ESP_BT_CONTROLLER_STATUS_ENABLED) { | 
					
						
							|  |  |  |         return ESP_ERR_INVALID_STATE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     bredr_sco_datapath_set(data_path); | 
					
						
							|  |  |  |     return ESP_OK; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-20 15:04:37 +08:00
										 |  |  | esp_err_t esp_ble_scan_dupilcate_list_flush(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (btdm_controller_status != ESP_BT_CONTROLLER_STATUS_ENABLED) { | 
					
						
							|  |  |  |         return ESP_ERR_INVALID_STATE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     btdm_controller_scan_duplicate_list_clear(); | 
					
						
							|  |  |  |     return ESP_OK; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-12 22:36:17 +08:00
										 |  |  | #endif /*  CONFIG_BT_ENABLED */
 |