aboutsummaryrefslogtreecommitdiffstats
path: root/net/netfilter/nf_flow_table_procfs.c
blob: 159b033a43e6061196140828c292238db78c9dec (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
// SPDX-License-Identifier: GPL-2.0-only
#include <linux/kernel.h>
#include <linux/proc_fs.h>
#include <net/netfilter/nf_flow_table.h>

static void *nf_flow_table_cpu_seq_start(struct seq_file *seq, loff_t *pos)
{
	struct net *net = seq_file_net(seq);
	int cpu;

	if (*pos == 0)
		return SEQ_START_TOKEN;

	for (cpu = *pos - 1; cpu < nr_cpu_ids; ++cpu) {
		if (!cpu_possible(cpu))
			continue;
		*pos = cpu + 1;
		return per_cpu_ptr(net->ft.stat, cpu);
	}

	return NULL;
}

static void *nf_flow_table_cpu_seq_next(struct seq_file *seq, void *v, loff_t *pos)
{
	struct net *net = seq_file_net(seq);
	int cpu;

	for (cpu = *pos; cpu < nr_cpu_ids; ++cpu) {
		if (!cpu_possible(cpu))
			continue;
		*pos = cpu + 1;
		return per_cpu_ptr(net->ft.stat, cpu);
	}
	(*pos)++;
	return NULL;
}

static void nf_flow_table_cpu_seq_stop(struct seq_file *seq, void *v)
{
}

static int nf_flow_table_cpu_seq_show(struct seq_file *seq, void *v)
{
	const struct nf_flow_table_stat *st = v;

	if (v == SEQ_START_TOKEN) {
		seq_puts(seq, "wq_add   wq_del   wq_stats\n");
		return 0;
	}

	seq_printf(seq, "%8d %8d %8d\n",
		   st->count_wq_add,
		   st->count_wq_del,
		   st->count_wq_stats
		);
	return 0;
}

static const struct seq_operations nf_flow_table_cpu_seq_ops = {
	.start	= nf_flow_table_cpu_seq_start,
	.next	= nf_flow_table_cpu_seq_next,
	.stop	= nf_flow_table_cpu_seq_stop,
	.show	= nf_flow_table_cpu_seq_show,
};

int nf_flow_table_init_proc(struct net *net)
{
	struct proc_dir_entry *pde;

	pde = proc_create_net("nf_flowtable", 0444, net->proc_net_stat,
			      &nf_flow_table_cpu_seq_ops,
			      sizeof(struct seq_net_private));
	return pde ? 0 : -ENOMEM;
}

void nf_flow_table_fini_proc(struct net *net)
{
	remove_proc_entry("nf_flowtable", net->proc_net_stat);
}