mirror of https://github.com/OISF/suricata
Add TILE-Gx mPIPE packet processing support.
The TILE-Gx processor includes a packet processing engine, called mPIPE, that can deliver packets directly into user space memory. It handles buffer allocation and load balancing (either static 5-tuple hashing, or dynamic flow affinity hashing are used here). The new packet source code is in source-mpipe.c and source-mpipe.h A new Tile runmode is added that configures the Suricata pipelines in worker mode, where each thread does the entire packet processing pipeline. It scales across all the Gx chips sizes of 9, 16, 36 or 72 cores. The new runmode is in runmode-tile.c and runmode-tile.h The configure script detects the TILE-Gx architecture and defines HAVE_MPIPE, which is then used to conditionally enable the code to support mPIPE packet processing. Suricata runs on TILE-Gx even without mPIPE support enabled. The Suricata Packet structures are allocated by the mPIPE hardware by allocating the Suricata Packet structure immediatley before the mPIPE packet buffer and then pushing the mPIPE packet buffer pointer onto the mPIPE buffer stack. This way, mPIPE writes the packet data into the buffer, returns the mPIPE packet buffer pointer, which is then converted into a Suricata Packet pointer for processing inside Suricata. When the Packet is freed, the buffer is returned to mPIPE's buffer stack, by setting ReleasePacket to an mPIPE release specific function. The code checks for the largest Huge page available in Linux when Suricata is started. TILE-Gx supports Huge pages sizes of 16MB, 64MB, 256MB, 1GB and 4GB. Suricata then divides one of those page into packet buffers for mPIPE. The code is not yet optimized for high performance. Performance improvements will follow shortly. The code was originally written by Tom Decanio and then further modified by Tilera. This code has been tested with Tilera's Multicore Developement Environment (MDE) version 4.1.5. The TILEncore-Gx36 (PCIe card) and TILEmpower-Gx (1U Rack mount).pull/465/head
parent
04f3f14541
commit
316190c6b9
@ -0,0 +1,269 @@
|
||||
/* Copyright (C) 2011-2013 Open Information Security Foundation
|
||||
*
|
||||
* You can copy, redistribute or modify this Program under the terms of
|
||||
* the GNU General Public License version 2 as published by the Free
|
||||
* Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* version 2 along with this program; if not, write to the Free Software
|
||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
||||
* 02110-1301, USA.
|
||||
*/
|
||||
|
||||
/**
|
||||
* \file
|
||||
*
|
||||
* \author Tom DeCanio <decanio.tom@gmail.com>
|
||||
* \author Ken Steele, Tilera Corporation <suricata@tilera.com>
|
||||
*
|
||||
* Tilera TILE-Gx runmode support
|
||||
*/
|
||||
|
||||
#include "suricata-common.h"
|
||||
#include "tm-threads.h"
|
||||
#include "conf.h"
|
||||
#include "runmodes.h"
|
||||
#include "runmode-tile.h"
|
||||
#include "log-httplog.h"
|
||||
#include "output.h"
|
||||
#include "source-mpipe.h"
|
||||
|
||||
#include "alert-fastlog.h"
|
||||
#include "alert-prelude.h"
|
||||
#include "alert-unified2-alert.h"
|
||||
#include "alert-debuglog.h"
|
||||
|
||||
#include "util-debug.h"
|
||||
#include "util-time.h"
|
||||
#include "util-cpu.h"
|
||||
#include "util-affinity.h"
|
||||
#include "util-device.h"
|
||||
|
||||
#ifdef HAVE_MPIPE
|
||||
/* Number of configured parallel pipelines. */
|
||||
unsigned int tile_num_pipelines;
|
||||
#endif
|
||||
|
||||
/*
|
||||
* runmode support for tilegx
|
||||
*/
|
||||
|
||||
static const char *mpipe_default_mode = "workers";
|
||||
|
||||
const char *RunModeTileMpipeGetDefaultMode(void)
|
||||
{
|
||||
return mpipe_default_mode;
|
||||
}
|
||||
|
||||
void RunModeTileMpipeRegister(void)
|
||||
{
|
||||
#ifdef HAVE_MPIPE
|
||||
RunModeRegisterNewRunMode(RUNMODE_TILERA_MPIPE, "workers",
|
||||
"Workers tilegx mpipe mode, each thread does all"
|
||||
" tasks from acquisition to logging",
|
||||
RunModeTileMpipeWorkers);
|
||||
mpipe_default_mode = "workers";
|
||||
#endif
|
||||
}
|
||||
|
||||
#ifdef HAVE_MPIPE
|
||||
|
||||
void *ParseMpipeConfig(const char *iface)
|
||||
{
|
||||
ConfNode *if_root;
|
||||
ConfNode *mpipe_node;
|
||||
MpipeIfaceConfig *aconf = SCMalloc(sizeof(*aconf));
|
||||
char *copymodestr;
|
||||
char *out_iface = NULL;
|
||||
|
||||
if (aconf == NULL) {
|
||||
return NULL;
|
||||
}
|
||||
|
||||
if (iface == NULL) {
|
||||
SCFree(aconf);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
strlcpy(aconf->iface, iface, sizeof(aconf->iface));
|
||||
|
||||
/* Find initial node */
|
||||
mpipe_node = ConfGetNode("mpipe.inputs");
|
||||
if (mpipe_node == NULL) {
|
||||
SCLogInfo("Unable to find mpipe config using default value");
|
||||
return aconf;
|
||||
}
|
||||
|
||||
if_root = ConfNodeLookupKeyValue(mpipe_node, "interface", iface);
|
||||
if (if_root == NULL) {
|
||||
SCLogInfo("Unable to find mpipe config for "
|
||||
"interface %s, using default value",
|
||||
iface);
|
||||
return aconf;
|
||||
}
|
||||
|
||||
if (ConfGetChildValue(if_root, "copy-iface", &out_iface) == 1) {
|
||||
if (strlen(out_iface) > 0) {
|
||||
aconf->out_iface = out_iface;
|
||||
}
|
||||
}
|
||||
aconf->copy_mode = MPIPE_COPY_MODE_NONE;
|
||||
if (ConfGetChildValue(if_root, "copy-mode", ©modestr) == 1) {
|
||||
if (aconf->out_iface == NULL) {
|
||||
SCLogInfo("Copy mode activated but no destination"
|
||||
" iface. Disabling feature");
|
||||
} else if (strlen(copymodestr) <= 0) {
|
||||
aconf->out_iface = NULL;
|
||||
} else if (strcmp(copymodestr, "ips") == 0) {
|
||||
SCLogInfo("MPIPE IPS mode activated %s->%s",
|
||||
iface,
|
||||
aconf->out_iface);
|
||||
aconf->copy_mode = MPIPE_COPY_MODE_IPS;
|
||||
} else if (strcmp(copymodestr, "tap") == 0) {
|
||||
SCLogInfo("MPIPE TAP mode activated %s->%s",
|
||||
iface,
|
||||
aconf->out_iface);
|
||||
aconf->copy_mode = MPIPE_COPY_MODE_TAP;
|
||||
} else {
|
||||
SCLogInfo("Invalid mode (no in tap, ips)");
|
||||
}
|
||||
}
|
||||
return aconf;
|
||||
}
|
||||
|
||||
/**
|
||||
* \brief RunModeTileMpipeWorkers set up to process all modules in each thread.
|
||||
*
|
||||
* \param de_ctx pointer to the Detection Engine
|
||||
* \param iface pointer to the name of the interface from which we will
|
||||
* fetch the packets
|
||||
* \retval 0 if all goes well. (If any problem is detected the engine will
|
||||
* exit())
|
||||
*/
|
||||
int RunModeTileMpipeWorkers(DetectEngineCtx *de_ctx)
|
||||
{
|
||||
SCEnter();
|
||||
char tname[32];
|
||||
char *thread_name;
|
||||
TmModule *tm_module;
|
||||
int pipe;
|
||||
|
||||
RunModeInitialize();
|
||||
|
||||
/* Available cpus */
|
||||
uint16_t ncpus = UtilCpuGetNumProcessorsOnline();
|
||||
|
||||
TimeModeSetLive();
|
||||
|
||||
unsigned int pipe_max = 1;
|
||||
if (ncpus > 1)
|
||||
pipe_max = ncpus - 1;
|
||||
|
||||
intmax_t threads;
|
||||
|
||||
if (ConfGetInt("mpipe.threads", &threads) == 1) {
|
||||
tile_num_pipelines = threads;
|
||||
} else {
|
||||
tile_num_pipelines = pipe_max;
|
||||
}
|
||||
SCLogInfo("%d Tilera worker threads", tile_num_pipelines);
|
||||
|
||||
ReceiveMpipeInit();
|
||||
|
||||
char *mpipe_dev = NULL;
|
||||
int nlive = LiveGetDeviceCount();
|
||||
if (nlive > 0) {
|
||||
SCLogInfo("Using %d live device(s).", nlive);
|
||||
/*mpipe_dev = LiveGetDevice(0);*/
|
||||
} else {
|
||||
/*
|
||||
* Attempt to get interface from config file
|
||||
* overrides -i from command line.
|
||||
*/
|
||||
if (ConfGet("mpipe.interface", &mpipe_dev) == 0) {
|
||||
if (ConfGet("mpipe.single_mpipe_dev", &mpipe_dev) == 0) {
|
||||
SCLogError(SC_ERR_RUNMODE, "Failed retrieving "
|
||||
"mpipe.single_mpipe_dev from Conf");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
for (pipe = 0; pipe < tile_num_pipelines; pipe++) {
|
||||
char *mpipe_devc;
|
||||
|
||||
if (nlive > 0) {
|
||||
mpipe_devc = SCStrdup("multi");
|
||||
} else {
|
||||
mpipe_devc = SCStrdup(mpipe_dev);
|
||||
}
|
||||
|
||||
snprintf(tname, sizeof(tname), "Worker%d", pipe+1);
|
||||
thread_name = SCStrdup(tname);
|
||||
|
||||
/* create the threads */
|
||||
ThreadVars *tv_worker =
|
||||
TmThreadCreatePacketHandler(thread_name,
|
||||
"packetpool", "packetpool",
|
||||
"packetpool", "packetpool",
|
||||
"pktacqloop");
|
||||
if (tv_worker == NULL) {
|
||||
printf("ERROR: TmThreadsCreate failed\n");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
tm_module = TmModuleGetByName("ReceiveMpipe");
|
||||
if (tm_module == NULL) {
|
||||
printf("ERROR: TmModuleGetByName failed for ReceiveMpipe\n");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
TmSlotSetFuncAppend(tv_worker, tm_module, (void *)mpipe_devc);
|
||||
|
||||
/* set affinity for worker */
|
||||
int pipe_cpu = pipe + 1;
|
||||
TmThreadSetCPUAffinity(tv_worker, pipe_cpu);
|
||||
|
||||
tm_module = TmModuleGetByName("DecodeMpipe");
|
||||
if (tm_module == NULL) {
|
||||
printf("ERROR: TmModuleGetByName DecodeMpipe failed\n");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
TmSlotSetFuncAppend(tv_worker, tm_module, NULL);
|
||||
|
||||
tm_module = TmModuleGetByName("StreamTcp");
|
||||
if (tm_module == NULL) {
|
||||
printf("ERROR: TmModuleGetByName StreamTcp failed\n");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
TmSlotSetFuncAppend(tv_worker, tm_module, NULL);
|
||||
|
||||
tm_module = TmModuleGetByName("Detect");
|
||||
if (tm_module == NULL) {
|
||||
printf("ERROR: TmModuleGetByName Detect failed\n");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
TmSlotSetFuncAppend(tv_worker, tm_module, (void *)de_ctx);
|
||||
|
||||
tm_module = TmModuleGetByName("RespondReject");
|
||||
if (tm_module == NULL) {
|
||||
printf("ERROR: TmModuleGetByName for RespondReject failed\n");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
TmSlotSetFuncAppend(tv_worker, tm_module, NULL);
|
||||
|
||||
SetupOutputs(tv_worker);
|
||||
|
||||
if (TmThreadSpawn(tv_worker) != TM_ECODE_OK) {
|
||||
printf("ERROR: TmThreadSpawn failed\n");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
#endif
|
@ -0,0 +1,41 @@
|
||||
/* Copyright (C) 2011-2013 Open Information Security Foundation
|
||||
*
|
||||
* You can copy, redistribute or modify this Program under the terms of
|
||||
* the GNU General Public License version 2 as published by the Free
|
||||
* Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* version 2 along with this program; if not, write to the Free Software
|
||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
||||
* 02110-1301, USA.
|
||||
*/
|
||||
|
||||
/**
|
||||
* \file
|
||||
*
|
||||
* \author Tom DeCanio <decanio.tom@gmail.com>
|
||||
* \author Ken Steele, Tilera Corporation <suricata@tilera.com>
|
||||
*
|
||||
* Tilera TILE-Gx runmode support
|
||||
*/
|
||||
|
||||
#ifndef __RUNMODE_TILE_H__
|
||||
#define __RUNMODE_TILE_H__
|
||||
|
||||
#include "suricata-common.h"
|
||||
|
||||
const char *RunModeTileMpipeGetDefaultMode(void);
|
||||
void RunModeTileMpipeRegister(void);
|
||||
|
||||
extern unsigned int tile_num_pipelines;
|
||||
|
||||
int RunModeTileMpipeWorkers(DetectEngineCtx *);
|
||||
|
||||
void *ParseMpipeConfig(const char *iface);
|
||||
|
||||
#endif /* __RUNMODE_TILE_H__ */
|
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,93 @@
|
||||
/* Copyright (C) 2011-2013 Open Information Security Foundation
|
||||
*
|
||||
* You can copy, redistribute or modify this Program under the terms of
|
||||
* the GNU General Public License version 2 as published by the Free
|
||||
* Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* version 2 along with this program; if not, write to the Free Software
|
||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
||||
* 02110-1301, USA.
|
||||
*/
|
||||
|
||||
/**
|
||||
* \file
|
||||
*
|
||||
* \author Tom DeCanio <decanio.tom@gmail.com>
|
||||
* \author Ken Steele, Tilera Corporation <suricata@tilera.com>
|
||||
*/
|
||||
|
||||
#ifndef __SOURCE_MPIPE_H__
|
||||
#define __SOURCE_MPIPE_H__
|
||||
|
||||
#ifdef HAVE_MPIPE
|
||||
|
||||
#include <gxio/mpipe.h>
|
||||
#include <tmc/cpus.h>
|
||||
|
||||
#define MPIPE_FREE_PACKET(p) MpipeFreePacket((p))
|
||||
|
||||
#define MPIPE_COPY_MODE_NONE 0
|
||||
#define MPIPE_COPY_MODE_TAP 1
|
||||
#define MPIPE_COPY_MODE_IPS 2
|
||||
|
||||
#define MPIPE_IFACE_NAME_LENGTH 8
|
||||
|
||||
typedef struct MpipeIfaceConfig_
|
||||
{
|
||||
char iface[MPIPE_IFACE_NAME_LENGTH];
|
||||
int copy_mode;
|
||||
char *out_iface;
|
||||
} MpipeIfaceConfig;
|
||||
|
||||
typedef struct MpipePeer_
|
||||
{
|
||||
int channel;
|
||||
char iface[MPIPE_IFACE_NAME_LENGTH];
|
||||
} MpipePeer;
|
||||
|
||||
/* per interface TAP/IPS configuration */
|
||||
typedef struct MpipePeerVars_
|
||||
{
|
||||
gxio_mpipe_equeue_t *peer_equeue;
|
||||
void (*ReleasePacket)(struct Packet_ *);
|
||||
int copy_mode;
|
||||
} MpipePeerVars;
|
||||
|
||||
/* per packet Mpipe vars */
|
||||
typedef struct MpipePacketVars_
|
||||
{
|
||||
/* TileGX mpipe stuff */
|
||||
struct {
|
||||
uint_reg_t channel : 5;
|
||||
uint_reg_t l2_size : 14;
|
||||
uint_reg_t size : 3;
|
||||
uint_reg_t bucket_id : 13;
|
||||
uint_reg_t nr : 1;
|
||||
uint_reg_t cs : 1;
|
||||
uint_reg_t va : 42;
|
||||
uint_reg_t stack_idx : 5;
|
||||
} idesc;
|
||||
|
||||
/* packetpool this was allocated from */
|
||||
uint8_t rank;
|
||||
|
||||
gxio_mpipe_equeue_t *peer_equeue;
|
||||
} MpipePacketVars;
|
||||
|
||||
int MpipeLiveRegisterDevice(char *);
|
||||
int MpipeLiveGetDeviceCount(void);
|
||||
char *MpipeLiveGetDevice(int);
|
||||
void MpipeFreePacket(void *arg);
|
||||
void TmModuleReceiveMpipeRegister (void);
|
||||
void TmModuleDecodeMpipeRegister (void);
|
||||
|
||||
TmEcode ReceiveMpipeInit(void);
|
||||
|
||||
#endif /* HAVE_MPIPE */
|
||||
#endif /* __SOURCE_MPIPE_H__ */
|
Loading…
Reference in New Issue