1 // SPDX-License-Identifier: GPL-2.0-or-later
2 /* Miscellaneous bits for the netfs support library.
4 * Copyright (C) 2022 Red Hat, Inc. All Rights Reserved.
8 #include <linux/module.h>
9 #include <linux/export.h>
10 #include <linux/mempool.h>
11 #include <linux/proc_fs.h>
12 #include <linux/seq_file.h>
14 #define CREATE_TRACE_POINTS
15 #include <trace/events/netfs.h>
17 MODULE_DESCRIPTION("Network fs support");
18 MODULE_AUTHOR("Red Hat, Inc.");
19 MODULE_LICENSE("GPL");
21 EXPORT_TRACEPOINT_SYMBOL(netfs_sreq);
24 module_param_named(debug, netfs_debug, uint, S_IWUSR | S_IRUGO);
25 MODULE_PARM_DESC(netfs_debug, "Netfs support debugging mask");
27 static struct kmem_cache *netfs_request_slab;
28 static struct kmem_cache *netfs_subrequest_slab;
29 mempool_t netfs_request_pool;
30 mempool_t netfs_subrequest_pool;
33 LIST_HEAD(netfs_io_requests);
34 DEFINE_SPINLOCK(netfs_proc_lock);
36 static const char *netfs_origins[nr__netfs_io_origin] = {
37 [NETFS_READAHEAD] = "RA",
38 [NETFS_READPAGE] = "RP",
39 [NETFS_READ_GAPS] = "RG",
40 [NETFS_READ_SINGLE] = "R1",
41 [NETFS_READ_FOR_WRITE] = "RW",
42 [NETFS_DIO_READ] = "DR",
43 [NETFS_WRITEBACK] = "WB",
44 [NETFS_WRITEBACK_SINGLE] = "W1",
45 [NETFS_WRITETHROUGH] = "WT",
46 [NETFS_UNBUFFERED_WRITE] = "UW",
47 [NETFS_DIO_WRITE] = "DW",
48 [NETFS_PGPRIV2_COPY_TO_CACHE] = "2C",
52 * Generate a list of I/O requests in /proc/fs/netfs/requests
54 static int netfs_requests_seq_show(struct seq_file *m, void *v)
56 struct netfs_io_request *rreq;
58 if (v == &netfs_io_requests) {
60 "REQUEST OR REF FL ERR OPS COVERAGE\n"
61 "======== == === == ==== === =========\n"
66 rreq = list_entry(v, struct netfs_io_request, proc_link);
68 "%08x %s %3d %2lx %4ld %3d @%04llx %llx/%llx",
70 netfs_origins[rreq->origin],
71 refcount_read(&rreq->ref),
75 rreq->start, rreq->submitted, rreq->len);
80 static void *netfs_requests_seq_start(struct seq_file *m, loff_t *_pos)
84 return seq_list_start_head(&netfs_io_requests, *_pos);
87 static void *netfs_requests_seq_next(struct seq_file *m, void *v, loff_t *_pos)
89 return seq_list_next(v, &netfs_io_requests, _pos);
92 static void netfs_requests_seq_stop(struct seq_file *m, void *v)
98 static const struct seq_operations netfs_requests_seq_ops = {
99 .start = netfs_requests_seq_start,
100 .next = netfs_requests_seq_next,
101 .stop = netfs_requests_seq_stop,
102 .show = netfs_requests_seq_show,
104 #endif /* CONFIG_PROC_FS */
106 static int __init netfs_init(void)
110 netfs_request_slab = kmem_cache_create("netfs_request",
111 sizeof(struct netfs_io_request), 0,
112 SLAB_HWCACHE_ALIGN | SLAB_ACCOUNT,
114 if (!netfs_request_slab)
117 if (mempool_init_slab_pool(&netfs_request_pool, 100, netfs_request_slab) < 0)
120 netfs_subrequest_slab = kmem_cache_create("netfs_subrequest",
121 sizeof(struct netfs_io_subrequest) + 16, 0,
122 SLAB_HWCACHE_ALIGN | SLAB_ACCOUNT,
124 if (!netfs_subrequest_slab)
127 if (mempool_init_slab_pool(&netfs_subrequest_pool, 100, netfs_subrequest_slab) < 0)
128 goto error_subreqpool;
130 if (!proc_mkdir("fs/netfs", NULL))
132 if (!proc_create_seq("fs/netfs/requests", S_IFREG | 0444, NULL,
133 &netfs_requests_seq_ops))
135 #ifdef CONFIG_FSCACHE_STATS
136 if (!proc_create_single("fs/netfs/stats", S_IFREG | 0444, NULL,
141 ret = fscache_init();
148 remove_proc_subtree("fs/netfs", NULL);
150 mempool_exit(&netfs_subrequest_pool);
152 kmem_cache_destroy(netfs_subrequest_slab);
154 mempool_exit(&netfs_request_pool);
156 kmem_cache_destroy(netfs_request_slab);
160 fs_initcall(netfs_init);
162 static void __exit netfs_exit(void)
165 remove_proc_subtree("fs/netfs", NULL);
166 mempool_exit(&netfs_subrequest_pool);
167 kmem_cache_destroy(netfs_subrequest_slab);
168 mempool_exit(&netfs_request_pool);
169 kmem_cache_destroy(netfs_request_slab);
171 module_exit(netfs_exit);