]> git.proxmox.com Git - mirror_frr.git/blobdiff - zebra/kernel_netlink.c
*: auto-convert to SPDX License IDs
[mirror_frr.git] / zebra / kernel_netlink.c
index 31d8294a0fb07d6fc92259fc2bc00ded2b9a5faa..bac75a641b107d81b54a68505fa0017a4ad2a420 100644 (file)
@@ -1,21 +1,6 @@
+// SPDX-License-Identifier: GPL-2.0-or-later
 /* Kernel communication using netlink interface.
  * Copyright (C) 1999 Kunihiro Ishiguro
- *
- * This file is part of GNU Zebra.
- *
- * GNU Zebra is free software; you can redistribute it and/or modify it
- * under the terms of the GNU General Public License as published by the
- * Free Software Foundation; either version 2, or (at your option) any
- * later version.
- *
- * GNU Zebra is distributed in the hope that it will be useful, but
- * WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
- * General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License along
- * with this program; see the file COPYING; if not, write to the Free Software
- * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
  */
 
 #include <zebra.h>
@@ -47,6 +32,7 @@
 #include "zebra/rt_netlink.h"
 #include "zebra/if_netlink.h"
 #include "zebra/rule_netlink.h"
+#include "zebra/tc_netlink.h"
 #include "zebra/netconf_netlink.h"
 #include "zebra/zebra_errors.h"
 
