| 
									
										
										
										
											2022-09-15 18:31:11 +08:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright 2022 The OpenSSL Project Authors. All Rights Reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Licensed under the Apache License 2.0 (the "License").  You may not use | 
					
						
							|  |  |  |  * this file except in compliance with the License.  You can obtain a copy | 
					
						
							|  |  |  |  * in the file LICENSE in the source distribution or at | 
					
						
							|  |  |  |  * https://www.openssl.org/source/license.html
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "internal/quic_cfq.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct quic_cfq_item_ex_st QUIC_CFQ_ITEM_EX; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct quic_cfq_item_ex_st { | 
					
						
							|  |  |  |     QUIC_CFQ_ITEM           public; | 
					
						
							|  |  |  |     QUIC_CFQ_ITEM_EX       *prev, *next; | 
					
						
							|  |  |  |     unsigned char          *encoded; | 
					
						
							|  |  |  |     cfq_free_cb            *free_cb; | 
					
						
							|  |  |  |     void                   *free_cb_arg; | 
					
						
							|  |  |  |     uint64_t                frame_type; | 
					
						
							|  |  |  |     size_t                  encoded_len; | 
					
						
							|  |  |  |     uint32_t                priority, pn_space; | 
					
						
							| 
									
										
										
										
											2022-10-13 00:44:40 +08:00
										 |  |  |     int                     state; | 
					
						
							| 
									
										
										
										
											2022-09-15 18:31:11 +08:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-13 00:44:40 +08:00
										 |  |  | uint64_t ossl_quic_cfq_item_get_frame_type(const QUIC_CFQ_ITEM *item) | 
					
						
							| 
									
										
										
										
											2022-09-15 18:31:11 +08:00
										 |  |  | { | 
					
						
							|  |  |  |     QUIC_CFQ_ITEM_EX *ex = (QUIC_CFQ_ITEM_EX *)item; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return ex->frame_type; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-13 00:44:40 +08:00
										 |  |  | const unsigned char *ossl_quic_cfq_item_get_encoded(const QUIC_CFQ_ITEM *item) | 
					
						
							| 
									
										
										
										
											2022-09-15 18:31:11 +08:00
										 |  |  | { | 
					
						
							|  |  |  |     QUIC_CFQ_ITEM_EX *ex = (QUIC_CFQ_ITEM_EX *)item; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return ex->encoded; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-13 00:44:40 +08:00
										 |  |  | size_t ossl_quic_cfq_item_get_encoded_len(const QUIC_CFQ_ITEM *item) | 
					
						
							| 
									
										
										
										
											2022-09-15 18:31:11 +08:00
										 |  |  | { | 
					
						
							|  |  |  |     QUIC_CFQ_ITEM_EX *ex = (QUIC_CFQ_ITEM_EX *)item; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return ex->encoded_len; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-13 00:44:40 +08:00
										 |  |  | int ossl_quic_cfq_item_get_state(const QUIC_CFQ_ITEM *item) | 
					
						
							| 
									
										
										
										
											2022-09-15 18:31:11 +08:00
										 |  |  | { | 
					
						
							|  |  |  |     QUIC_CFQ_ITEM_EX *ex = (QUIC_CFQ_ITEM_EX *)item; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return ex->state; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-13 00:44:40 +08:00
										 |  |  | uint32_t ossl_quic_cfq_item_get_pn_space(const QUIC_CFQ_ITEM *item) | 
					
						
							| 
									
										
										
										
											2022-09-15 18:31:11 +08:00
										 |  |  | { | 
					
						
							|  |  |  |     QUIC_CFQ_ITEM_EX *ex = (QUIC_CFQ_ITEM_EX *)item; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return ex->pn_space; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct quic_cfq_item_list_st { | 
					
						
							|  |  |  |     QUIC_CFQ_ITEM_EX *head, *tail; | 
					
						
							|  |  |  | } QUIC_CFQ_ITEM_LIST; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct quic_cfq_st { | 
					
						
							|  |  |  |     /* 
 | 
					
						
							|  |  |  |      * Invariant: A CFQ item is always in exactly one of these lists, never more | 
					
						
							|  |  |  |      * or less than one. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * Invariant: The list the CFQ item is determined exactly by the state field | 
					
						
							|  |  |  |      * of the item. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     QUIC_CFQ_ITEM_LIST                      new_list, tx_list, free_list; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int compare(const QUIC_CFQ_ITEM_EX *a, const QUIC_CFQ_ITEM_EX *b) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (a->pn_space < b->pn_space) | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     else if (a->pn_space > b->pn_space) | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (a->priority > b->priority) | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     else if (a->priority < b->priority) | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void list_remove(QUIC_CFQ_ITEM_LIST *l, QUIC_CFQ_ITEM_EX *n) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (l->head == n) | 
					
						
							|  |  |  |         l->head = n->next; | 
					
						
							|  |  |  |     if (l->tail == n) | 
					
						
							|  |  |  |         l->tail = n->prev; | 
					
						
							|  |  |  |     if (n->prev != NULL) | 
					
						
							|  |  |  |         n->prev->next = n->next; | 
					
						
							|  |  |  |     if (n->next != NULL) | 
					
						
							|  |  |  |         n->next->prev = n->prev; | 
					
						
							|  |  |  |     n->prev = n->next = NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void list_insert_head(QUIC_CFQ_ITEM_LIST *l, QUIC_CFQ_ITEM_EX *n) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     n->next = l->head; | 
					
						
							|  |  |  |     n->prev = NULL; | 
					
						
							|  |  |  |     l->head = n; | 
					
						
							|  |  |  |     if (n->next != NULL) | 
					
						
							|  |  |  |         n->next->prev = n; | 
					
						
							|  |  |  |     if (l->tail == NULL) | 
					
						
							|  |  |  |         l->tail = n; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void list_insert_tail(QUIC_CFQ_ITEM_LIST *l, QUIC_CFQ_ITEM_EX *n) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     n->prev = l->tail; | 
					
						
							|  |  |  |     n->next = NULL; | 
					
						
							|  |  |  |     l->tail = n; | 
					
						
							|  |  |  |     if (n->prev != NULL) | 
					
						
							|  |  |  |         n->prev->next = n; | 
					
						
							|  |  |  |     if (l->head == NULL) | 
					
						
							|  |  |  |         l->head = n; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void list_insert_after(QUIC_CFQ_ITEM_LIST *l, | 
					
						
							|  |  |  |                               QUIC_CFQ_ITEM_EX *ref, | 
					
						
							|  |  |  |                               QUIC_CFQ_ITEM_EX *n) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     n->prev = ref; | 
					
						
							|  |  |  |     n->next = ref->next; | 
					
						
							|  |  |  |     if (ref->next != NULL) | 
					
						
							|  |  |  |         ref->next->prev = n; | 
					
						
							|  |  |  |     ref->next = n; | 
					
						
							|  |  |  |     if (l->tail == ref) | 
					
						
							|  |  |  |         l->tail = n; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void list_insert_sorted(QUIC_CFQ_ITEM_LIST *l, QUIC_CFQ_ITEM_EX *n, | 
					
						
							|  |  |  |                                int (*cmp)(const QUIC_CFQ_ITEM_EX *a, | 
					
						
							|  |  |  |                                           const QUIC_CFQ_ITEM_EX *b)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QUIC_CFQ_ITEM_EX *p = l->head, *pprev = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (p == NULL) { | 
					
						
							|  |  |  |         l->head = l->tail = n; | 
					
						
							|  |  |  |         n->prev = n->next = NULL; | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (; p != NULL && cmp(p, n) < 0; pprev = p, p = p->next); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (p == NULL) | 
					
						
							|  |  |  |         list_insert_tail(l, n); | 
					
						
							|  |  |  |     else if (pprev == NULL) | 
					
						
							|  |  |  |         list_insert_head(l, n); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         list_insert_after(l, pprev, n); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QUIC_CFQ *ossl_quic_cfq_new(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QUIC_CFQ *cfq = OPENSSL_zalloc(sizeof(*cfq)); | 
					
						
							| 
									
										
										
										
											2022-10-13 00:44:40 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-15 18:31:11 +08:00
										 |  |  |     if (cfq == NULL) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return cfq; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void clear_item(QUIC_CFQ_ITEM_EX *item) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (item->free_cb != NULL) { | 
					
						
							|  |  |  |         item->free_cb(item->encoded, item->encoded_len, item->free_cb_arg); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         item->free_cb       = NULL; | 
					
						
							|  |  |  |         item->encoded       = NULL; | 
					
						
							|  |  |  |         item->encoded_len   = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     item->state = -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void free_list_items(QUIC_CFQ_ITEM_LIST *l) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QUIC_CFQ_ITEM_EX *p, *pnext; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (p = l->head; p != NULL; p = pnext) { | 
					
						
							|  |  |  |         pnext = p->next; | 
					
						
							|  |  |  |         clear_item(p); | 
					
						
							|  |  |  |         OPENSSL_free(p); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ossl_quic_cfq_free(QUIC_CFQ *cfq) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (cfq == NULL) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     free_list_items(&cfq->new_list); | 
					
						
							|  |  |  |     free_list_items(&cfq->tx_list); | 
					
						
							|  |  |  |     free_list_items(&cfq->free_list); | 
					
						
							|  |  |  |     OPENSSL_free(cfq); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static QUIC_CFQ_ITEM_EX *cfq_get_free(QUIC_CFQ *cfq) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QUIC_CFQ_ITEM_EX *item = cfq->free_list.head; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (item != NULL) | 
					
						
							|  |  |  |         return item; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     item = OPENSSL_zalloc(sizeof(*item)); | 
					
						
							|  |  |  |     if (item == NULL) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     item->state = -1; | 
					
						
							|  |  |  |     list_insert_tail(&cfq->free_list, item); | 
					
						
							|  |  |  |     return item; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QUIC_CFQ_ITEM *ossl_quic_cfq_add_frame(QUIC_CFQ            *cfq, | 
					
						
							|  |  |  |                                        uint32_t             priority, | 
					
						
							|  |  |  |                                        uint32_t             pn_space, | 
					
						
							|  |  |  |                                        uint64_t             frame_type, | 
					
						
							|  |  |  |                                        const unsigned char *encoded, | 
					
						
							|  |  |  |                                        size_t               encoded_len, | 
					
						
							|  |  |  |                                        cfq_free_cb         *free_cb, | 
					
						
							|  |  |  |                                        void                *free_cb_arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QUIC_CFQ_ITEM_EX *item = cfq_get_free(cfq); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (item == NULL) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     item->priority      = priority; | 
					
						
							|  |  |  |     item->frame_type    = frame_type; | 
					
						
							|  |  |  |     item->pn_space      = pn_space; | 
					
						
							|  |  |  |     item->encoded       = (unsigned char *)encoded; | 
					
						
							|  |  |  |     item->encoded_len   = encoded_len; | 
					
						
							|  |  |  |     item->free_cb       = free_cb; | 
					
						
							|  |  |  |     item->free_cb_arg   = free_cb_arg; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     item->state = QUIC_CFQ_STATE_NEW; | 
					
						
							|  |  |  |     list_remove(&cfq->free_list, item); | 
					
						
							|  |  |  |     list_insert_sorted(&cfq->new_list, item, compare); | 
					
						
							|  |  |  |     return &item->public; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ossl_quic_cfq_mark_tx(QUIC_CFQ *cfq, QUIC_CFQ_ITEM *item) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QUIC_CFQ_ITEM_EX *ex = (QUIC_CFQ_ITEM_EX *)item; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (ex->state) { | 
					
						
							|  |  |  |     case QUIC_CFQ_STATE_NEW: | 
					
						
							|  |  |  |         list_remove(&cfq->new_list, ex); | 
					
						
							|  |  |  |         list_insert_tail(&cfq->tx_list, ex); | 
					
						
							|  |  |  |         ex->state = QUIC_CFQ_STATE_TX; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case QUIC_CFQ_STATE_TX: | 
					
						
							|  |  |  |         break; /* nothing to do */ | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         assert(0); /* invalid state (e.g. in free state) */ | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ossl_quic_cfq_mark_lost(QUIC_CFQ *cfq, QUIC_CFQ_ITEM *item, | 
					
						
							|  |  |  |                              uint32_t priority) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QUIC_CFQ_ITEM_EX *ex = (QUIC_CFQ_ITEM_EX *)item; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (ex->state) { | 
					
						
							|  |  |  |     case QUIC_CFQ_STATE_NEW: | 
					
						
							|  |  |  |         if (priority != UINT32_MAX && priority != ex->priority) { | 
					
						
							|  |  |  |             list_remove(&cfq->new_list, ex); | 
					
						
							|  |  |  |             ex->priority = priority; | 
					
						
							|  |  |  |             list_insert_sorted(&cfq->new_list, ex, compare); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; /* nothing to do */ | 
					
						
							|  |  |  |     case QUIC_CFQ_STATE_TX: | 
					
						
							|  |  |  |         if (priority != UINT32_MAX) | 
					
						
							|  |  |  |             ex->priority = priority; | 
					
						
							|  |  |  |         list_remove(&cfq->tx_list, ex); | 
					
						
							|  |  |  |         list_insert_sorted(&cfq->new_list, ex, compare); | 
					
						
							|  |  |  |         ex->state = QUIC_CFQ_STATE_NEW; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         assert(0); /* invalid state (e.g. in free state) */ | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Releases a CFQ item. The item may be in either state (NEW or TX) prior to the | 
					
						
							|  |  |  |  * call. The QUIC_CFQ_ITEM pointer must not be used following this call. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void ossl_quic_cfq_release(QUIC_CFQ *cfq, QUIC_CFQ_ITEM *item) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QUIC_CFQ_ITEM_EX *ex = (QUIC_CFQ_ITEM_EX *)item; | 
					
						
							| 
									
										
										
										
											2022-10-13 00:44:40 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-15 18:31:11 +08:00
										 |  |  |     switch (ex->state) { | 
					
						
							|  |  |  |     case QUIC_CFQ_STATE_NEW: | 
					
						
							|  |  |  |         list_remove(&cfq->new_list, ex); | 
					
						
							|  |  |  |         list_insert_tail(&cfq->free_list, ex); | 
					
						
							|  |  |  |         clear_item(ex); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case QUIC_CFQ_STATE_TX: | 
					
						
							|  |  |  |         list_remove(&cfq->tx_list, ex); | 
					
						
							|  |  |  |         list_insert_tail(&cfq->free_list, ex); | 
					
						
							|  |  |  |         clear_item(ex); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         assert(0); /* invalid state (e.g. in free state) */ | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-13 00:44:40 +08:00
										 |  |  | QUIC_CFQ_ITEM *ossl_quic_cfq_get_priority_head(const QUIC_CFQ *cfq, | 
					
						
							|  |  |  |                                                uint32_t pn_space) | 
					
						
							| 
									
										
										
										
											2022-09-15 18:31:11 +08:00
										 |  |  | { | 
					
						
							|  |  |  |     QUIC_CFQ_ITEM_EX *item = cfq->new_list.head; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (; item != NULL && item->pn_space != pn_space; item = item->next); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-15 19:48:50 +08:00
										 |  |  |     if (item == NULL) | 
					
						
							| 
									
										
										
										
											2022-10-13 00:44:40 +08:00
										 |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2022-09-15 19:48:50 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-15 18:31:11 +08:00
										 |  |  |     return &item->public; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-13 00:44:40 +08:00
										 |  |  | QUIC_CFQ_ITEM *ossl_quic_cfq_item_get_priority_next(const QUIC_CFQ_ITEM *item, | 
					
						
							| 
									
										
										
										
											2022-09-15 18:31:11 +08:00
										 |  |  |                                                     uint32_t pn_space) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QUIC_CFQ_ITEM_EX *ex = (QUIC_CFQ_ITEM_EX *)item; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (ex == NULL) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |      ex = ex->next; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |      for (; ex != NULL && ex->pn_space != pn_space; ex = ex->next); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-15 19:48:50 +08:00
										 |  |  |      if (ex == NULL) | 
					
						
							|  |  |  |          return NULL; /* ubsan */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-15 18:31:11 +08:00
										 |  |  |      return &ex->public; | 
					
						
							|  |  |  | } |