suricata
detect-ftpdata.c
Go to the documentation of this file.
1/* Copyright (C) 2017-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 Eric Leblond <eric@regit.org>
22 *
23 * Match on ftp command used to trigger a ftp data transfer
24 */
25
26#include "suricata-common.h"
27#include "util-unittest.h"
28
29#include "detect-parse.h"
30#include "detect-engine.h"
31#include "detect-engine-state.h"
32
33#include "app-layer-ftp.h"
34
35#include "detect-ftpdata.h"
36
37/**
38 * \brief Regex for parsing our keyword options
39 */
40#define PARSE_REGEX "^\\s*(stor|retr)\\s*$"
41static DetectParseRegex parse_regex;
42
43/* Prototypes of functions registered in DetectFtpdataRegister below */
44static int DetectFtpdataMatch(DetectEngineThreadCtx *,
45 Flow *, uint8_t, void *, void *,
46 const Signature *, const SigMatchCtx *);
47static int DetectFtpdataSetup (DetectEngineCtx *, Signature *, const char *);
48static void DetectFtpdataFree (DetectEngineCtx *, void *);
49#ifdef UNITTESTS
50static void DetectFtpdataRegisterTests (void);
51#endif
52static int g_ftpdata_buffer_id = 0;
53
54/**
55 * \brief Registration function for ftpcommand: keyword
56 *
57 * This function is called once in the 'lifetime' of the engine.
58 */
60 /* keyword name: this is how the keyword is used in a rule */
61 sigmatch_table[DETECT_FTPDATA].name = "ftpdata_command";
62 /* description: listed in "suricata --list-keywords=all" */
63 sigmatch_table[DETECT_FTPDATA].desc = "match FTP command triggering a FTP data channel";
64 sigmatch_table[DETECT_FTPDATA].url = "/rules/ftp-keywords.html#ftpdata-command";
65 sigmatch_table[DETECT_FTPDATA].AppLayerTxMatch = DetectFtpdataMatch;
66 /* setup function is called during signature parsing, when the ftpcommand
67 * keyword is encountered in the rule */
68 sigmatch_table[DETECT_FTPDATA].Setup = DetectFtpdataSetup;
69 /* free function is called when the detect engine is freed. Normally at
70 * shutdown, but also during rule reloads. */
71 sigmatch_table[DETECT_FTPDATA].Free = DetectFtpdataFree;
72 /* registers unittests into the system */
73#ifdef UNITTESTS
74 sigmatch_table[DETECT_FTPDATA].RegisterTests = DetectFtpdataRegisterTests;
75#endif
78
81 g_ftpdata_buffer_id = DetectBufferTypeGetByName("ftpdata_command");
82
83 /* set up the PCRE for keyword parsing */
85}
86
87/**
88 * \brief This function is used to check matches from the FTP App Layer Parser
89 *
90 * \param t pointer to thread vars
91 * \param det_ctx pointer to the pattern matcher thread
92 * \param p pointer to the current packet
93 * \param m pointer to the sigmatch
94 * \retval 0 no match
95 * \retval 1 match
96 */
97static int DetectFtpdataMatch(DetectEngineThreadCtx *det_ctx,
98 Flow *f, uint8_t flags,
99 void *state, void *txv,
100 const Signature *s, const SigMatchCtx *m)
101{
102 const DetectFtpdataData *ftpcommandd = (const DetectFtpdataData *) m;
103 const FtpDataState *ftp_state = (const FtpDataState *)state;
104
105 if (ftp_state == NULL)
106 return 0;
107
108 if (ftpcommandd->command == ftp_state->command) {
109 /* Only match if the flow is in the good direction */
110 if ((flags & STREAM_TOSERVER) && (ftpcommandd->command == FTP_COMMAND_RETR)) {
111 return 0;
112 } else if ((flags & STREAM_TOCLIENT) && (ftpcommandd->command == FTP_COMMAND_STOR)) {
113 return 0;
114 }
115 return 1;
116 }
117
118 return 0;
119}
120
121/**
122 * \brief This function is used to parse ftpcommand options passed via ftpcommand: keyword
123 *
124 * \param ftpcommandstr Pointer to the user provided ftpcommand options
125 *
126 * \retval ftpcommandd pointer to DetectFtpdataData on success
127 * \retval NULL on failure
128 */
129static DetectFtpdataData *DetectFtpdataParse(const char *ftpcommandstr)
130{
131 DetectFtpdataData *ftpcommandd = NULL;
132 char arg1[5] = "";
133 size_t pcre2len;
134 pcre2_match_data *match = NULL;
135
136 int ret = DetectParsePcreExec(&parse_regex, &match, ftpcommandstr, 0, 0);
137 if (ret != 2) {
138 SCLogError("parse error, ret %" PRId32 "", ret);
139 goto error;
140 }
141
142 pcre2len = sizeof(arg1);
143 int res = pcre2_substring_copy_bynumber(match, 1, (PCRE2_UCHAR8 *)arg1, &pcre2len);
144 if (res < 0) {
145 SCLogError("pcre2_substring_copy_bynumber failed");
146 goto error;
147 }
148 SCLogDebug("Arg1 \"%s\"", arg1);
149
150 ftpcommandd = SCMalloc(sizeof (DetectFtpdataData));
151 if (unlikely(ftpcommandd == NULL))
152 goto error;
153 if (!strcmp(arg1, "stor")) {
154 ftpcommandd->command = FTP_COMMAND_STOR;
155 } else if (!strcmp(arg1, "retr")) {
156 ftpcommandd->command = FTP_COMMAND_RETR;
157 } else {
158 SCLogError("Invalid command value");
159 goto error;
160 }
161
162 pcre2_match_data_free(match);
163 return ftpcommandd;
164
165error:
166 if (match) {
167 pcre2_match_data_free(match);
168 }
169 if (ftpcommandd)
170 SCFree(ftpcommandd);
171 return NULL;
172}
173
174/**
175 * \brief parse the options from the 'ftpcommand' keyword in the rule into
176 * the Signature data structure.
177 *
178 * \param de_ctx pointer to the Detection Engine Context
179 * \param s pointer to the Current Signature
180 * \param str pointer to the user provided ftpcommand options
181 *
182 * \retval 0 on Success
183 * \retval -1 on Failure
184 */
185static int DetectFtpdataSetup(DetectEngineCtx *de_ctx, Signature *s, const char *str)
186{
188 return -1;
189
190 DetectFtpdataData *ftpcommandd = DetectFtpdataParse(str);
191 if (ftpcommandd == NULL)
192 return -1;
193
195 g_ftpdata_buffer_id) == NULL) {
196 DetectFtpdataFree(de_ctx, ftpcommandd);
197 return -1;
198 }
199 return 0;
200}
201
202/**
203 * \brief this function will free memory associated with DetectFtpdataData
204 *
205 * \param ptr pointer to DetectFtpdataData
206 */
207static void DetectFtpdataFree(DetectEngineCtx *de_ctx, void *ptr) {
208 DetectFtpdataData *ftpcommandd = (DetectFtpdataData *)ptr;
209
210 /* do more specific cleanup here, if needed */
211
212 SCFree(ftpcommandd);
213}
214
215#ifdef UNITTESTS
216
217static int DetectFtpdataParseTest01(void)
218{
219 DetectFtpdataData *ftpcommandd = DetectFtpdataParse("stor");
220 FAIL_IF_NULL(ftpcommandd);
221 FAIL_IF(!(ftpcommandd->command == FTP_COMMAND_STOR));
222 DetectFtpdataFree(NULL, ftpcommandd);
223 PASS;
224}
225
226static int DetectFtpdataSignatureTest01(void)
227{
230
231 Signature *sig = DetectEngineAppendSig(de_ctx, "alert ip any any -> any any (ftpdata_command:stor; sid:1; rev:1;)");
232 FAIL_IF_NULL(sig);
233 sig = DetectEngineAppendSig(de_ctx, "alert ip any any -> any any (ftpdata_command:retr; sid:2; rev:1;)");
234 FAIL_IF_NULL(sig);
235 sig = DetectEngineAppendSig(de_ctx, "alert ip any any -> any any (ftpdata_command:xxx; sid:3; rev:1;)");
236 FAIL_IF_NOT_NULL(sig);
237
239 PASS;
240}
241
242/**
243 * \brief this function registers unit tests for DetectFtpdata
244 */
245static void DetectFtpdataRegisterTests(void)
246{
247 UtRegisterTest("DetectFtpdataParseTest01", DetectFtpdataParseTest01);
248 UtRegisterTest("DetectFtpdataSignatureTest01",
249 DetectFtpdataSignatureTest01);
250}
251#endif /* UNITTESTS */
@ ALPROTO_FTPDATA
uint8_t flags
Definition decode-gre.h:0
DetectEngineCtx * DetectEngineCtxInit(void)
void DetectEngineCtxFree(DetectEngineCtx *)
Free a DetectEngineCtx::
Signature * DetectEngineAppendSig(DetectEngineCtx *, const char *)
Parse and append a Signature into the Detection Engine Context signature list.
Data structures and function prototypes for keeping state for the detection engine.
void DetectAppLayerInspectEngineRegister(const char *name, AppProto alproto, uint32_t dir, int progress, InspectEngineFuncPtr Callback, InspectionBufferGetDataPtr GetData)
Registers an app inspection engine.
int DetectBufferTypeGetByName(const char *name)
uint8_t DetectEngineInspectGenericList(DetectEngineCtx *de_ctx, DetectEngineThreadCtx *det_ctx, const struct DetectEngineAppInspectionEngine_ *engine, const Signature *s, Flow *f, uint8_t flags, void *alstate, void *txv, uint64_t tx_id)
Do the content inspection & validation for a signature.
void DetectFtpdataRegister(void)
Registration function for ftpcommand: keyword.
#define PARSE_REGEX
Regex for parsing our keyword options.
void DetectSetupParseRegexes(const char *parse_str, DetectParseRegex *detect_parse)
int SCDetectSignatureSetAppProto(Signature *s, AppProto alproto)
int DetectParsePcreExec(DetectParseRegex *parse_regex, pcre2_match_data **match, const char *str, int start_offset, int options)
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_TOCLIENT
Definition detect.h:272
#define SIG_FLAG_TOSERVER
Definition detect.h:271
SCMutex m
Definition flow-hash.h:6
DetectEngineCtx * de_ctx
#define FAIL_IF_NULL(expr)
Fail a test if expression evaluates to NULL.
void UtRegisterTest(const char *name, int(*TestFn)(void))
Register unit test.
#define PASS
Pass the test.
#define FAIL_IF(expr)
Fail a test if expression evaluates to true.
#define FAIL_IF_NOT_NULL(expr)
Fail a test if expression evaluates to non-NULL.
main detection engine ctx
Definition detect.h:932
FtpRequestCommand command
Flow data structure.
Definition flow.h:356
FtpRequestCommand command
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
void(* Free)(DetectEngineCtx *, void *)
Definition detect.h:1446
const char * desc
Definition detect.h:1461
int(* AppLayerTxMatch)(DetectEngineThreadCtx *, Flow *, uint8_t flags, void *alstate, void *txv, const Signature *, const SigMatchCtx *)
Definition detect.h:1424
void(* RegisterTests)(void)
Definition detect.h:1448
const char * name
Definition detect.h:1459
Signature container.
Definition detect.h:668
#define str(s)
#define SCLogDebug(...)
Definition util-debug.h:275
#define SCLogError(...)
Macro used to log ERROR messages.
Definition util-debug.h:267
#define SCMalloc(sz)
Definition util-mem.h:47
#define SCFree(p)
Definition util-mem.h:61
#define unlikely(expr)