suricata
detect-icmpv6-mtu.c
Go to the documentation of this file.
1 /* Copyright (C) 2020 Open Information Security Foundation
2  *
3  * You can copy, redistribute or modify this Program under the terms of
4  * the GNU General Public License version 2 as published by the Free
5  * Software Foundation.
6  *
7  * This program is distributed in the hope that it will be useful,
8  * but WITHOUT ANY WARRANTY; without even the implied warranty of
9  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10  * GNU General Public License for more details.
11  *
12  * You should have received a copy of the GNU General Public License
13  * version 2 along with this program; if not, write to the Free Software
14  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
15  * 02110-1301, USA.
16  */
17 
18 /**
19  * \file
20  *
21  * \author Philippe Antoine <p.antoine@catenacyber.fr>
22  *
23  */
24 
25 #include "suricata-common.h"
26 
27 #include "detect.h"
28 #include "detect-parse.h"
29 
30 #include "detect-icmpv6-mtu.h"
31 #include "detect-engine-uint.h"
32 
33 /* prototypes */
34 static int DetectICMPv6mtuMatch (DetectEngineThreadCtx *, Packet *,
35  const Signature *, const SigMatchCtx *);
36 static int DetectICMPv6mtuSetup (DetectEngineCtx *, Signature *, const char *);
38 #ifdef UNITTESTS
40 #endif
41 static int PrefilterSetupIcmpv6mtu(DetectEngineCtx *de_ctx, SigGroupHead *sgh);
42 static bool PrefilterIcmpv6mtuIsPrefilterable(const Signature *s);
43 
44 /**
45  * \brief Registration function for icmpv6.mtu: keyword
46  */
47 
49 {
50  sigmatch_table[DETECT_ICMPV6MTU].name = "icmpv6.mtu";
51  sigmatch_table[DETECT_ICMPV6MTU].desc = "match on ICMPv6 MTU field";
52  sigmatch_table[DETECT_ICMPV6MTU].url = "/rules/header-keywords.html#icmpv6mtu";
53  sigmatch_table[DETECT_ICMPV6MTU].Match = DetectICMPv6mtuMatch;
54  sigmatch_table[DETECT_ICMPV6MTU].Setup = DetectICMPv6mtuSetup;
56 #ifdef UNITTESTS
58 #endif
59  sigmatch_table[DETECT_ICMPV6MTU].SupportsPrefilter = PrefilterIcmpv6mtuIsPrefilterable;
60  sigmatch_table[DETECT_ICMPV6MTU].SetupPrefilter = PrefilterSetupIcmpv6mtu;
61 }
62 
63 // returns 0 on no mtu, and 1 if mtu
64 static inline int DetectICMPv6mtuGetValue(Packet *p, uint32_t *picmpv6mtu)
65 {
66  if (!(PacketIsICMPv6(p)))
67  return 0;
68  const ICMPV6Hdr *icmpv6h = PacketGetICMPv6(p);
69  if (ICMPV6_GET_CODE(icmpv6h) != 0)
70  return 0;
71  if (!(ICMPV6_HAS_MTU(icmpv6h)))
72  return 0;
73 
74  *picmpv6mtu = ICMPV6_GET_MTU(icmpv6h);
75  return 1;
76 }
77 
78 /**
79  * \brief This function is used to match ICMPV6 MTU rule option on a packet with those passed via icmpv6.mtu:
80  *
81  * \param det_ctx pointer to the pattern matcher thread
82  * \param p pointer to the current packet
83  * \param s pointer to the signature unused
84  * \param ctx pointer to the signature match context
85  *
86  * \retval 0 no match
87  * \retval 1 match
88  */
89 static int DetectICMPv6mtuMatch (DetectEngineThreadCtx *det_ctx, Packet *p,
90  const Signature *s, const SigMatchCtx *ctx)
91 {
93 
94  uint32_t picmpv6mtu;
95  if (DetectICMPv6mtuGetValue(p, &picmpv6mtu) == 0) {
96  return 0;
97  }
98 
99  const DetectU32Data *du32 = (const DetectU32Data *)ctx;
100  return DetectU32Match(picmpv6mtu, du32);
101 }
102 
103 /**
104  * \brief this function is used to attach the parsed icmpv6.mtu data into the current signature
105  *
106  * \param de_ctx pointer to the Detection Engine Context
107  * \param s pointer to the Current Signature
108  * \param icmpv6mtustr pointer to the user provided icmpv6.mtu options
109  *
110  * \retval 0 on Success
111  * \retval -1 on Failure
112  */
113 static int DetectICMPv6mtuSetup (DetectEngineCtx *de_ctx, Signature *s, const char *icmpv6mtustr)
114 {
115  DetectU32Data *icmpv6mtud = DetectU32Parse(icmpv6mtustr);
116  if (icmpv6mtud == NULL)
117  return -1;
118 
120  DETECT_SM_LIST_MATCH) == NULL) {
121  DetectICMPv6mtuFree(de_ctx, icmpv6mtud);
122  return -1;
123  }
126 
127  return 0;
128 }
129 
130 /**
131  * \brief this function will free memory associated with DetectU32Data
132  *
133  * \param ptr pointer to DetectU32Data
134  */
136 {
137  rs_detect_u32_free(ptr);
138 }
139 
140 /* prefilter code */
141 
142 static void
143 PrefilterPacketIcmpv6mtuMatch(DetectEngineThreadCtx *det_ctx, Packet *p, const void *pectx)
144 {
146 
147  uint32_t picmpv6mtu;
148  if (DetectICMPv6mtuGetValue(p, &picmpv6mtu) == 0) {
149  return;
150  }
151 
152  /* during setup Suricata will automatically see if there is another
153  * check that can be added: alproto, sport or dport */
154  const PrefilterPacketHeaderCtx *ctx = pectx;
155  if (!PrefilterPacketHeaderExtraMatch(ctx, p))
156  return;
157 
158  /* if we match, add all the sigs that use this prefilter. This means
159  * that these will be inspected further */
160  DetectU32Data du32;
161  du32.mode = ctx->v1.u8[0];
162  du32.arg1 = ctx->v1.u32[1];
163  du32.arg2 = ctx->v1.u32[2];
164  if (DetectU32Match(picmpv6mtu, &du32))
165  {
166  SCLogDebug("packet matches icmpv6.mtu/hl %u", picmpv6mtu);
167  PrefilterAddSids(&det_ctx->pmq, ctx->sigs_array, ctx->sigs_cnt);
168  }
169 }
170 
171 static int PrefilterSetupIcmpv6mtu(DetectEngineCtx *de_ctx, SigGroupHead *sgh)
172 {
174  PrefilterPacketU32Set, PrefilterPacketU32Compare, PrefilterPacketIcmpv6mtuMatch);
175 }
176 
177 static bool PrefilterIcmpv6mtuIsPrefilterable(const Signature *s)
178 {
179  return PrefilterIsPrefilterableById(s, DETECT_ICMPV6MTU);
180 }
181 
182 #ifdef UNITTESTS
183 #include "tests/detect-icmpv6-mtu.c"
184 #endif
detect-engine-uint.h
SigTableElmt_::url
const char * url
Definition: detect.h:1304
SIG_MASK_REQUIRE_REAL_PKT
#define SIG_MASK_REQUIRE_REAL_PKT
Definition: detect.h:306
DetectU32Match
int DetectU32Match(const uint32_t parg, const DetectUintData_u32 *du32)
Definition: detect-engine-uint.c:31
SigTableElmt_::desc
const char * desc
Definition: detect.h:1303
sigmatch_table
SigTableElmt * sigmatch_table
Definition: detect-parse.c:127
SigTableElmt_::Free
void(* Free)(DetectEngineCtx *, void *)
Definition: detect.h:1291
DETECT_PROTO_IPV6
#define DETECT_PROTO_IPV6
Definition: detect-engine-proto.h:34
SigTableElmt_::name
const char * name
Definition: detect.h:1301
PKT_IS_PSEUDOPKT
#define PKT_IS_PSEUDOPKT(p)
return 1 if the packet is a pseudo packet
Definition: decode.h:1326
SigGroupHead_
Container for matching data for a signature group.
Definition: detect.h:1457
SCLogDebug
#define SCLogDebug(...)
Definition: util-debug.h:269
DetectU32Parse
DetectUintData_u32 * DetectU32Parse(const char *u32str)
This function is used to parse u32 options passed via some u32 keyword.
Definition: detect-engine-uint.c:45
PrefilterPacketU32Set
void PrefilterPacketU32Set(PrefilterPacketHeaderValue *v, void *smctx)
Definition: detect-engine-uint.c:51
DetectICMPv6mtuRegisterTests
void DetectICMPv6mtuRegisterTests(void)
this function registers unit tests for DetectICMPv6mtu
Definition: detect-icmpv6-mtu.c:47
DetectEngineThreadCtx_::pmq
PrefilterRuleStore pmq
Definition: detect.h:1197
ctx
struct Thresholds ctx
DetectEngineCtx_
main detection engine ctx
Definition: detect.h:841
detect-icmpv6-mtu.h
ICMPV6_HAS_MTU
#define ICMPV6_HAS_MTU(icmp6h)
Definition: decode-icmpv6.h:114
SigTableElmt_::Setup
int(* Setup)(DetectEngineCtx *, Signature *, const char *)
Definition: detect.h:1286
detect-icmpv6-mtu.c
SigTableElmt_::SetupPrefilter
int(* SetupPrefilter)(DetectEngineCtx *de_ctx, struct SigGroupHead_ *sgh)
Definition: detect.h:1289
PrefilterPacketHeaderCtx_
Definition: detect-engine-prefilter-common.h:35
DetectICMPv6mtuRegister
void DetectICMPv6mtuRegister(void)
Registration function for icmpv6.mtu: keyword.
Definition: detect-icmpv6-mtu.c:48
de_ctx
DetectEngineCtx * de_ctx
Definition: fuzz_siginit.c:17
DetectEngineThreadCtx_
Definition: detect.h:1090
detect.h
DETECT_SM_LIST_MATCH
@ DETECT_SM_LIST_MATCH
Definition: detect.h:114
Signature_::flags
uint32_t flags
Definition: detect.h:602
Packet_
Definition: decode.h:479
DETECT_ICMPV6MTU
@ DETECT_ICMPV6MTU
Definition: detect-engine-register.h:291
DetectICMPv6mtuFree
void DetectICMPv6mtuFree(DetectEngineCtx *de_ctx, void *)
this function will free memory associated with DetectU32Data
Definition: detect-icmpv6-mtu.c:135
ICMPV6_GET_CODE
#define ICMPV6_GET_CODE(icmp6h)
Definition: decode-icmpv6.h:103
PrefilterSetupPacketHeader
int PrefilterSetupPacketHeader(DetectEngineCtx *de_ctx, SigGroupHead *sgh, int sm_type, SignatureMask mask, void(*Set)(PrefilterPacketHeaderValue *v, void *), bool(*Compare)(PrefilterPacketHeaderValue v, void *), void(*Match)(DetectEngineThreadCtx *det_ctx, Packet *p, const void *pectx))
Definition: detect-engine-prefilter-common.c:404
PrefilterPacketU32Compare
bool PrefilterPacketU32Compare(PrefilterPacketHeaderValue v, void *smctx)
Definition: detect-engine-uint.c:60
SigTableElmt_::Match
int(* Match)(DetectEngineThreadCtx *, Packet *, const Signature *, const SigMatchCtx *)
Definition: detect.h:1269
SigMatchCtx_
Used to start a pointer to SigMatch context Should never be dereferenced without casting to something...
Definition: detect.h:344
ICMPV6Hdr_
Definition: decode-icmpv6.h:129
DetectProto_::flags
uint8_t flags
Definition: detect-engine-proto.h:38
Signature_::proto
DetectProto proto
Definition: detect.h:620
suricata-common.h
ICMPV6_GET_MTU
#define ICMPV6_GET_MTU(icmp6h)
Definition: decode-icmpv6.h:116
DetectU32Data
DetectUintData_u32 DetectU32Data
Definition: detect-engine-uint.h:41
SigTableElmt_::SupportsPrefilter
bool(* SupportsPrefilter)(const Signature *s)
Definition: detect.h:1288
detect-parse.h
Signature_
Signature container.
Definition: detect.h:601
SigMatchAppendSMToList
SigMatch * SigMatchAppendSMToList(DetectEngineCtx *de_ctx, Signature *s, uint16_t type, SigMatchCtx *ctx, const int list)
Append a SigMatch to the list type.
Definition: detect-parse.c:436
DEBUG_VALIDATE_BUG_ON
#define DEBUG_VALIDATE_BUG_ON(exp)
Definition: util-validate.h:102
SigTableElmt_::RegisterTests
void(* RegisterTests)(void)
Definition: detect.h:1293
SIG_FLAG_REQUIRE_PACKET
#define SIG_FLAG_REQUIRE_PACKET
Definition: detect.h:250