blob: e42402f180b75a1b7d9abe8e8548cb1caf76a2e0 [file] [log] [blame]
Greg Kroah-Hartmanb2441312017-11-01 15:07:57 +01001/* SPDX-License-Identifier: GPL-2.0 */
Tom Herbert37dd0242014-10-03 15:48:09 -07002#ifndef __NET_GUE_H
3#define __NET_GUE_H
4
Tom Herbert5024c332014-11-04 09:06:53 -08005/* Definitions for the GUE header, standard and private flags, lengths
6 * of optional fields are below.
7 *
8 * Diagram of GUE header:
9 *
10 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
11 * |Ver|C| Hlen | Proto/ctype | Standard flags |P|
12 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
13 * | |
14 * ~ Fields (optional) ~
15 * | |
16 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
17 * | Private flags (optional, P bit is set) |
18 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
19 * | |
20 * ~ Private fields (optional) ~
21 * | |
22 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
23 *
Aiden Leong26ac10b2020-06-22 20:04:58 -070024 * C bit indicates control message when set, data message when unset.
Tom Herbert5024c332014-11-04 09:06:53 -080025 * For a control message, proto/ctype is interpreted as a type of
26 * control message. For data messages, proto/ctype is the IP protocol
27 * of the next header.
28 *
29 * P bit indicates private flags field is present. The private flags
30 * may refer to options placed after this field.
31 */
32
Tom Herbert37dd0242014-10-03 15:48:09 -070033struct guehdr {
34 union {
35 struct {
36#if defined(__LITTLE_ENDIAN_BITFIELD)
Tom Herbert5024c332014-11-04 09:06:53 -080037 __u8 hlen:5,
38 control:1,
39 version:2;
Tom Herbert37dd0242014-10-03 15:48:09 -070040#elif defined (__BIG_ENDIAN_BITFIELD)
Tom Herbert5024c332014-11-04 09:06:53 -080041 __u8 version:2,
42 control:1,
43 hlen:5;
Tom Herbert37dd0242014-10-03 15:48:09 -070044#else
45#error "Please fix <asm/byteorder.h>"
46#endif
Xin Long20080972017-12-10 16:56:00 +080047 __u8 proto_ctype;
48 __be16 flags;
Tom Herbert37dd0242014-10-03 15:48:09 -070049 };
Xin Long20080972017-12-10 16:56:00 +080050 __be32 word;
Tom Herbert37dd0242014-10-03 15:48:09 -070051 };
52};
53
Tom Herbert5024c332014-11-04 09:06:53 -080054/* Standard flags in GUE header */
55
56#define GUE_FLAG_PRIV htons(1<<0) /* Private flags are in options */
57#define GUE_LEN_PRIV 4
58
59#define GUE_FLAGS_ALL (GUE_FLAG_PRIV)
60
61/* Private flags in the private option extension */
62
Vandana BN88405682019-07-01 19:46:10 +053063#define GUE_PFLAG_REMCSUM htonl(1U << 31)
Tom Herbertc1aa8342014-11-04 09:06:55 -080064#define GUE_PLEN_REMCSUM 4
65
66#define GUE_PFLAGS_ALL (GUE_PFLAG_REMCSUM)
Tom Herbert5024c332014-11-04 09:06:53 -080067
68/* Functions to compute options length corresponding to flags.
69 * If we ever have a lot of flags this can be potentially be
70 * converted to a more optimized algorithm (table lookup
71 * for instance).
72 */
73static inline size_t guehdr_flags_len(__be16 flags)
74{
75 return ((flags & GUE_FLAG_PRIV) ? GUE_LEN_PRIV : 0);
76}
77
78static inline size_t guehdr_priv_flags_len(__be32 flags)
79{
80 return 0;
81}
82
83/* Validate standard and private flags. Returns non-zero (meaning invalid)
84 * if there is an unknown standard or private flags, or the options length for
85 * the flags exceeds the options length specific in hlen of the GUE header.
86 */
Xin Long20080972017-12-10 16:56:00 +080087static inline int validate_gue_flags(struct guehdr *guehdr, size_t optlen)
Tom Herbert5024c332014-11-04 09:06:53 -080088{
Xin Long20080972017-12-10 16:56:00 +080089 __be16 flags = guehdr->flags;
Tom Herbert5024c332014-11-04 09:06:53 -080090 size_t len;
Tom Herbert5024c332014-11-04 09:06:53 -080091
92 if (flags & ~GUE_FLAGS_ALL)
93 return 1;
94
95 len = guehdr_flags_len(flags);
96 if (len > optlen)
97 return 1;
98
99 if (flags & GUE_FLAG_PRIV) {
100 /* Private flags are last four bytes accounted in
101 * guehdr_flags_len
102 */
Xin Long20080972017-12-10 16:56:00 +0800103 __be32 pflags = *(__be32 *)((void *)&guehdr[1] +
104 len - GUE_LEN_PRIV);
Tom Herbert5024c332014-11-04 09:06:53 -0800105
Xin Long20080972017-12-10 16:56:00 +0800106 if (pflags & ~GUE_PFLAGS_ALL)
Tom Herbert5024c332014-11-04 09:06:53 -0800107 return 1;
108
Xin Long20080972017-12-10 16:56:00 +0800109 len += guehdr_priv_flags_len(pflags);
Tom Herbert5024c332014-11-04 09:06:53 -0800110 if (len > optlen)
111 return 1;
112 }
113
114 return 0;
115}
116
Tom Herbert37dd0242014-10-03 15:48:09 -0700117#endif