1*1c60b9acSAndroid Build Coastguard Worker /* 2*1c60b9acSAndroid Build Coastguard Worker * libwebsockets - small server side websockets and web server implementation 3*1c60b9acSAndroid Build Coastguard Worker * 4*1c60b9acSAndroid Build Coastguard Worker * Copyright (C) 2010 - 2021 Andy Green <[email protected]> 5*1c60b9acSAndroid Build Coastguard Worker * 6*1c60b9acSAndroid Build Coastguard Worker * Permission is hereby granted, free of charge, to any person obtaining a copy 7*1c60b9acSAndroid Build Coastguard Worker * of this software and associated documentation files (the "Software"), to 8*1c60b9acSAndroid Build Coastguard Worker * deal in the Software without restriction, including without limitation the 9*1c60b9acSAndroid Build Coastguard Worker * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or 10*1c60b9acSAndroid Build Coastguard Worker * sell copies of the Software, and to permit persons to whom the Software is 11*1c60b9acSAndroid Build Coastguard Worker * furnished to do so, subject to the following conditions: 12*1c60b9acSAndroid Build Coastguard Worker * 13*1c60b9acSAndroid Build Coastguard Worker * The above copyright notice and this permission notice shall be included in 14*1c60b9acSAndroid Build Coastguard Worker * all copies or substantial portions of the Software. 15*1c60b9acSAndroid Build Coastguard Worker * 16*1c60b9acSAndroid Build Coastguard Worker * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 17*1c60b9acSAndroid Build Coastguard Worker * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 18*1c60b9acSAndroid Build Coastguard Worker * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 19*1c60b9acSAndroid Build Coastguard Worker * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 20*1c60b9acSAndroid Build Coastguard Worker * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING 21*1c60b9acSAndroid Build Coastguard Worker * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS 22*1c60b9acSAndroid Build Coastguard Worker * IN THE SOFTWARE. 23*1c60b9acSAndroid Build Coastguard Worker */ 24*1c60b9acSAndroid Build Coastguard Worker 25*1c60b9acSAndroid Build Coastguard Worker #define lwsl_cache lwsl_debug 26*1c60b9acSAndroid Build Coastguard Worker #define lwsl_hexdump_cache lwsl_hexdump_debug 27*1c60b9acSAndroid Build Coastguard Worker 28*1c60b9acSAndroid Build Coastguard Worker #define LWS_CACHE_MAX_LEVELS 3 29*1c60b9acSAndroid Build Coastguard Worker 30*1c60b9acSAndroid Build Coastguard Worker /* 31*1c60b9acSAndroid Build Coastguard Worker * If we need structure inside the cache tag names, use this character as a 32*1c60b9acSAndroid Build Coastguard Worker * separator 33*1c60b9acSAndroid Build Coastguard Worker */ 34*1c60b9acSAndroid Build Coastguard Worker #define LWSCTAG_SEP '|' 35*1c60b9acSAndroid Build Coastguard Worker 36*1c60b9acSAndroid Build Coastguard Worker /* 37*1c60b9acSAndroid Build Coastguard Worker * Our synthetic cache result items all have tags starting with this char 38*1c60b9acSAndroid Build Coastguard Worker */ 39*1c60b9acSAndroid Build Coastguard Worker #define META_ITEM_LEADING '!' 40*1c60b9acSAndroid Build Coastguard Worker 41*1c60b9acSAndroid Build Coastguard Worker typedef struct lws_cache_ttl_item_heap { 42*1c60b9acSAndroid Build Coastguard Worker lws_dll2_t list_expiry; 43*1c60b9acSAndroid Build Coastguard Worker lws_dll2_t list_lru; 44*1c60b9acSAndroid Build Coastguard Worker 45*1c60b9acSAndroid Build Coastguard Worker lws_usec_t expiry; 46*1c60b9acSAndroid Build Coastguard Worker size_t key_len; 47*1c60b9acSAndroid Build Coastguard Worker size_t size; 48*1c60b9acSAndroid Build Coastguard Worker 49*1c60b9acSAndroid Build Coastguard Worker /* 50*1c60b9acSAndroid Build Coastguard Worker * len + key_len + 1 bytes of data overcommitted, user object first 51*1c60b9acSAndroid Build Coastguard Worker * so it is well-aligned, then the NUL-terminated key name 52*1c60b9acSAndroid Build Coastguard Worker */ 53*1c60b9acSAndroid Build Coastguard Worker } lws_cache_ttl_item_heap_t; 54*1c60b9acSAndroid Build Coastguard Worker 55*1c60b9acSAndroid Build Coastguard Worker /* this is a "base class", all cache implementations have one at the start */ 56*1c60b9acSAndroid Build Coastguard Worker 57*1c60b9acSAndroid Build Coastguard Worker typedef struct lws_cache_ttl_lru { 58*1c60b9acSAndroid Build Coastguard Worker struct lws_cache_creation_info info; 59*1c60b9acSAndroid Build Coastguard Worker lws_sorted_usec_list_t sul; 60*1c60b9acSAndroid Build Coastguard Worker struct lws_cache_ttl_lru *child; 61*1c60b9acSAndroid Build Coastguard Worker uint64_t current_footprint; 62*1c60b9acSAndroid Build Coastguard Worker } lws_cache_ttl_lru_t; 63*1c60b9acSAndroid Build Coastguard Worker 64*1c60b9acSAndroid Build Coastguard Worker /* 65*1c60b9acSAndroid Build Coastguard Worker * The heap-backed cache uses lws_dll2 linked-lists to track items that are 66*1c60b9acSAndroid Build Coastguard Worker * in it. 67*1c60b9acSAndroid Build Coastguard Worker */ 68*1c60b9acSAndroid Build Coastguard Worker 69*1c60b9acSAndroid Build Coastguard Worker typedef struct lws_cache_ttl_lru_heap { 70*1c60b9acSAndroid Build Coastguard Worker lws_cache_ttl_lru_t cache; 71*1c60b9acSAndroid Build Coastguard Worker 72*1c60b9acSAndroid Build Coastguard Worker lws_dll2_owner_t items_expiry; 73*1c60b9acSAndroid Build Coastguard Worker lws_dll2_owner_t items_lru; 74*1c60b9acSAndroid Build Coastguard Worker } lws_cache_ttl_lru_t_heap_t; 75*1c60b9acSAndroid Build Coastguard Worker 76*1c60b9acSAndroid Build Coastguard Worker /* 77*1c60b9acSAndroid Build Coastguard Worker * We want to be able to work with a large file-backed implementation even on 78*1c60b9acSAndroid Build Coastguard Worker * devices that don't have heap to track what is in it. It means if lookups 79*1c60b9acSAndroid Build Coastguard Worker * reach this cache layer, we will be scanning a potentially large file. 80*1c60b9acSAndroid Build Coastguard Worker * 81*1c60b9acSAndroid Build Coastguard Worker * L1 caching of lookups (including null result list) reduces the expense of 82*1c60b9acSAndroid Build Coastguard Worker * this on average. We keep a copy of the last computed earliest expiry. 83*1c60b9acSAndroid Build Coastguard Worker * 84*1c60b9acSAndroid Build Coastguard Worker * We can't keep an open file handle here. Because other processes may change 85*1c60b9acSAndroid Build Coastguard Worker * the cookie file by deleting and replacing it, we have to open it fresh each 86*1c60b9acSAndroid Build Coastguard Worker * time. 87*1c60b9acSAndroid Build Coastguard Worker */ 88*1c60b9acSAndroid Build Coastguard Worker typedef struct lws_cache_nscookiejar { 89*1c60b9acSAndroid Build Coastguard Worker lws_cache_ttl_lru_t cache; 90*1c60b9acSAndroid Build Coastguard Worker 91*1c60b9acSAndroid Build Coastguard Worker lws_usec_t earliest_expiry; 92*1c60b9acSAndroid Build Coastguard Worker } lws_cache_nscookiejar_t; 93*1c60b9acSAndroid Build Coastguard Worker 94*1c60b9acSAndroid Build Coastguard Worker void 95*1c60b9acSAndroid Build Coastguard Worker lws_cache_clear_matches(lws_dll2_owner_t *results_owner); 96*1c60b9acSAndroid Build Coastguard Worker 97*1c60b9acSAndroid Build Coastguard Worker void 98*1c60b9acSAndroid Build Coastguard Worker lws_cache_schedule(struct lws_cache_ttl_lru *cache, sul_cb_t cb, lws_usec_t e); 99