# Licensed to the Apache Software Foundation (ASF) under one or more # contributor license agreements. See the NOTICE file distributed with # this work for additional information regarding copyright ownership. # The ASF licenses this file to You 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. # package Apache::TestConfig; #not TestConfigC on purpose use strict; use warnings FATAL => 'all'; use Config; use Apache::TestConfig (); use Apache::TestConfigPerl (); use Apache::TestTrace; use File::Find qw(finddepth); sub cmodule_find { my($self, $mod) = @_; return unless $mod =~ /^mod_(\w+)\.c$/; my $sym = $1; my $dir = $File::Find::dir; my $file = catfile $dir, $mod; unless ($self->{APXS}) { $self->{cmodules_disabled}->{$mod} = "no apxs configured"; return; } my $fh = Symbol::gensym(); open $fh, $file or die "open $file: $!"; my $v = <$fh>; if ($v =~ /^\#define\s+HTTPD_TEST_REQUIRE_APACHE\s+(\d+)\s*$/) { #define HTTPD_TEST_REQUIRE_APACHE 1 unless ($self->{server}->{rev} == $1) { my $reason = "requires Apache version $1"; $self->{cmodules_disabled}->{$mod} = $reason; notice "$mod $reason, skipping."; return; } } elsif ($v =~ /^\#define\s+HTTPD_TEST_REQUIRE_APACHE\s+(\d\.\d+(\.\d+)?)/) { #define HTTPD_TEST_REQUIRE_APACHE 2.1 my $wanted = $1; (my $current) = $self->{server}->{version} =~ m:^Apache/(\d\.\d+\.\d+):; if (Apache::Test::normalize_vstring($current) < Apache::Test::normalize_vstring($wanted)) { my $reason = "requires Apache version $wanted"; $self->{cmodules_disabled}->{$mod} = $reason; notice "$mod $reason, skipping."; return; } } close $fh; push @{ $self->{cmodules} }, { name => "mod_$sym", sym => "${sym}_module", dir => $dir, subdir => basename $dir, }; } sub cmodules_configure { my($self, $dir) = @_; $self->{cmodules_disabled} = {}; #for have_module to check $dir ||= catfile $self->{vars}->{top_dir}, 'c-modules'; unless (-d $dir) { return; } $self->{cmodules_dir} = $dir; finddepth(sub { cmodule_find($self, $_) }, $dir); unless ($self->{APXS}) { warning "cannot build c-modules without apxs"; return; } $self->cmodules_generate_include; $self->cmodules_write_makefiles; $self->cmodules_compile; $self->cmodules_httpd_conf; } sub cmodules_makefile_vars { return < "", 2 => "") : (1 => "", 2 => ".libs/"); sub cmodules_build_so { my($self, $name) = @_; $name = "mod_$name" unless $name =~ /^mod_/; my $libdir = $self->server->version_of(\%lib_dir); my $lib = "$libdir$name.so"; } sub cmodules_write_makefiles { my $self = shift; my $modules = $self->{cmodules}; for (@$modules) { $self->cmodules_write_makefile($_); } my $file = catfile $self->{cmodules_dir}, 'Makefile'; my $fh = $self->genfile($file); print $fh $self->cmodules_makefile_vars; my @dirs = map { $_->{subdir} } @$modules; my @targets = qw(clean); my @libs; for my $dir (@dirs) { for my $targ (@targets) { print $fh "$dir-$targ:\n\tcd $dir && \$(MAKE) $targ\n\n"; } my $lib = $self->cmodules_build_so($dir); my $cfile = "$dir/mod_$dir.c"; push @libs, "$dir/$lib"; print $fh "$libs[-1]: $cfile\n\tcd $dir && \$(MAKE) $lib\n\n"; } for my $targ (@targets) { print $fh "$targ: ", (map { "$_-$targ " } @dirs), "\n\n"; } print $fh "all: @libs\n\n"; close $fh or die "close $file: $!"; } sub cmodules_write_makefile { my ($self, $mod) = @_; my $write = \&{"cmodules_write_makefile_$^O"}; $write = \&cmodules_write_makefile_default unless defined &$write; $write->($self, $mod); } sub cmodules_write_makefile_default { my($self, $mod) = @_; my $dversion = $self->server->dversion; my $name = $mod->{name}; my $makefile = catfile $mod->{dir}, 'Makefile'; my $extra = $ENV{EXTRA_CFLAGS} || ''; debug "writing $makefile"; my $lib = $self->cmodules_build_so($name); my $fh = $self->genfile($makefile); print $fh <{APXS} all: $lib $lib: $name.c \$(APXS) $dversion $extra -I$self->{cmodules_dir} -c $name.c clean: -rm -rf $name.o $name.lo $name.slo $name.la $name.i $name.s $name.gcno .libs EOF close $fh or die "close $makefile: $!"; } sub cmodules_write_makefile_aix { my($self, $mod) = @_; my $dversion = $self->server->dversion; my $name = $mod->{name}; my $makefile = catfile $mod->{dir}, 'Makefile'; my $apxsflags = ''; # # Only do this for Apache 1.* # if ($self->server->{rev} == 1) { $apxsflags = "-Wl,-bE:$name.exp"; my $expfile = catfile $mod->{dir}, "$name.exp"; if (! -f $expfile) { my $fh = Symbol::gensym(); $name =~ /^mod_(\w+)(?:\.c)?$/; my $sym = $1 . '_module'; open $fh, ">$expfile" or die "open $expfile: $!"; print $fh "$sym\n"; close $fh; } } my $extra = $ENV{EXTRA_CFLAGS} || ''; debug "writing $makefile"; my $lib = $self->cmodules_build_so($name); my $fh = Symbol::gensym(); open $fh, ">$makefile" or die "open $makefile: $!"; print $fh <{APXS} APXSFLAGS=$apxsflags all: $lib $lib: $name.c \$(APXS) $dversion $extra -I$self->{cmodules_dir} \$(APXSFLAGS) -c $name.c clean: -rm -rf $name.o $name.lo $name.slo $name.la .libs EOF close $fh or die "close $makefile: $!"; } sub cmodules_write_makefile_MSWin32 { my($self, $mod) = @_; my $dversion = $self->server->dversion; my $name = $mod->{name}; my $makefile = "$mod->{dir}/Makefile"; debug "writing $makefile"; my $extras = ''; my $lib = $self->cmodules_build_so($name); $extras = ' -llibhttpd -p ' if ($self->server->{rev} != 1); my $goners = join ' ', (map {$name . '.' . $_} qw(exp lib so lo)); my $fh = Symbol::gensym(); open $fh, ">$makefile" or die "open $makefile: $!"; my $extra = $ENV{EXTRA_CFLAGS} || ''; debug "writing $makefile"; print $fh <{APXS} all: $lib $lib: $name.c \$(APXS) $dversion $extra -I$self->{cmodules_dir} $extras -c $name.c clean: -erase $goners EOF close $fh or die "close $makefile: $!"; } sub cmodules_make { my $self = shift; my $targ = shift || 'all'; my $cmd = "cd $self->{cmodules_dir} && $Config{make} $targ"; debug $cmd; system $cmd; if ($?) { die "Failed to build c-modules"; } } sub cmodules_compile { shift->cmodules_make('all'); } sub cmodules_httpd_conf { my $self = shift; my @args; for my $mod (@{ $self->{cmodules} }) { my $dir = $mod->{dir}; my $lib = $self->cmodules_build_so($mod->{name}); my $so = "$dir/$lib"; next unless -e $so; $self->preamble(LoadModule => "$mod->{sym} $so"); my $cname = "$mod->{name}.c"; my $cfile = "$dir/$cname"; $self->{modules}->{$cname} = 1; $self->add_module_config($cfile, \@args); } $self->postamble(\@args) if @args; } sub cmodules_clean { my $self = shift; my $dir = $self->{cmodules_dir}; return unless $dir and -e "$dir/Makefile"; unless ($self->{clean_level} > 1) { #skip t/TEST -conf warning "skipping rebuild of c-modules; run t/TEST -clean to force"; return; } $self->cmodules_make('clean'); for my $mod (@{ $self->{cmodules} }) { my $makefile = "$mod->{dir}/Makefile"; debug "unlink $makefile"; unlink $makefile; } unlink "$dir/Makefile"; } #try making it easier for test modules to compile with both 1.x and 2.x sub cmodule_define_name { my $name = shift; $name eq 'NULL' ? $name : "APACHE_HTTPD_TEST_\U$name"; } sub cmodule_define { my $hook = cmodule_define_name(@_); "#ifndef $hook\n#define $hook NULL\n#endif\n"; } my @cmodule_config_names = qw(per_dir_create per_dir_merge per_srv_create per_srv_merge commands); my @cmodule_config_defines = map { cmodule_define($_); } @cmodule_config_names; my $cmodule_config_extra = "#ifndef APACHE_HTTPD_TEST_EXTRA_HOOKS\n". "#define APACHE_HTTPD_TEST_EXTRA_HOOKS(p) do { } while (0)\n". "#endif\n"; my $cmodule_config_hooks = join ",\n ", map { cmodule_define_name($_); } @cmodule_config_names; my @cmodule_phases = qw(post_read_request translate_name header_parser access_checker check_user_id auth_checker type_checker fixups handler log_transaction child_init); my $cmodule_hooks_1 = join ",\n ", map { cmodule_define_name($_); } qw(translate_name check_user_id auth_checker access_checker type_checker fixups log_transaction header_parser child_init NULL post_read_request); my $cmodule_template_1 = <<"EOF", static const handler_rec name ## _handlers[] = { {#name, APACHE_HTTPD_TEST_HANDLER}, /* ok if handler is NULL */ {NULL} }; module MODULE_VAR_EXPORT name ## _module = { STANDARD_MODULE_STUFF, NULL, /* initializer */ $cmodule_config_hooks, name ## _handlers, /* handlers */ $cmodule_hooks_1 } EOF my @cmodule_hooks = map { my $hook = cmodule_define_name($_); < $cmodule_template_1, 2 => $cmodule_template_2); sub cmodules_module_template { my $self = shift; my $template = $self->server->version_of(\%cmodule_templates); chomp $template; $template =~ s,$, \\,mg; $template =~ s, \\$,,s; local $" = ', '; return <genfile($file); while (read Apache::TestConfigC::DATA, my $buf, 1024) { print $fh $buf; } print $fh @cmodule_hook_defines, @cmodule_config_defines; print $fh $cmodule_config_extra; print $fh $self->cmodules_module_template; close $fh; } package Apache::TestConfigC; #Apache/TestConfig.pm also has __DATA__ 1; __DATA__ #ifndef APACHE_HTTPD_TEST_H #define APACHE_HTTPD_TEST_H /* headers present in both 1.x and 2.x */ #include "httpd.h" #include "http_config.h" #include "http_protocol.h" #include "http_request.h" #include "http_log.h" #include "http_main.h" #include "http_core.h" #include "ap_config.h" #ifdef APACHE1 #define AP_METHOD_BIT 1 typedef size_t apr_size_t; typedef array_header apr_array_header_t; #define APR_OFF_T_FMT "ld" #define APR_SIZE_T_FMT "lu" #endif /* APACHE1 */ #ifdef APACHE2 #ifndef APACHE_HTTPD_TEST_HOOK_ORDER #define APACHE_HTTPD_TEST_HOOK_ORDER APR_HOOK_MIDDLE #endif #include "ap_compat.h" #endif /* APACHE2 */ #endif /* APACHE_HTTPD_TEST_H */