Pet*_*itt 15 linux network-programming kernel-module linux-kernel
我一直在查看net/core/dev.c和其他文件,试图找出如何获取当前配置的网络设备列表,并证明它有点难以找到.
最终目标是能够使用dev.c中的dev_get_stats获取网络设备统计信息,但我需要知道当前接口,以便我可以获取net_device结构来传入.我必须在内核中执行此操作,因为我我正在编写一个模块,它添加了一个新的/ proc /条目,它与当前网络设备的一些统计信息有关,所以我可以从内核中完成这个.
如果有人能指出我如何获得接口,那将非常感激.
caf*_*caf 17
鉴于您要struct net *net
识别您感兴趣的网络命名空间,您应该抓住dev_base_lock
并使用for_each_netdev()
:
read_lock(&dev_base_lock);
for_each_netdev(net, dev) {
/* Inspect dev */
}
read_unlock(&dev_base_lock);
Run Code Online (Sandbox Code Playgroud)
(在较新的内核中,您可以使用RCU,但在这种情况下,这可能是一个过度复杂的问题).
要获取net
要使用的命名空间,您应该使用以下命令注册proc
文件register_pernet_subsys()
:
static const struct file_operations foostats_seq_fops = {
.owner = THIS_MODULE,
.open = foostats_seq_open,
.read = seq_read,
.llseek = seq_lseek,
.release = foostats_seq_release,
};
static int foo_proc_init_net(struct net *net)
{
if (!proc_net_fops_create(net, "foostats", S_IRUGO,
&foostats_seq_fops))
return -ENOMEM;
return 0;
}
static void foo_proc_exit_net(struct net *net)
{
proc_net_remove(net, "foostats");
}
static struct pernet_operations foo_proc_ops = {
.init = foo_proc_init_net,
.exit = foo_proc_exit_net,
};
register_pernet_subsys(&foo_proc_ops)
Run Code Online (Sandbox Code Playgroud)
在您的foostats_seq_open()
函数中,您对net
命名空间进行引用,并将其放在release函数中:
static int foostats_seq_open(struct inode *inode, struct file *file)
{
int err;
struct net *net;
err = -ENXIO;
net = get_proc_net(inode);
if (net == NULL)
goto err_net;
err = single_open(file, foostats_seq_show, net);
if (err < 0)
goto err_open;
return 0;
err_open:
put_net(net);
err_net:
return err;
}
static int foostats_seq_release(struct inode *inode, struct file *file)
{
struct net *net = ((struct seq_file *)file->private_data)->private;
put_net(net);
return single_release(inode, file);
}
Run Code Online (Sandbox Code Playgroud)
foostats_seq_show()
然后该函数可以获取net
,遍历设备,收集统计信息并生成输出:
static int sockstat6_seq_show(struct seq_file *seq, void *v)
{
struct net *net = seq->private;
struct net_device *dev;
int foostat, barstat;
read_lock(&dev_base_lock);
for_each_netdev(net, dev) {
/* Inspect dev */
}
read_unlock(&dev_base_lock);
seq_printf(seq, "Foo: %d\n", foostat);
seq_printf(seq, "Bar: %d\n", barstat);
return 0;
}
Run Code Online (Sandbox Code Playgroud)
eat*_*ter 16
这应该做的伎俩:
#include <linux/netdevice.h>
struct net_device *dev;
read_lock(&dev_base_lock);
dev = first_net_device(&init_net);
while (dev) {
printk(KERN_INFO "found [%s]\n", dev->name);
dev = next_net_device(dev);
}
read_unlock(&dev_base_lock);
Run Code Online (Sandbox Code Playgroud)