]> git.proxmox.com Git - mirror_frr.git/blame_incremental - bgpd/bgp_rd.c
Merge pull request #13649 from donaldsharp/unlock_the_node_or_else
[mirror_frr.git] / bgpd / bgp_rd.c
... / ...
CommitLineData
1// SPDX-License-Identifier: GPL-2.0-or-later
2/* BGP RD definitions for BGP-based VPNs (IP/EVPN)
3 * -- brought over from bgpd/bgp_mplsvpn.c
4 * Copyright (C) 2000 Kunihiro Ishiguro <kunihiro@zebra.org>
5 */
6
7#include <zebra.h>
8#include "command.h"
9#include "log.h"
10#include "prefix.h"
11#include "memory.h"
12#include "stream.h"
13#include "filter.h"
14#include "frrstr.h"
15
16#include "lib/printfrr.h"
17
18#include "bgpd/bgpd.h"
19#include "bgpd/bgp_rd.h"
20#include "bgpd/bgp_attr.h"
21
22#ifdef ENABLE_BGP_VNC
23#include "bgpd/rfapi/rfapi_backend.h"
24#endif
25
26uint16_t decode_rd_type(const uint8_t *pnt)
27{
28 uint16_t v;
29
30 v = ((uint16_t)*pnt++ << 8);
31#ifdef ENABLE_BGP_VNC
32 /*
33 * VNC L2 stores LHI in lower byte, so omit it
34 */
35 if (v != RD_TYPE_VNC_ETH)
36 v |= (uint16_t)*pnt;
37#else /* duplicate code for clarity */
38 v |= (uint16_t)*pnt;
39#endif
40 return v;
41}
42
43void encode_rd_type(uint16_t v, uint8_t *pnt)
44{
45 *((uint16_t *)pnt) = htons(v);
46}
47
48/* type == RD_TYPE_AS */
49void decode_rd_as(const uint8_t *pnt, struct rd_as *rd_as)
50{
51 rd_as->as = (uint16_t)*pnt++ << 8;
52 rd_as->as |= (uint16_t)*pnt++;
53 ptr_get_be32(pnt, &rd_as->val);
54}
55
56/* type == RD_TYPE_AS4 */
57void decode_rd_as4(const uint8_t *pnt, struct rd_as *rd_as)
58{
59 pnt = ptr_get_be32(pnt, &rd_as->as);
60 rd_as->val = ((uint16_t)*pnt++ << 8);
61 rd_as->val |= (uint16_t)*pnt;
62}
63
64/* type == RD_TYPE_IP */
65void decode_rd_ip(const uint8_t *pnt, struct rd_ip *rd_ip)
66{
67 memcpy(&rd_ip->ip, pnt, 4);
68 pnt += 4;
69
70 rd_ip->val = ((uint16_t)*pnt++ << 8);
71 rd_ip->val |= (uint16_t)*pnt;
72}
73
74#ifdef ENABLE_BGP_VNC
75/* type == RD_TYPE_VNC_ETH */
76void decode_rd_vnc_eth(const uint8_t *pnt, struct rd_vnc_eth *rd_vnc_eth)
77{
78 rd_vnc_eth->type = RD_TYPE_VNC_ETH;
79 rd_vnc_eth->local_nve_id = pnt[1];
80 memcpy(rd_vnc_eth->macaddr.octet, pnt + 2, ETH_ALEN);
81}
82#endif
83
84int str2prefix_rd(const char *str, struct prefix_rd *prd)
85{
86 int ret = 0, type = RD_TYPE_UNDEFINED;
87 char *p, *p2;
88 struct stream *s = NULL;
89 char *half = NULL;
90 struct in_addr addr;
91 as_t as_val;
92
93 prd->family = AF_UNSPEC;
94 prd->prefixlen = 64;
95
96 p = strchr(str, ':');
97 if (!p)
98 goto out;
99
100 /* a second ':' is accepted */
101 p2 = strchr(p + 1, ':');
102 if (p2) {
103 /* type is in first part */
104 half = XMALLOC(MTYPE_TMP, (p - str) + 1);
105 memcpy(half, str, (p - str));
106 half[p - str] = '\0';
107 type = atoi(half);
108 if (type != RD_TYPE_AS && type != RD_TYPE_IP &&
109 type != RD_TYPE_AS4)
110 goto out;
111 XFREE(MTYPE_TMP, half);
112 half = XMALLOC(MTYPE_TMP, (p2 - p));
113 memcpy(half, p + 1, (p2 - p - 1));
114 half[p2 - p - 1] = '\0';
115 p = p2 + 1;
116 } else {
117 half = XMALLOC(MTYPE_TMP, (p - str) + 1);
118 memcpy(half, str, (p - str));
119 half[p - str] = '\0';
120 }
121 if (!all_digit(p + 1))
122 goto out;
123
124 s = stream_new(RD_BYTES);
125
126 /* if it is an AS format or an IP */
127 if (asn_str2asn(half, &as_val)) {
128 if (as_val > UINT16_MAX) {
129 stream_putw(s, RD_TYPE_AS4);
130 stream_putl(s, as_val);
131 stream_putw(s, atol(p + 1));
132 if (type != RD_TYPE_UNDEFINED && type != RD_TYPE_AS4)
133 goto out;
134 } else {
135 stream_putw(s, RD_TYPE_AS);
136 stream_putw(s, as_val);
137 stream_putl(s, atol(p + 1));
138 if (type != RD_TYPE_UNDEFINED && type != RD_TYPE_AS)
139 goto out;
140 }
141 } else if (inet_aton(half, &addr)) {
142 stream_putw(s, RD_TYPE_IP);
143 stream_put_in_addr(s, &addr);
144 stream_putw(s, atol(p + 1));
145 if (type != RD_TYPE_UNDEFINED && type != RD_TYPE_IP)
146 goto out;
147 } else
148 goto out;
149 memcpy(prd->val, s->data, 8);
150 ret = 1;
151
152out:
153 if (s)
154 stream_free(s);
155 XFREE(MTYPE_TMP, half);
156 return ret;
157}
158
159char *prefix_rd2str(const struct prefix_rd *prd, char *buf, size_t size,
160 enum asnotation_mode asnotation)
161{
162 const uint8_t *pnt;
163 uint16_t type;
164 struct rd_as rd_as;
165 struct rd_ip rd_ip;
166 int len = 0;
167
168 assert(size >= RD_ADDRSTRLEN);
169
170 pnt = prd->val;
171
172 type = decode_rd_type(pnt);
173
174 if (type == RD_TYPE_AS) {
175 decode_rd_as(pnt + 2, &rd_as);
176 len += snprintfrr(buf + len, size - len, ASN_FORMAT(asnotation),
177 &rd_as.as);
178 snprintfrr(buf + len, size - len, ":%u", rd_as.val);
179 return buf;
180 } else if (type == RD_TYPE_AS4) {
181 decode_rd_as4(pnt + 2, &rd_as);
182 len += snprintfrr(buf + len, size - len, ASN_FORMAT(asnotation),
183 &rd_as.as);
184 snprintfrr(buf + len, size - len, ":%u", rd_as.val);
185 return buf;
186 } else if (type == RD_TYPE_IP) {
187 decode_rd_ip(pnt + 2, &rd_ip);
188 snprintfrr(buf, size, "%pI4:%hu", &rd_ip.ip, rd_ip.val);
189 return buf;
190 }
191#ifdef ENABLE_BGP_VNC
192 else if (type == RD_TYPE_VNC_ETH) {
193 snprintf(buf, size, "LHI:%d, %02x:%02x:%02x:%02x:%02x:%02x",
194 *(pnt + 1), /* LHI */
195 *(pnt + 2), /* MAC[0] */
196 *(pnt + 3), *(pnt + 4), *(pnt + 5), *(pnt + 6),
197 *(pnt + 7));
198
199 return buf;
200 }
201#endif
202
203 snprintf(buf, size, "Unknown Type: %d", type);
204 return buf;
205}
206
207void form_auto_rd(struct in_addr router_id,
208 uint16_t rd_id,
209 struct prefix_rd *prd)
210{
211 char buf[100];
212
213 prd->family = AF_UNSPEC;
214 prd->prefixlen = 64;
215 snprintfrr(buf, sizeof(buf), "%pI4:%hu", &router_id, rd_id);
216 (void)str2prefix_rd(buf, prd);
217}
218
219static ssize_t printfrr_prd_asnotation(struct fbuf *buf,
220 struct printfrr_eargs *ea,
221 const void *ptr,
222 enum asnotation_mode asnotation)
223{
224 char rd_buf[RD_ADDRSTRLEN];
225
226 if (!ptr)
227 return bputs(buf, "(null)");
228
229 prefix_rd2str(ptr, rd_buf, sizeof(rd_buf), asnotation);
230
231 return bputs(buf, rd_buf);
232}
233
234printfrr_ext_autoreg_p("RDP", printfrr_prd);
235static ssize_t printfrr_prd(struct fbuf *buf, struct printfrr_eargs *ea,
236 const void *ptr)
237{
238 return printfrr_prd_asnotation(buf, ea, ptr, ASNOTATION_PLAIN);
239}
240
241printfrr_ext_autoreg_p("RDD", printfrr_prd_dot);
242static ssize_t printfrr_prd_dot(struct fbuf *buf, struct printfrr_eargs *ea,
243 const void *ptr)
244{
245 return printfrr_prd_asnotation(buf, ea, ptr, ASNOTATION_DOT);
246}
247
248printfrr_ext_autoreg_p("RDE", printfrr_prd_dotplus);
249static ssize_t printfrr_prd_dotplus(struct fbuf *buf, struct printfrr_eargs *ea,
250 const void *ptr)
251{
252 return printfrr_prd_asnotation(buf, ea, ptr, ASNOTATION_DOTPLUS);
253}