182636622SMatthias Ringwald /* 282636622SMatthias Ringwald * Copyright (C) 2014 BlueKitchen GmbH 382636622SMatthias Ringwald * 482636622SMatthias Ringwald * Redistribution and use in source and binary forms, with or without 582636622SMatthias Ringwald * modification, are permitted provided that the following conditions 682636622SMatthias Ringwald * are met: 782636622SMatthias Ringwald * 882636622SMatthias Ringwald * 1. Redistributions of source code must retain the above copyright 982636622SMatthias Ringwald * notice, this list of conditions and the following disclaimer. 1082636622SMatthias Ringwald * 2. Redistributions in binary form must reproduce the above copyright 1182636622SMatthias Ringwald * notice, this list of conditions and the following disclaimer in the 1282636622SMatthias Ringwald * documentation and/or other materials provided with the distribution. 1382636622SMatthias Ringwald * 3. Neither the name of the copyright holders nor the names of 1482636622SMatthias Ringwald * contributors may be used to endorse or promote products derived 1582636622SMatthias Ringwald * from this software without specific prior written permission. 1682636622SMatthias Ringwald * 4. Any redistribution, use, or modification is done solely for 1782636622SMatthias Ringwald * personal benefit and not for any commercial purpose or for 1882636622SMatthias Ringwald * monetary gain. 1982636622SMatthias Ringwald * 2082636622SMatthias Ringwald * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS 2182636622SMatthias Ringwald * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 2282636622SMatthias Ringwald * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 2382636622SMatthias Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MATTHIAS 2482636622SMatthias Ringwald * RINGWALD OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, 2582636622SMatthias Ringwald * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 2682636622SMatthias Ringwald * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS 2782636622SMatthias Ringwald * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 2882636622SMatthias Ringwald * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 2982636622SMatthias Ringwald * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF 3082636622SMatthias Ringwald * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 3182636622SMatthias Ringwald * SUCH DAMAGE. 3282636622SMatthias Ringwald * 3382636622SMatthias Ringwald * Please inquire about commercial licensing options at 3482636622SMatthias Ringwald * [email protected] 3582636622SMatthias Ringwald * 3682636622SMatthias Ringwald */ 3782636622SMatthias Ringwald 3882636622SMatthias Ringwald /* 3982636622SMatthias Ringwald * run_loop.h 4082636622SMatthias Ringwald * 4182636622SMatthias Ringwald * Created by Matthias Ringwald on 6/6/09. 4282636622SMatthias Ringwald */ 4382636622SMatthias Ringwald 4480e33422SMatthias Ringwald #ifndef btstack_run_loop_H 4580e33422SMatthias Ringwald #define btstack_run_loop_H 4682636622SMatthias Ringwald 477907f069SMatthias Ringwald #include "btstack_config.h" 4882636622SMatthias Ringwald 49d58a1b5fSMatthias Ringwald #include "btstack_bool.h" 5082636622SMatthias Ringwald #include "btstack_linked_list.h" 5182636622SMatthias Ringwald 5282636622SMatthias Ringwald #include <stdint.h> 5382636622SMatthias Ringwald 5482636622SMatthias Ringwald #if defined __cplusplus 5582636622SMatthias Ringwald extern "C" { 5682636622SMatthias Ringwald #endif 5782636622SMatthias Ringwald 587cd5ef95SMatthias Ringwald 597cd5ef95SMatthias Ringwald /** 607cd5ef95SMatthias Ringwald * Callback types for run loop data sources 617cd5ef95SMatthias Ringwald */ 627cd5ef95SMatthias Ringwald typedef enum { 637cd5ef95SMatthias Ringwald DATA_SOURCE_CALLBACK_POLL = 1 << 0, 647cd5ef95SMatthias Ringwald DATA_SOURCE_CALLBACK_READ = 1 << 1, 657cd5ef95SMatthias Ringwald DATA_SOURCE_CALLBACK_WRITE = 1 << 2, 667cd5ef95SMatthias Ringwald } btstack_data_source_callback_type_t; 677cd5ef95SMatthias Ringwald 68ec820d77SMatthias Ringwald typedef struct btstack_data_source { 693aad8623SMatthias Ringwald // linked item 7082636622SMatthias Ringwald btstack_linked_item_t item; 713aad8623SMatthias Ringwald 723aad8623SMatthias Ringwald // item to watch in run loop 733aad8623SMatthias Ringwald union { 743aad8623SMatthias Ringwald // file descriptor for posix systems 755ed06181SMatthias Ringwald int fd; 763aad8623SMatthias Ringwald // handle on windows 773aad8623SMatthias Ringwald void * handle; 78398a95ecSMatthias Ringwald } source; 793aad8623SMatthias Ringwald 805ed06181SMatthias Ringwald // callback to call for enabled callback types 817cd5ef95SMatthias Ringwald void (*process)(struct btstack_data_source *ds, btstack_data_source_callback_type_t callback_type); 823aad8623SMatthias Ringwald 835ed06181SMatthias Ringwald // flags storing enabled callback types 845ed06181SMatthias Ringwald uint16_t flags; 853aad8623SMatthias Ringwald 86ec820d77SMatthias Ringwald } btstack_data_source_t; 8782636622SMatthias Ringwald 88eb886013SMatthias Ringwald typedef struct btstack_timer_source { 8982636622SMatthias Ringwald btstack_linked_item_t item; 90f316a845SMatthias Ringwald // timeout in system ticks (HAVE_EMBEDDED_TICK) or milliseconds (HAVE_EMBEDDED_TIME_MS) 91f316a845SMatthias Ringwald uint32_t timeout; 92fd939756SMatthias Ringwald // will be called when timer fired 93fd939756SMatthias Ringwald void (*process)(struct btstack_timer_source *ts); 94fd939756SMatthias Ringwald void * context; 95ec820d77SMatthias Ringwald } btstack_timer_source_t; 9682636622SMatthias Ringwald 97ec820d77SMatthias Ringwald typedef struct btstack_run_loop { 9882636622SMatthias Ringwald void (*init)(void); 990d70dd62SMatthias Ringwald void (*add_data_source)(btstack_data_source_t * data_source); 100d58a1b5fSMatthias Ringwald bool (*remove_data_source)(btstack_data_source_t * data_source); 1010d70dd62SMatthias Ringwald void (*enable_data_source_callbacks)(btstack_data_source_t * data_source, uint16_t callbacks); 1020d70dd62SMatthias Ringwald void (*disable_data_source_callbacks)(btstack_data_source_t * data_source, uint16_t callbacks); 103ec820d77SMatthias Ringwald void (*set_timer)(btstack_timer_source_t * timer, uint32_t timeout_in_ms); 104ec820d77SMatthias Ringwald void (*add_timer)(btstack_timer_source_t *timer); 105d58a1b5fSMatthias Ringwald bool (*remove_timer)(btstack_timer_source_t *timer); 10682636622SMatthias Ringwald void (*execute)(void); 10782636622SMatthias Ringwald void (*dump_timer)(void); 10882636622SMatthias Ringwald uint32_t (*get_time_ms)(void); 109528a4a3bSMatthias Ringwald } btstack_run_loop_t; 11082636622SMatthias Ringwald 111*796f7837SMatthias Ringwald 112*796f7837SMatthias Ringwald /* 113*796f7837SMatthias Ringwald * BTstack Run Loop Base Implementation 114*796f7837SMatthias Ringwald * Portable implementation of timer and data source management as base for platform specific implementations 115*796f7837SMatthias Ringwald */ 116*796f7837SMatthias Ringwald 117*796f7837SMatthias Ringwald // private data (access only by run loop implementations) 118*796f7837SMatthias Ringwald extern btstack_linked_list_t btstack_run_loop_base_timers; 119*796f7837SMatthias Ringwald extern btstack_linked_list_t btstack_run_loop_base_data_sources; 120*796f7837SMatthias Ringwald 121*796f7837SMatthias Ringwald /** 122*796f7837SMatthias Ringwald * @brief Init 123*796f7837SMatthias Ringwald */ 124*796f7837SMatthias Ringwald void btstack_run_loop_base_init(void); 125*796f7837SMatthias Ringwald 126*796f7837SMatthias Ringwald /** 127*796f7837SMatthias Ringwald * @brief Add timer source. 128*796f7837SMatthias Ringwald * @param timer to add 129*796f7837SMatthias Ringwald */ 130*796f7837SMatthias Ringwald void btstack_run_loop_base_add_timer(btstack_timer_source_t * timer); 131*796f7837SMatthias Ringwald 132*796f7837SMatthias Ringwald /** 133*796f7837SMatthias Ringwald * @brief Remove timer source. 134*796f7837SMatthias Ringwald * @param timer to remove 135*796f7837SMatthias Ringwald * @returns true if timer was removed 136*796f7837SMatthias Ringwald */ 137*796f7837SMatthias Ringwald bool btstack_run_loop_base_remove_timer(btstack_timer_source_t * timer); 138*796f7837SMatthias Ringwald 139*796f7837SMatthias Ringwald /** 140*796f7837SMatthias Ringwald * @brief Process timers: remove expired timers from list and call their process function 141*796f7837SMatthias Ringwald * @param now 142*796f7837SMatthias Ringwald */ 143*796f7837SMatthias Ringwald void btstack_run_loop_base_process_timers(uint32_t now); 144*796f7837SMatthias Ringwald 145*796f7837SMatthias Ringwald /** 146*796f7837SMatthias Ringwald * @brief Dump list of timers via log_info 147*796f7837SMatthias Ringwald */ 148*796f7837SMatthias Ringwald void btstack_run_loop_base_dump_timer(void); 149*796f7837SMatthias Ringwald 150*796f7837SMatthias Ringwald /** 151*796f7837SMatthias Ringwald * @brief Get time until first timer fires 152*796f7837SMatthias Ringwald * @returns -1 if no timers, time until next timeout otherwise 153*796f7837SMatthias Ringwald */ 154*796f7837SMatthias Ringwald int32_t btstack_run_loop_base_get_time_until_timeout(uint32_t now); 155*796f7837SMatthias Ringwald 156*796f7837SMatthias Ringwald /** 157*796f7837SMatthias Ringwald * @brief Add data source to run loop 158*796f7837SMatthias Ringwald * @param data_source to add 159*796f7837SMatthias Ringwald */ 160*796f7837SMatthias Ringwald void btstack_run_loop_base_add_data_source(btstack_data_source_t * data_source); 161*796f7837SMatthias Ringwald 162*796f7837SMatthias Ringwald /** 163*796f7837SMatthias Ringwald * @brief Remove data source from run loop 164*796f7837SMatthias Ringwald * @param data_source to remove 165*796f7837SMatthias Ringwald * @returns true if data srouce was removed 166*796f7837SMatthias Ringwald */ 167*796f7837SMatthias Ringwald bool btstack_run_loop_base_remove_data_source(btstack_data_source_t * data_source); 168*796f7837SMatthias Ringwald 169*796f7837SMatthias Ringwald /** 170*796f7837SMatthias Ringwald * @brief Enable callbacks for a data source 171*796f7837SMatthias Ringwald * @param data_source to remove 172*796f7837SMatthias Ringwald * @param callback types to enable 173*796f7837SMatthias Ringwald */ 174*796f7837SMatthias Ringwald void btstack_run_loop_base_enable_data_source_callbacks(btstack_data_source_t * data_source, uint16_t callbacks); 175*796f7837SMatthias Ringwald 176*796f7837SMatthias Ringwald /** 177*796f7837SMatthias Ringwald * @brief Enable callbacks for a data source 178*796f7837SMatthias Ringwald * @param data_source to remove 179*796f7837SMatthias Ringwald * @param callback types to disable 180*796f7837SMatthias Ringwald */ 181*796f7837SMatthias Ringwald void btstack_run_loop_base_disable_data_source_callbacks(btstack_data_source_t * data_source, uint16_t callbacks); 182*796f7837SMatthias Ringwald 183*796f7837SMatthias Ringwald /** 184*796f7837SMatthias Ringwald * @brief Poll data sources. It calls the procss function for all data sources where DATA_SOURCE_CALLBACK_POLL is set 185*796f7837SMatthias Ringwald */ 186*796f7837SMatthias Ringwald void btstack_run_loop_base_poll_data_sources(void); 187*796f7837SMatthias Ringwald 18882636622SMatthias Ringwald 18982636622SMatthias Ringwald /* API_START */ 19082636622SMatthias Ringwald 19182636622SMatthias Ringwald /** 19282636622SMatthias Ringwald * @brief Init main run loop. Must be called before any other run loop call. 19382636622SMatthias Ringwald * 194528a4a3bSMatthias Ringwald * Use btstack_run_loop_$(btstack_run_loop_TYPE)_get_instance() from btstack_run_loop_$(btstack_run_loop_TYPE).h to get instance 19582636622SMatthias Ringwald */ 196528a4a3bSMatthias Ringwald void btstack_run_loop_init(const btstack_run_loop_t * run_loop); 19782636622SMatthias Ringwald 19882636622SMatthias Ringwald /** 19982636622SMatthias Ringwald * @brief Set timer based on current time in milliseconds. 20082636622SMatthias Ringwald */ 201896424b7SMatthias Ringwald void btstack_run_loop_set_timer(btstack_timer_source_t * ts, uint32_t timeout_in_ms); 20282636622SMatthias Ringwald 20382636622SMatthias Ringwald /** 20482636622SMatthias Ringwald * @brief Set callback that will be executed when timer expires. 20582636622SMatthias Ringwald */ 206ec820d77SMatthias Ringwald void btstack_run_loop_set_timer_handler(btstack_timer_source_t * ts, void (*process)(btstack_timer_source_t *_ts)); 20782636622SMatthias Ringwald 20882636622SMatthias Ringwald /** 209fd939756SMatthias Ringwald * @brief Set context for this timer 210fd939756SMatthias Ringwald */ 211fd939756SMatthias Ringwald void btstack_run_loop_set_timer_context(btstack_timer_source_t * ts, void * context); 212fd939756SMatthias Ringwald 213fd939756SMatthias Ringwald /** 214fd939756SMatthias Ringwald * @brief Get context for this timer 215fd939756SMatthias Ringwald */ 216fd939756SMatthias Ringwald void * btstack_run_loop_get_timer_context(btstack_timer_source_t * ts); 217fd939756SMatthias Ringwald 218fd939756SMatthias Ringwald /** 219fd939756SMatthias Ringwald * @brief Add timer source. 22082636622SMatthias Ringwald */ 221ec820d77SMatthias Ringwald void btstack_run_loop_add_timer(btstack_timer_source_t * timer); 222fd939756SMatthias Ringwald 223fd939756SMatthias Ringwald /** 224fd939756SMatthias Ringwald * @brief Remove timer source. 225fd939756SMatthias Ringwald */ 226ec820d77SMatthias Ringwald int btstack_run_loop_remove_timer(btstack_timer_source_t * timer); 22782636622SMatthias Ringwald 22882636622SMatthias Ringwald /** 22982636622SMatthias Ringwald * @brief Get current time in ms 23082636622SMatthias Ringwald * @note 32-bit ms counter will overflow after approx. 52 days 23182636622SMatthias Ringwald */ 232528a4a3bSMatthias Ringwald uint32_t btstack_run_loop_get_time_ms(void); 23382636622SMatthias Ringwald 23482636622SMatthias Ringwald /** 235*796f7837SMatthias Ringwald * @brief Dump timers using log_info 236*796f7837SMatthias Ringwald */ 237*796f7837SMatthias Ringwald void btstack_run_loop_timer_dump(void); 238*796f7837SMatthias Ringwald 239*796f7837SMatthias Ringwald 240*796f7837SMatthias Ringwald /** 24182636622SMatthias Ringwald * @brief Set data source callback. 24282636622SMatthias Ringwald */ 243896424b7SMatthias Ringwald void btstack_run_loop_set_data_source_handler(btstack_data_source_t * data_source, void (*process)(btstack_data_source_t *_ds, btstack_data_source_callback_type_t callback_type)); 24482636622SMatthias Ringwald 24582636622SMatthias Ringwald /** 2463a5c43eeSMatthias Ringwald * @brief Set data source file descriptor. 247896424b7SMatthias Ringwald * @param data_source 248896424b7SMatthias Ringwald * @param fd file descriptor 2493a5c43eeSMatthias Ringwald * @note No effect if port doensn't have file descriptors 2503a5c43eeSMatthias Ringwald */ 251896424b7SMatthias Ringwald void btstack_run_loop_set_data_source_fd(btstack_data_source_t * data_source, int fd); 2523a5c43eeSMatthias Ringwald 2533a5c43eeSMatthias Ringwald /** 2543a5c43eeSMatthias Ringwald * @brief Get data source file descriptor. 255896424b7SMatthias Ringwald * @param data_source 2563a5c43eeSMatthias Ringwald */ 257896424b7SMatthias Ringwald int btstack_run_loop_get_data_source_fd(btstack_data_source_t * data_source); 2583a5c43eeSMatthias Ringwald 259f04a41aeSMatthias Ringwald /** 260f04a41aeSMatthias Ringwald * @brief Set data source file descriptor. 261f04a41aeSMatthias Ringwald * @param data_source 262f04a41aeSMatthias Ringwald * @param handle 263f04a41aeSMatthias Ringwald * @note No effect if port doensn't have file descriptors 264f04a41aeSMatthias Ringwald */ 265f04a41aeSMatthias Ringwald void btstack_run_loop_set_data_source_handle(btstack_data_source_t * data_source, void * handle); 266f04a41aeSMatthias Ringwald 267f04a41aeSMatthias Ringwald /** 268f04a41aeSMatthias Ringwald * @brief Get data source file descriptor. 269f04a41aeSMatthias Ringwald * @param data_source 270f04a41aeSMatthias Ringwald */ 271f04a41aeSMatthias Ringwald void * btstack_run_loop_get_data_source_handle(btstack_data_source_t * data_source); 272f04a41aeSMatthias Ringwald 2733a5c43eeSMatthias Ringwald /** 274896424b7SMatthias Ringwald * @brief Enable callbacks for a data source 275896424b7SMatthias Ringwald * @param data_source to remove 276896424b7SMatthias Ringwald * @param callback types to enable 27782636622SMatthias Ringwald */ 27824ced5a6SMatthias Ringwald void btstack_run_loop_enable_data_source_callbacks(btstack_data_source_t * data_source, uint16_t callbacks); 279896424b7SMatthias Ringwald 280896424b7SMatthias Ringwald /** 281896424b7SMatthias Ringwald * @brief Enable callbacks for a data source 282896424b7SMatthias Ringwald * @param data_source to remove 283896424b7SMatthias Ringwald * @param callback types to disable 284896424b7SMatthias Ringwald */ 28524ced5a6SMatthias Ringwald void btstack_run_loop_disable_data_source_callbacks(btstack_data_source_t * data_source, uint16_t callbacks); 286896424b7SMatthias Ringwald 287896424b7SMatthias Ringwald /** 288896424b7SMatthias Ringwald * @brief Add data source to run loop 289896424b7SMatthias Ringwald * @param data_source to add 290896424b7SMatthias Ringwald */ 291896424b7SMatthias Ringwald void btstack_run_loop_add_data_source(btstack_data_source_t * data_source); 292896424b7SMatthias Ringwald 293896424b7SMatthias Ringwald /** 294896424b7SMatthias Ringwald * @brief Remove data source from run loop 295896424b7SMatthias Ringwald * @param data_source to remove 296896424b7SMatthias Ringwald */ 297896424b7SMatthias Ringwald int btstack_run_loop_remove_data_source(btstack_data_source_t * data_source); 29882636622SMatthias Ringwald 29982636622SMatthias Ringwald /** 30082636622SMatthias Ringwald * @brief Execute configured run loop. This function does not return. 30182636622SMatthias Ringwald */ 302528a4a3bSMatthias Ringwald void btstack_run_loop_execute(void); 30382636622SMatthias Ringwald 3043a2e2107SMatthias Ringwald /** 3053a2e2107SMatthias Ringwald * @brief De-Init Run Loop 3063a2e2107SMatthias Ringwald */ 3073a2e2107SMatthias Ringwald void btstack_run_loop_deinit(void); 3083a2e2107SMatthias Ringwald 30982636622SMatthias Ringwald /* API_END */ 31082636622SMatthias Ringwald 311*796f7837SMatthias Ringwald 312*796f7837SMatthias Ringwald 31382636622SMatthias Ringwald #if defined __cplusplus 31482636622SMatthias Ringwald } 31582636622SMatthias Ringwald #endif 31682636622SMatthias Ringwald 31780e33422SMatthias Ringwald #endif // btstack_run_loop_H 318