162306a36Sopenharmony_ci#ifndef INTERNAL_IO_SLIST_H 262306a36Sopenharmony_ci#define INTERNAL_IO_SLIST_H 362306a36Sopenharmony_ci 462306a36Sopenharmony_ci#include <linux/io_uring_types.h> 562306a36Sopenharmony_ci 662306a36Sopenharmony_ci#define __wq_list_for_each(pos, head) \ 762306a36Sopenharmony_ci for (pos = (head)->first; pos; pos = (pos)->next) 862306a36Sopenharmony_ci 962306a36Sopenharmony_ci#define wq_list_for_each(pos, prv, head) \ 1062306a36Sopenharmony_ci for (pos = (head)->first, prv = NULL; pos; prv = pos, pos = (pos)->next) 1162306a36Sopenharmony_ci 1262306a36Sopenharmony_ci#define wq_list_for_each_resume(pos, prv) \ 1362306a36Sopenharmony_ci for (; pos; prv = pos, pos = (pos)->next) 1462306a36Sopenharmony_ci 1562306a36Sopenharmony_ci#define wq_list_empty(list) (READ_ONCE((list)->first) == NULL) 1662306a36Sopenharmony_ci 1762306a36Sopenharmony_ci#define INIT_WQ_LIST(list) do { \ 1862306a36Sopenharmony_ci (list)->first = NULL; \ 1962306a36Sopenharmony_ci} while (0) 2062306a36Sopenharmony_ci 2162306a36Sopenharmony_cistatic inline void wq_list_add_after(struct io_wq_work_node *node, 2262306a36Sopenharmony_ci struct io_wq_work_node *pos, 2362306a36Sopenharmony_ci struct io_wq_work_list *list) 2462306a36Sopenharmony_ci{ 2562306a36Sopenharmony_ci struct io_wq_work_node *next = pos->next; 2662306a36Sopenharmony_ci 2762306a36Sopenharmony_ci pos->next = node; 2862306a36Sopenharmony_ci node->next = next; 2962306a36Sopenharmony_ci if (!next) 3062306a36Sopenharmony_ci list->last = node; 3162306a36Sopenharmony_ci} 3262306a36Sopenharmony_ci 3362306a36Sopenharmony_cistatic inline void wq_list_add_tail(struct io_wq_work_node *node, 3462306a36Sopenharmony_ci struct io_wq_work_list *list) 3562306a36Sopenharmony_ci{ 3662306a36Sopenharmony_ci node->next = NULL; 3762306a36Sopenharmony_ci if (!list->first) { 3862306a36Sopenharmony_ci list->last = node; 3962306a36Sopenharmony_ci WRITE_ONCE(list->first, node); 4062306a36Sopenharmony_ci } else { 4162306a36Sopenharmony_ci list->last->next = node; 4262306a36Sopenharmony_ci list->last = node; 4362306a36Sopenharmony_ci } 4462306a36Sopenharmony_ci} 4562306a36Sopenharmony_ci 4662306a36Sopenharmony_cistatic inline void wq_list_add_head(struct io_wq_work_node *node, 4762306a36Sopenharmony_ci struct io_wq_work_list *list) 4862306a36Sopenharmony_ci{ 4962306a36Sopenharmony_ci node->next = list->first; 5062306a36Sopenharmony_ci if (!node->next) 5162306a36Sopenharmony_ci list->last = node; 5262306a36Sopenharmony_ci WRITE_ONCE(list->first, node); 5362306a36Sopenharmony_ci} 5462306a36Sopenharmony_ci 5562306a36Sopenharmony_cistatic inline void wq_list_cut(struct io_wq_work_list *list, 5662306a36Sopenharmony_ci struct io_wq_work_node *last, 5762306a36Sopenharmony_ci struct io_wq_work_node *prev) 5862306a36Sopenharmony_ci{ 5962306a36Sopenharmony_ci /* first in the list, if prev==NULL */ 6062306a36Sopenharmony_ci if (!prev) 6162306a36Sopenharmony_ci WRITE_ONCE(list->first, last->next); 6262306a36Sopenharmony_ci else 6362306a36Sopenharmony_ci prev->next = last->next; 6462306a36Sopenharmony_ci 6562306a36Sopenharmony_ci if (last == list->last) 6662306a36Sopenharmony_ci list->last = prev; 6762306a36Sopenharmony_ci last->next = NULL; 6862306a36Sopenharmony_ci} 6962306a36Sopenharmony_ci 7062306a36Sopenharmony_cistatic inline void __wq_list_splice(struct io_wq_work_list *list, 7162306a36Sopenharmony_ci struct io_wq_work_node *to) 7262306a36Sopenharmony_ci{ 7362306a36Sopenharmony_ci list->last->next = to->next; 7462306a36Sopenharmony_ci to->next = list->first; 7562306a36Sopenharmony_ci INIT_WQ_LIST(list); 7662306a36Sopenharmony_ci} 7762306a36Sopenharmony_ci 7862306a36Sopenharmony_cistatic inline bool wq_list_splice(struct io_wq_work_list *list, 7962306a36Sopenharmony_ci struct io_wq_work_node *to) 8062306a36Sopenharmony_ci{ 8162306a36Sopenharmony_ci if (!wq_list_empty(list)) { 8262306a36Sopenharmony_ci __wq_list_splice(list, to); 8362306a36Sopenharmony_ci return true; 8462306a36Sopenharmony_ci } 8562306a36Sopenharmony_ci return false; 8662306a36Sopenharmony_ci} 8762306a36Sopenharmony_ci 8862306a36Sopenharmony_cistatic inline void wq_stack_add_head(struct io_wq_work_node *node, 8962306a36Sopenharmony_ci struct io_wq_work_node *stack) 9062306a36Sopenharmony_ci{ 9162306a36Sopenharmony_ci node->next = stack->next; 9262306a36Sopenharmony_ci stack->next = node; 9362306a36Sopenharmony_ci} 9462306a36Sopenharmony_ci 9562306a36Sopenharmony_cistatic inline void wq_list_del(struct io_wq_work_list *list, 9662306a36Sopenharmony_ci struct io_wq_work_node *node, 9762306a36Sopenharmony_ci struct io_wq_work_node *prev) 9862306a36Sopenharmony_ci{ 9962306a36Sopenharmony_ci wq_list_cut(list, node, prev); 10062306a36Sopenharmony_ci} 10162306a36Sopenharmony_ci 10262306a36Sopenharmony_cistatic inline 10362306a36Sopenharmony_cistruct io_wq_work_node *wq_stack_extract(struct io_wq_work_node *stack) 10462306a36Sopenharmony_ci{ 10562306a36Sopenharmony_ci struct io_wq_work_node *node = stack->next; 10662306a36Sopenharmony_ci 10762306a36Sopenharmony_ci stack->next = node->next; 10862306a36Sopenharmony_ci return node; 10962306a36Sopenharmony_ci} 11062306a36Sopenharmony_ci 11162306a36Sopenharmony_cistatic inline struct io_wq_work *wq_next_work(struct io_wq_work *work) 11262306a36Sopenharmony_ci{ 11362306a36Sopenharmony_ci if (!work->list.next) 11462306a36Sopenharmony_ci return NULL; 11562306a36Sopenharmony_ci 11662306a36Sopenharmony_ci return container_of(work->list.next, struct io_wq_work, list); 11762306a36Sopenharmony_ci} 11862306a36Sopenharmony_ci 11962306a36Sopenharmony_ci#endif // INTERNAL_IO_SLIST_H 120