1 /* SPDX-License-Identifier: LGPL-2.1-only */
2 /*
3  * Copyright (c) 2010 Thomas Graf <tgraf@suug.ch>
4  */
5 
6 #include "nl-default.h"
7 
8 #include <linux/ipv6.h>
9 #include <linux/snmp.h>
10 
11 #include <netlink/netlink.h>
12 #include <netlink/attr.h>
13 #include <netlink/route/rtnl.h>
14 #include <netlink/route/link/inet6.h>
15 
16 #include "nl-route.h"
17 #include "link-api.h"
18 #include "nl-priv-static-route/nl-priv-static-route.h"
19 
20 #define I6_ADDR_GEN_MODE_UNKNOWN	UINT8_MAX
21 
22 struct inet6_data
23 {
24 	uint32_t		i6_flags;
25 	struct ifla_cacheinfo	i6_cacheinfo;
26 	uint32_t		i6_conf[DEVCONF_MAX];
27 	struct in6_addr		i6_token;
28 	uint8_t			i6_conf_len;
29 	uint8_t			i6_addr_gen_mode;
30 };
31 
inet6_alloc(struct rtnl_link * link)32 static void *inet6_alloc(struct rtnl_link *link)
33 {
34 	struct inet6_data *i6;
35 
36 	i6 = calloc(1, sizeof(struct inet6_data));
37 	if (i6)
38 		i6->i6_addr_gen_mode = I6_ADDR_GEN_MODE_UNKNOWN;
39 
40 	return i6;
41 }
42 
inet6_clone(struct rtnl_link * link,void * data)43 static void *inet6_clone(struct rtnl_link *link, void *data)
44 {
45 	struct inet6_data *i6;
46 
47 	if ((i6 = inet6_alloc(link)))
48 		memcpy(i6, data, sizeof(*i6));
49 
50 	return i6;
51 }
52 
inet6_free(struct rtnl_link * link,void * data)53 static void inet6_free(struct rtnl_link *link, void *data)
54 {
55 	free(data);
56 }
57 
58 static struct nla_policy inet6_policy[IFLA_INET6_MAX+1] = {
59 	[IFLA_INET6_FLAGS]		= { .type = NLA_U32 },
60 	[IFLA_INET6_CACHEINFO]		= { .minlen = sizeof(struct ifla_cacheinfo) },
61 	[IFLA_INET6_CONF]		= { .minlen = 4 },
62 	[IFLA_INET6_STATS]		= { .minlen = 8 },
63 	[IFLA_INET6_ICMP6STATS]		= { .minlen = 8 },
64 	[IFLA_INET6_TOKEN]		= { .minlen = sizeof(struct in6_addr) },
65 	[IFLA_INET6_ADDR_GEN_MODE]	= { .type = NLA_U8 },
66 };
67 
68 static const uint8_t map_stat_id_from_IPSTATS_MIB_v1[__IPSTATS_MIB_MAX] = {
69 	/* 14a196807482e6fc74f15fc03176d5c08880588f^:include/linux/snmp.h
70 	 * version before the API change in commit 14a196807482e6fc74f15fc03176d5c08880588f.
71 	 * This version was valid since commit edf391ff17232f097d72441c9ad467bcb3b5db18, which
72 	 * predates support for parsing IFLA_PROTINFO in libnl3. Such an even older meaning of
73 	 * the flags is not supported in libnl3. */
74 	[ 1] = RTNL_LINK_IP6_INPKTS,                    /* IPSTATS_MIB_INPKTS                   */
75 	[ 2] = RTNL_LINK_IP6_INHDRERRORS,               /* IPSTATS_MIB_INHDRERRORS              */
76 	[ 3] = RTNL_LINK_IP6_INTOOBIGERRORS,            /* IPSTATS_MIB_INTOOBIGERRORS           */
77 	[ 4] = RTNL_LINK_IP6_INNOROUTES,                /* IPSTATS_MIB_INNOROUTES               */
78 	[ 5] = RTNL_LINK_IP6_INADDRERRORS,              /* IPSTATS_MIB_INADDRERRORS             */
79 	[ 6] = RTNL_LINK_IP6_INUNKNOWNPROTOS,           /* IPSTATS_MIB_INUNKNOWNPROTOS          */
80 	[ 7] = RTNL_LINK_IP6_INTRUNCATEDPKTS,           /* IPSTATS_MIB_INTRUNCATEDPKTS          */
81 	[ 8] = RTNL_LINK_IP6_INDISCARDS,                /* IPSTATS_MIB_INDISCARDS               */
82 	[ 9] = RTNL_LINK_IP6_INDELIVERS,                /* IPSTATS_MIB_INDELIVERS               */
83 	[10] = RTNL_LINK_IP6_OUTFORWDATAGRAMS,          /* IPSTATS_MIB_OUTFORWDATAGRAMS         */
84 	[11] = RTNL_LINK_IP6_OUTPKTS,                   /* IPSTATS_MIB_OUTPKTS                  */
85 	[12] = RTNL_LINK_IP6_OUTDISCARDS,               /* IPSTATS_MIB_OUTDISCARDS              */
86 	[13] = RTNL_LINK_IP6_OUTNOROUTES,               /* IPSTATS_MIB_OUTNOROUTES              */
87 	[14] = RTNL_LINK_IP6_REASMTIMEOUT,              /* IPSTATS_MIB_REASMTIMEOUT             */
88 	[15] = RTNL_LINK_IP6_REASMREQDS,                /* IPSTATS_MIB_REASMREQDS               */
89 	[16] = RTNL_LINK_IP6_REASMOKS,                  /* IPSTATS_MIB_REASMOKS                 */
90 	[17] = RTNL_LINK_IP6_REASMFAILS,                /* IPSTATS_MIB_REASMFAILS               */
91 	[18] = RTNL_LINK_IP6_FRAGOKS,                   /* IPSTATS_MIB_FRAGOKS                  */
92 	[19] = RTNL_LINK_IP6_FRAGFAILS,                 /* IPSTATS_MIB_FRAGFAILS                */
93 	[20] = RTNL_LINK_IP6_FRAGCREATES,               /* IPSTATS_MIB_FRAGCREATES              */
94 	[21] = RTNL_LINK_IP6_INMCASTPKTS,               /* IPSTATS_MIB_INMCASTPKTS              */
95 	[22] = RTNL_LINK_IP6_OUTMCASTPKTS,              /* IPSTATS_MIB_OUTMCASTPKTS             */
96 	[23] = RTNL_LINK_IP6_INBCASTPKTS,               /* IPSTATS_MIB_INBCASTPKTS              */
97 	[24] = RTNL_LINK_IP6_OUTBCASTPKTS,              /* IPSTATS_MIB_OUTBCASTPKTS             */
98 	[25] = RTNL_LINK_IP6_INOCTETS,                  /* IPSTATS_MIB_INOCTETS                 */
99 	[26] = RTNL_LINK_IP6_OUTOCTETS,                 /* IPSTATS_MIB_OUTOCTETS                */
100 	[27] = RTNL_LINK_IP6_INMCASTOCTETS,             /* IPSTATS_MIB_INMCASTOCTETS            */
101 	[28] = RTNL_LINK_IP6_OUTMCASTOCTETS,            /* IPSTATS_MIB_OUTMCASTOCTETS           */
102 	[29] = RTNL_LINK_IP6_INBCASTOCTETS,             /* IPSTATS_MIB_INBCASTOCTETS            */
103 	[30] = RTNL_LINK_IP6_OUTBCASTOCTETS,            /* IPSTATS_MIB_OUTBCASTOCTETS           */
104 };
105 
106 static const uint8_t map_stat_id_from_IPSTATS_MIB_v2[__IPSTATS_MIB_MAX] = {
107 	/* d8ec26d7f8287f5788a494f56e8814210f0e64be:include/uapi/linux/snmp.h
108 	 * version since the API change in commit 14a196807482e6fc74f15fc03176d5c08880588f */
109 	[ 1] = RTNL_LINK_IP6_INPKTS,                    /* IPSTATS_MIB_INPKTS                   */
110 	[ 2] = RTNL_LINK_IP6_INOCTETS,                  /* IPSTATS_MIB_INOCTETS                 */
111 	[ 3] = RTNL_LINK_IP6_INDELIVERS,                /* IPSTATS_MIB_INDELIVERS               */
112 	[ 4] = RTNL_LINK_IP6_OUTFORWDATAGRAMS,          /* IPSTATS_MIB_OUTFORWDATAGRAMS         */
113 	[ 5] = RTNL_LINK_IP6_OUTPKTS,                   /* IPSTATS_MIB_OUTPKTS                  */
114 	[ 6] = RTNL_LINK_IP6_OUTOCTETS,                 /* IPSTATS_MIB_OUTOCTETS                */
115 	[ 7] = RTNL_LINK_IP6_INHDRERRORS,               /* IPSTATS_MIB_INHDRERRORS              */
116 	[ 8] = RTNL_LINK_IP6_INTOOBIGERRORS,            /* IPSTATS_MIB_INTOOBIGERRORS           */
117 	[ 9] = RTNL_LINK_IP6_INNOROUTES,                /* IPSTATS_MIB_INNOROUTES               */
118 	[10] = RTNL_LINK_IP6_INADDRERRORS,              /* IPSTATS_MIB_INADDRERRORS             */
119 	[11] = RTNL_LINK_IP6_INUNKNOWNPROTOS,           /* IPSTATS_MIB_INUNKNOWNPROTOS          */
120 	[12] = RTNL_LINK_IP6_INTRUNCATEDPKTS,           /* IPSTATS_MIB_INTRUNCATEDPKTS          */
121 	[13] = RTNL_LINK_IP6_INDISCARDS,                /* IPSTATS_MIB_INDISCARDS               */
122 	[14] = RTNL_LINK_IP6_OUTDISCARDS,               /* IPSTATS_MIB_OUTDISCARDS              */
123 	[15] = RTNL_LINK_IP6_OUTNOROUTES,               /* IPSTATS_MIB_OUTNOROUTES              */
124 	[16] = RTNL_LINK_IP6_REASMTIMEOUT,              /* IPSTATS_MIB_REASMTIMEOUT             */
125 	[17] = RTNL_LINK_IP6_REASMREQDS,                /* IPSTATS_MIB_REASMREQDS               */
126 	[18] = RTNL_LINK_IP6_REASMOKS,                  /* IPSTATS_MIB_REASMOKS                 */
127 	[19] = RTNL_LINK_IP6_REASMFAILS,                /* IPSTATS_MIB_REASMFAILS               */
128 	[20] = RTNL_LINK_IP6_FRAGOKS,                   /* IPSTATS_MIB_FRAGOKS                  */
129 	[21] = RTNL_LINK_IP6_FRAGFAILS,                 /* IPSTATS_MIB_FRAGFAILS                */
130 	[22] = RTNL_LINK_IP6_FRAGCREATES,               /* IPSTATS_MIB_FRAGCREATES              */
131 	[23] = RTNL_LINK_IP6_INMCASTPKTS,               /* IPSTATS_MIB_INMCASTPKTS              */
132 	[24] = RTNL_LINK_IP6_OUTMCASTPKTS,              /* IPSTATS_MIB_OUTMCASTPKTS             */
133 	[25] = RTNL_LINK_IP6_INBCASTPKTS,               /* IPSTATS_MIB_INBCASTPKTS              */
134 	[26] = RTNL_LINK_IP6_OUTBCASTPKTS,              /* IPSTATS_MIB_OUTBCASTPKTS             */
135 	[27] = RTNL_LINK_IP6_INMCASTOCTETS,             /* IPSTATS_MIB_INMCASTOCTETS            */
136 	[28] = RTNL_LINK_IP6_OUTMCASTOCTETS,            /* IPSTATS_MIB_OUTMCASTOCTETS           */
137 	[29] = RTNL_LINK_IP6_INBCASTOCTETS,             /* IPSTATS_MIB_INBCASTOCTETS            */
138 	[30] = RTNL_LINK_IP6_OUTBCASTOCTETS,            /* IPSTATS_MIB_OUTBCASTOCTETS           */
139 	[31] = RTNL_LINK_IP6_CSUMERRORS,                /* IPSTATS_MIB_CSUMERRORS               */
140 	[32] = RTNL_LINK_IP6_NOECTPKTS,                 /* IPSTATS_MIB_NOECTPKTS                */
141 	[33] = RTNL_LINK_IP6_ECT1PKTS,                  /* IPSTATS_MIB_ECT1PKTS                 */
142 	[34] = RTNL_LINK_IP6_ECT0PKTS,                  /* IPSTATS_MIB_ECT0PKTS                 */
143 	[35] = RTNL_LINK_IP6_CEPKTS,                    /* IPSTATS_MIB_CEPKTS                   */
144 	[36] = RTNL_LINK_REASM_OVERLAPS,                /* IPSTATS_MIB_REASM_OVERLAPS           */
145 };
146 
147 const uint8_t *const _nltst_map_stat_id_from_IPSTATS_MIB_v2 = map_stat_id_from_IPSTATS_MIB_v2;
148 
inet6_parse_protinfo(struct rtnl_link * link,struct nlattr * attr,void * data)149 static int inet6_parse_protinfo(struct rtnl_link *link, struct nlattr *attr,
150 				void *data)
151 {
152 	struct inet6_data *i6 = data;
153 	struct nlattr *tb[IFLA_INET6_MAX+1];
154 	int err;
155 
156 	err = nla_parse_nested(tb, IFLA_INET6_MAX, attr, inet6_policy);
157 	if (err < 0)
158 		return err;
159 	if (tb[IFLA_INET6_CONF] && nla_len(tb[IFLA_INET6_CONF]) % 4)
160 		return -EINVAL;
161 	if (tb[IFLA_INET6_STATS] && nla_len(tb[IFLA_INET6_STATS]) % 8)
162 		return -EINVAL;
163 	if (tb[IFLA_INET6_ICMP6STATS] && nla_len(tb[IFLA_INET6_ICMP6STATS]) % 8)
164 		return -EINVAL;
165 
166 	if (tb[IFLA_INET6_FLAGS])
167 		i6->i6_flags = nla_get_u32(tb[IFLA_INET6_FLAGS]);
168 
169 	if (tb[IFLA_INET6_CACHEINFO])
170 		nla_memcpy(&i6->i6_cacheinfo, tb[IFLA_INET6_CACHEINFO],
171 			   sizeof(i6->i6_cacheinfo));
172 
173 	if (tb[IFLA_INET6_CONF]) {
174 		i6->i6_conf_len = _NL_MIN(ARRAY_SIZE(i6->i6_conf),
175 				      nla_len(tb[IFLA_INET6_CONF]) /
176 					      sizeof(i6->i6_conf[0]));
177 		nla_memcpy(&i6->i6_conf, tb[IFLA_INET6_CONF],
178 			   sizeof(i6->i6_conf[0]) * i6->i6_conf_len);
179 	}
180 
181 	if (tb[IFLA_INET6_TOKEN])
182 		nla_memcpy(&i6->i6_token, tb[IFLA_INET6_TOKEN],
183 		           sizeof(struct in6_addr));
184 
185 	if (tb[IFLA_INET6_ADDR_GEN_MODE])
186 		i6->i6_addr_gen_mode = nla_get_u8 (tb[IFLA_INET6_ADDR_GEN_MODE]);
187 
188 	/*
189 	 * Due to 32bit data alignment, these addresses must be copied to an
190 	 * aligned location prior to access.
191 	 */
192 	if (tb[IFLA_INET6_STATS]) {
193 		unsigned char *cnt = nla_data(tb[IFLA_INET6_STATS]);
194 		uint64_t stat;
195 		int i;
196 		int len = nla_len(tb[IFLA_INET6_STATS]) / 8;
197 		const uint8_t *map_stat_id = map_stat_id_from_IPSTATS_MIB_v2;
198 
199 		if (len < 32 ||
200 		    (tb[IFLA_INET6_ICMP6STATS] && nla_len(tb[IFLA_INET6_ICMP6STATS]) < 6)) {
201 			/* kernel commit 14a196807482e6fc74f15fc03176d5c08880588f reordered the values.
202 			 * The later commit 6a5dc9e598fe90160fee7de098fa319665f5253e added values
203 			 * IPSTATS_MIB_CSUMERRORS/ICMP6_MIB_CSUMERRORS. If the netlink is shorter
204 			 * then this, assume that the kernel uses the previous meaning of the
205 			 * enumeration. */
206 			map_stat_id = map_stat_id_from_IPSTATS_MIB_v1;
207 		}
208 
209 		len = _NL_MIN(__IPSTATS_MIB_MAX, len);
210 		for (i = 1; i < len; i++) {
211 			memcpy(&stat, &cnt[i * sizeof(stat)], sizeof(stat));
212 			rtnl_link_set_stat(link, map_stat_id[i], stat);
213 		}
214 	}
215 
216 	if (tb[IFLA_INET6_ICMP6STATS]) {
217 #define _NL_ICMP6_MIB_MAX 6
218 		unsigned char *cnt = nla_data(tb[IFLA_INET6_ICMP6STATS]);
219 		uint64_t stat;
220 		int i;
221 		int len = _NL_MIN(_NL_ICMP6_MIB_MAX, nla_len(tb[IFLA_INET6_ICMP6STATS]) / 8);
222 
223 		_NL_STATIC_ASSERT (__ICMP6_MIB_MAX >= _NL_ICMP6_MIB_MAX);
224 		_NL_STATIC_ASSERT (RTNL_LINK_ICMP6_CSUMERRORS - RTNL_LINK_ICMP6_INMSGS + 1 == 5);
225 
226 		for (i = 1; i < len; i++) {
227 			memcpy(&stat, &cnt[i * sizeof(stat)], sizeof(stat));
228 			rtnl_link_set_stat(link, RTNL_LINK_ICMP6_INMSGS + i - 1,
229 					   stat);
230 		}
231 	}
232 
233 	return 0;
234 }
235 
inet6_fill_af(struct rtnl_link * link,struct nl_msg * msg,void * data)236 static int inet6_fill_af(struct rtnl_link *link, struct nl_msg *msg, void *data)
237 {
238 	struct inet6_data *id = data;
239 
240 	if (id->i6_addr_gen_mode != I6_ADDR_GEN_MODE_UNKNOWN)
241 		NLA_PUT_U8(msg, IFLA_INET6_ADDR_GEN_MODE, id->i6_addr_gen_mode);
242 
243 	return 0;
244 
245 nla_put_failure:
246 	return -NLE_MSGSIZE;
247 }
248 
249 /* These live in include/net/if_inet6.h and should be moved to include/linux */
250 #define IF_RA_OTHERCONF	0x80
251 #define IF_RA_MANAGED	0x40
252 #define IF_RA_RCVD	0x20
253 #define IF_RS_SENT	0x10
254 #define IF_READY	0x80000000
255 
256 static const struct trans_tbl inet6_flags[] = {
257 	__ADD(IF_RA_OTHERCONF, ra_otherconf),
258 	__ADD(IF_RA_MANAGED, ra_managed),
259 	__ADD(IF_RA_RCVD, ra_rcvd),
260 	__ADD(IF_RS_SENT, rs_sent),
261 	__ADD(IF_READY, ready),
262 };
263 
rtnl_link_inet6_flags2str(int flags,char * buf,size_t len)264 char *rtnl_link_inet6_flags2str(int flags, char *buf, size_t len)
265 {
266 	return __flags2str(flags, buf, len, inet6_flags,
267 			   ARRAY_SIZE(inet6_flags));
268 }
269 
rtnl_link_inet6_str2flags(const char * name)270 int rtnl_link_inet6_str2flags(const char *name)
271 {
272 	return __str2flags(name, inet6_flags, ARRAY_SIZE(inet6_flags));
273 }
274 
275 static const struct trans_tbl inet6_devconf[] = {
276 	__ADD(DEVCONF_FORWARDING, forwarding),
277 	__ADD(DEVCONF_HOPLIMIT, hoplimit),
278 	__ADD(DEVCONF_MTU6, mtu6),
279 	__ADD(DEVCONF_ACCEPT_RA, accept_ra),
280 	__ADD(DEVCONF_ACCEPT_REDIRECTS, accept_redirects),
281 	__ADD(DEVCONF_AUTOCONF, autoconf),
282 	__ADD(DEVCONF_DAD_TRANSMITS, dad_transmits),
283 	__ADD(DEVCONF_RTR_SOLICITS, rtr_solicits),
284 	__ADD(DEVCONF_RTR_SOLICIT_INTERVAL, rtr_solicit_interval),
285 	__ADD(DEVCONF_RTR_SOLICIT_DELAY, rtr_solicit_delay),
286 	__ADD(DEVCONF_USE_TEMPADDR, use_tempaddr),
287 	__ADD(DEVCONF_TEMP_VALID_LFT, temp_valid_lft),
288 	__ADD(DEVCONF_TEMP_PREFERED_LFT, temp_prefered_lft),
289 	__ADD(DEVCONF_REGEN_MAX_RETRY, regen_max_retry),
290 	__ADD(DEVCONF_MAX_DESYNC_FACTOR, max_desync_factor),
291 	__ADD(DEVCONF_MAX_ADDRESSES, max_addresses),
292 	__ADD(DEVCONF_FORCE_MLD_VERSION, force_mld_version),
293 	__ADD(DEVCONF_ACCEPT_RA_DEFRTR, accept_ra_defrtr),
294 	__ADD(DEVCONF_ACCEPT_RA_PINFO, accept_ra_pinfo),
295 	__ADD(DEVCONF_ACCEPT_RA_RTR_PREF, accept_ra_rtr_pref),
296 	__ADD(DEVCONF_RTR_PROBE_INTERVAL, rtr_probe_interval),
297 	__ADD(DEVCONF_ACCEPT_RA_RT_INFO_MAX_PLEN, accept_ra_rt_info),
298 	__ADD(DEVCONF_PROXY_NDP, proxy_ndp),
299 	__ADD(DEVCONF_OPTIMISTIC_DAD, optimistic_dad),
300 	__ADD(DEVCONF_ACCEPT_SOURCE_ROUTE, accept_source_route),
301 	__ADD(DEVCONF_MC_FORWARDING, mc_forwarding),
302 	__ADD(DEVCONF_DISABLE_IPV6, disable_ipv6),
303 	__ADD(DEVCONF_ACCEPT_DAD, accept_dad),
304 	__ADD(DEVCONF_FORCE_TLLAO, force_tllao),
305 };
306 
inet6_devconf2str(int type,char * buf,size_t len)307 static char *inet6_devconf2str(int type, char *buf, size_t len)
308 {
309 	return __type2str(type, buf, len, inet6_devconf,
310 			  ARRAY_SIZE(inet6_devconf));
311 }
312 
313 static const struct trans_tbl inet6_addr_gen_mode[] = {
314 	__ADD(IN6_ADDR_GEN_MODE_EUI64, eui64),
315 	__ADD(IN6_ADDR_GEN_MODE_NONE, none),
316 	__ADD(IN6_ADDR_GEN_MODE_STABLE_PRIVACY, stable_privacy),
317 };
318 
rtnl_link_inet6_addrgenmode2str(uint8_t mode,char * buf,size_t len)319 const char *rtnl_link_inet6_addrgenmode2str(uint8_t mode, char *buf, size_t len)
320 {
321 	return __type2str(mode, buf, len, inet6_addr_gen_mode,
322 			  ARRAY_SIZE(inet6_addr_gen_mode));
323 }
324 
rtnl_link_inet6_str2addrgenmode(const char * mode)325 uint8_t rtnl_link_inet6_str2addrgenmode(const char *mode)
326 {
327 	return (uint8_t) __str2type(mode, inet6_addr_gen_mode,
328 			            ARRAY_SIZE(inet6_addr_gen_mode));
329 }
330 
inet6_dump_details(struct rtnl_link * link,struct nl_dump_params * p,void * data)331 static void inet6_dump_details(struct rtnl_link *link,
332 				struct nl_dump_params *p, void *data)
333 {
334 	struct inet6_data *i6 = data;
335 	struct nl_addr *addr;
336 	int i, n = 0;
337 	char buf[64];
338 
339 	nl_dump_line(p, "    ipv6 max-reasm-len %s",
340 	             nl_size2str(i6->i6_cacheinfo.max_reasm_len, buf, sizeof(buf)));
341 
342 	nl_dump(p, " <%s>\n",
343 	        rtnl_link_inet6_flags2str(i6->i6_flags, buf, sizeof(buf)));
344 
345 	nl_dump_line(p, "      create-stamp %.2fs reachable-time %s",
346 	             (double) i6->i6_cacheinfo.tstamp / 100.,
347 	             nl_msec2str(i6->i6_cacheinfo.reachable_time, buf, sizeof(buf)));
348 
349 	nl_dump(p, " retrans-time %s\n",
350 	        nl_msec2str(i6->i6_cacheinfo.retrans_time, buf, sizeof(buf)));
351 
352 	addr = nl_addr_build(AF_INET6, &i6->i6_token, sizeof(i6->i6_token));
353 	nl_dump(p, "      token %s\n",
354 	        nl_addr2str(addr, buf, sizeof(buf)));
355 	nl_addr_put(addr);
356 
357 	nl_dump(p, "      link-local address mode %s\n",
358 	        rtnl_link_inet6_addrgenmode2str(i6->i6_addr_gen_mode,
359 	                                        buf, sizeof(buf)));
360 
361 	nl_dump_line(p, "      devconf:\n");
362 	nl_dump_line(p, "      ");
363 
364 	for (i = 0; i < (int) i6->i6_conf_len; i++) {
365 		char buf2[64];
366 		uint32_t value = i6->i6_conf[i];
367 		int x, offset;
368 
369 		switch (i) {
370 		case DEVCONF_TEMP_VALID_LFT:
371 		case DEVCONF_TEMP_PREFERED_LFT:
372 			nl_msec2str((uint64_t) value * 1000., buf2, sizeof(buf2));
373 			break;
374 
375 		case DEVCONF_RTR_PROBE_INTERVAL:
376 		case DEVCONF_RTR_SOLICIT_INTERVAL:
377 		case DEVCONF_RTR_SOLICIT_DELAY:
378 			nl_msec2str(value, buf2, sizeof(buf2));
379 			break;
380 
381 		default:
382 			snprintf(buf2, sizeof(buf2), "%u", value);
383 			break;
384 		}
385 
386 		inet6_devconf2str(i, buf, sizeof(buf));
387 
388 		offset = 23 - strlen(buf2);
389 		if (offset < 0)
390 			offset = 0;
391 
392 		for (x = strlen(buf); x < offset; x++)
393 			buf[x] = ' ';
394 
395 		_nl_strncpy_trunc(&buf[offset], buf2, sizeof(buf) - offset);
396 
397 		nl_dump_line(p, "%s", buf);
398 
399 		if (++n == 3) {
400 			nl_dump(p, "\n");
401 			nl_dump_line(p, "      ");
402 			n = 0;
403 		} else
404 			nl_dump(p, "  ");
405 	}
406 
407 	if (n != 0)
408 		nl_dump(p, "\n");
409 }
410 
inet6_dump_stats(struct rtnl_link * link,struct nl_dump_params * p,void * data)411 static void inet6_dump_stats(struct rtnl_link *link,
412 			     struct nl_dump_params *p, void *data)
413 {
414 	double octets;
415 	char *octetsUnit;
416 
417 	nl_dump(p, "    IPv6:       InPkts           InOctets     "
418 		   "    InDiscards         InDelivers\n");
419 	nl_dump(p, "    %18" PRIu64 " ", link->l_stats[RTNL_LINK_IP6_INPKTS]);
420 
421 	octets = nl_cancel_down_bytes(link->l_stats[RTNL_LINK_IP6_INOCTETS],
422 				      &octetsUnit);
423 	if (octets)
424 		nl_dump(p, "%14.2f %3s ", octets, octetsUnit);
425 	else
426 		nl_dump(p, "%16u B ", 0);
427 
428 	nl_dump(p, "%18" PRIu64 " %18" PRIu64 "\n",
429 		link->l_stats[RTNL_LINK_IP6_INDISCARDS],
430 		link->l_stats[RTNL_LINK_IP6_INDELIVERS]);
431 
432 	nl_dump(p, "               OutPkts          OutOctets     "
433 		   "   OutDiscards        OutForwards\n");
434 
435 	nl_dump(p, "    %18" PRIu64 " ", link->l_stats[RTNL_LINK_IP6_OUTPKTS]);
436 
437 	octets = nl_cancel_down_bytes(link->l_stats[RTNL_LINK_IP6_OUTOCTETS],
438 				      &octetsUnit);
439 	if (octets)
440 		nl_dump(p, "%14.2f %3s ", octets, octetsUnit);
441 	else
442 		nl_dump(p, "%16u B ", 0);
443 
444 	nl_dump(p, "%18" PRIu64 " %18" PRIu64 "\n",
445 		link->l_stats[RTNL_LINK_IP6_OUTDISCARDS],
446 		link->l_stats[RTNL_LINK_IP6_OUTFORWDATAGRAMS]);
447 
448 	nl_dump(p, "           InMcastPkts      InMcastOctets     "
449 		   "   InBcastPkts     InBcastOctests\n");
450 
451 	nl_dump(p, "    %18" PRIu64 " ", link->l_stats[RTNL_LINK_IP6_INMCASTPKTS]);
452 
453 	octets = nl_cancel_down_bytes(link->l_stats[RTNL_LINK_IP6_INMCASTOCTETS],
454 				      &octetsUnit);
455 	if (octets)
456 		nl_dump(p, "%14.2f %3s ", octets, octetsUnit);
457 	else
458 		nl_dump(p, "%16u B ", 0);
459 
460 	nl_dump(p, "%18" PRIu64 " ", link->l_stats[RTNL_LINK_IP6_INBCASTPKTS]);
461 	octets = nl_cancel_down_bytes(link->l_stats[RTNL_LINK_IP6_INBCASTOCTETS],
462 				      &octetsUnit);
463 	if (octets)
464 		nl_dump(p, "%14.2f %3s\n", octets, octetsUnit);
465 	else
466 		nl_dump(p, "%16u B\n", 0);
467 
468 	nl_dump(p, "          OutMcastPkts     OutMcastOctets     "
469 		   "  OutBcastPkts    OutBcastOctests\n");
470 
471 	nl_dump(p, "    %18" PRIu64 " ", link->l_stats[RTNL_LINK_IP6_OUTMCASTPKTS]);
472 
473 	octets = nl_cancel_down_bytes(link->l_stats[RTNL_LINK_IP6_OUTMCASTOCTETS],
474 				      &octetsUnit);
475 	if (octets)
476 		nl_dump(p, "%14.2f %3s ", octets, octetsUnit);
477 	else
478 		nl_dump(p, "%16u B ", 0);
479 
480 	nl_dump(p, "%18" PRIu64 " ", link->l_stats[RTNL_LINK_IP6_OUTBCASTPKTS]);
481 	octets = nl_cancel_down_bytes(link->l_stats[RTNL_LINK_IP6_OUTBCASTOCTETS],
482 				      &octetsUnit);
483 	if (octets)
484 		nl_dump(p, "%14.2f %3s\n", octets, octetsUnit);
485 	else
486 		nl_dump(p, "%16u B\n", 0);
487 
488 	nl_dump(p, "              ReasmOKs         ReasmFails     "
489 		   "    ReasmReqds       ReasmTimeout\n");
490 	nl_dump(p, "    %18" PRIu64 " %18" PRIu64 " %18" PRIu64 " %18" PRIu64 "\n",
491 		link->l_stats[RTNL_LINK_IP6_REASMOKS],
492 		link->l_stats[RTNL_LINK_IP6_REASMFAILS],
493 		link->l_stats[RTNL_LINK_IP6_REASMREQDS],
494 		link->l_stats[RTNL_LINK_IP6_REASMTIMEOUT]);
495 
496 	nl_dump(p, "               FragOKs          FragFails    "
497 		   "    FragCreates\n");
498 	nl_dump(p, "    %18" PRIu64 " %18" PRIu64 " %18" PRIu64 "\n",
499 		link->l_stats[RTNL_LINK_IP6_FRAGOKS],
500 		link->l_stats[RTNL_LINK_IP6_FRAGFAILS],
501 		link->l_stats[RTNL_LINK_IP6_FRAGCREATES]);
502 
503 	nl_dump(p, "           InHdrErrors      InTooBigErrors   "
504 		   "     InNoRoutes       InAddrErrors\n");
505 	nl_dump(p, "    %18" PRIu64 " %18" PRIu64 " %18" PRIu64 " %18" PRIu64 "\n",
506 		link->l_stats[RTNL_LINK_IP6_INHDRERRORS],
507 		link->l_stats[RTNL_LINK_IP6_INTOOBIGERRORS],
508 		link->l_stats[RTNL_LINK_IP6_INNOROUTES],
509 		link->l_stats[RTNL_LINK_IP6_INADDRERRORS]);
510 
511 	nl_dump(p, "       InUnknownProtos     InTruncatedPkts   "
512 		   "    OutNoRoutes       InCsumErrors\n");
513 	nl_dump(p, "    %18" PRIu64 " %18" PRIu64 " %18" PRIu64 " %18" PRIu64 "\n",
514 		link->l_stats[RTNL_LINK_IP6_INUNKNOWNPROTOS],
515 		link->l_stats[RTNL_LINK_IP6_INTRUNCATEDPKTS],
516 		link->l_stats[RTNL_LINK_IP6_OUTNOROUTES],
517 		link->l_stats[RTNL_LINK_IP6_CSUMERRORS]);
518 
519 	nl_dump(p, "           InNoECTPkts          InECT1Pkts   "
520 		   "     InECT0Pkts           InCEPkts\n");
521 	nl_dump(p, "    %18" PRIu64 " %18" PRIu64 " %18" PRIu64 " %18" PRIu64 "\n",
522 		link->l_stats[RTNL_LINK_IP6_NOECTPKTS],
523 		link->l_stats[RTNL_LINK_IP6_ECT1PKTS],
524 		link->l_stats[RTNL_LINK_IP6_ECT0PKTS],
525 		link->l_stats[RTNL_LINK_IP6_CEPKTS]);
526 
527 	nl_dump(p, "    ICMPv6:     InMsgs           InErrors        "
528 		   "    OutMsgs          OutErrors       InCsumErrors\n");
529 	nl_dump(p, "    %18" PRIu64 " %18" PRIu64 " %18" PRIu64 " %18" PRIu64 " %18" PRIu64 "\n",
530 		link->l_stats[RTNL_LINK_ICMP6_INMSGS],
531 		link->l_stats[RTNL_LINK_ICMP6_INERRORS],
532 		link->l_stats[RTNL_LINK_ICMP6_OUTMSGS],
533 		link->l_stats[RTNL_LINK_ICMP6_OUTERRORS],
534 		link->l_stats[RTNL_LINK_ICMP6_CSUMERRORS]);
535 }
536 
537 static const struct nla_policy protinfo_policy = {
538 	.type			= NLA_NESTED,
539 };
540 
541 static struct rtnl_link_af_ops inet6_ops = {
542 	.ao_family			= AF_INET6,
543 	.ao_alloc			= &inet6_alloc,
544 	.ao_clone			= &inet6_clone,
545 	.ao_free			= &inet6_free,
546 	.ao_parse_protinfo		= &inet6_parse_protinfo,
547 	.ao_parse_af			= &inet6_parse_protinfo,
548 	.ao_fill_af			= &inet6_fill_af,
549 	.ao_dump[NL_DUMP_DETAILS]	= &inet6_dump_details,
550 	.ao_dump[NL_DUMP_STATS]		= &inet6_dump_stats,
551 	.ao_protinfo_policy		= &protinfo_policy,
552 };
553 
554 /**
555  * Return IPv6 specific flags
556  * @arg link		Link object
557  * @arg out_flags	Flags on success
558  *
559  * Returns the link's IPv6 flags.
560  *
561  * @return 0 on success
562  * @return -NLE_NOATTR configuration setting not available
563  */
rtnl_link_inet6_get_flags(struct rtnl_link * link,uint32_t * out_flags)564 int rtnl_link_inet6_get_flags(struct rtnl_link *link, uint32_t* out_flags)
565 {
566 	struct inet6_data *id = NULL;
567 
568 	if (!(id = rtnl_link_af_data(link, &inet6_ops)))
569 		return -NLE_NOATTR;
570 
571 	*out_flags = id->i6_flags;
572 	return 0;
573 }
574 
575 /**
576  * Set IPv6 specific flags
577  * @arg link		Link object
578  * @arg flags		Flags to set
579  *
580  * Sets the link's IPv6 specific flags. Overwrites currently set flags.
581  *
582  * @return 0 on success
583  * @return -NLE_NOMEM could not allocate inet6 data
584  */
rtnl_link_inet6_set_flags(struct rtnl_link * link,uint32_t flags)585 int rtnl_link_inet6_set_flags(struct rtnl_link *link, uint32_t flags)
586 {
587 	struct inet6_data *id;
588 
589 	if (!(id = rtnl_link_af_alloc(link, &inet6_ops)))
590 		return -NLE_NOMEM;
591 
592 	id->i6_flags = flags;
593 	return 0;
594 }
595 
596 /**
597  * Get IPv6 tokenized interface identifier
598  * @arg link		Link object
599  * @arg token		Tokenized interface identifier on success
600  *
601  * Returns the link's IPv6 tokenized interface identifier.
602  *
603  * @return 0 on success
604  * @return -NLE_NOMEM  failure to allocate struct nl_addr result
605  * @return -NLE_NOATTR configuration setting not available
606  * @return -NLE_NOADDR tokenized interface identifier is not set
607  */
rtnl_link_inet6_get_token(struct rtnl_link * link,struct nl_addr ** addr)608 int rtnl_link_inet6_get_token(struct rtnl_link *link, struct nl_addr **addr)
609 {
610 	struct inet6_data *id;
611 
612 	if (!(id = rtnl_link_af_data(link, &inet6_ops)))
613 		return -NLE_NOATTR;
614 
615 	*addr = nl_addr_build(AF_INET6, &id->i6_token, sizeof(id->i6_token));
616 	if (!*addr)
617 		return -NLE_NOMEM;
618 	if (nl_addr_iszero(*addr)) {
619 		nl_addr_put(*addr);
620 		*addr = NULL;
621 		return -NLE_NOADDR;
622 	}
623 
624 	return 0;
625 }
626 
627 /**
628  * Set IPv6 tokenized interface identifier
629  * @arg link		Link object
630  * @arg token		Tokenized interface identifier
631  *
632  * Sets the link's IPv6 tokenized interface identifier.
633  *
634  * @return 0 on success
635  * @return -NLE_NOMEM could not allocate inet6 data
636  * @return -NLE_INVAL addr is not a valid inet6 address
637  */
rtnl_link_inet6_set_token(struct rtnl_link * link,struct nl_addr * addr)638 int rtnl_link_inet6_set_token(struct rtnl_link *link, struct nl_addr *addr)
639 {
640 	struct inet6_data *id;
641 
642 	if ((nl_addr_get_family(addr) != AF_INET6) ||
643 	    (nl_addr_get_len(addr) != sizeof(id->i6_token)))
644 		return -NLE_INVAL;
645 
646 	if (!(id = rtnl_link_af_alloc(link, &inet6_ops)))
647 		return -NLE_NOMEM;
648 
649 	memcpy(&id->i6_token, nl_addr_get_binary_addr(addr),
650 	       sizeof(id->i6_token));
651 	return 0;
652 }
653 
654 /**
655  * Get IPv6 link-local address generation mode
656  * @arg link		Link object
657  * @arg mode		Generation mode on success
658  *
659  * Returns the link's IPv6 link-local address generation mode.
660  *
661  * @return 0 on success
662  * @return -NLE_NOATTR configuration setting not available
663  * @return -NLE_INVAL generation mode unknown. If the link was received via
664  *                    netlink, it means that address generation mode is not
665  *                    supported by the kernel.
666  */
rtnl_link_inet6_get_addr_gen_mode(struct rtnl_link * link,uint8_t * mode)667 int rtnl_link_inet6_get_addr_gen_mode(struct rtnl_link *link, uint8_t *mode)
668 {
669 	struct inet6_data *id;
670 
671 	if (!(id = rtnl_link_af_data(link, &inet6_ops)))
672 		return -NLE_NOATTR;
673 
674 	if (id->i6_addr_gen_mode == I6_ADDR_GEN_MODE_UNKNOWN)
675 		return -NLE_INVAL;
676 
677 	*mode = id->i6_addr_gen_mode;
678 	return 0;
679 }
680 
681 /**
682  * Set IPv6 link-local address generation mode
683  * @arg link		Link object
684  * @arg mode		Generation mode
685  *
686  * Sets the link's IPv6 link-local address generation mode.
687  *
688  * @return 0 on success
689  * @return -NLE_NOMEM could not allocate inet6 data
690  */
rtnl_link_inet6_set_addr_gen_mode(struct rtnl_link * link,uint8_t mode)691 int rtnl_link_inet6_set_addr_gen_mode(struct rtnl_link *link, uint8_t mode)
692 {
693 	struct inet6_data *id;
694 
695 	if (!(id = rtnl_link_af_alloc(link, &inet6_ops)))
696 		return -NLE_NOMEM;
697 
698 	id->i6_addr_gen_mode = mode;
699 	return 0;
700 }
701 
702 /**
703  * Get value of a ipv6 link configuration setting
704  * @arg link		Link object
705  * @arg cfgid		Configuration identifier
706  * @arg res		Result pointer
707  *
708  * Stores the value of the specified configuration setting in the provided
709  * result pointer.
710  *
711  * @return 0 on success or a negative error code.
712  * @return -NLE_RANGE cfgid is out of range or not provided by kernel.
713  * @return -NLE_NOATTR configuration setting not available
714  */
rtnl_link_inet6_get_conf(struct rtnl_link * link,unsigned int cfgid,uint32_t * res)715 int rtnl_link_inet6_get_conf(struct rtnl_link *link, unsigned int cfgid,
716 			     uint32_t *res)
717 {
718 	struct inet6_data *id;
719 
720 	if (!(id = rtnl_link_af_data(link, &inet6_ops)))
721 		return -NLE_NOATTR;
722 
723 	if (cfgid >= id->i6_conf_len)
724 		return -NLE_RANGE;
725 
726 	*res = id->i6_conf[cfgid];
727 
728 	return 0;
729 }
730 
731 
inet6_init(void)732 static void _nl_init inet6_init(void)
733 {
734 	rtnl_link_af_register(&inet6_ops);
735 }
736 
inet6_exit(void)737 static void _nl_exit inet6_exit(void)
738 {
739 	rtnl_link_af_unregister(&inet6_ops);
740 }
741