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
Eric Leblond 14 years ago committed by Victor Julien
parent 559b5db7df
commit 37ee483b75

@ -185,7 +185,7 @@ util-optimize.h \
util-privs.c util-privs.h \ util-privs.c util-privs.h \
util-decode-asn1.c util-decode-asn1.h \ util-decode-asn1.c util-decode-asn1.h \
util-ringbuffer.c util-ringbuffer.h \ util-ringbuffer.c util-ringbuffer.h \
util-validate.h \ util-validate.h util-affinity.h util-affinity.c \
util-memcmp.c util-memcmp.h \ util-memcmp.c util-memcmp.h \
util-proto-name.c util-proto-name.h \ util-proto-name.c util-proto-name.h \
tm-modules.c tm-modules.h \ tm-modules.c tm-modules.h \

@ -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…
Cancel
Save