Получение списка сетевых устройств внутри ядра Linux - PullRequest
15 голосов
/ 21 декабря 2010

Я просматривал net / core / dev.c и другие файлы, чтобы попытаться выяснить, как получить список сетевых устройств, которые в данный момент настроены, и найти его немного сложно.

Конечная цель - получить статистику сетевых устройств с помощью dev_get_stats в dev.c, но мне нужно знать текущие интерфейсы, чтобы я мог получить структуру net_device для передачи. Я должен сделать это внутриЯдро, поскольку я пишу модуль, который добавляет новую запись / proc /, которая относится к некоторой статистике с текущих сетевых устройств, поэтому из того, что я могу собрать, это должно быть сделано внутри ядра.

Если кто-то могукажите мне, как получить интерфейсы, это будет высоко ценится.

Ответы [ 2 ]

17 голосов
/ 21 декабря 2010

Учитывая 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);

(В более новых ядрах вы можете использовать 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)

В вашей функции 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);
}

Функция 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;
}
16 голосов
/ 21 декабря 2010

Это должно сработать:

#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);
...