@@ -114,6 +100,15 @@ static const struct message nlmsg_str[] = {{RTM_NEWROUTE, "RTM_NEWROUTE"},
                                           {RTM_NEWTUNNEL, "RTM_NEWTUNNEL"},
                                           {RTM_DELTUNNEL, "RTM_DELTUNNEL"},
                                           {RTM_GETTUNNEL, "RTM_GETTUNNEL"},
+                                          {RTM_NEWQDISC, "RTM_NEWQDISC"},
+                                          {RTM_DELQDISC, "RTM_DELQDISC"},
+                                          {RTM_GETQDISC, "RTM_GETQDISC"},
+                                          {RTM_NEWTCLASS, "RTM_NEWTCLASS"},
+                                          {RTM_DELTCLASS, "RTM_DELTCLASS"},
+                                          {RTM_GETTCLASS, "RTM_GETTCLASS"},
+                                          {RTM_NEWTFILTER, "RTM_NEWTFILTER"},
+                                          {RTM_DELTFILTER, "RTM_DELTFILTER"},
+                                          {RTM_GETTFILTER, "RTM_GETTFILTER"},
                                           {0}};
 
 static const struct message rtproto_str[] = {
@@ -192,13 +187,13 @@ struct nl_batch {
 
        const struct zebra_dplane_info *zns;
 
-       struct dplane_ctx_q ctx_list;
+       struct dplane_ctx_list_head ctx_list;
 
        /*
         * Pointer to the queue of completed contexts outbound back
         * towards the dataplane module.
         */
-       struct dplane_ctx_q *ctx_out_q;
+       struct dplane_ctx_list_head *ctx_out_q;
 };
 
 int netlink_config_write_helper(struct vty *vty)
@@ -413,6 +408,15 @@ static int netlink_information_fetch(struct nlmsghdr *h, ns_id_t ns_id,
                return netlink_nexthop_change(h, ns_id, startup);
        case RTM_DELNEXTHOP:
                return netlink_nexthop_change(h, ns_id, startup);
+       case RTM_NEWQDISC:
+       case RTM_DELQDISC:
+               return netlink_qdisc_change(h, ns_id, startup);
+       case RTM_NEWTCLASS:
+       case RTM_DELTCLASS:
+               return netlink_tclass_change(h, ns_id, startup);
+       case RTM_NEWTFILTER:
+       case RTM_DELTFILTER:
+               return netlink_tfilter_change(h, ns_id, startup);
 
        /* Messages handled in the dplane thread */
        case RTM_NEWADDR:
@@ -1023,12 +1027,18 @@ static int netlink_parse_error(const struct nlsock *nl, struct nlmsghdr *h,
                return 1;
        }
 
-       /* Deal with errors that occur because of races in link handling. */
-       if (is_cmd
-           && ((msg_type == RTM_DELROUTE
-                && (-errnum == ENODEV || -errnum == ESRCH))
-               || (msg_type == RTM_NEWROUTE
-                   && (-errnum == ENETDOWN || -errnum == EEXIST)))) {
+       /*
+        * Deal with errors that occur because of races in link handling
+        * or types are not supported in kernel.
+        */
+       if (is_cmd &&
+           ((msg_type == RTM_DELROUTE &&
+             (-errnum == ENODEV || -errnum == ESRCH)) ||
+            (msg_type == RTM_NEWROUTE &&
+             (-errnum == ENETDOWN || -errnum == EEXIST)) ||
+            ((msg_type == RTM_NEWTUNNEL || msg_type == RTM_DELTUNNEL ||
+              msg_type == RTM_GETTUNNEL) &&
+             (-errnum == EOPNOTSUPP)))) {
                if (IS_ZEBRA_DEBUG_KERNEL)
                        zlog_debug("%s: error: %s type=%s(%u), seq=%u, pid=%u",
                                   nl->name, safe_strerror(-errnum),
@@ -1421,10 +1431,11 @@ static void nl_batch_reset(struct nl_batch *bth)
        bth->msgcnt = 0;
        bth->zns = NULL;
 
-       TAILQ_INIT(&(bth->ctx_list));
+       dplane_ctx_q_init(&(bth->ctx_list));
 }
 
-static void nl_batch_init(struct nl_batch *bth, struct dplane_ctx_q *ctx_out_q)
+static void nl_batch_init(struct nl_batch *bth,
+                         struct dplane_ctx_list_head *ctx_out_q)
 {
        /*
         * If the size of the buffer has changed, free and then allocate a new
@@ -1613,27 +1624,41 @@ static enum netlink_msg_status nl_put_msg(struct nl_batch *bth,
 
        case DPLANE_OP_INTF_ADDR_ADD:
        case DPLANE_OP_INTF_ADDR_DEL:
-       case DPLANE_OP_INTF_NETCONFIG:
        case DPLANE_OP_NONE:
                return FRR_NETLINK_ERROR;
 
+       case DPLANE_OP_INTF_NETCONFIG:
+               return netlink_put_intf_netconfig(bth, ctx);
+
        case DPLANE_OP_INTF_INSTALL:
        case DPLANE_OP_INTF_UPDATE:
        case DPLANE_OP_INTF_DELETE:
                return netlink_put_intf_update_msg(bth, ctx);
+
+       case DPLANE_OP_TC_QDISC_INSTALL:
+       case DPLANE_OP_TC_QDISC_UNINSTALL:
+               return netlink_put_tc_qdisc_update_msg(bth, ctx);
+       case DPLANE_OP_TC_CLASS_ADD:
+       case DPLANE_OP_TC_CLASS_DELETE:
+       case DPLANE_OP_TC_CLASS_UPDATE:
+               return netlink_put_tc_class_update_msg(bth, ctx);
+       case DPLANE_OP_TC_FILTER_ADD:
+       case DPLANE_OP_TC_FILTER_DELETE:
+       case DPLANE_OP_TC_FILTER_UPDATE:
+               return netlink_put_tc_filter_update_msg(bth, ctx);
        }
 
        return FRR_NETLINK_ERROR;
 }
 
-void kernel_update_multi(struct dplane_ctx_q *ctx_list)
+void kernel_update_multi(struct dplane_ctx_list_head *ctx_list)
 {
        struct nl_batch batch;
        struct zebra_dplane_ctx *ctx;
-       struct dplane_ctx_q handled_list;
+       struct dplane_ctx_list_head handled_list;
        enum netlink_msg_status res;
 
-       TAILQ_INIT(&handled_list);
+       dplane_ctx_q_init(&handled_list);
        nl_batch_init(&batch, &handled_list);
 
        while (true) {
@@ -1664,7 +1689,7 @@ void kernel_update_multi(struct dplane_ctx_q *ctx_list)
 
        nl_batch_send(&batch);
 
-       TAILQ_INIT(ctx_list);
+       dplane_ctx_q_init(ctx_list);
        dplane_ctx_list_append(ctx_list, &handled_list);
 }
 
@@ -1734,15 +1759,16 @@ void kernel_init(struct zebra_ns *zns)
         * RTNLGRP_XXX to a bit position for ourself
         */
        groups = RTMGRP_LINK                   |
-               RTMGRP_IPV4_ROUTE              |
-               RTMGRP_IPV4_IFADDR             |
-               RTMGRP_IPV6_ROUTE              |
-               RTMGRP_IPV6_IFADDR             |
-               RTMGRP_IPV4_MROUTE             |
-               RTMGRP_NEIGH                   |
-               ((uint32_t) 1 << (RTNLGRP_IPV4_RULE - 1)) |
-               ((uint32_t) 1 << (RTNLGRP_IPV6_RULE - 1)) |
-               ((uint32_t) 1 << (RTNLGRP_NEXTHOP - 1));
+                       RTMGRP_IPV4_ROUTE              |
+                       RTMGRP_IPV4_IFADDR             |
+                       RTMGRP_IPV6_ROUTE              |
+                       RTMGRP_IPV6_IFADDR             |
+                       RTMGRP_IPV4_MROUTE             |
+                       RTMGRP_NEIGH                   |
+                       ((uint32_t) 1 << (RTNLGRP_IPV4_RULE - 1)) |
+                       ((uint32_t) 1 << (RTNLGRP_IPV6_RULE - 1)) |
+                       ((uint32_t) 1 << (RTNLGRP_NEXTHOP - 1))   |
+                       ((uint32_t) 1 << (RTNLGRP_TC - 1));
 
        dplane_groups = (RTMGRP_LINK            |
                         RTMGRP_IPV4_IFADDR     |
@@ -1902,7 +1928,7 @@ static void kernel_nlsock_fini(struct nlsock *nls)
 
 void kernel_terminate(struct zebra_ns *zns, bool complete)
 {
-       thread_cancel(&zns->t_netlink);
+       THREAD_OFF(zns->t_netlink);
 
        kernel_nlsock_fini(&zns->netlink);