summaryrefslogtreecommitdiffstats
path: root/pimd/pim_addr.h
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-09 13:16:35 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-09 13:16:35 +0000
commite2bbf175a2184bd76f6c54ccf8456babeb1a46fc (patch)
treef0b76550d6e6f500ada964a3a4ee933a45e5a6f1 /pimd/pim_addr.h
parentInitial commit. (diff)
downloadfrr-e2bbf175a2184bd76f6c54ccf8456babeb1a46fc.tar.xz
frr-e2bbf175a2184bd76f6c54ccf8456babeb1a46fc.zip
Adding upstream version 9.1.upstream/9.1
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'pimd/pim_addr.h')
-rw-r--r--pimd/pim_addr.h158
1 files changed, 158 insertions, 0 deletions
diff --git a/pimd/pim_addr.h b/pimd/pim_addr.h
new file mode 100644
index 0000000..94c63bb
--- /dev/null
+++ b/pimd/pim_addr.h
@@ -0,0 +1,158 @@
+// SPDX-License-Identifier: GPL-2.0-or-later
+/*
+ * PIM address generalizations
+ * Copyright (C) 2022 David Lamparter for NetDEF, Inc.
+ */
+
+#ifndef _PIMD_PIM_ADDR_H
+#define _PIMD_PIM_ADDR_H
+
+#include "jhash.h"
+#include "prefix.h"
+
+/* clang-format off */
+
+#if PIM_IPV == 4
+typedef struct in_addr pim_addr;
+
+#define PIM_ADDRSTRLEN INET_ADDRSTRLEN
+#define PIM_AF AF_INET
+#define PIM_AFI AFI_IP
+#define PIM_PROTO_REG IPPROTO_RAW
+#define PIM_IPADDR IPADDR_V4
+#define ipaddr_pim ipaddr_v4
+#define PIM_MAX_BITLEN IPV4_MAX_BITLEN
+#define PIM_AF_NAME "ip"
+#define PIM_AF_DBG "pim"
+#define GM_AF_DBG "igmp"
+#define PIM_MROUTE_DBG "mroute"
+#define PIMREG "pimreg"
+#define GM "IGMP"
+#define IPPROTO_GM IPPROTO_IGMP
+
+#define PIM_ADDR_FUNCNAME(name) ipv4_##name
+
+union pimprefixptr {
+ prefixtype(pimprefixptr, struct prefix, p)
+ prefixtype(pimprefixptr, struct prefix_ipv4, p4)
+} TRANSPARENT_UNION;
+
+union pimprefixconstptr {
+ prefixtype(pimprefixconstptr, const struct prefix, p)
+ prefixtype(pimprefixconstptr, const struct prefix_ipv4, p4)
+} TRANSPARENT_UNION;
+
+#else
+typedef struct in6_addr pim_addr;
+
+#define PIM_ADDRSTRLEN INET6_ADDRSTRLEN
+#define PIM_AF AF_INET6
+#define PIM_AFI AFI_IP6
+#define PIM_PROTO_REG IPPROTO_PIM
+#define PIM_IPADDR IPADDR_V6
+#define ipaddr_pim ipaddr_v6
+#define PIM_MAX_BITLEN IPV6_MAX_BITLEN
+#define PIM_AF_NAME "ipv6"
+#define PIM_AF_DBG "pimv6"
+#define GM_AF_DBG "mld"
+#define PIM_MROUTE_DBG "mroute6"
+#define PIMREG "pim6reg"
+#define GM "MLD"
+#define IPPROTO_GM IPPROTO_ICMPV6
+
+#define PIM_ADDR_FUNCNAME(name) ipv6_##name
+
+union pimprefixptr {
+ prefixtype(pimprefixptr, struct prefix, p)
+ prefixtype(pimprefixptr, struct prefix_ipv6, p6)
+} TRANSPARENT_UNION;
+
+union pimprefixconstptr {
+ prefixtype(pimprefixconstptr, const struct prefix, p)
+ prefixtype(pimprefixconstptr, const struct prefix_ipv6, p6)
+} TRANSPARENT_UNION;
+#endif
+
+/* for assignment/initialization (C99 compound literal)
+ * named PIMADDR_ANY (not PIM_ADDR_ANY) to match INADDR_ANY
+ */
+#define PIMADDR_ANY (pim_addr){ }
+
+/* clang-format on */
+
+static inline bool pim_addr_is_any(pim_addr addr)
+{
+ pim_addr zero = {};
+
+ return memcmp(&addr, &zero, sizeof(zero)) == 0;
+}
+
+static inline int pim_addr_cmp(pim_addr a, pim_addr b)
+{
+ return memcmp(&a, &b, sizeof(a));
+}
+
+static inline void pim_addr_to_prefix(union pimprefixptr out, pim_addr in)
+{
+ out.p->family = PIM_AF;
+ out.p->prefixlen = PIM_MAX_BITLEN;
+ memcpy(out.p->u.val, &in, sizeof(in));
+}
+
+static inline pim_addr pim_addr_from_prefix(union pimprefixconstptr in)
+{
+ pim_addr ret;
+
+ if (in.p->family != PIM_AF)
+ return PIMADDR_ANY;
+
+ memcpy(&ret, in.p->u.val, sizeof(ret));
+ return ret;
+}
+
+static inline uint8_t pim_addr_scope(const pim_addr addr)
+{
+ return PIM_ADDR_FUNCNAME(mcast_scope)(&addr);
+}
+
+static inline bool pim_addr_nofwd(const pim_addr addr)
+{
+ return PIM_ADDR_FUNCNAME(mcast_nofwd)(&addr);
+}
+
+static inline bool pim_addr_ssm(const pim_addr addr)
+{
+ return PIM_ADDR_FUNCNAME(mcast_ssm)(&addr);
+}
+
+/* don't use this struct directly, use the pim_sgaddr typedef */
+struct _pim_sgaddr {
+ pim_addr grp;
+ pim_addr src;
+};
+
+typedef struct _pim_sgaddr pim_sgaddr;
+
+static inline int pim_sgaddr_cmp(const pim_sgaddr a, const pim_sgaddr b)
+{
+ /* memcmp over the entire struct = memcmp(grp) + memcmp(src) */
+ return memcmp(&a, &b, sizeof(a));
+}
+
+static inline uint32_t pim_sgaddr_hash(const pim_sgaddr a, uint32_t initval)
+{
+ return jhash2((uint32_t *)&a, sizeof(a) / sizeof(uint32_t), initval);
+}
+
+#ifdef _FRR_ATTRIBUTE_PRINTFRR
+#pragma FRR printfrr_ext "%pPA" (pim_addr *)
+#pragma FRR printfrr_ext "%pSG" (pim_sgaddr *)
+#endif
+
+/*
+ * There is no pim_sgaddr2str(). This is intentional. Instead, use:
+ * snprintfrr(buf, sizeof(buf), "%pPA", sgaddr)
+ * (and note that snprintfrr is implicit for vty_out and zlog_*)
+ */
+
+#endif /* _PIMD_PIM_ADDR_H */