mirror of https://github.com/OISF/suricata
Add affinity util function and related files
This patch adds two new files which implement advanced affinity settings. Signed-off-by: Eric Leblond <eric@regit.org>remotes/origin/master-1.1.x
parent
559b5db7df
commit
37ee483b75
@ -0,0 +1,243 @@
|
||||
/* Copyright (C) 2010 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 Eric Leblond <eric@regit.org>
|
||||
*
|
||||
* CPU affinity related code and helper.
|
||||
*/
|
||||
|
||||
#include "suricata-common.h"
|
||||
#define _THREAD_AFFINITY
|
||||
#include "util-affinity.h"
|
||||
#include "util-cpu.h"
|
||||
#include "conf.h"
|
||||
#include "threads.h"
|
||||
#include "queue.h"
|
||||
#include "runmodes.h"
|
||||
|
||||
ThreadsAffinityType thread_affinity[MAX_CPU_SET] = {
|
||||
{
|
||||
.name = "receive_cpu_set",
|
||||
.mode_flag = EXCLUSIVE_AFFINITY,
|
||||
.prio = PRIO_MEDIUM,
|
||||
.lcpu = 0,
|
||||
},
|
||||
{
|
||||
.name = "decode_cpu_set",
|
||||
.mode_flag = BALANCED_AFFINITY,
|
||||
.prio = PRIO_MEDIUM,
|
||||
.lcpu = 0,
|
||||
},
|
||||
{
|
||||
.name = "stream_cpu_set",
|
||||
.mode_flag = BALANCED_AFFINITY,
|
||||
.prio = PRIO_MEDIUM,
|
||||
.lcpu = 0,
|
||||
},
|
||||
{
|
||||
.name = "detect_cpu_set",
|
||||
.mode_flag = EXCLUSIVE_AFFINITY,
|
||||
.prio = PRIO_MEDIUM,
|
||||
.lcpu = 0,
|
||||
},
|
||||
{
|
||||
.name = "verdict_cpu_set",
|
||||
.mode_flag = BALANCED_AFFINITY,
|
||||
.prio = PRIO_MEDIUM,
|
||||
.lcpu = 0,
|
||||
},
|
||||
{
|
||||
.name = "reject_cpu_set",
|
||||
.mode_flag = BALANCED_AFFINITY,
|
||||
.prio = PRIO_MEDIUM,
|
||||
.lcpu = 0,
|
||||
},
|
||||
{
|
||||
.name = "output_cpu_set",
|
||||
.mode_flag = BALANCED_AFFINITY,
|
||||
.prio = PRIO_MEDIUM,
|
||||
.lcpu = 0,
|
||||
},
|
||||
};
|
||||
|
||||
/**
|
||||
* \brief find affinity by its name
|
||||
* \retval a pointer to the affinity or NULL if not found
|
||||
*/
|
||||
ThreadsAffinityType * GetAffinityTypeFromName(const char *name) {
|
||||
int i;
|
||||
for (i = 0; i < MAX_CPU_SET; i++) {
|
||||
if (!strcmp(thread_affinity[i].name, name)) {
|
||||
return &thread_affinity[i];
|
||||
}
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static void AffinitySetupInit()
|
||||
{
|
||||
int i, j;
|
||||
int ncpu = UtilCpuGetNumProcessorsConfigured();
|
||||
|
||||
SCLogDebug("Initialize affinity setup\n");
|
||||
/* be conservative relatively to OS: use all cpus by default */
|
||||
for (i = 0; i < MAX_CPU_SET; i++) {
|
||||
cpu_set_t *cs = &thread_affinity[i].cpu_set;
|
||||
CPU_ZERO(cs);
|
||||
for (j = 0; j < ncpu; j++) {
|
||||
CPU_SET(j, cs);
|
||||
}
|
||||
}
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
/**
|
||||
* \brief Extract cpu affinity configuration from current config file
|
||||
*/
|
||||
|
||||
void AffinitySetupLoadFromConfig()
|
||||
{
|
||||
ConfNode *root = ConfGetNode("threading.cpu_affinity");
|
||||
ConfNode *affinity;
|
||||
|
||||
AffinitySetupInit();
|
||||
|
||||
SCLogDebug("Load affinity from config\n");
|
||||
if (root == NULL) {
|
||||
SCLogInfo("can't get cpu_affinity node");
|
||||
return;
|
||||
}
|
||||
|
||||
TAILQ_FOREACH(affinity, &root->head, next) {
|
||||
ThreadsAffinityType *taf = GetAffinityTypeFromName(affinity->val);
|
||||
ConfNode *node = NULL;
|
||||
ConfNode *lnode;
|
||||
|
||||
if (taf == NULL) {
|
||||
SCLogError(SC_ERR_INVALID_ARGUMENT, "unknown cpu_affinity type");
|
||||
exit(EXIT_FAILURE);
|
||||
} else {
|
||||
SCLogInfo("Found affinity definition for \"%s\"",
|
||||
affinity->val);
|
||||
}
|
||||
CPU_ZERO(&taf->cpu_set);
|
||||
|
||||
node = ConfNodeLookupChild(affinity->head.tqh_first, "cpu");
|
||||
|
||||
if (node == NULL) {
|
||||
SCLogInfo("unable to find 'cpu'");
|
||||
} else {
|
||||
TAILQ_FOREACH(lnode, &node->head, next) {
|
||||
int i;
|
||||
long int a,b;
|
||||
int stop = 0;
|
||||
if (!strcmp(lnode->val, "all")) {
|
||||
a = 0;
|
||||
b = UtilCpuGetNumProcessorsConfigured();
|
||||
stop = 1;
|
||||
} else if (index(lnode->val, '-') != NULL) {
|
||||
char *sep = index(lnode->val, '-');
|
||||
char *end;
|
||||
a = strtoul(lnode->val, &end, 10);
|
||||
if (end != sep) {
|
||||
SCLogError(SC_ERR_INVALID_ARGUMENT,
|
||||
"invalid cpu range (start invalid): \"%s\"",
|
||||
lnode->val);
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
b = strtol(sep + 1, &end, 10);
|
||||
if (end != sep + strlen(sep)) {
|
||||
SCLogError(SC_ERR_INVALID_ARGUMENT,
|
||||
"invalid cpu range (end invalid): \"%s\"",
|
||||
lnode->val);
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
if (a > b) {
|
||||
SCLogError(SC_ERR_INVALID_ARGUMENT,
|
||||
"invalid cpu range (bad order): \"%s\"",
|
||||
lnode->val);
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
} else {
|
||||
char *end;
|
||||
a = strtoul(lnode->val, &end, 10);
|
||||
if (end != lnode->val + strlen(lnode->val)) {
|
||||
SCLogError(SC_ERR_INVALID_ARGUMENT,
|
||||
"invalid cpu range (not an integer): \"%s\"",
|
||||
lnode->val);
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
b = a;
|
||||
}
|
||||
for (i = a; i<= b; i++) {
|
||||
CPU_SET(i, &taf->cpu_set);
|
||||
}
|
||||
if (stop)
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
node = ConfNodeLookupChild(affinity->head.tqh_first, "mode");
|
||||
if (node != NULL) {
|
||||
if (!strcmp(node->val, "exclusive")) {
|
||||
taf->mode_flag = EXCLUSIVE_AFFINITY;
|
||||
} else if (!strcmp(node->val, "balanced")) {
|
||||
taf->mode_flag = BALANCED_AFFINITY;
|
||||
} else {
|
||||
SCLogError(SC_ERR_INVALID_ARGUMENT, "unknown cpu_affinity node");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
}
|
||||
|
||||
node = ConfNodeLookupChild(affinity->head.tqh_first, "prio");
|
||||
if (node == NULL)
|
||||
continue;
|
||||
if (!strcmp(node->val, "low")) {
|
||||
taf->prio = PRIO_LOW;
|
||||
} else if (!strcmp(node->val, "medium")) {
|
||||
taf->prio = PRIO_MEDIUM;
|
||||
} else if (!strcmp(node->val, "high")) {
|
||||
taf->prio = PRIO_HIGH;
|
||||
} else {
|
||||
SCLogError(SC_ERR_INVALID_ARGUMENT, "unknown cpu_affinity prio");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* \brief Return next cpu to use for a given thread family
|
||||
* \retval the cpu to used given by its id
|
||||
*/
|
||||
int AffinityGetNextCPU(ThreadsAffinityType *taf)
|
||||
{
|
||||
int ncpu = taf->lcpu;
|
||||
while (!CPU_ISSET(ncpu, &taf->cpu_set)) {
|
||||
ncpu++;
|
||||
if (ncpu >= UtilCpuGetNumProcessorsOnline())
|
||||
ncpu = 0;
|
||||
}
|
||||
taf->lcpu = ncpu + 1;
|
||||
if (taf->lcpu >= UtilCpuGetNumProcessorsOnline())
|
||||
taf->lcpu = 0;
|
||||
return ncpu;
|
||||
}
|
@ -0,0 +1,63 @@
|
||||
/* Copyright (C) 2010 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 Eric Leblond <eric@regit.org>
|
||||
*/
|
||||
|
||||
#ifndef __UTIL_AFFINITY_H__
|
||||
#define __UTIL_AFFINITY_H__
|
||||
#include "suricata-common.h"
|
||||
|
||||
enum {
|
||||
RECEIVE_CPU_SET,
|
||||
DECODE_CPU_SET,
|
||||
STREAM_CPU_SET,
|
||||
DETECT_CPU_SET,
|
||||
VERDICT_CPU_SET,
|
||||
REJECT_CPU_SET,
|
||||
OUTPUT_CPU_SET,
|
||||
MAX_CPU_SET
|
||||
};
|
||||
|
||||
enum {
|
||||
BALANCED_AFFINITY,
|
||||
EXCLUSIVE_AFFINITY,
|
||||
MAX_AFFINITY
|
||||
};
|
||||
|
||||
typedef struct ThreadsAffinityType_ {
|
||||
char *name;
|
||||
cpu_set_t cpu_set;
|
||||
uint8_t mode_flag;
|
||||
uint8_t prio;
|
||||
uint16_t lcpu; /* use by exclusive mode */
|
||||
} ThreadsAffinityType;
|
||||
|
||||
/** store thread affinity mode for all type of threads */
|
||||
#ifndef _THREAD_AFFINITY
|
||||
ThreadsAffinityType thread_affinity[MAX_CPU_SET];
|
||||
#endif
|
||||
|
||||
void AffinitySetupLoadFromConfig();
|
||||
ThreadsAffinityType * GetAffinityTypeFromName(const char *name);
|
||||
|
||||
int AffinityGetNextCPU(ThreadsAffinityType *taf);
|
||||
|
||||
#endif /* __UTIL_AFFINITY_H__ */
|
Loading…
Reference in New Issue