1 /** @file
2 @brief IPv4 related functions
3
4 This is not to be included by the application.
5 */
6
7 /*
8 * Copyright (c) 2016 Intel Corporation
9 *
10 * SPDX-License-Identifier: Apache-2.0
11 */
12
13 #ifndef __IPV4_H
14 #define __IPV4_H
15
16 #include <zephyr/types.h>
17
18 #include <zephyr/net/net_ip.h>
19 #include <zephyr/net/net_pkt.h>
20 #include <zephyr/net/net_if.h>
21 #include <zephyr/net/net_context.h>
22
23 #define NET_IPV4_IHL_MASK 0x0F
24 #define NET_IPV4_DSCP_MASK 0xFC
25 #define NET_IPV4_DSCP_OFFSET 2
26 #define NET_IPV4_ECN_MASK 0x03
27
28 /* IPv4 Options */
29 #define NET_IPV4_OPTS_EO 0 /* End of Options */
30 #define NET_IPV4_OPTS_NOP 1 /* No operation */
31 #define NET_IPV4_OPTS_RR 7 /* Record Route */
32 #define NET_IPV4_OPTS_TS 68 /* Timestamp */
33 #define NET_IPV4_OPTS_RA 148 /* Router Alert */
34
35 /* IPv4 Options Timestamp flags */
36 #define NET_IPV4_TS_OPT_TS_ONLY 0 /* Timestamp only */
37 #define NET_IPV4_TS_OPT_TS_ADDR 1 /* Timestamp and address */
38 #define NET_IPV4_TS_OPT_TS_PRES 3 /* Timestamp prespecified hops*/
39
40 #define NET_IPV4_HDR_OPTNS_MAX_LEN 40
41
42 /* Fragment bits */
43 #define NET_IPV4_MF BIT(0) /* More fragments */
44 #define NET_IPV4_DF BIT(1) /* Do not fragment */
45
46 #define NET_IPV4_IGMP_QUERY 0x11 /* Membership query */
47 #define NET_IPV4_IGMP_REPORT_V1 0x12 /* v1 Membership report */
48 #define NET_IPV4_IGMP_REPORT_V2 0x16 /* v2 Membership report */
49 #define NET_IPV4_IGMP_LEAVE 0x17 /* v2 Leave group */
50 #define NET_IPV4_IGMP_REPORT_V3 0x22 /* v3 Membership report */
51
52 struct net_ipv4_igmp_v2_query {
53 /* IGMP message type */
54 uint8_t type;
55 /* Max response code */
56 uint8_t max_rsp;
57 /* 16-bit ones' complement of the entire message */
58 uint16_t chksum;
59 /* The multicast address being queried */
60 struct in_addr address;
61 } __packed;
62
63 struct net_ipv4_igmp_v2_report {
64 /* IGMP message type */
65 uint8_t type;
66 /* Max response code */
67 uint8_t max_rsp;
68 /* 16-bit ones' complement of the entire message */
69 uint16_t chksum;
70 /* The multicast address being queried */
71 struct in_addr address;
72 } __packed;
73
74 struct net_ipv4_igmp_v3_query {
75 /* IGMP message type */
76 uint8_t type;
77 /* Max response code */
78 uint8_t max_rsp;
79 /* 16-bit ones' complement of the entire message */
80 uint16_t chksum;
81 /* The multicast address being queried */
82 struct in_addr address;
83 /* Reserved field, ignore */
84 uint8_t reserved: 4;
85 /* Suppress Router-side Processing Flag */
86 uint8_t suppress: 1;
87 /* Querier's Robustness Variable */
88 uint8_t qrv: 3;
89 /* Querier's Query Interval Code */
90 uint8_t qqic;
91 /* Number of Source Addresses */
92 uint16_t sources_len;
93 } __packed;
94
95 struct net_ipv4_igmp_v3_group_record {
96 /* Record type */
97 uint8_t type;
98 /* Aux Data Len */
99 uint8_t aux_len;
100 /* Number of Source Addresses */
101 uint16_t sources_len;
102 /* The multicast address to report to*/
103 struct in_addr address;
104 } __packed;
105
106 struct net_ipv4_igmp_v3_report {
107 /* IGMP message type */
108 uint8_t type;
109 /* Reserved field, ignore */
110 uint8_t reserved_1;
111 /* 16-bit ones' complement of the entire message */
112 uint16_t chksum;
113 /* Reserved field, ignore */
114 uint16_t reserved_2;
115 /* Number of Group Records */
116 uint16_t groups_len;
117 } __packed;
118
119 /**
120 * @brief Create IPv4 packet in provided net_pkt with option to set all the
121 * caller settable values.
122 *
123 * @param pkt Network packet
124 * @param src Source IPv4 address
125 * @param dst Destination IPv4 address
126 * @param tos Type of service
127 * @param id Fragment id
128 * @param flags Fragmentation flags
129 * @param offset Fragment offset
130 *
131 * @return 0 on success, negative errno otherwise.
132 */
133 #if defined(CONFIG_NET_NATIVE_IPV4)
134 int net_ipv4_create_full(struct net_pkt *pkt,
135 const struct in_addr *src,
136 const struct in_addr *dst,
137 uint8_t tos,
138 uint16_t id,
139 uint8_t flags,
140 uint16_t offset);
141 #else
net_ipv4_create_full(struct net_pkt * pkt,const struct in_addr * src,const struct in_addr * dst,uint8_t tos,uint16_t id,uint8_t flags,uint16_t offset)142 static inline int net_ipv4_create_full(struct net_pkt *pkt,
143 const struct in_addr *src,
144 const struct in_addr *dst,
145 uint8_t tos,
146 uint16_t id,
147 uint8_t flags,
148 uint16_t offset)
149 {
150 ARG_UNUSED(pkt);
151 ARG_UNUSED(src);
152 ARG_UNUSED(dst);
153 ARG_UNUSED(tos);
154 ARG_UNUSED(id);
155 ARG_UNUSED(flags);
156 ARG_UNUSED(offset);
157
158 return -ENOTSUP;
159 }
160 #endif
161
162 /**
163 * @brief Create IPv4 packet in provided net_pkt.
164 *
165 * @param pkt Network packet
166 * @param src Source IPv4 address
167 * @param dst Destination IPv4 address
168 *
169 * @return 0 on success, negative errno otherwise.
170 */
171 #if defined(CONFIG_NET_NATIVE_IPV4)
172 int net_ipv4_create(struct net_pkt *pkt,
173 const struct in_addr *src,
174 const struct in_addr *dst);
175 #else
net_ipv4_create(struct net_pkt * pkt,const struct in_addr * src,const struct in_addr * dst)176 static inline int net_ipv4_create(struct net_pkt *pkt,
177 const struct in_addr *src,
178 const struct in_addr *dst)
179 {
180 ARG_UNUSED(pkt);
181 ARG_UNUSED(src);
182 ARG_UNUSED(dst);
183
184 return -ENOTSUP;
185 }
186 #endif
187
188 /**
189 * @brief Finalize IPv4 packet. It should be called right before
190 * sending the packet and after all the data has been added into
191 * the packet. This function will set the length of the
192 * packet and calculate the higher protocol checksum if needed.
193 *
194 * @param pkt Network packet
195 * @param next_header_proto Protocol type of the next header after IPv4 header.
196 *
197 * @return 0 on success, negative errno otherwise.
198 */
199 #if defined(CONFIG_NET_NATIVE_IPV4)
200 int net_ipv4_finalize(struct net_pkt *pkt, uint8_t next_header_proto);
201 #else
net_ipv4_finalize(struct net_pkt * pkt,uint8_t next_header_proto)202 static inline int net_ipv4_finalize(struct net_pkt *pkt,
203 uint8_t next_header_proto)
204 {
205 ARG_UNUSED(pkt);
206 ARG_UNUSED(next_header_proto);
207
208 return -ENOTSUP;
209 }
210 #endif
211
212 /**
213 * @typedef net_ipv4_parse_hdr_options_cb_t
214 * @brief IPv4 header options handle callback
215 *
216 * @details The callback is called when parser encounter
217 * supported options.
218 *
219 * @param opt_type Option type
220 * @param opt_data Option data
221 * @param opt_len Option length
222 * @param user_data Userdata given in net_ipv4_parse_hdr_options()
223 *
224 * @return 0 on success, negative otherwise.
225 */
226 typedef int (*net_ipv4_parse_hdr_options_cb_t)(uint8_t opt_type,
227 uint8_t *opt_data,
228 uint8_t opt_len,
229 void *user_data);
230
231 /**
232 * @brief Parse IPv4 header options.
233 * Parse the IPv4 header options and call the callback with
234 * options type, data and length along with user_data.
235 *
236 * @param pkt Network packet
237 * @param cb callback to handle IPv4 header options
238 * @param user_data User data
239 *
240 * @return 0 on success, negative otherwise.
241 */
242 #if defined(CONFIG_NET_IPV4_HDR_OPTIONS)
243 int net_ipv4_parse_hdr_options(struct net_pkt *pkt,
244 net_ipv4_parse_hdr_options_cb_t cb,
245 void *user_data);
246 #else
net_ipv4_parse_hdr_options(struct net_pkt * pkt,net_ipv4_parse_hdr_options_cb_t cb,void * user_data)247 static inline int net_ipv4_parse_hdr_options(struct net_pkt *pkt,
248 net_ipv4_parse_hdr_options_cb_t cb,
249 void *user_data)
250 {
251 ARG_UNUSED(pkt);
252 ARG_UNUSED(cb);
253 ARG_UNUSED(user_data);
254
255 return -ENOTSUP;
256 }
257 #endif
258
259 /**
260 * @brief Decode DSCP value from ToS field.
261 *
262 * @param tos ToS field value from the IPv4 header.
263 *
264 * @return Decoded DSCP value.
265 */
net_ipv4_get_dscp(uint8_t tos)266 static inline uint8_t net_ipv4_get_dscp(uint8_t tos)
267 {
268 return (tos & NET_IPV4_DSCP_MASK) >> NET_IPV4_DSCP_OFFSET;
269 }
270
271 /**
272 * @brief Encode DSCP value into ToS field.
273 *
274 * @param tos A pointer to the ToS field.
275 * @param dscp DSCP value to set.
276 */
net_ipv4_set_dscp(uint8_t * tos,uint8_t dscp)277 static inline void net_ipv4_set_dscp(uint8_t *tos, uint8_t dscp)
278 {
279 *tos &= ~NET_IPV4_DSCP_MASK;
280 *tos |= (dscp << NET_IPV4_DSCP_OFFSET) & NET_IPV4_DSCP_MASK;
281 }
282
283 /**
284 * @brief Convert DSCP value to priority.
285 *
286 * @param dscp DSCP value.
287 */
net_ipv4_dscp_to_priority(uint8_t dscp)288 static inline uint8_t net_ipv4_dscp_to_priority(uint8_t dscp)
289 {
290 return dscp >> 3;
291 }
292
293 /**
294 * @brief Decode ECN value from ToS field.
295 *
296 * @param tos ToS field value from the IPv4 header.
297 *
298 * @return Decoded ECN value.
299 */
net_ipv4_get_ecn(uint8_t tos)300 static inline uint8_t net_ipv4_get_ecn(uint8_t tos)
301 {
302 return tos & NET_IPV4_ECN_MASK;
303 }
304
305 /**
306 * @brief Encode ECN value into ToS field.
307 *
308 * @param tos A pointer to the ToS field.
309 * @param ecn ECN value to set.
310 */
net_ipv4_set_ecn(uint8_t * tos,uint8_t ecn)311 static inline void net_ipv4_set_ecn(uint8_t *tos, uint8_t ecn)
312 {
313 *tos &= ~NET_IPV4_ECN_MASK;
314 *tos |= ecn & NET_IPV4_ECN_MASK;
315 }
316
317 #if defined(CONFIG_NET_IPV4_FRAGMENT)
318 /** Store pending IPv4 fragment information that is needed for reassembly. */
319 struct net_ipv4_reassembly {
320 /** IPv4 source address of the fragment */
321 struct in_addr src;
322
323 /** IPv4 destination address of the fragment */
324 struct in_addr dst;
325
326 /**
327 * Timeout for cancelling the reassembly. The timer is used
328 * also to detect if this reassembly slot is used or not.
329 */
330 struct k_work_delayable timer;
331
332 /** Pointers to pending fragments */
333 struct net_pkt *pkt[CONFIG_NET_IPV4_FRAGMENT_MAX_PKT];
334
335 /** IPv4 fragment identification */
336 uint16_t id;
337 uint8_t protocol;
338 };
339 #else
340 struct net_ipv4_reassembly;
341 #endif
342
343 /**
344 * @typedef net_ipv4_frag_cb_t
345 * @brief Callback used while iterating over pending IPv4 fragments.
346 *
347 * @param reass IPv4 fragment reassembly struct
348 * @param user_data A valid pointer on some user data or NULL
349 */
350 typedef void (*net_ipv4_frag_cb_t)(struct net_ipv4_reassembly *reass, void *user_data);
351
352 /**
353 * @brief Go through all the currently pending IPv4 fragments.
354 *
355 * @param cb Callback to call for each pending IPv4 fragment.
356 * @param user_data User specified data or NULL.
357 */
358 void net_ipv4_frag_foreach(net_ipv4_frag_cb_t cb, void *user_data);
359
360 /**
361 * @brief Prepare packet for sending, this will split up a packet that is too large to send into
362 * multiple fragments so that it can be sent. It will also update PMTU destination cache if it
363 * is enabled.
364 *
365 * @param pkt Network packet
366 *
367 * @return Return verdict about the packet.
368 */
369 enum net_verdict net_ipv4_prepare_for_send(struct net_pkt *pkt);
370
371 #if defined(CONFIG_NET_NATIVE_IPV4)
372 /**
373 * @brief Initialises IPv4
374 */
375 void net_ipv4_init(void);
376
377 /**
378 * @brief Handles IPv4 fragmented packets.
379 *
380 * @param pkt Network head packet.
381 * @param hdr The IPv4 header of the current packet
382 *
383 * @return Return verdict about the packet.
384 */
385 #if defined(CONFIG_NET_IPV4_FRAGMENT)
386 enum net_verdict net_ipv4_handle_fragment_hdr(struct net_pkt *pkt, struct net_ipv4_hdr *hdr);
387 #else
net_ipv4_handle_fragment_hdr(struct net_pkt * pkt,struct net_ipv4_hdr * hdr)388 static inline enum net_verdict net_ipv4_handle_fragment_hdr(struct net_pkt *pkt,
389 struct net_ipv4_hdr *hdr)
390 {
391 ARG_UNUSED(pkt);
392 ARG_UNUSED(hdr);
393
394 return NET_DROP;
395 }
396 #endif /* CONFIG_NET_IPV4_FRAGMENT */
397
398 #if defined(CONFIG_NET_IPV4_FRAGMENT)
399 enum net_verdict net_ipv4_prepare_for_send_fragment(struct net_pkt *pkt);
400 #endif
401
402 /**
403 * @brief Sets up fragment buffers for usage, should only be called by the SYS_INIT() handler in
404 * net_core.c
405 */
406 #if defined(CONFIG_NET_IPV4_FRAGMENT)
407 void net_ipv4_setup_fragment_buffers(void);
408 #else
net_ipv4_setup_fragment_buffers(void)409 static inline void net_ipv4_setup_fragment_buffers(void)
410 {
411 }
412 #endif /* CONFIG_NET_IPV4_FRAGMENT */
413 #else
414 #define net_ipv4_init(...)
415 #endif /* CONFIG_NET_NATIVE_IPV4 */
416
417 /**
418 * @brief Starts address conflict detection for an IPv4 address.
419 *
420 * @param iface Network interface the address belongs to.
421 * @param ifaddr IPv4 address to probe.
422 *
423 * @return 0 on success, negative otherwise.
424 */
425 int net_ipv4_acd_start(struct net_if *iface, struct net_if_addr *ifaddr);
426
427 /**
428 * @brief Cancel address conflict detection for an IPv4 address.
429 *
430 * @param iface Network interface the address belongs to.
431 * @param ifaddr IPv4 address to probe.
432 */
433 void net_ipv4_acd_cancel(struct net_if *iface, struct net_if_addr *ifaddr);
434
435 /**
436 * @brief Notify no conflict was detected for an IPv4 address.
437 *
438 * @param iface Network interface the address belongs to.
439 * @param ifaddr IPv4 address.
440 */
441 void net_if_ipv4_acd_succeeded(struct net_if *iface, struct net_if_addr *ifaddr);
442
443 /**
444 * @brief Notify conflict for an IPv4 address.
445 *
446 * @param iface Network interface the address belongs to.
447 * @param ifaddr IPv4 address.
448 */
449 void net_if_ipv4_acd_failed(struct net_if *iface, struct net_if_addr *ifaddr);
450
451 /**
452 * @brief Initialize IPv4 address conflict detection module.
453 */
454 void net_ipv4_acd_init(void);
455
456 /**
457 * @brief Process ARP packet in terms of conflict detection.
458 *
459 * @param iface Network interface the packet was received on.
460 * @param pkt ARP packet to process.
461 *
462 * @return Return verdict about the packet.
463 */
464 enum net_verdict net_ipv4_acd_input(struct net_if *iface, struct net_pkt *pkt);
465
466 #endif /* __IPV4_H */
467