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 38fe5a6c4eSMilanka Ringwald /** 39fe5a6c4eSMilanka Ringwald * @title Run Loop 4082636622SMatthias Ringwald * 4182636622SMatthias Ringwald */ 4282636622SMatthias Ringwald 4380e33422SMatthias Ringwald #ifndef btstack_run_loop_H 4480e33422SMatthias Ringwald #define btstack_run_loop_H 4582636622SMatthias Ringwald 467907f069SMatthias Ringwald #include "btstack_config.h" 4782636622SMatthias Ringwald 48d58a1b5fSMatthias Ringwald #include "btstack_bool.h" 4982636622SMatthias Ringwald #include "btstack_linked_list.h" 5082636622SMatthias Ringwald 5182636622SMatthias Ringwald #include <stdint.h> 5282636622SMatthias Ringwald 5382636622SMatthias Ringwald #if defined __cplusplus 5482636622SMatthias Ringwald extern "C" { 5582636622SMatthias Ringwald #endif 5682636622SMatthias Ringwald 577cd5ef95SMatthias Ringwald /** 587cd5ef95SMatthias Ringwald * Callback types for run loop data sources 597cd5ef95SMatthias Ringwald */ 607cd5ef95SMatthias Ringwald typedef enum { 617cd5ef95SMatthias Ringwald DATA_SOURCE_CALLBACK_POLL = 1 << 0, 627cd5ef95SMatthias Ringwald DATA_SOURCE_CALLBACK_READ = 1 << 1, 637cd5ef95SMatthias Ringwald DATA_SOURCE_CALLBACK_WRITE = 1 << 2, 647cd5ef95SMatthias Ringwald } btstack_data_source_callback_type_t; 657cd5ef95SMatthias Ringwald 66ec820d77SMatthias Ringwald typedef struct btstack_data_source { 673aad8623SMatthias Ringwald // linked item 6882636622SMatthias Ringwald btstack_linked_item_t item; 693aad8623SMatthias Ringwald 703aad8623SMatthias Ringwald // item to watch in run loop 713aad8623SMatthias Ringwald union { 723aad8623SMatthias Ringwald // file descriptor for posix systems 735ed06181SMatthias Ringwald int fd; 743aad8623SMatthias Ringwald // handle on windows 753aad8623SMatthias Ringwald void * handle; 76398a95ecSMatthias Ringwald } source; 773aad8623SMatthias Ringwald 785ed06181SMatthias Ringwald // callback to call for enabled callback types 797cd5ef95SMatthias Ringwald void (*process)(struct btstack_data_source *ds, btstack_data_source_callback_type_t callback_type); 803aad8623SMatthias Ringwald 815ed06181SMatthias Ringwald // flags storing enabled callback types 825ed06181SMatthias Ringwald uint16_t flags; 833aad8623SMatthias Ringwald 84ec820d77SMatthias Ringwald } btstack_data_source_t; 8582636622SMatthias Ringwald 86eb886013SMatthias Ringwald typedef struct btstack_timer_source { 8782636622SMatthias Ringwald btstack_linked_item_t item; 88f316a845SMatthias Ringwald // timeout in system ticks (HAVE_EMBEDDED_TICK) or milliseconds (HAVE_EMBEDDED_TIME_MS) 89f316a845SMatthias Ringwald uint32_t timeout; 90fd939756SMatthias Ringwald // will be called when timer fired 91fd939756SMatthias Ringwald void (*process)(struct btstack_timer_source *ts); 92fd939756SMatthias Ringwald void * context; 93ec820d77SMatthias Ringwald } btstack_timer_source_t; 9482636622SMatthias Ringwald 95ec820d77SMatthias Ringwald typedef struct btstack_run_loop { 9682636622SMatthias Ringwald void (*init)(void); 970d70dd62SMatthias Ringwald void (*add_data_source)(btstack_data_source_t * data_source); 98d58a1b5fSMatthias Ringwald bool (*remove_data_source)(btstack_data_source_t * data_source); 990d70dd62SMatthias Ringwald void (*enable_data_source_callbacks)(btstack_data_source_t * data_source, uint16_t callbacks); 1000d70dd62SMatthias Ringwald void (*disable_data_source_callbacks)(btstack_data_source_t * data_source, uint16_t callbacks); 101ec820d77SMatthias Ringwald void (*set_timer)(btstack_timer_source_t * timer, uint32_t timeout_in_ms); 102ec820d77SMatthias Ringwald void (*add_timer)(btstack_timer_source_t *timer); 103d58a1b5fSMatthias Ringwald bool (*remove_timer)(btstack_timer_source_t *timer); 10482636622SMatthias Ringwald void (*execute)(void); 10582636622SMatthias Ringwald void (*dump_timer)(void); 10682636622SMatthias Ringwald uint32_t (*get_time_ms)(void); 107528a4a3bSMatthias Ringwald } btstack_run_loop_t; 10882636622SMatthias Ringwald 109796f7837SMatthias Ringwald 110796f7837SMatthias Ringwald /* 111796f7837SMatthias Ringwald * BTstack Run Loop Base Implementation 112796f7837SMatthias Ringwald * Portable implementation of timer and data source management as base for platform specific implementations 113796f7837SMatthias Ringwald */ 114796f7837SMatthias Ringwald 115796f7837SMatthias Ringwald // private data (access only by run loop implementations) 116796f7837SMatthias Ringwald extern btstack_linked_list_t btstack_run_loop_base_timers; 117796f7837SMatthias Ringwald extern btstack_linked_list_t btstack_run_loop_base_data_sources; 118796f7837SMatthias Ringwald 119796f7837SMatthias Ringwald /** 120796f7837SMatthias Ringwald * @brief Init 121796f7837SMatthias Ringwald */ 122796f7837SMatthias Ringwald void btstack_run_loop_base_init(void); 123796f7837SMatthias Ringwald 124796f7837SMatthias Ringwald /** 125796f7837SMatthias Ringwald * @brief Add timer source. 126796f7837SMatthias Ringwald * @param timer to add 127796f7837SMatthias Ringwald */ 128796f7837SMatthias Ringwald void btstack_run_loop_base_add_timer(btstack_timer_source_t * timer); 129796f7837SMatthias Ringwald 130796f7837SMatthias Ringwald /** 131796f7837SMatthias Ringwald * @brief Remove timer source. 132796f7837SMatthias Ringwald * @param timer to remove 133796f7837SMatthias Ringwald * @returns true if timer was removed 134796f7837SMatthias Ringwald */ 135796f7837SMatthias Ringwald bool btstack_run_loop_base_remove_timer(btstack_timer_source_t * timer); 136796f7837SMatthias Ringwald 137796f7837SMatthias Ringwald /** 138796f7837SMatthias Ringwald * @brief Process timers: remove expired timers from list and call their process function 139796f7837SMatthias Ringwald * @param now 140796f7837SMatthias Ringwald */ 141796f7837SMatthias Ringwald void btstack_run_loop_base_process_timers(uint32_t now); 142796f7837SMatthias Ringwald 143796f7837SMatthias Ringwald /** 144796f7837SMatthias Ringwald * @brief Dump list of timers via log_info 145796f7837SMatthias Ringwald */ 146796f7837SMatthias Ringwald void btstack_run_loop_base_dump_timer(void); 147796f7837SMatthias Ringwald 148796f7837SMatthias Ringwald /** 149796f7837SMatthias Ringwald * @brief Get time until first timer fires 150796f7837SMatthias Ringwald * @returns -1 if no timers, time until next timeout otherwise 151796f7837SMatthias Ringwald */ 152796f7837SMatthias Ringwald int32_t btstack_run_loop_base_get_time_until_timeout(uint32_t now); 153796f7837SMatthias Ringwald 154796f7837SMatthias Ringwald /** 155796f7837SMatthias Ringwald * @brief Add data source to run loop 156796f7837SMatthias Ringwald * @param data_source to add 157796f7837SMatthias Ringwald */ 158796f7837SMatthias Ringwald void btstack_run_loop_base_add_data_source(btstack_data_source_t * data_source); 159796f7837SMatthias Ringwald 160796f7837SMatthias Ringwald /** 161796f7837SMatthias Ringwald * @brief Remove data source from run loop 162796f7837SMatthias Ringwald * @param data_source to remove 163796f7837SMatthias Ringwald * @returns true if data srouce was removed 164796f7837SMatthias Ringwald */ 165796f7837SMatthias Ringwald bool btstack_run_loop_base_remove_data_source(btstack_data_source_t * data_source); 166796f7837SMatthias Ringwald 167796f7837SMatthias Ringwald /** 168796f7837SMatthias Ringwald * @brief Enable callbacks for a data source 169796f7837SMatthias Ringwald * @param data_source to remove 170*b45b7749SMilanka Ringwald * @param callback_types to enable 171796f7837SMatthias Ringwald */ 172*b45b7749SMilanka Ringwald void btstack_run_loop_base_enable_data_source_callbacks(btstack_data_source_t * data_source, uint16_t callback_types); 173796f7837SMatthias Ringwald 174796f7837SMatthias Ringwald /** 175796f7837SMatthias Ringwald * @brief Enable callbacks for a data source 176796f7837SMatthias Ringwald * @param data_source to remove 177*b45b7749SMilanka Ringwald * @param callback_types to disable 178796f7837SMatthias Ringwald */ 179*b45b7749SMilanka Ringwald void btstack_run_loop_base_disable_data_source_callbacks(btstack_data_source_t * data_source, uint16_t callback_types); 180796f7837SMatthias Ringwald 181796f7837SMatthias Ringwald /** 182796f7837SMatthias Ringwald * @brief Poll data sources. It calls the procss function for all data sources where DATA_SOURCE_CALLBACK_POLL is set 183796f7837SMatthias Ringwald */ 184796f7837SMatthias Ringwald void btstack_run_loop_base_poll_data_sources(void); 185796f7837SMatthias Ringwald 18682636622SMatthias Ringwald 18782636622SMatthias Ringwald /* API_START */ 18882636622SMatthias Ringwald 18982636622SMatthias Ringwald /** 19082636622SMatthias Ringwald * @brief Init main run loop. Must be called before any other run loop call. 19182636622SMatthias Ringwald * 192528a4a3bSMatthias Ringwald * Use btstack_run_loop_$(btstack_run_loop_TYPE)_get_instance() from btstack_run_loop_$(btstack_run_loop_TYPE).h to get instance 19382636622SMatthias Ringwald */ 194528a4a3bSMatthias Ringwald void btstack_run_loop_init(const btstack_run_loop_t * run_loop); 19582636622SMatthias Ringwald 19682636622SMatthias Ringwald /** 19782636622SMatthias Ringwald * @brief Set timer based on current time in milliseconds. 19882636622SMatthias Ringwald */ 199*b45b7749SMilanka Ringwald void btstack_run_loop_set_timer(btstack_timer_source_t * timer, uint32_t timeout_in_ms); 20082636622SMatthias Ringwald 20182636622SMatthias Ringwald /** 20282636622SMatthias Ringwald * @brief Set callback that will be executed when timer expires. 20382636622SMatthias Ringwald */ 204*b45b7749SMilanka Ringwald void btstack_run_loop_set_timer_handler(btstack_timer_source_t * timer, void (*process)(btstack_timer_source_t * _timer)); 20582636622SMatthias Ringwald 20682636622SMatthias Ringwald /** 207fd939756SMatthias Ringwald * @brief Set context for this timer 208fd939756SMatthias Ringwald */ 209*b45b7749SMilanka Ringwald void btstack_run_loop_set_timer_context(btstack_timer_source_t * timer, void * context); 210fd939756SMatthias Ringwald 211fd939756SMatthias Ringwald /** 212fd939756SMatthias Ringwald * @brief Get context for this timer 213fd939756SMatthias Ringwald */ 214*b45b7749SMilanka Ringwald void * btstack_run_loop_get_timer_context(btstack_timer_source_t * timer); 215fd939756SMatthias Ringwald 216fd939756SMatthias Ringwald /** 217fd939756SMatthias Ringwald * @brief Add timer source. 21882636622SMatthias Ringwald */ 219ec820d77SMatthias Ringwald void btstack_run_loop_add_timer(btstack_timer_source_t * timer); 220fd939756SMatthias Ringwald 221fd939756SMatthias Ringwald /** 222fd939756SMatthias Ringwald * @brief Remove timer source. 223fd939756SMatthias Ringwald */ 224ec820d77SMatthias Ringwald int btstack_run_loop_remove_timer(btstack_timer_source_t * timer); 22582636622SMatthias Ringwald 22682636622SMatthias Ringwald /** 22782636622SMatthias Ringwald * @brief Get current time in ms 22882636622SMatthias Ringwald * @note 32-bit ms counter will overflow after approx. 52 days 22982636622SMatthias Ringwald */ 230528a4a3bSMatthias Ringwald uint32_t btstack_run_loop_get_time_ms(void); 23182636622SMatthias Ringwald 23282636622SMatthias Ringwald /** 233796f7837SMatthias Ringwald * @brief Dump timers using log_info 234796f7837SMatthias Ringwald */ 235796f7837SMatthias Ringwald void btstack_run_loop_timer_dump(void); 236796f7837SMatthias Ringwald 237796f7837SMatthias Ringwald 238796f7837SMatthias Ringwald /** 23982636622SMatthias Ringwald * @brief Set data source callback. 24082636622SMatthias Ringwald */ 241*b45b7749SMilanka Ringwald void btstack_run_loop_set_data_source_handler(btstack_data_source_t * data_source, void (*process)(btstack_data_source_t * _data_source, btstack_data_source_callback_type_t callback_type)); 24282636622SMatthias Ringwald 24382636622SMatthias Ringwald /** 2443a5c43eeSMatthias Ringwald * @brief Set data source file descriptor. 245896424b7SMatthias Ringwald * @param data_source 246896424b7SMatthias Ringwald * @param fd file descriptor 2473a5c43eeSMatthias Ringwald * @note No effect if port doensn't have file descriptors 2483a5c43eeSMatthias Ringwald */ 249896424b7SMatthias Ringwald void btstack_run_loop_set_data_source_fd(btstack_data_source_t * data_source, int fd); 2503a5c43eeSMatthias Ringwald 2513a5c43eeSMatthias Ringwald /** 2523a5c43eeSMatthias Ringwald * @brief Get data source file descriptor. 253896424b7SMatthias Ringwald * @param data_source 2543a5c43eeSMatthias Ringwald */ 255896424b7SMatthias Ringwald int btstack_run_loop_get_data_source_fd(btstack_data_source_t * data_source); 2563a5c43eeSMatthias Ringwald 257f04a41aeSMatthias Ringwald /** 258f04a41aeSMatthias Ringwald * @brief Set data source file descriptor. 259f04a41aeSMatthias Ringwald * @param data_source 260f04a41aeSMatthias Ringwald * @param handle 261f04a41aeSMatthias Ringwald * @note No effect if port doensn't have file descriptors 262f04a41aeSMatthias Ringwald */ 263f04a41aeSMatthias Ringwald void btstack_run_loop_set_data_source_handle(btstack_data_source_t * data_source, void * handle); 264f04a41aeSMatthias Ringwald 265f04a41aeSMatthias Ringwald /** 266f04a41aeSMatthias Ringwald * @brief Get data source file descriptor. 267f04a41aeSMatthias Ringwald * @param data_source 268f04a41aeSMatthias Ringwald */ 269f04a41aeSMatthias Ringwald void * btstack_run_loop_get_data_source_handle(btstack_data_source_t * data_source); 270f04a41aeSMatthias Ringwald 2713a5c43eeSMatthias Ringwald /** 272896424b7SMatthias Ringwald * @brief Enable callbacks for a data source 273896424b7SMatthias Ringwald * @param data_source to remove 274896424b7SMatthias Ringwald * @param callback types to enable 27582636622SMatthias Ringwald */ 27624ced5a6SMatthias Ringwald void btstack_run_loop_enable_data_source_callbacks(btstack_data_source_t * data_source, uint16_t callbacks); 277896424b7SMatthias Ringwald 278896424b7SMatthias Ringwald /** 279896424b7SMatthias Ringwald * @brief Enable callbacks for a data source 280896424b7SMatthias Ringwald * @param data_source to remove 281896424b7SMatthias Ringwald * @param callback types to disable 282896424b7SMatthias Ringwald */ 28324ced5a6SMatthias Ringwald void btstack_run_loop_disable_data_source_callbacks(btstack_data_source_t * data_source, uint16_t callbacks); 284896424b7SMatthias Ringwald 285896424b7SMatthias Ringwald /** 286896424b7SMatthias Ringwald * @brief Add data source to run loop 287896424b7SMatthias Ringwald * @param data_source to add 288896424b7SMatthias Ringwald */ 289896424b7SMatthias Ringwald void btstack_run_loop_add_data_source(btstack_data_source_t * data_source); 290896424b7SMatthias Ringwald 291896424b7SMatthias Ringwald /** 292896424b7SMatthias Ringwald * @brief Remove data source from run loop 293896424b7SMatthias Ringwald * @param data_source to remove 294896424b7SMatthias Ringwald */ 295896424b7SMatthias Ringwald int btstack_run_loop_remove_data_source(btstack_data_source_t * data_source); 29682636622SMatthias Ringwald 29782636622SMatthias Ringwald /** 29882636622SMatthias Ringwald * @brief Execute configured run loop. This function does not return. 29982636622SMatthias Ringwald */ 300528a4a3bSMatthias Ringwald void btstack_run_loop_execute(void); 30182636622SMatthias Ringwald 3023a2e2107SMatthias Ringwald /** 3033a2e2107SMatthias Ringwald * @brief De-Init Run Loop 3043a2e2107SMatthias Ringwald */ 3053a2e2107SMatthias Ringwald void btstack_run_loop_deinit(void); 3063a2e2107SMatthias Ringwald 30782636622SMatthias Ringwald /* API_END */ 30882636622SMatthias Ringwald 309796f7837SMatthias Ringwald 310796f7837SMatthias Ringwald 31182636622SMatthias Ringwald #if defined __cplusplus 31282636622SMatthias Ringwald } 31382636622SMatthias Ringwald #endif 31482636622SMatthias Ringwald 31580e33422SMatthias Ringwald #endif // btstack_run_loop_H 316