]> git.netwichtig.de Git - user/henk/code/inspircd.git/blob - configure
b81a7809a1346c69f2e7d28b892455b7a6267507
[user/henk/code/inspircd.git] / configure
1 #!/usr/bin/env perl
2
3 #
4 # InspIRCd -- Internet Relay Chat Daemon
5 #
6 #   Copyright (C) 2009-2010 Daniel De Graaf <danieldg@inspircd.org>
7 #   Copyright (C) 2007, 2009 Dennis Friis <peavey@inspircd.org>
8 #   Copyright (C) 2003, 2006-2008 Craig Edwards <craigedwards@brainbox.cc>
9 #   Copyright (C) 2006-2008 Robin Burchell <robin+git@viroteck.net>
10 #   Copyright (C) 2008 Thomas Stagner <aquanight@inspircd.org>
11 #   Copyright (C) 2007 John Brooks <john.brooks@dereferenced.net>
12 #   Copyright (C) 2006 Oliver Lupton <oliverlupton@gmail.com>
13 #   Copyright (C) 2003-2006 Craig McLure <craig@chatspike.net>
14 #
15 # This file is part of InspIRCd.  InspIRCd is free software: you can
16 # redistribute it and/or modify it under the terms of the GNU General Public
17 # License as published by the Free Software Foundation, version 2.
18 #
19 # This program is distributed in the hope that it will be useful, but WITHOUT
20 # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
21 # FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more
22 # details.
23 #
24 # You should have received a copy of the GNU General Public License
25 # along with this program.  If not, see <http://www.gnu.org/licenses/>.
26 #
27
28
29 BEGIN {
30         require 5.10.0;
31 }
32
33 use feature ':5.10';
34 use strict;
35 use warnings FATAL => qw(all);
36
37 use File::Basename        qw(basename);
38 use File::Copy            ();
39 use File::Spec::Functions qw(rel2abs);
40 use FindBin               qw($RealDir);
41 use Getopt::Long          qw(GetOptions);
42 use POSIX                 qw(getgid getuid);
43
44 use lib $RealDir;
45 use make::common;
46 use make::configure;
47 use make::console;
48 use make::directive;
49
50 my ($opt_binary_dir,
51     $opt_config_dir,
52     $opt_data_dir,
53     $opt_development,
54     $opt_disable_interactive,
55     $opt_distribution_label,
56     $opt_gid,
57     $opt_log_dir,
58     $opt_manual_dir,
59     $opt_module_dir,
60     $opt_prefix,
61     $opt_socketengine,
62     $opt_system,
63     $opt_uid);
64
65 sub list_extras ();
66
67 sub enable_extras (@);
68
69 sub disable_extras (@);
70
71 my @opt_enableextras;
72 my @opt_disableextras;
73
74 GetOptions(
75         'clean'  => \&cmd_clean,
76         'help'   => \&cmd_help,
77         'update' => \&cmd_update,
78
79         'development'          => \$opt_development,
80         'disable-interactive'  => \$opt_disable_interactive,
81         'distribution-label=s' => \$opt_distribution_label,
82         'binary-dir=s'         => \$opt_binary_dir,
83         'config-dir=s'         => \$opt_config_dir,
84         'data-dir=s'           => \$opt_data_dir,
85         'gid=s'                => \$opt_gid,
86         'log-dir=s'            => \$opt_log_dir,
87         'manual-dir=s'         => \$opt_manual_dir,
88         'module-dir=s'         => \$opt_module_dir,
89         'prefix=s'             => \$opt_prefix,
90         'socketengine=s'       => \$opt_socketengine,
91         'system'               => \$opt_system,
92         'uid=s'                => \$opt_uid,
93
94         # TODO: when the modulemanager rewrite is done these should be removed.
95         'disable-extras=s@' => \@opt_disableextras,
96         'enable-extras=s@'  => \@opt_enableextras,
97         'list-extras'       => sub { list_extras; exit 0; },
98 );
99
100 if (scalar(@opt_enableextras) + scalar(@opt_disableextras) > 0) {
101         @opt_enableextras = split /,/, join(',', @opt_enableextras);
102         @opt_disableextras = split /,/, join(',', @opt_disableextras);
103         enable_extras(@opt_enableextras);
104         disable_extras(@opt_disableextras);
105         list_extras;
106         print "Remember: YOU are responsible for making sure any libraries needed have been installed!\n";
107         exit 0;
108 }
109
110 our $interactive = !(
111         !-t STDIN ||
112         !-t STDOUT ||
113         defined $opt_binary_dir ||
114         defined $opt_config_dir ||
115         defined $opt_data_dir ||
116         defined $opt_development ||
117         defined $opt_disable_interactive ||
118         defined $opt_distribution_label ||
119         defined $opt_gid ||
120         defined $opt_log_dir ||
121         defined $opt_manual_dir ||
122         defined $opt_module_dir ||
123         defined $opt_prefix ||
124         defined $opt_socketengine ||
125         defined $opt_system ||
126         defined $opt_uid
127 );
128
129 my %version = get_version();
130 print_format "<|BOLD Configuring InspIRCd $version{FULL} on $^O.|>\n";
131
132 our %config;
133 if ($interactive) {
134         %config = read_configure_cache();
135         run_test CONFIGURE_CACHE_FILE, %config;
136         if (!defined $config{VERSION}) {
137                 $config{VERSION} = CONFIGURE_CACHE_VERSION;
138         } elsif ($config{VERSION} != CONFIGURE_CACHE_VERSION) {
139                 print_warning "ignoring contents of ${\CONFIGURE_CACHE_FILE} as it was generated by an incompatible version of $0!";
140                 %config = ('VERSION', CONFIGURE_CACHE_VERSION);
141         }
142 }
143
144 $config{CXX} = find_compiler($config{CXX} // $ENV{CXX});
145 unless ($config{CXX}) {
146         print "A suitable C++ compiler could not be detected on your system!\n";
147         print "Set the CXX environment variable to the compiler binary path if this is incorrect.\n";
148         exit 1; 
149 }
150 my %compiler = get_compiler_info($config{CXX});
151
152 $config{HAS_CLOCK_GETTIME} = run_test 'clock_gettime()', test_file($config{CXX}, 'clock_gettime.cpp', '-lrt');
153 $config{HAS_EVENTFD} = run_test 'eventfd()', test_file($config{CXX}, 'eventfd.cpp');
154
155 if ($config{HAS_EPOLL} = run_test 'epoll', test_header($config{CXX}, 'sys/epoll.h')) {
156         $config{SOCKETENGINE} //= 'epoll';
157 }
158
159 if ($config{HAS_KQUEUE} = run_test 'kqueue', test_file($config{CXX}, 'kqueue.cpp')) {
160         $config{SOCKETENGINE} //= 'kqueue';
161 }
162
163 if ($config{HAS_PORTS} = run_test 'Solaris IOCP', test_header($config{CXX}, 'port.h')) {
164         $config{SOCKETENGINE} //= 'ports';
165 }
166
167 if ($config{HAS_POLL} = run_test 'poll', test_header($config{CXX}, 'poll.h')) {
168         $config{SOCKETENGINE} //= 'poll';
169 }
170
171 # Select is available on all platforms
172 $config{HAS_SELECT} = 1;
173 $config{SOCKETENGINE} //= 'select';
174
175 if (defined $opt_socketengine) {
176         my $cfgkey = 'HAS_' . uc $opt_socketengine;
177         if ($config{$cfgkey} && -f "src/socketengines/socketengine_$opt_socketengine.cpp") {
178                 $config{SOCKETENGINE} = $opt_socketengine;
179         } else {
180                 print "Unable to use a socket engine which is not supported on this platform ($opt_socketengine)!\n";
181                 print "Available socket engines are:";
182                 foreach (<src/socketengines/socketengine_*.cpp>) {
183                         s/src\/socketengines\/socketengine_(\w+)\.cpp/$1/;
184                         print " $1" if $config{'HAS_' . uc $1};
185                 }
186                 print "\n";     
187                 exit 1;
188         }
189 }
190
191 # If the user has specified a distribution label then we use it in
192 # place of the label from src/version.sh or Git.
193 if (defined $opt_distribution_label) {
194         $version{LABEL} = $opt_distribution_label;
195 }
196
197 if (defined $opt_system) {
198         $config{BASE_DIR}   = $opt_prefix     // '/var/lib/inspircd';
199         $config{BINARY_DIR} = $opt_binary_dir // '/usr/sbin';
200         $config{CONFIG_DIR} = $opt_config_dir // '/etc/inspircd';
201         $config{DATA_DIR}   = $opt_data_dir   // '/var/inspircd';
202         $config{LOG_DIR}    = $opt_module_dir // '/var/log/inspircd';
203         $config{MANUAL_DIR} = $opt_manual_dir // '/usr/share/man/man1';
204         $config{MODULE_DIR} = $opt_module_dir // '/usr/lib/inspircd';
205 } else {
206         $config{BASE_DIR}   = $opt_prefix     // $config{BASE_DIR}   // rel2abs 'run';
207         $config{BINARY_DIR} = $opt_binary_dir // $config{BINARY_DIR} // rel2abs $config{BASE_DIR} . '/bin';
208         $config{CONFIG_DIR} = $opt_config_dir // $config{CONFIG_DIR} // rel2abs $config{BASE_DIR} . '/conf';
209         $config{DATA_DIR}   = $opt_data_dir   // $config{DATA_DIR}   // rel2abs $config{BASE_DIR} . '/data';
210         $config{LOG_DIR}    = $opt_log_dir    // $config{LOG_DIR}    // rel2abs $config{BASE_DIR} . '/logs';
211         $config{MANUAL_DIR} = $opt_manual_dir // $config{MANUAL_DIR} // rel2abs $config{BASE_DIR} . '/manuals';
212         $config{MODULE_DIR} = $opt_module_dir // $config{MODULE_DIR} // rel2abs $config{BASE_DIR} . '/modules';
213 }
214
215 # Parse --gid=123 or --gid=foo and extract the group id.
216 my @group;
217 if (defined $opt_gid) {
218         @group = $opt_gid =~ /^\d+$/ ? getgrgid($opt_gid) : getgrnam($opt_gid);
219         print_error "there is no '$opt_gid' group on this system!" unless @group;
220 } else {
221         @group = $opt_system ? getgrnam('irc') : getgrgid($config{GID} // getgid());
222         print_error "you need to specify a group to run as using '--gid [id|name]'!" unless @group;
223 }
224 $config{GROUP} = $group[0];
225 $config{GID}   = $group[2];
226
227 # Parse --uid=123 or --uid=foo and extract the user id.
228 my @user;
229 if (defined $opt_uid) {
230         @user = $opt_uid =~ /^\d+$/ ? getpwuid($opt_uid) : getpwnam($opt_uid);
231         print_error "there is no '$opt_uid' user on this system!" unless @user;
232 } else {
233         @user = $opt_system ? getpwnam('irc') : getpwuid($config{UID} // getuid());
234         print_error "you need to specify a user to run as using '--uid [id|name]'!" unless @user;
235 }
236 $config{USER} = $user[0];
237 $config{UID}  = $user[2];
238
239 # Clear the screen.
240 system 'tput', 'clear' if $interactive;
241
242 # Check that the user actually wants this version.
243 if ($version{LABEL} ne 'release') {
244         print_warning <<'EOW';
245 You are building a development version. This contains code which has
246 not been tested as heavily and may contain various faults which could seriously
247 affect the running of your server. It is recommended that you use a stable
248 version instead.
249
250 You can obtain the latest stable version from http://www.inspircd.org/ or by
251 running `git checkout insp20` if you are installing from Git.
252 EOW
253         if (!prompt_bool $interactive, 'I understand this warning and want to continue anyway.', $opt_development // 0) {
254                 say STDERR 'If you understand this warning and still want to continue pass the --development flag.' unless $interactive;
255                 exit 1;
256         }
257 }
258
259 # Configure directory settings.
260 my $question = <<"EOQ";
261 Currently, InspIRCd is configured with the following paths:
262
263 <|BOLD Base:|>   $config{BASE_DIR}
264 <|BOLD Binary:|> $config{BINARY_DIR}
265 <|BOLD Config:|> $config{CONFIG_DIR}
266 <|BOLD Data:|>   $config{DATA_DIR}
267 <|BOLD Log:|>    $config{LOG_DIR}
268 <|BOLD Manual:|> $config{MANUAL_DIR}
269 <|BOLD Module:|> $config{MODULE_DIR}
270
271 Do you want to change these settings?
272 EOQ
273 if (prompt_bool $interactive, $question, 0) {
274         my $original_base_dir = $config{BASE_DIR};
275         $config{BASE_DIR} = prompt_dir $interactive, 'In what directory do you wish to install the InspIRCd base?', $config{BASE_DIR};
276         foreach my $key (qw(BINARY_DIR CONFIG_DIR DATA_DIR LOG_DIR MANUAL_DIR MODULE_DIR)) {
277                 $config{$key} =~ s/^\Q$original_base_dir\E/$config{BASE_DIR}/;
278         }
279         $config{BINARY_DIR} = prompt_dir $interactive, 'In what directory should the InspIRCd binary be placed?', $config{BINARY_DIR};
280         $config{CONFIG_DIR} = prompt_dir $interactive, 'In what directory are configuration files to be stored?', $config{CONFIG_DIR};
281         $config{DATA_DIR}   = prompt_dir $interactive, 'In what directory are variable data files to be stored?', $config{DATA_DIR};
282         $config{LOG_DIR}    = prompt_dir $interactive, 'In what directory are log files to be stored?',           $config{LOG_DIR};
283         $config{MANUAL_DIR} = prompt_dir $interactive, 'In what directory are manual pages to be placed?',        $config{MANUAL_DIR};
284         $config{MODULE_DIR} = prompt_dir $interactive, 'In what directory are modules to be placed?',             $config{MODULE_DIR};
285 }
286
287 # Configure module settings.
288 $question = <<'EOQ';
289 Currently, InspIRCd is configured to automatically enable all available extra modules.
290
291 Would you like to enable extra modules manually?
292 EOQ
293 if (prompt_bool $interactive, $question, 0) {
294         foreach my $extra (<src/modules/extra/m_*.cpp>) {
295                 my $module_name = basename $extra, '.cpp';
296                 if (prompt_bool $interactive, "Would you like to enable $module_name?", 0) {
297                         enable_extras "$module_name.cpp";
298                 }
299         }
300 } else {
301         # TODO: finish modulemanager rewrite and replace this code with:
302         # system './modulemanager', 'enable', '--auto';
303         enable_extras 'm_ssl_gnutls.cpp' unless system 'pkg-config --exists gnutls >/dev/null 2>&1';
304         enable_extras 'm_ssl_mbedtls.cpp' if -e '/usr/include/mbedtls/ssl.h';
305         enable_extras 'm_ssl_openssl.cpp' unless system 'pkg-config --exists openssl >/dev/null 2>&1';
306 }
307
308 # Generate SSL certificates.
309 if (<src/modules/m_ssl_*.cpp> && prompt_bool $interactive, 'Would you like to generate SSL certificates now?', $interactive) {
310         system './tools/genssl', 'auto';
311 }
312
313 write_configure_cache %config;
314 parse_templates \%config, \%compiler, \%version;
315
316 print_format <<"EOM";
317
318 Configuration is complete! You have chosen to build with the following settings:
319
320 <|GREEN Compiler:|>
321   <|GREEN Binary:|>  $config{CXX}
322   <|GREEN Name:|>    $compiler{NAME}
323   <|GREEN Version:|> $compiler{VERSION}
324
325 <|GREEN Extra Modules:|>
326 EOM
327
328 for my $file (<src/modules/m_*>) {
329         my $module = basename $file, '.cpp';
330         say "  * $module" if -l $file;
331 }
332
333 print_format <<"EOM";
334
335 <|GREEN Paths:|>
336   <|GREEN Base:|>   $config{BASE_DIR}
337   <|GREEN Binary:|> $config{BINARY_DIR}
338   <|GREEN Config:|> $config{CONFIG_DIR}
339   <|GREEN Data:|>   $config{DATA_DIR}
340   <|GREEN Log:|>    $config{LOG_DIR}
341   <|GREEN Manual:|> $config{MANUAL_DIR}
342   <|GREEN Module:|> $config{MODULE_DIR}
343
344 <|GREEN Execution Group:|> $config{GROUP} ($config{GID})
345 <|GREEN Execution User:|>  $config{USER} ($config{UID})
346 <|GREEN Socket Engine:|>   $config{SOCKETENGINE}
347
348 To build with these settings run '<|GREEN make -j${\get_cpu_count}|>' now.
349
350 EOM
351
352 # Routine to list out the extra/ modules that have been enabled.
353 # Note: when getting any filenames out and comparing, it's important to lc it if the
354 # file system is not case-sensitive (== Epoc, MacOS, OS/2 (incl DOS/DJGPP), VMS, Win32
355 # (incl NetWare, Symbian)). Cygwin may or may not be case-sensitive, depending on
356 # configuration, however, File::Spec does not currently tell us (it assumes Unix behavior).
357 sub list_extras () {
358         use File::Spec;
359         # @_ not used
360         my $srcdir = File::Spec->catdir("src", "modules");
361         my $abs_srcdir = File::Spec->rel2abs($srcdir);
362         local $_;
363         my $dd;
364         opendir $dd, File::Spec->catdir($abs_srcdir, "extra") or die (File::Spec->catdir($abs_srcdir, "extra") . ": $!\n");
365         my @extras = map { File::Spec->case_tolerant() ? lc($_) : $_ } (readdir($dd));
366         closedir $dd;
367         undef $dd;
368         opendir $dd, $abs_srcdir or die "$abs_srcdir: $!\n";
369         my @sources = map { File::Spec->case_tolerant() ? lc($_) : $_ } (readdir($dd));
370         closedir $dd;
371         undef $dd;
372         my $maxlen = (sort { $b <=> $a } (map {length($_)} (@extras)))[0];
373         my %extras = ();
374 EXTRA:  for my $extra (@extras) {
375                 next if (File::Spec->curdir() eq $extra || File::Spec->updir() eq $extra);
376                 my $abs_extra = File::Spec->catfile($abs_srcdir, "extra", $extra);
377                 my $abs_source = File::Spec->catfile($abs_srcdir, $extra);
378                 next unless ($extra =~ m/\.(cpp|h)$/ || (-d $abs_extra)); # C++ Source/Header, or directory
379                 if (-l $abs_source) {
380                         # Symlink, is it in the right place?
381                         my $targ = readlink($abs_source);
382                         my $abs_targ = File::Spec->rel2abs($targ, $abs_srcdir);
383                         if ($abs_targ eq $abs_extra) {
384                                 $extras{$extra} = "\e[32;1menabled\e[0m";
385                         } else {
386                                 $extras{$extra} = sprintf("\e[31;1mwrong symlink target (%s)\e[0m", $abs_targ);
387                         }
388                 } elsif (-e $abs_source) {
389                         my ($devext, $inoext) = stat($abs_extra);
390                         my ($devsrc, $inosrc, undef, $lnksrc) = stat($abs_source);
391                         if ($lnksrc > 1) {
392                                 if ($devsrc == $devext && $inosrc == $inoext) {
393                                         $extras{$extra} = "\e[32;1menabled\e[0m";
394                                 } else {
395                                         $extras{$extra} = sprintf("\e[31;1mwrong hardlink target (%d:%d)\e[0m", $devsrc, $inosrc);
396                                 }
397                         } else {
398                                 open my $extfd, "<", $abs_extra;
399                                 open my $srcfd, "<", $abs_source;
400                                 local $/ = undef;
401                                 if (scalar(<$extfd>) eq scalar(<$srcfd>)) {
402                                         $extras{$extra} = "\e[32;1menabled\e[0m";
403                                 } else {
404                                         $extras{$extra} = sprintf("\e[31;1mout of synch (re-copy)\e[0m");
405                                 }
406                         }
407                 } else {
408                         $extras{$extra} = "\e[33;1mdisabled\e[0m";
409                 }
410         }
411         # Now let's add dependency info
412         for my $extra (keys(%extras)) {
413                 next unless $extras{$extra} =~ m/enabled/; # only process enabled extras.
414                 my $abs_extra = File::Spec->catfile($abs_srcdir, "extra", $extra);
415                 my @deps = split /\s+/, get_directive($abs_extra, 'ModDep', '');
416                 for my $dep (@deps) {
417                         if (exists($extras{$dep})) {
418                                 my $ref = \$extras{$dep}; # Take reference.
419                                 if ($$ref !~ m/needed by/) {
420                                         # First dependency found.
421                                         if ($$ref =~ m/enabled/) {
422                                                 $$ref .= " (needed by \e[32;1m$extra\e[0m";
423                                         } else {
424                                                 $$ref =~ s/\e\[.*?m//g; # Strip out previous coloring. Will be set in bold+red+blink later.
425                                                 $$ref .= " (needed by \e[0;32;1;5m$extra\e[0;31;1;5m";
426                                         }
427                                 } else {
428                                         if ($$ref =~ m/enabled/) {
429                                                 $$ref .= ", \e[32;1m$extra\e[0m";
430                                         } else {
431                                                 $$ref .= ", \e[0;32;1;5m$extra\e[0;31;1;5m";
432                                         }
433                                 }
434                         }
435                 }
436         }
437         for my $extra (sort {$a cmp $b} keys(%extras)) {
438                 my $text = $extras{$extra};
439                 if ($text =~ m/needed by/ && $text !~ m/enabled/) {
440                         printf "\e[31;1;5m%-*s = %s%s\e[0m\n", $maxlen, $extra, $text, ($text =~ m/needed by/ ? ")" : "");
441                 } else {
442                         printf "%-*s = %s%s\n", $maxlen, $extra, $text, ($text =~ m/needed by/ ? "\e[0m)" : "");
443                 }
444         }
445         return keys(%extras) if wantarray; # Can be used by manage_extras.
446 }
447
448 sub enable_extras (@) {
449         my (@extras) = @_;
450         for my $extra (@extras) {
451                 my $extrapath = "src/modules/extra/$extra";
452                 if (!-e $extrapath) {
453                         print STDERR "Cannot enable \e[32;1m$extra\e[0m : No such file or directory in src/modules/extra\n";
454                         next;
455                 }
456                 my $source = "src/modules/$extra";
457                 if (-e $source) {
458                         print STDERR "Cannot enable \e[32;1m$extra\e[0m : destination in src/modules exists (might already be enabled?)\n";
459                         next;
460                 }
461                 # Get dependencies, and add them to be processed.
462                 my @deps = split /\s+/, get_directive($extrapath, 'ModDep', '');
463                 for my $dep (@deps) {
464                         next if scalar(grep { $_ eq $dep } (@extras)) > 0; # Skip if we're going to be enabling it anyway.
465                         if (!-e "src/modules/$dep" && !-e "include/$dep") {
466                                 if (-e "src/modules/extra/$dep") {
467                                         print STDERR "Will also enable extra \e[32;1m$dep\e[0m (needed by \e[32;1m$extra\e[0m)\n";
468                                         push @extras, $dep;
469                                 } else {
470                                         print STDERR "\e[33;1mWARNING:\e[0m module \e[32;1m$extra\e[0m might be missing dependency \e[32;1m$dep\e[0m - YOU are responsible for satisfying it!\n";
471                                 }
472                         }
473                 }
474                 print "Enabling $extra ... \n";
475                 symlink "extra/$extra", $source or print STDERR "$source: Cannot link to 'extra/$extra': $!\n";
476         }
477 }
478
479 sub disable_extras (@)
480 {
481         opendir my $dd, "src/modules/extra/";
482         my @files = readdir($dd);
483         closedir $dd;
484         my (@extras) = @_;
485 EXTRA:  for my $extra (@extras) {
486                 my $extrapath = "src/modules/extra/$extra";
487                 my $source = "src/modules/$extra";
488                 if (!-e $extrapath) {
489                         print STDERR "Cannot disable \e[32;1m$extra\e[0m : Is not an extra\n";
490                         next;
491                 }
492                 if ((! -l $source) || readlink($source) ne "extra/$extra") {
493                         print STDERR "Cannot disable \e[32;1m$extra\e[0m : Source is not a link or doesn't refer to the right file. Remove manually if this is in error.\n";
494                         next;
495                 }
496                 # Check if anything needs this.
497                 for my $file (@files) {
498                         my @deps = split /\s+/, get_directive("src/modules/extra/$file", 'ModDep', '');
499                         # File depends on this extra...
500                         if (scalar(grep { $_ eq $extra } @deps) > 0) {
501                                 # And is both enabled and not about to be disabled.
502                                 if (-e "src/modules/$file" && scalar(grep { $_ eq $file } @extras) < 1) {
503                                         print STDERR "Cannot disable \e[32;1m$extra\e[0m : is needed by \e[32;1m$file\e[0m\n";
504                                         next EXTRA;
505                                 }
506                         }
507                 }
508                 # Now remove.
509                 print "Disabling $extra ... \n";
510                 unlink "src/modules/$extra" or print STDERR "Cannot disable \e[32;1m$extra\e[0m : $!\n";
511         }
512 }