aboutsummaryrefslogtreecommitdiff
path: root/src/core/nginx.c
diff options
context:
space:
mode:
authorMaxim Dounin <mdounin@mdounin.ru>2016-01-11 19:23:17 +0300
committerMaxim Dounin <mdounin@mdounin.ru>2016-01-11 19:23:17 +0300
commitd0bf684ab6643edc18ab65e2c40bf88862ead8e5 (patch)
tree44aed9a9d8e61216267e62b2b3a5e80664641ad3 /src/core/nginx.c
parentaf647a3da2eb9ff558655424868f82f259de94f1 (diff)
downloadnginx-d0bf684ab6643edc18ab65e2c40bf88862ead8e5.tar.gz
nginx-d0bf684ab6643edc18ab65e2c40bf88862ead8e5.zip
Core: worker_cpu_affinity auto.
If enabled, workers are bound to available CPUs, each worker to once CPU in order. If there are more workers than available CPUs, remaining are bound in a loop, starting again from the first available CPU. The optional mask parameter defines which CPUs are available for automatic binding. In collaboration with Vladimir Homutov.
Diffstat (limited to 'src/core/nginx.c')
-rw-r--r--src/core/nginx.c43
1 files changed, 41 insertions, 2 deletions
diff --git a/src/core/nginx.c b/src/core/nginx.c
index 33355870f..64db3818e 100644
--- a/src/core/nginx.c
+++ b/src/core/nginx.c
@@ -961,6 +961,7 @@ ngx_core_module_create_conf(ngx_cycle_t *cycle)
* ccf->pid = NULL;
* ccf->oldpid = NULL;
* ccf->priority = 0;
+ * ccf->cpu_affinity_auto = 0;
* ccf->cpu_affinity_n = 0;
* ccf->cpu_affinity = NULL;
*/
@@ -1002,7 +1003,8 @@ ngx_core_module_init_conf(ngx_cycle_t *cycle, void *conf)
#if (NGX_HAVE_CPU_AFFINITY)
- if (ccf->cpu_affinity_n
+ if (!ccf->cpu_affinity_auto
+ && ccf->cpu_affinity_n
&& ccf->cpu_affinity_n != 1
&& ccf->cpu_affinity_n != (ngx_uint_t) ccf->worker_processes)
{
@@ -1273,7 +1275,24 @@ ngx_set_cpu_affinity(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
value = cf->args->elts;
- for (n = 1; n < cf->args->nelts; n++) {
+ if (ngx_strcmp(value[1].data, "auto") == 0) {
+
+ if (cf->args->nelts > 3) {
+ ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
+ "invalid number of arguments in "
+ "\"worker_cpu_affinity\" directive");
+ return NGX_CONF_ERROR;
+ }
+
+ ccf->cpu_affinity_auto = 1;
+ mask[0] = (uint64_t) -1 >> (64 - ngx_min(64, ngx_ncpu));
+ n = 2;
+
+ } else {
+ n = 1;
+ }
+
+ for ( /* void */ ; n < cf->args->nelts; n++) {
if (value[n].len > 64) {
ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
@@ -1323,6 +1342,8 @@ ngx_set_cpu_affinity(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
uint64_t
ngx_get_cpu_affinity(ngx_uint_t n)
{
+ uint64_t mask;
+ ngx_uint_t i;
ngx_core_conf_t *ccf;
ccf = (ngx_core_conf_t *) ngx_get_conf(ngx_cycle->conf_ctx,
@@ -1332,6 +1353,24 @@ ngx_get_cpu_affinity(ngx_uint_t n)
return 0;
}
+ if (ccf->cpu_affinity_auto) {
+ mask = ccf->cpu_affinity[ccf->cpu_affinity_n - 1];
+
+ if (mask == 0) {
+ return 0;
+ }
+
+ for (i = 0; /* void */ ; i++) {
+ if ((mask & ((uint64_t) 1 << (i % 64))) && n-- == 0) {
+ break;
+ }
+
+ /* void */
+ }
+
+ return (uint64_t) 1 << (i % 64);
+ }
+
if (ccf->cpu_affinity_n > n) {
return ccf->cpu_affinity[n];
}