1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
|
// 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;
typedef struct prefix_ipv4 prefix_pim;
#define PIM_ADDRSTRLEN INET_ADDRSTRLEN
#define PIM_AF AF_INET
#define PIM_AFI AFI_IP
#define PIM_PROTO_REG IPPROTO_RAW
#define PIM_IANA_AFI IANA_AFI_IPV4
#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 PIM_AF_ROUTER "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 {
uniontype(pimprefixptr, struct prefix, p)
uniontype(pimprefixptr, struct prefix_ipv4, p4)
} TRANSPARENT_UNION;
union pimprefixconstptr {
uniontype(pimprefixconstptr, const struct prefix, p)
uniontype(pimprefixconstptr, const struct prefix_ipv4, p4)
} TRANSPARENT_UNION;
#else
typedef struct in6_addr pim_addr;
typedef struct prefix_ipv6 prefix_pim;
#define PIM_ADDRSTRLEN INET6_ADDRSTRLEN
#define PIM_AF AF_INET6
#define PIM_AFI AFI_IP6
#define PIM_PROTO_REG IPPROTO_PIM
#define PIM_IANA_AFI IANA_AFI_IPV6
#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 PIM_AF_ROUTER "pim6"
#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 {
uniontype(pimprefixptr, struct prefix, p)
uniontype(pimprefixptr, struct prefix_ipv6, p6)
} TRANSPARENT_UNION;
union pimprefixconstptr {
uniontype(pimprefixconstptr, const struct prefix, p)
uniontype(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 */
|