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 */
34static int DetectICMPv6mtuMatch (DetectEngineThreadCtx *, Packet *,
35 const Signature *, const SigMatchCtx *);
36static int DetectICMPv6mtuSetup (DetectEngineCtx *, Signature *, const char *);
38#ifdef UNITTESTS
40#endif
41static int PrefilterSetupIcmpv6mtu(DetectEngineCtx *de_ctx, SigGroupHead *sgh);
42static 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
64static 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 */
89static 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 */
113static 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 SCDetectU32Free(ptr);
138}
139
140/* prefilter code */
141
142static void
143PrefilterPacketIcmpv6mtuMatch(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
171static int PrefilterSetupIcmpv6mtu(DetectEngineCtx *de_ctx, SigGroupHead *sgh)
172{
174 PrefilterPacketU32Set, PrefilterPacketU32Compare, PrefilterPacketIcmpv6mtuMatch);
175}
176
177static bool PrefilterIcmpv6mtuIsPrefilterable(const Signature *s)
178{
179 return PrefilterIsPrefilterableById(s, DETECT_ICMPV6MTU);
180}
181
182#ifdef UNITTESTS
184#endif
#define ICMPV6_GET_CODE(icmp6h)
#define ICMPV6_GET_MTU(icmp6h)
#define ICMPV6_HAS_MTU(icmp6h)
#define PKT_IS_PSEUDOPKT(p)
return 1 if the packet is a pseudo packet
Definition decode.h:1321
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))
#define DETECT_PROTO_IPV6
DetectUintData_u32 * DetectU32Parse(const char *u32str)
This function is used to parse u32 options passed via some u32 keyword.
int DetectU32Match(const uint32_t parg, const DetectUintData_u32 *du32)
void PrefilterPacketU32Set(PrefilterPacketHeaderValue *v, void *smctx)
bool PrefilterPacketU32Compare(PrefilterPacketHeaderValue v, void *smctx)
DetectUintData_u32 DetectU32Data
void DetectICMPv6mtuRegisterTests(void)
this function registers unit tests for DetectICMPv6mtu
void DetectICMPv6mtuFree(DetectEngineCtx *de_ctx, void *)
this function will free memory associated with DetectU32Data
void DetectICMPv6mtuRegister(void)
Registration function for icmpv6.mtu: keyword.
SigMatch * SCSigMatchAppendSMToList(DetectEngineCtx *de_ctx, Signature *s, uint16_t type, SigMatchCtx *ctx, const int list)
Append a SigMatch to the list type.
SigTableElmt * sigmatch_table
#define SIG_FLAG_REQUIRE_PACKET
Definition detect.h:254
#define SIG_MASK_REQUIRE_REAL_PKT
Definition detect.h:316
@ DETECT_SM_LIST_MATCH
Definition detect.h:117
DetectEngineCtx * de_ctx
struct Thresholds ctx
main detection engine ctx
Definition detect.h:932
PrefilterRuleStore pmq
Definition detect.h:1349
Container for matching data for a signature group.
Definition detect.h:1629
Used to start a pointer to SigMatch context Should never be dereferenced without casting to something...
Definition detect.h:351
const char * url
Definition detect.h:1462
int(* Setup)(DetectEngineCtx *, Signature *, const char *)
Definition detect.h:1441
int(* SetupPrefilter)(DetectEngineCtx *de_ctx, struct SigGroupHead_ *sgh)
Definition detect.h:1444
void(* Free)(DetectEngineCtx *, void *)
Definition detect.h:1446
const char * desc
Definition detect.h:1461
void(* RegisterTests)(void)
Definition detect.h:1448
int(* Match)(DetectEngineThreadCtx *, Packet *, const Signature *, const SigMatchCtx *)
Definition detect.h:1421
const char * name
Definition detect.h:1459
bool(* SupportsPrefilter)(const Signature *s)
Definition detect.h:1443
Signature container.
Definition detect.h:668
uint32_t flags
Definition detect.h:669
DetectProto proto
Definition detect.h:687
#define SCLogDebug(...)
Definition util-debug.h:275
#define DEBUG_VALIDATE_BUG_ON(exp)