Изменить флаги моста с помощью libnl - PullRequest
2 голосов
/ 11 июня 2019

Я пытаюсь создать мост с помощью libnl и изменить его флаги.Код:

#include <netlink/netlink.h>
#include <netlink/route/link.h>
#include <netlink/route/link/bridge.h>
#include <linux/if.h>

#define BRIDGE_NAME "brr"
int main()
{
    struct rtnl_link *link;
    struct rtnl_link *change;
    struct nl_sock *sk;

    int err;

    sk = nl_socket_alloc();
    if ((err = nl_connect(sk, NETLINK_ROUTE)) < 0) {
        nl_perror(err, "Unable to connect socket");
        return err;
    }

    link = rtnl_link_bridge_alloc();

    rtnl_link_set_name(link, BRIDGE_NAME);
    rtnl_link_set_family(link, AF_BRIDGE);
    rtnl_link_set_flags(link, IFF_UP);

    unsigned int flag = 0;

    if ((err = rtnl_link_add(sk, link, NLM_F_CREATE)) < 0)
    {
        nl_perror(err, "Unable to add link");
        return err;
    }

    rtnl_link_put(link);
    link = NULL;

    if (rtnl_link_get_kernel(sk, 0, BRIDGE_NAME, &link) < 0)
    {
        printf("can't get link\n");
        return -1;
    }


    change = rtnl_link_alloc();
    rtnl_link_unset_flags(change, IFF_UP);

    if ((err = rtnl_link_change(sk, link, change, 0)) < 0)
        printf("can't change flags: %s(%d)\n", nl_geterror(err), err);

    rtnl_link_put(link);
    rtnl_link_put(change);
    return 0;
}

Как я компилирую:

gcc bridge-test.c -o createbr  $(pkg-config --cflags --libs libnl-3.0 libnl-genl-3.0 libnl-route-3.0)

И я получаю:

Operation not supported(-10)

Это не ошибка libnl, она возвращается из ядра.

Я на довольно старом Debian, но я не думаю, что это проблема:

$ uname -a
Linux debian 4.9.0-8-amd64 #1 SMP Debian 4.9.144-3 (2019-02-02) x86_64 GNU/Linux

версия libnl:

$ aptitude versions libnl-3-dev
i   3.4.0-1                                                                                            

Проблема возникает, когда япытаюсь удалить флаг IFF_UP.Я делаю это неправильно?

1 Ответ

0 голосов
/ 11 июня 2019

ПРАВИЛЬНЫЙ ОТВЕТ

Это ошибка в самой libnl-3.4.0. Обратите внимание на эту тему: http://lists.infradead.org/pipermail/libnl/2017-November/thread.html#2384

Уже исправлено https://github.com/thom311/libnl/commit/3c427d0f4fcca91a661d85d21cdea89d41271c6c

СТАРЫЙ ОТВЕТ, ЭТО НЕПРАВИЛЬНЫЙ ОТВЕТ

После дня поиска в Google (я разместил этот вопрос как акт отчаяния), я думаю, что нашел ответ.

Создать мост в Linux:

$ sudo ip link add brr type bridge

Теперь посмотрите, что iproute2 отправляет ядру, когда вы пытаетесь его записать:

sudo strace -s 100 -f -o out -x ip link set brr down
...
1644  sendmsg(3, {msg_name={sa_family=AF_NETLINK, nl_pid=0, nl_groups=00000000}, msg_namelen=12, msg_iov=[{iov_base={{len=32, type=RTM_NEWLINK, flags=NLM_F_REQUEST|NLM_F_ACK, seq=1560223693, pid=0}, {ifi_family=AF_UNSPEC, ifi_type=ARPHRD_NETROM, ifi_index=if_nametoindex("brr"), ifi_flags=0, ifi_change=0x1}}, iov_len=32}], msg_iovlen=1, msg_controllen=0, msg_flags=0}, 0) = 32
...

Теперь попробуйте отключить интерфейс, используя приведенный выше код. Strace даст нам:

...
1563  sendmsg(3, {msg_name={sa_family=AF_NETLINK, nl_pid=0, nl_groups=00000000}, msg_namelen=12, msg_iov=[{iov_base={{len=32, type=RTM_SETLINK, flags=NLM_F_REQUEST|NLM_F_ACK, seq=0, pid=0}, {ifi_family=AF_BRIDGE, ifi_type=ARPHRD_NETROM, ifi_index=if_nametoindex("brr"), ifi_flags=IFF_BROADCAST|IFF_RUNNING|IFF_MULTICAST|IFF_LOWER_UP, ifi_change=0x1}}, iov_len=32}], msg_iovlen=1, msg_controllen=0, msg_flags=0}, 0) = 32
...

Эти две строки почти одинаковы с одним исключением:

iproute2:
ifi_family=AF_UNSPEC
mycode:
ifi_family=AF_BRIDGE

и запрос обрабатывается двумя разными функциями в ядре:

rtnl_register(PF_UNSPEC, RTM_SETLINK, rtnl_setlink, NULL, 0);
...
rtnl_register(PF_BRIDGE, RTM_SETLINK, rtnl_bridge_setlink, NULL, 0);

Эти строки из net/core/rtnetlink.c

Итак, установка семейства AF_UNSPEC для исходной ссылки решает проблему. Это не совсем очевидно, хотя.

UPDATE: Вы не можете изменить специфичные для моста флаги таким образом.

...