// SPDX-License-Identifier: GPL-2.0-or-later /* * NET3: Garbage Collector For AF_UNIX sockets * * Garbage Collector: * Copyright (C) Barak A. Pearlmutter. * * Chopped about by Alan Cox 22/3/96 to make it fit the AF_UNIX socket problem. * If it doesn't work blame me, it worked when Barak sent it. * * Assumptions: * * - object w/ a bit * - free list * * Current optimizations: * * - explicit stack instead of recursion * - tail recurse on first born instead of immediate push/pop * - we gather the stuff that should not be killed into tree * and stack is just a path from root to the current pointer. * * Future optimizations: * * - don't just push entire root set; process in place * * Fixes: * Alan Cox 07 Sept 1997 Vmalloc internal stack as needed. * Cope with changing max_files. * Al Viro 11 Oct 1998 * Graph may have cycles. That is, we can send the descriptor * of foo to bar and vice versa. Current code chokes on that. * Fix: move SCM_RIGHTS ones into the separate list and then * skb_free() them all instead of doing explicit fput's. * Another problem: since fput() may block somebody may * create a new unix_socket when we are in the middle of sweep * phase. Fix: revert the logic wrt MARKED. Mark everything * upon the beginning and unmark non-junk ones. * * [12 Oct 1998] AAARGH! New code purges all SCM_RIGHTS * sent to connect()'ed but still not accept()'ed sockets. * Fixed. Old code had slightly different problem here: * extra fput() in situation when we passed the descriptor via * such socket and closed it (descriptor). That would happen on * each unix_gc() until the accept(). Since the struct file in * question would go to the free list and might be reused... * That might be the reason of random oopses on filp_close() * in unrelated processes. * * AV 28 Feb 1999 * Kill the explicit allocation of stack. Now we keep the tree * with root in dummy + pointer (gc_current) to one of the nodes. * Stack is represented as path from gc_current to dummy. Unmark * now means "add to tree". Push == "make it a son of gc_current". * Pop == "move gc_current to parent". We keep only pointers to * parents (->gc_tree). * AV 1 Mar 1999 * Damn. Added missing check for ->dead in listen queues scanning. * * Miklos Szeredi 25 Jun 2007 * Reimplement with a cycle collecting algorithm. This should * solve several problems with the previous code, like being racy * wrt receive and holding up unrelated socket operations. */ #include <linux/kernel.h> #include <linux/string.h> #include <linux/socket.h> #include <linux/un.h> #include <linux/net.h> #include <linux/fs.h> #include <linux/skbuff.h> #include <linux/netdevice.h> #include <linux/file.h> #include <linux/proc_fs.h> #include <linux/mutex.h> #include <linux/wait.h> #include <net/sock.h> #include <net/af_unix.h> #include <net/scm.h> #include <net/tcp_states.h> struct unix_sock *unix_get_socket(struct file *filp) { … } static struct unix_vertex *unix_edge_successor(struct unix_edge *edge) { … } static bool unix_graph_maybe_cyclic; static bool unix_graph_grouped; static void unix_update_graph(struct unix_vertex *vertex) { … } static LIST_HEAD(unix_unvisited_vertices); enum unix_vertex_index { … }; static unsigned long unix_vertex_unvisited_index = …; static void unix_add_edge(struct scm_fp_list *fpl, struct unix_edge *edge) { … } static void unix_del_edge(struct scm_fp_list *fpl, struct unix_edge *edge) { … } static void unix_free_vertices(struct scm_fp_list *fpl) { … } static DEFINE_SPINLOCK(unix_gc_lock); unsigned int unix_tot_inflight; void unix_add_edges(struct scm_fp_list *fpl, struct unix_sock *receiver) { … } void unix_del_edges(struct scm_fp_list *fpl) { … } void unix_update_edges(struct unix_sock *receiver) { … } int unix_prepare_fpl(struct scm_fp_list *fpl) { … } void unix_destroy_fpl(struct scm_fp_list *fpl) { … } static bool unix_vertex_dead(struct unix_vertex *vertex) { … } static void unix_collect_skb(struct list_head *scc, struct sk_buff_head *hitlist) { … } static bool unix_scc_cyclic(struct list_head *scc) { … } static LIST_HEAD(unix_visited_vertices); static unsigned long unix_vertex_grouped_index = …; static void __unix_walk_scc(struct unix_vertex *vertex, unsigned long *last_index, struct sk_buff_head *hitlist) { … } static void unix_walk_scc(struct sk_buff_head *hitlist) { … } static void unix_walk_scc_fast(struct sk_buff_head *hitlist) { … } static bool gc_in_progress; static void __unix_gc(struct work_struct *work) { … } static DECLARE_WORK(unix_gc_work, __unix_gc); void unix_gc(void) { … } #define UNIX_INFLIGHT_TRIGGER_GC … #define UNIX_INFLIGHT_SANE_USER … void wait_for_unix_gc(struct scm_fp_list *fpl) { … }