| /* Copyright 2015 greenbytes GmbH (https://www.greenbytes.de) |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #include <apr_optional.h> |
| #include <apr_optional_hooks.h> |
| #include <apr_want.h> |
| |
| #include <httpd.h> |
| #include <http_log.h> |
| |
| #include "mod_h2.h" |
| |
| #include <nghttp2/nghttp2.h> |
| #include "h2_stream.h" |
| #include "h2_alt_svc.h" |
| #include "h2_conn.h" |
| #include "h2_task.h" |
| #include "h2_session.h" |
| #include "h2_config.h" |
| #include "h2_ctx.h" |
| #include "h2_h2.h" |
| #include "h2_switch.h" |
| #include "h2_version.h" |
| |
| |
| static void h2_hooks(apr_pool_t *pool); |
| |
| AP_DECLARE_MODULE(http2) = { |
| STANDARD20_MODULE_STUFF, |
| NULL, |
| NULL, |
| h2_config_create_svr, /* func to create per server config */ |
| h2_config_merge, /* func to merge per server config */ |
| h2_cmds, /* command handlers */ |
| h2_hooks |
| }; |
| |
| /* The module initialization. Called once as apache hook, before any multi |
| * processing (threaded or not) happens. It is typically at least called twice, |
| * see |
| * http://wiki.apache.org/httpd/ModuleLife |
| * Since the first run is just a "practise" run, we want to initialize for real |
| * only on the second try. This defeats the purpose of the first dry run a bit, |
| * since apache wants to verify that a new configuration actually will work. |
| * So if we have trouble with the configuration, this will only be detected |
| * when the server has already switched. |
| * On the other hand, when we initialize lib nghttp2, all possible crazy things |
| * might happen and this might even eat threads. So, better init on the real |
| * invocation, for now at least. |
| */ |
| static int h2_post_config(apr_pool_t *p, apr_pool_t *plog, |
| apr_pool_t *ptemp, server_rec *s) |
| { |
| void *data = NULL; |
| const char *mod_h2_init_key = "mod_h2_init_counter"; |
| nghttp2_info *ngh2; |
| apr_status_t status; |
| (void)plog;(void)ptemp; |
| |
| apr_pool_userdata_get(&data, mod_h2_init_key, s->process->pool); |
| if ( data == NULL ) { |
| ap_log_error( APLOG_MARK, APLOG_DEBUG, 0, s, |
| "initializing post config dry run"); |
| apr_pool_userdata_set((const void *)1, mod_h2_init_key, |
| apr_pool_cleanup_null, s->process->pool); |
| return APR_SUCCESS; |
| } |
| |
| ngh2 = nghttp2_version(0); |
| ap_log_error( APLOG_MARK, APLOG_INFO, 0, s, |
| "mod_http2 (v%s, nghttp2 %s), initializing...", |
| MOD_HTTP2_VERSION, ngh2? ngh2->version_str : "unknown"); |
| |
| switch (h2_conn_mpm_type()) { |
| case H2_MPM_EVENT: |
| case H2_MPM_WORKER: |
| /* all fine, we know these ones */ |
| break; |
| case H2_MPM_PREFORK: |
| /* ok, we now know how to handle that one */ |
| break; |
| case H2_MPM_UNKNOWN: |
| /* ??? */ |
| ap_log_error(APLOG_MARK, APLOG_DEBUG, 0, s, |
| "post_config: mpm type unknown"); |
| break; |
| } |
| |
| status = h2_h2_init(p, s); |
| if (status == APR_SUCCESS) { |
| status = h2_switch_init(p, s); |
| } |
| |
| return status; |
| } |
| |
| /* Runs once per created child process. Perform any process |
| * related initionalization here. |
| */ |
| static void h2_child_init(apr_pool_t *pool, server_rec *s) |
| { |
| /* Set up our connection processing */ |
| apr_status_t status = h2_conn_child_init(pool, s); |
| if (status != APR_SUCCESS) { |
| ap_log_error(APLOG_MARK, APLOG_ERR, status, s, |
| APLOGNO(02949) "initializing connection handling"); |
| } |
| } |
| |
| /* Install this module into the apache2 infrastructure. |
| */ |
| static void h2_hooks(apr_pool_t *pool) |
| { |
| static const char *const mod_ssl[] = { "mod_ssl.c", NULL}; |
| |
| ap_log_perror(APLOG_MARK, APLOG_INFO, 0, pool, "installing hooks"); |
| |
| /* Run once after configuration is set, but before mpm children initialize. |
| */ |
| ap_hook_post_config(h2_post_config, mod_ssl, NULL, APR_HOOK_MIDDLE); |
| |
| /* Run once after a child process has been created. |
| */ |
| ap_hook_child_init(h2_child_init, NULL, NULL, APR_HOOK_MIDDLE); |
| |
| h2_h2_register_hooks(); |
| h2_switch_register_hooks(); |
| h2_task_register_hooks(); |
| |
| h2_alt_svc_register_hooks(); |
| |
| } |
| |
| |