suricata
main.c
Go to the documentation of this file.
1 /* Copyright (C) 2024 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 #include "suricata.h"
19 #include "runmodes.h"
20 #include "conf.h"
21 #include "pcap.h"
22 #include "runmode-lib.h"
23 #include "tm-threads.h"
24 #include "threadvars.h"
25 #include "action-globals.h"
26 #include "packet.h"
27 
28 static int worker_id = 1;
29 
30 /**
31  * Struct to pass arguments into a worker thread.
32  */
33 struct WorkerArgs {
36 };
37 
38 /**
39  * Release packet callback.
40  *
41  * If there is any cleanup that needs to be done when Suricata is done
42  * with a packet, this is the place to do it.
43  *
44  * Important: If using a custom release function, you must also
45  * release or free the packet.
46  *
47  * Optionally this is where you would handle IPS like functionality
48  * such as forwarding the packet, or triggering some other mechanism
49  * to forward the packet.
50  */
51 static void ReleasePacket(Packet *p)
52 {
54  SCLogNotice("Dropping packet!");
55  }
56 
57  /* As we overode the default release function, we must release or
58  * free the packet. */
60 }
61 
62 /**
63  * Suricata worker thread in library mode.
64  * The functions should be wrapped in an API layer.
65  */
66 static void *SimpleWorker(void *arg)
67 {
68  struct WorkerArgs *args = arg;
69  ThreadVars *tv = args->tv;
70 
71  /* Start worker. */
72  if (SCRunModeLibSpawnWorker(tv) != 0) {
73  pthread_exit(NULL);
74  }
75 
76  /* Replay pcap. */
77  pcap_t *fp = pcap_open_offline(args->pcap_filename, NULL);
78  if (fp == NULL) {
79  pthread_exit(NULL);
80  }
81 
82  LiveDevice *device = LiveGetDevice("lib0");
83  assert(device != NULL);
84 
85  int datalink = pcap_datalink(fp);
86  int count = 0;
87  struct pcap_pkthdr pkthdr;
88  const u_char *packet;
89  while ((packet = pcap_next(fp, &pkthdr)) != NULL) {
90 
91  /* Have we been asked to stop? */
93  goto done;
94  }
95 
97  if (unlikely(p == NULL)) {
98  /* Memory allocation error. */
99  goto done;
100  }
101 
102  /* If we are processing a PCAP and it is the first packet we need to set the timestamp. */
103  SCTime_t timestamp = SCTIME_FROM_TIMEVAL(&pkthdr.ts);
104  if (count == 0) {
106  }
107 
108  /* Setup the packet, these will become functions to avoid
109  * internal Packet access. */
111  SCPacketSetTime(p, SCTIME_FROM_TIMEVAL(&pkthdr.ts));
112  SCPacketSetDatalink(p, datalink);
113  SCPacketSetLiveDevice(p, device);
114  SCPacketSetReleasePacket(p, ReleasePacket);
115 
116  if (PacketSetData(p, packet, pkthdr.len) == -1) {
118  goto done;
119  }
120 
121  if (TmThreadsSlotProcessPkt(tv, tv->tm_slots, p) != TM_ECODE_OK) {
123  goto done;
124  }
125 
126  (void)SC_ATOMIC_ADD(device->pkts, 1);
127  count++;
128  }
129 
130 done:
131  pcap_close(fp);
132 
133  /* Stop the engine. */
134  EngineStop();
135 
136  /* Cleanup.
137  *
138  * Note that there is some thread synchronization between this
139  * function and SuricataShutdown such that they must be run
140  * concurrently at this time before either will exit. */
142 
143  SCLogNotice("Worker thread exiting");
144  pthread_exit(NULL);
145 }
146 
147 int main(int argc, char **argv)
148 {
149  SuricataPreInit(argv[0]);
150 
151  /* Parse command line options. This is optional, you could
152  * directly configure Suricata through the Conf API. */
153  SCParseCommandLine(argc, argv);
154 
155  /* Find our list of pcap files, after the "--". */
156  while (argc) {
157  bool end = strncmp(argv[0], "--", 2) == 0;
158  argv++;
159  argc--;
160  if (end) {
161  break;
162  }
163  }
164  if (argc == 0) {
165  fprintf(stderr, "ERROR: No PCAP files provided\n");
166  return 1;
167  }
168 
169  /* Set the runmode to library mode. Perhaps in the future this
170  * should be done in some library bootstrap function. */
172 
173  /* Validate/finalize the runmode. */
174  if (SCFinalizeRunMode() != TM_ECODE_OK) {
175  exit(EXIT_FAILURE);
176  }
177 
178  /* Handle internal runmodes. Typically you wouldn't do this as a
179  * library user, however this example is showing how to replicate
180  * the Suricata application with the library. */
181  switch (SCStartInternalRunMode(argc, argv)) {
182  case TM_ECODE_DONE:
183  exit(EXIT_SUCCESS);
184  case TM_ECODE_FAILED:
185  exit(EXIT_FAILURE);
186  }
187 
188  /* Load configuration file, could be done earlier but must be done
189  * before SuricataInit, but even then its still optional as you
190  * may be programmatically configuration Suricata. */
191  if (SCLoadYamlConfig() != TM_ECODE_OK) {
192  exit(EXIT_FAILURE);
193  }
194 
195  /* Set "offline" runmode to replay a pcap in library mode. */
196  if (!SCConfSetFromString("runmode=offline", 1)) {
197  exit(EXIT_FAILURE);
198  }
199 
200  /* Force logging to the current directory. */
201  SCConfSetFromString("default-log-dir=.", 1);
202 
203  if (LiveRegisterDevice("lib0") < 0) {
204  fprintf(stderr, "LiveRegisterDevice failed");
205  exit(1);
206  }
207 
208  SuricataInit();
209 
210  /* Create and start worker on its own thread, passing the PCAP
211  * file as argument. This needs to be done in between SuricataInit
212  * and SuricataPostInit. */
213  pthread_t worker;
215  if (!tv) {
216  FatalError("Failed to create ThreadVars");
217  }
218  struct WorkerArgs args = {
219  .tv = tv,
220  .pcap_filename = argv[argc - 1],
221  };
222  if (pthread_create(&worker, NULL, SimpleWorker, &args) != 0) {
223  exit(EXIT_FAILURE);
224  }
225 
227 
228  /* Run the main loop, this just waits for the worker thread to
229  * call EngineStop signalling Suricata that it is done reading the
230  * pcap. */
232 
233  /* Shutdown engine. */
234  SCLogNotice("Shutting down");
235 
236  /* Note that there is some thread synchronization between this
237  * function and SCTmThreadsSlotPktAcqLoopFinish that require them
238  * to be run concurrently at this time. */
240  GlobalsDestroy();
241 
242  return EXIT_SUCCESS;
243 }
PacketCheckAction
bool PacketCheckAction(const Packet *p, const uint8_t a)
Definition: packet.c:49
tm-threads.h
SuricataMainLoop
void SuricataMainLoop(void)
Definition: suricata.c:2822
LiveRegisterDevice
int LiveRegisterDevice(const char *dev)
Add a pcap device for monitoring and create structure.
Definition: util-device.c:126
PacketFreeOrRelease
void PacketFreeOrRelease(Packet *p)
Return a packet to where it was allocated.
Definition: decode.c:250
unlikely
#define unlikely(expr)
Definition: util-optimize.h:35
SCPacketSetLiveDevice
void SCPacketSetLiveDevice(Packet *p, LiveDevice *device)
Set a packets live device.
Definition: packet.c:170
SuricataInit
void SuricataInit(void)
Definition: suricata.c:2912
TM_ECODE_DONE
@ TM_ECODE_DONE
Definition: tm-threads-common.h:83
action-globals.h
SuricataPostInit
void SuricataPostInit(void)
Definition: suricata.c:3009
LiveDevice_
Definition: util-device.h:41
SC_ATOMIC_ADD
#define SC_ATOMIC_ADD(name, val)
add a value to our atomic variable
Definition: util-atomic.h:332
SURICATA_STOP
#define SURICATA_STOP
Definition: suricata.h:89
SCParseCommandLine
TmEcode SCParseCommandLine(int argc, char **argv)
Definition: suricata.c:1321
WorkerArgs
Definition: main.c:33
SuricataShutdown
void SuricataShutdown(void)
Definition: suricata.c:2998
TmqhOutputPacketpool
void TmqhOutputPacketpool(ThreadVars *t, Packet *p)
Definition: tmqh-packetpool.c:314
TM_ECODE_FAILED
@ TM_ECODE_FAILED
Definition: tm-threads-common.h:82
RUNMODE_LIB
@ RUNMODE_LIB
Definition: runmodes.h:40
TM_ECODE_OK
@ TM_ECODE_OK
Definition: tm-threads-common.h:81
SCPacketSetSource
void SCPacketSetSource(Packet *p, enum PktSrcEnum source)
Set packet source.
Definition: packet.c:185
SCRunmodeSet
void SCRunmodeSet(SCRunMode run_mode)
Set the current run mode.
Definition: suricata.c:271
PKT_SRC_WIRE
@ PKT_SRC_WIRE
Definition: decode.h:52
main
int main(int argc, char **argv)
Definition: main.c:20
ThreadVars_::tm_slots
struct TmSlot_ * tm_slots
Definition: threadvars.h:96
LiveGetDevice
LiveDevice * LiveGetDevice(const char *name)
Get a pointer to the device at idx.
Definition: util-device.c:248
ThreadVars_
Per thread variable structure.
Definition: threadvars.h:58
SCTIME_FROM_TIMEVAL
#define SCTIME_FROM_TIMEVAL(tv)
Definition: util-time.h:79
SCRunModeLibSpawnWorker
int SCRunModeLibSpawnWorker(void *td)
start the "fake" worker.
Definition: runmode-lib.c:95
Packet_
Definition: decode.h:477
conf.h
SCPacketSetTime
void SCPacketSetTime(Packet *p, SCTime_t ts)
Set the timestamp for a packet.
Definition: packet.c:180
SCTime_t
Definition: util-time.h:40
runmodes.h
SCConfSetFromString
int SCConfSetFromString(const char *input, int final)
Set a configuration parameter from a string.
Definition: conf.c:249
TmThreadsInitThreadsTimestamp
void TmThreadsInitThreadsTimestamp(const SCTime_t ts)
Definition: tm-threads.c:2314
packet.h
WorkerArgs::pcap_filename
char * pcap_filename
Definition: main.c:35
ACTION_DROP
#define ACTION_DROP
Definition: action-globals.h:30
SCStartInternalRunMode
int SCStartInternalRunMode(int argc, char **argv)
Definition: suricata.c:2314
SCPacketSetReleasePacket
void SCPacketSetReleasePacket(Packet *p, void(*ReleasePacket)(Packet *p))
Set a packet release function.
Definition: packet.c:165
FatalError
#define FatalError(...)
Definition: util-debug.h:502
EngineStop
void EngineStop(void)
make sure threads can stop the engine by calling this function. Purpose: pcap file mode needs to be a...
Definition: suricata.c:452
tv
ThreadVars * tv
Definition: fuzz_decodepcapfile.c:32
threadvars.h
SCLoadYamlConfig
TmEcode SCLoadYamlConfig(void)
Definition: suricata.c:964
SCRunModeLibCreateThreadVars
ThreadVars * SCRunModeLibCreateThreadVars(int worker_id)
Create ThreadVars for use by a user provided thread.
Definition: runmode-lib.c:59
runmode-lib.h
PacketSetData
int PacketSetData(Packet *p, const uint8_t *pktdata, uint32_t pktlen)
Set data for Packet and set length when zero copy is used.
Definition: decode.c:813
suricata.h
SCPacketSetDatalink
void SCPacketSetDatalink(Packet *p, int datalink)
Set a packets data link type.
Definition: packet.c:175
WorkerArgs::tv
ThreadVars * tv
Definition: main.c:34
SCLogNotice
#define SCLogNotice(...)
Macro used to log NOTICE messages.
Definition: util-debug.h:237
SuricataPreInit
void SuricataPreInit(const char *progname)
Definition: suricata.c:2903
SCTmThreadsSlotPktAcqLoopFinish
bool SCTmThreadsSlotPktAcqLoopFinish(ThreadVars *tv)
Definition: tm-threads.c:273
PacketGetFromQueueOrAlloc
Packet * PacketGetFromQueueOrAlloc(void)
Get a packet. We try to get a packet from the packetpool first, but if that is empty we alloc a packe...
Definition: decode.c:267
SCFinalizeRunMode
int SCFinalizeRunMode(void)
Definition: suricata.c:2370
GlobalsDestroy
void GlobalsDestroy(void)
Definition: suricata.c:372
suricata_ctl_flags
volatile uint8_t suricata_ctl_flags
Definition: suricata.c:171