Axe several perl interpreter instances: they may be useful in currently

unsupported threaded environment, but now they complicate code:
*) perl_clone() requires at least duplicating nginx stash;
*) the multiplicity requires to re-evalute all precompiled subroutines
   and nginx stash in new interpreter context.
This commit is contained in:
Igor Sysoev 2006-11-26 14:35:27 +00:00
parent bdd1a85885
commit 97fd87d7ff
3 changed files with 14 additions and 176 deletions

View file

@ -36,11 +36,7 @@ if test -n "$NGX_PERL_VER"; then
echo " + perl interpreter multiplicity found" echo " + perl interpreter multiplicity found"
fi fi
if $NGX_PERL -V:useithreads | grep define > /dev/null; then if $NGX_PERL -V:useithreads | grep undef > /dev/null; then
have=NGX_HAVE_PERL_CLONE . auto/have
echo " + perl_clone() found"
else
# FreeBSD port wants to link with -pthread non-threaded perl # FreeBSD port wants to link with -pthread non-threaded perl
ngx_perl_ldopts=`echo $ngx_perl_ldopts | sed 's/ -pthread//'` ngx_perl_ldopts=`echo $ngx_perl_ldopts | sed 's/ -pthread//'`
fi fi

View file

@ -11,11 +11,6 @@
typedef struct { typedef struct {
PerlInterpreter **free_perls;
ngx_uint_t interp;
ngx_uint_t nalloc;
ngx_uint_t interp_max;
PerlInterpreter *perl; PerlInterpreter *perl;
ngx_str_t modules; ngx_str_t modules;
ngx_array_t requires; ngx_array_t requires;
@ -45,12 +40,6 @@ static ngx_int_t ngx_http_perl_ssi(ngx_http_request_t *r,
ngx_http_ssi_ctx_t *ssi_ctx, ngx_str_t **params); ngx_http_ssi_ctx_t *ssi_ctx, ngx_str_t **params);
#endif #endif
static ngx_int_t
ngx_http_perl_get_interpreter(ngx_http_perl_main_conf_t *pmcf,
PerlInterpreter **perl, ngx_log_t *log);
static ngx_inline void
ngx_http_perl_free_interpreter(ngx_http_perl_main_conf_t *pmcf,
PerlInterpreter *perl);
static char *ngx_http_perl_init_interpreter(ngx_conf_t *cf, static char *ngx_http_perl_init_interpreter(ngx_conf_t *cf,
ngx_http_perl_main_conf_t *pmcf); ngx_http_perl_main_conf_t *pmcf);
static PerlInterpreter * static PerlInterpreter *
@ -72,16 +61,11 @@ static char *ngx_http_perl_require(ngx_conf_t *cf, ngx_command_t *cmd,
void *conf); void *conf);
static char *ngx_http_perl(ngx_conf_t *cf, ngx_command_t *cmd, void *conf); static char *ngx_http_perl(ngx_conf_t *cf, ngx_command_t *cmd, void *conf);
static char *ngx_http_perl_set(ngx_conf_t *cf, ngx_command_t *cmd, void *conf); static char *ngx_http_perl_set(ngx_conf_t *cf, ngx_command_t *cmd, void *conf);
static char *ngx_http_perl_interp_max_unsupported(ngx_conf_t *cf, void *post,
void *data);
#if (NGX_HAVE_PERL_CLONE || NGX_HAVE_PERL_MULTIPLICITY)
static void ngx_http_perl_cleanup_perl(void *data);
#endif
static void ngx_http_perl_cleanup_sv(void *data); static void ngx_http_perl_cleanup_sv(void *data);
#if (NGX_HAVE_PERL_MULTIPLICITY)
static ngx_conf_post_handler_pt ngx_http_perl_interp_max_p = static void ngx_http_perl_cleanup_perl(void *data);
ngx_http_perl_interp_max_unsupported; #endif
static ngx_command_t ngx_http_perl_commands[] = { static ngx_command_t ngx_http_perl_commands[] = {
@ -100,13 +84,6 @@ static ngx_command_t ngx_http_perl_commands[] = {
0, 0,
NULL }, NULL },
{ ngx_string("perl_interp_max"),
NGX_HTTP_MAIN_CONF|NGX_CONF_TAKE1,
ngx_conf_set_num_slot,
NGX_HTTP_MAIN_CONF_OFFSET,
offsetof(ngx_http_perl_main_conf_t, interp_max),
&ngx_http_perl_interp_max_p },
{ ngx_string("perl"), { ngx_string("perl"),
NGX_HTTP_LOC_CONF|NGX_CONF_TAKE1, NGX_HTTP_LOC_CONF|NGX_CONF_TAKE1,
ngx_http_perl, ngx_http_perl,
@ -229,16 +206,9 @@ ngx_http_perl_handle_request(ngx_http_request_t *r)
pmcf = ngx_http_get_module_main_conf(r, ngx_http_perl_module); pmcf = ngx_http_get_module_main_conf(r, ngx_http_perl_module);
rc = ngx_http_perl_get_interpreter(pmcf, &ctx->perl, r->connection->log);
if (rc != NGX_OK) {
ngx_http_finalize_request(r, rc);
return;
}
{ {
dTHXa(ctx->perl); dTHXa(pmcf->perl);
if (ctx->next == NULL) { if (ctx->next == NULL) {
plcf = ngx_http_get_module_loc_conf(r, ngx_http_perl_module); plcf = ngx_http_get_module_loc_conf(r, ngx_http_perl_module);
@ -255,8 +225,6 @@ ngx_http_perl_handle_request(ngx_http_request_t *r)
} }
ngx_http_perl_free_interpreter(pmcf, ctx->perl);
if (rc > 600) { if (rc > 600) {
rc = NGX_OK; rc = NGX_OK;
} }
@ -300,7 +268,6 @@ ngx_http_perl_variable(ngx_http_request_t *r, ngx_http_variable_value_t *v,
ngx_http_perl_variable_t *pv = (ngx_http_perl_variable_t *) data; ngx_http_perl_variable_t *pv = (ngx_http_perl_variable_t *) data;
ngx_int_t rc; ngx_int_t rc;
ngx_uint_t recursive;
ngx_str_t value; ngx_str_t value;
ngx_http_perl_ctx_t *ctx; ngx_http_perl_ctx_t *ctx;
ngx_http_perl_main_conf_t *pmcf; ngx_http_perl_main_conf_t *pmcf;
@ -317,38 +284,21 @@ ngx_http_perl_variable(ngx_http_request_t *r, ngx_http_variable_value_t *v,
} }
ngx_http_set_ctx(r, ctx, ngx_http_perl_module); ngx_http_set_ctx(r, ctx, ngx_http_perl_module);
pmcf = ngx_http_get_module_main_conf(r, ngx_http_perl_module);
rc = ngx_http_perl_get_interpreter(pmcf, &ctx->perl,
r->connection->log);
if (rc != NGX_OK) {
return rc;
}
recursive = 0;
} else {
pmcf = NULL;
recursive = 1;
} }
pmcf = ngx_http_get_module_main_conf(r, ngx_http_perl_module);
value.data = NULL; value.data = NULL;
{ {
dTHXa(ctx->perl); dTHXa(pmcf->perl);
rc = ngx_http_perl_call_handler(aTHX_ r, pv->sub, NULL, rc = ngx_http_perl_call_handler(aTHX_ r, pv->sub, NULL,
&pv->handler, &value); &pv->handler, &value);
} }
if (recursive == 0) {
ngx_http_perl_free_interpreter(pmcf, ctx->perl);
}
if (value.data) { if (value.data) {
v->len = value.len; v->len = value.len;
v->valid = 1; v->valid = 1;
@ -385,8 +335,6 @@ ngx_http_perl_ssi(ngx_http_request_t *r, ngx_http_ssi_ctx_t *ssi_ctx,
ngx_log_debug0(NGX_LOG_DEBUG_HTTP, r->connection->log, 0, ngx_log_debug0(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
"perl ssi handler"); "perl ssi handler");
pmcf = ngx_http_get_module_main_conf(r, ngx_http_perl_module);
ctx = ngx_http_get_module_ctx(r, ngx_http_perl_module); ctx = ngx_http_get_module_ctx(r, ngx_http_perl_module);
if (ctx == NULL) { if (ctx == NULL) {
@ -398,11 +346,7 @@ ngx_http_perl_ssi(ngx_http_request_t *r, ngx_http_ssi_ctx_t *ssi_ctx,
ngx_http_set_ctx(r, ctx, ngx_http_perl_module); ngx_http_set_ctx(r, ctx, ngx_http_perl_module);
} }
rc = ngx_http_perl_get_interpreter(pmcf, &ctx->perl, r->connection->log); pmcf = ngx_http_get_module_main_conf(r, ngx_http_perl_module);
if (rc != NGX_OK) {
return rc;
}
ctx->ssi = ssi_ctx; ctx->ssi = ssi_ctx;
@ -411,7 +355,7 @@ ngx_http_perl_ssi(ngx_http_request_t *r, ngx_http_ssi_ctx_t *ssi_ctx,
{ {
dTHXa(ctx->perl); dTHXa(pmcf->perl);
#if 0 #if 0
@ -440,8 +384,6 @@ ngx_http_perl_ssi(ngx_http_request_t *r, ngx_http_ssi_ctx_t *ssi_ctx,
} }
ngx_http_perl_free_interpreter(pmcf, ctx->perl);
ctx->filename.data = NULL; ctx->filename.data = NULL;
ctx->redirect_uri.len = 0; ctx->redirect_uri.len = 0;
ctx->ssi = NULL; ctx->ssi = NULL;
@ -454,46 +396,10 @@ ngx_http_perl_ssi(ngx_http_request_t *r, ngx_http_ssi_ctx_t *ssi_ctx,
#endif #endif
static ngx_int_t
ngx_http_perl_get_interpreter(ngx_http_perl_main_conf_t *pmcf,
PerlInterpreter **perl, ngx_log_t *log)
{
if (pmcf->interp) {
pmcf->interp--;
*perl = pmcf->free_perls[pmcf->interp];
return NGX_OK;
}
if (pmcf->nalloc < pmcf->interp_max) {
*perl = ngx_http_perl_create_interpreter(pmcf, log);
if (*perl) {
return NGX_OK;
}
return NGX_HTTP_INTERNAL_SERVER_ERROR;
}
ngx_log_error(NGX_LOG_ALERT, log, 0, "no free perl interpreter");
return NGX_HTTP_SERVICE_UNAVAILABLE;
}
static ngx_inline void
ngx_http_perl_free_interpreter(ngx_http_perl_main_conf_t *pmcf,
PerlInterpreter *perl)
{
pmcf->free_perls[pmcf->interp++] = perl;
}
static char * static char *
ngx_http_perl_init_interpreter(ngx_conf_t *cf, ngx_http_perl_main_conf_t *pmcf) ngx_http_perl_init_interpreter(ngx_conf_t *cf, ngx_http_perl_main_conf_t *pmcf)
{ {
#if (NGX_HAVE_PERL_CLONE || NGX_HAVE_PERL_MULTIPLICITY) #if (NGX_HAVE_PERL_MULTIPLICITY)
ngx_pool_cleanup_t *cln; ngx_pool_cleanup_t *cln;
cln = ngx_pool_cleanup_add(cf->pool, 0); cln = ngx_pool_cleanup_add(cf->pool, 0);
@ -517,7 +423,7 @@ ngx_http_perl_init_interpreter(ngx_conf_t *cf, ngx_http_perl_main_conf_t *pmcf)
} }
} }
#if !(NGX_HAVE_PERL_CLONE || NGX_HAVE_PERL_MULTIPLICITY) #if !(NGX_HAVE_PERL_MULTIPLICITY)
if (perl) { if (perl) {
if (ngx_http_perl_run_requires(aTHX_ &pmcf->requires, cf->log) if (ngx_http_perl_run_requires(aTHX_ &pmcf->requires, cf->log)
@ -542,7 +448,7 @@ ngx_http_perl_init_interpreter(ngx_conf_t *cf, ngx_http_perl_main_conf_t *pmcf)
return NGX_CONF_ERROR; return NGX_CONF_ERROR;
} }
#if (NGX_HAVE_PERL_CLONE || NGX_HAVE_PERL_MULTIPLICITY) #if (NGX_HAVE_PERL_MULTIPLICITY)
cln->handler = ngx_http_perl_cleanup_perl; cln->handler = ngx_http_perl_cleanup_perl;
cln->data = pmcf->perl; cln->data = pmcf->perl;
@ -569,32 +475,6 @@ ngx_http_perl_create_interpreter(ngx_http_perl_main_conf_t *pmcf,
ngx_log_debug0(NGX_LOG_DEBUG_HTTP, log, 0, "create perl interpreter"); ngx_log_debug0(NGX_LOG_DEBUG_HTTP, log, 0, "create perl interpreter");
#if (NGX_HAVE_PERL_CLONE)
if (pmcf->perl) {
perl = perl_clone(pmcf->perl, CLONEf_KEEP_PTR_TABLE);
if (perl == NULL) {
ngx_log_error(NGX_LOG_ALERT, log, 0, "perl_clone() failed");
return NULL;
}
{
dTHXa(perl);
ptr_table_free(PL_ptr_table);
PL_ptr_table = NULL;
}
pmcf->nalloc++;
return perl;
}
#endif
perl = perl_alloc(); perl = perl_alloc();
if (perl == NULL) { if (perl == NULL) {
ngx_log_error(NGX_LOG_ALERT, log, 0, "perl_alloc() failed"); ngx_log_error(NGX_LOG_ALERT, log, 0, "perl_alloc() failed");
@ -649,8 +529,6 @@ ngx_http_perl_create_interpreter(ngx_http_perl_main_conf_t *pmcf,
} }
pmcf->nalloc++;
return perl; return perl;
fail: fail:
@ -824,8 +702,6 @@ ngx_http_perl_create_main_conf(ngx_conf_t *cf)
return NGX_CONF_ERROR; return NGX_CONF_ERROR;
} }
pmcf->interp_max = NGX_CONF_UNSET_UINT;
if (ngx_array_init(&pmcf->requires, cf->pool, 1, sizeof(u_char *)) if (ngx_array_init(&pmcf->requires, cf->pool, 1, sizeof(u_char *))
!= NGX_OK) != NGX_OK)
{ {
@ -841,33 +717,17 @@ ngx_http_perl_init_main_conf(ngx_conf_t *cf, void *conf)
{ {
ngx_http_perl_main_conf_t *pmcf = conf; ngx_http_perl_main_conf_t *pmcf = conf;
#if (NGX_HAVE_PERL_CLONE || NGX_HAVE_PERL_MULTIPLICITY)
ngx_conf_init_uint_value(pmcf->interp_max, 10);
#else
ngx_conf_init_uint_value(pmcf->interp_max, 1);
#endif
pmcf->free_perls = ngx_pcalloc(cf->pool,
pmcf->interp_max * sizeof(PerlInterpreter *));
if (pmcf->free_perls == NULL) {
return NGX_CONF_ERROR;
}
if (pmcf->perl == NULL) { if (pmcf->perl == NULL) {
if (ngx_http_perl_init_interpreter(cf, pmcf) != NGX_CONF_OK) { if (ngx_http_perl_init_interpreter(cf, pmcf) != NGX_CONF_OK) {
return NGX_CONF_ERROR; return NGX_CONF_ERROR;
} }
} }
#if !(NGX_HAVE_PERL_CLONE)
ngx_http_perl_free_interpreter(pmcf, pmcf->perl);
#endif
return NGX_CONF_OK; return NGX_CONF_OK;
} }
#if (NGX_HAVE_PERL_CLONE || NGX_HAVE_PERL_MULTIPLICITY) #if (NGX_HAVE_PERL_MULTIPLICITY)
static void static void
ngx_http_perl_cleanup_perl(void *data) ngx_http_perl_cleanup_perl(void *data)
@ -1123,19 +983,3 @@ ngx_http_perl_set(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
return NGX_CONF_OK; return NGX_CONF_OK;
} }
static char *
ngx_http_perl_interp_max_unsupported(ngx_conf_t *cf, void *post, void *data)
{
#if (NGX_HAVE_PERL_CLONE || NGX_HAVE_PERL_MULTIPLICITY)
return NGX_CONF_OK;
#else
return "to use perl_interp_max you have to build perl with "
"-Dusemultiplicity or -Dusethreads options";
#endif
}

View file

@ -20,8 +20,6 @@
typedef ngx_http_request_t *nginx; typedef ngx_http_request_t *nginx;
typedef struct { typedef struct {
PerlInterpreter *perl;
ngx_str_t filename; ngx_str_t filename;
ngx_str_t redirect_uri; ngx_str_t redirect_uri;
ngx_str_t redirect_args; ngx_str_t redirect_args;