]> git.netwichtig.de Git - user/henk/code/inspircd.git/blobdiff - make/configure.pm
m_spanningtree Remove SpanningTreeUtilities* fields and parameters
[user/henk/code/inspircd.git] / make / configure.pm
index dd2dc90896bd68af3e9596d577adbb90fbc889db..ba67545ac36eb2fe41ec87d0338668fa6de6f072 100644 (file)
 #
-# Copyright 2002-2007 The ChatSpike Development Team
-# <brain@chatspike.net>
-# <Craig@chatspike.net>
+# InspIRCd -- Internet Relay Chat Daemon
 #
-# Licensed under GPL, please see the COPYING file
-# for more information
+#   Copyright (C) 2012 Peter Powell <petpow@saberuk.com>
+#   Copyright (C) 2008 Robin Burchell <robin+git@viroteck.net>
+#   Copyright (C) 2007-2008 Craig Edwards <craigedwards@brainbox.cc>
+#   Copyright (C) 2008 Thomas Stagner <aquanight@inspircd.org>
+#   Copyright (C) 2007 Dennis Friis <peavey@inspircd.org>
 #
+# This file is part of InspIRCd.  InspIRCd is free software: you can
+# redistribute it and/or modify it under the terms of the GNU General Public
+# License as published by the Free Software Foundation, version 2.
+#
+# This program is distributed in the hope that it will be useful, but WITHOUT
+# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
+# FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more
+# details.
+#
+# You should have received a copy of the GNU General Public License
+# along with this program.  If not, see <http://www.gnu.org/licenses/>.
+#
+
 
 package make::configure;
+
+require 5.8.0;
+
+use strict;
+use warnings FATAL => qw(all);
+
 use Exporter 'import';
 use POSIX;
 use make::utilities;
-@EXPORT = qw(promptnumeric dumphash is_dir getmodules getrevision getcompilerflags getlinkerflags getdependencies resolve_directory yesno showhelp promptstring);
+our @EXPORT = qw(get_compiler_info find_compiler run_test test_file test_header promptnumeric dumphash getmodules getrevision get_property yesno showhelp promptstring_s module_installed);
+
+my $revision;
+
+sub get_compiler_info($) {
+        my %info = (NAME => shift, VERSION => '0.0');
+        my $version = `$info{NAME} -v 2>&1`;
+               return (ERROR => 1) if $?;
+        if ($version =~ /(?:clang|llvm)\sversion\s(\d+\.\d+)/i) {
+                $info{NAME} = 'Clang';
+                $info{VERSION} = $1;
+                $info{UNSUPPORTED} = $1 lt '3.0';
+                $info{REASON} = 'Clang 2.9 and older do not have adequate C++ support.';
+        } elsif ($version =~ /gcc\sversion\s(\d+\.\d+)/i) {
+                $info{NAME} = 'GCC';
+                $info{VERSION} = $1;
+                $info{UNSUPPORTED} = $1 lt '4.1';
+                $info{REASON} = 'GCC 4.0 and older do not have adequate C++ support.';
+        } elsif ($version =~ /(?:icc|icpc)\sversion\s(\d+\.\d+).\d+\s\(gcc\sversion\s(\d+\.\d+).\d+/i) {
+                $info{NAME} = 'ICC';
+                $info{VERSION} = $1;
+                $info{UNSUPPORTED} = $2 lt '4.1';
+                $info{REASON} = "ICC $1 (GCC $2 compatibility mode) does not have adequate C++ support."
+        }
+        return %info;
+}  
+
+sub find_compiler {
+       foreach my $compiler ('c++', 'g++', 'clang++', 'icpc') {
+               return $compiler unless system "$compiler -v > /dev/null 2>&1";
+               if ($^O eq 'Darwin') {
+                       return $compiler unless system "xcrun $compiler -v > /dev/null 2>&1";
+               }
+       }
+       return "";
+}
+
+sub run_test($$) {
+       my ($what, $result) = @_;
+       print "Checking whether $what is available... ";
+       print $result ? "yes\n" : "no\n";
+       return $result;
+}
 
-my $no_svn = 0;
+sub test_file($$;$) {
+       my ($cc, $file, $args) = @_;
+       my $status = 0;
+       $args ||= '';
+       $status ||= system "$cc -o __test_$file make/test/$file $args >/dev/null 2>&1";
+       $status ||= system "./__test_$file >/dev/null 2>&1";
+       unlink  "./__test_$file";
+       return !$status;
+}
+
+sub test_header($$;$) {
+       my ($cc, $header, $args) = @_;
+       $args ||= '';
+       open(CC, "| $cc -E - $args >/dev/null 2>&1") or return 0;
+       print CC "#include <$header>";
+       close(CC);
+       return !$?;
+}
 
 sub yesno {
        my ($flag,$prompt) = @_;
-       print "$prompt [\033[1;32m$main::config{$flag}\033[0m] -> ";
-       chomp($tmp = <STDIN>);
+       print "$prompt [\e[1;32m$main::config{$flag}\e[0m] -> ";
+       chomp(my $tmp = <STDIN>);
        if ($tmp eq "") { $tmp = $main::config{$flag} }
        if (($tmp eq "") || ($tmp =~ /^y/i))
        {
@@ -31,97 +110,58 @@ sub yesno {
        return;
 }
 
-sub resolve_directory
+sub get_property($$;$)
 {
-       my $ret = $_[0];
-       eval
-       {
-               use File::Spec;
-               $ret = File::Spec->rel2abs($_[0]);
-       };
-       return $ret;
-}
-
-sub getrevision {
-       if ($no_svn)
-       {
-               return "0";
-       }
-       my $data = `svn info`;
-       if ($data eq "")
-       {
-               $no_svn = 1;
-               $rev = "0";
-               return $rev;
-       }
-       $data =~ /Revision: (\d+)/;
-       my $rev = $1;
-       if (!defined($rev))
-       {
-               $rev = "0";
-       }
-       return $rev;
-}
-
-sub getcompilerflags {
-       my ($file) = @_;
-       open(FLAGS, $file);
-       while (<FLAGS>) {
-               if ($_ =~ /^\/\* \$CompileFlags: (.+) \*\/$/) {
-                       close(FLAGS);
-                       return translate_functions($1,$file);
+       my ($file, $property, $default) = @_;
+       open(MODULE, $file) or return $default;
+       while (<MODULE>) {
+               if ($_ =~ /^\/\* \$(\S+): (.+) \*\/$/) {
+                       next unless $1 eq $property;
+                       close(MODULE);
+                       return translate_functions($2, $file);
                }
        }
-       close(FLAGS);
-       return undef;
+       close(MODULE);
+       return defined $default ? $default : '';
 }
 
-sub getlinkerflags {
-       my ($file) = @_;
-       open(FLAGS, $file);
-       while (<FLAGS>) {
-               if ($_ =~ /^\/\* \$LinkerFlags: (.+) \*\/$/) {
-                       close(FLAGS);
-                       return translate_functions($1,$file);
-               }
-       }
-       close(FLAGS);
-       return undef;
-}
-
-sub getdependencies {
-       my ($file) = @_;
-       open(FLAGS, $file);
-       while (<FLAGS>) {
-               if ($_ =~ /^\/\* \$ModDep: (.+) \*\/$/) {
-                       close(FLAGS);
-                       return translate_functions($1,$file);
-               }
-       }
-       close(FLAGS);
-       return undef;
+sub getrevision {
+       return $revision if defined $revision;
+       chomp(my $tags = `git describe --tags 2>/dev/null`);
+       $revision = $tags || 'release';
+       return $revision;
 }
 
-
 sub getmodules
 {
+       my ($silent) = @_;
+
        my $i = 0;
-       print "Detecting modules ";
-       opendir(DIRHANDLE, "src/modules");
-       foreach $name (sort readdir(DIRHANDLE))
+
+       if (!$silent)
+       {
+               print "Detecting modules ";
+       }
+
+       opendir(DIRHANDLE, "src/modules") or die("WTF, missing src/modules!");
+       foreach my $name (sort readdir(DIRHANDLE))
        {
                if ($name =~ /^m_(.+)\.cpp$/)
                {
-                       $mod = $1;
-                       if ($mod !~ /_static$/)
+                       my $mod = $1;
+                       $main::modlist[$i++] = $mod;
+                       if (!$silent)
                        {
-                               $main::modlist[$i++] = $mod;
                                print ".";
                        }
                }
        }
        closedir(DIRHANDLE);
-       print "\nOk, $i modules.\n";
+
+       if (!$silent)
+       {
+               print "\nOk, $i modules.\n";
+       }
 }
 
 sub promptnumeric($$)
@@ -131,8 +171,8 @@ sub promptnumeric($$)
        while (!$continue)
        {
                print "Please enter the maximum $prompt?\n";
-               print "[\033[1;32m$main::config{$configitem}\033[0m] -> ";
-               chomp($var = <STDIN>);
+               print "[\e[1;32m$main::config{$configitem}\e[0m] -> ";
+               chomp(my $var = <STDIN>);
                if ($var eq "")
                {
                        $var = $main::config{$configitem};
@@ -148,12 +188,19 @@ sub promptnumeric($$)
        }
 }
 
-sub promptstring($$)
+sub module_installed($)
+{
+       my $module = shift;
+       eval("use $module;");
+       return !$@;
+}
+
+sub promptstring_s($$)
 {
        my ($prompt,$default) = @_;
        my $var;
        print "$prompt\n";
-       print "[\033[1;32m$default\033[0m] -> ";
+       print "[\e[1;32m$default\e[0m] -> ";
        chomp($var = <STDIN>);
        $var = $default if $var eq "";
        print "\n";
@@ -162,56 +209,29 @@ sub promptstring($$)
 
 sub dumphash()
 {
-       print "\n\033[1;32mPre-build configuration is complete!\033[0m\n\n";
-       print "\033[0mBase install path:\033[1;32m\t\t$main::config{BASE_DIR}\033[0m\n";
-       print "\033[0mConfig path:\033[1;32m\t\t\t$main::config{CONFIG_DIR}\033[0m\n";
-       print "\033[0mModule path:\033[1;32m\t\t\t$main::config{MODULE_DIR}\033[0m\n";
-       print "\033[0mLibrary path:\033[1;32m\t\t\t$main::config{LIBRARY_DIR}\033[0m\n";
-       print "\033[0mMax connections:\033[1;32m\t\t$main::config{MAX_CLIENT}\033[0m\n";
-       print "\033[0mMax nickname length:\033[1;32m\t\t$main::config{NICK_LENGT}\033[0m\n";
-       print "\033[0mMax channel length:\033[1;32m\t\t$main::config{CHAN_LENGT}\033[0m\n";
-       print "\033[0mMax mode length:\033[1;32m\t\t$main::config{MAXI_MODES}\033[0m\n";
-       print "\033[0mMax ident length:\033[1;32m\t\t$main::config{MAX_IDENT}\033[0m\n";
-       print "\033[0mMax quit length:\033[1;32m\t\t$main::config{MAX_QUIT}\033[0m\n";
-       print "\033[0mMax topic length:\033[1;32m\t\t$main::config{MAX_TOPIC}\033[0m\n";
-       print "\033[0mMax kick length:\033[1;32m\t\t$main::config{MAX_KICK}\033[0m\n";
-       print "\033[0mMax name length:\033[1;32m\t\t$main::config{MAX_GECOS}\033[0m\n";
-       print "\033[0mMax away length:\033[1;32m\t\t$main::config{MAX_AWAY}\033[0m\n";
-       print "\033[0mGCC Version Found:\033[1;32m\t\t$main::config{GCCVER}.x\033[0m\n";
-       print "\033[0mCompiler program:\033[1;32m\t\t$main::config{CC}\033[0m\n";
-       print "\033[0mStatic modules:\033[1;32m\t\t\t$main::config{STATIC_LINK}\033[0m\n";
-       print "\033[0mIPv6 Support:\033[1;32m\t\t\t$main::config{IPV6}\033[0m\n";
-       print "\033[0mIPv6 to IPv4 Links:\033[1;32m\t\t$main::config{SUPPORT_IP6LINKS}\033[0m\n";
-       print "\033[0mGnuTLS Support:\033[1;32m\t\t\t$main::config{USE_GNUTLS}\033[0m\n";
-       print "\033[0mOpenSSL Support:\033[1;32m\t\t$main::config{USE_OPENSSL}\033[0m\n\n";
-}
-
-sub is_dir
-{
-       my ($path) = @_;
-       if (chdir($path))
-       {
-               chdir($main::this);
-               return 1;
-       }
-       else
-       {
-               # Just in case..
-               chdir($main::this);
-               return 0;
-       }
+       print "\n\e[1;32mPre-build configuration is complete!\e[0m\n\n";
+       print "\e[0mBase install path:\e[1;32m\t\t$main::config{BASE_DIR}\e[0m\n";
+       print "\e[0mConfig path:\e[1;32m\t\t\t$main::config{CONFIG_DIR}\e[0m\n";
+       print "\e[0mData path:\e[1;32m\t\t\t$main::config{DATA_DIR}\e[0m\n";
+       print "\e[0mLog path:\e[1;32m\t\t\t$main::config{LOG_DIR}\e[0m\n";
+       print "\e[0mModule path:\e[1;32m\t\t\t$main::config{MODULE_DIR}\e[0m\n";
+       print "\e[0mCompiler:\e[1;32m\t\t\t$main::cxx{NAME} $main::cxx{VERSION}\e[0m\n";
+       print "\e[0mSocket engine:\e[1;32m\t\t\t$main::config{SOCKETENGINE}\e[0m\n";
+       print "\e[0mGnuTLS support:\e[1;32m\t\t\t$main::config{USE_GNUTLS}\e[0m\n";
+       print "\e[0mOpenSSL support:\e[1;32m\t\t$main::config{USE_OPENSSL}\e[0m\n";
 }
 
 sub showhelp
 {
-       chomp($PWD = `pwd`);
-       print "Usage: configure [options]
-
-*** NOTE: NON-INTERACTIVE CONFIGURE IS *NOT* SUPPORTED BY THE ***
-*** INSPIRCD DEVELOPMENT TEAM. DO NOT ASK FOR HELP REGARDING  ***
-***     NON-INTERACTIVE CONFIGURE ON THE FORUMS OR ON IRC!    ***
-
-Options: [defaults in brackets after descriptions]
+       chomp(my $PWD = `pwd`);
+       my (@socketengines, $SELIST);
+       foreach (<src/socketengines/socketengine_*.cpp>) {
+               s/src\/socketengines\/socketengine_(\w+)\.cpp/$1/;
+               push(@socketengines, $1);
+       }
+       $SELIST = join(", ", @socketengines);
+       print <<EOH;
+Usage: configure [options]
 
 When no options are specified, interactive
 configuration is started and you must specify
@@ -219,59 +239,37 @@ any required values manually. If one or more
 options are specified, non-interactive configuration
 is started, and any omitted values are defaulted.
 
-Arguments with a single \"-\" symbol, as in
-InspIRCd 1.0.x, are also allowed.
+Arguments with a single \"-\" symbol are also allowed.
 
-  --disable-interactive        Sets no options intself, but
+  --disable-interactive        Sets no options itself, but
                                will disable any interactive prompting.
   --update                     Update makefiles and dependencies
-  --modupdate                  Detect new modules and write makefiles
-  --svnupdate {--rebuild}      Update working copy via subversion
-                                {and optionally rebuild if --rebuild
-                                 is also specified}
   --clean                      Remove .config.cache file and go interactive
   --enable-gnutls              Enable GnuTLS module [no]
   --enable-openssl             Enable OpenSSL module [no]
-  --with-nick-length=[n]       Specify max. nick length [32]
-  --with-channel-length=[n]    Specify max. channel length [64]
-  --with-max-clients=[n]       Specify maximum number of users
-                               which may connect locally
-  --enable-optimization=[n]    Optimize using -O[n] gcc flag
-  --enable-epoll               Enable epoll() where supported [set]
-  --enable-kqueue              Enable kqueue() where supported [set]
-  --disable-epoll              Do not enable epoll(), fall back
-                               to select() [not set]
-  --disable-kqueue             Do not enable kqueue(), fall back
-                               to select() [not set]
-  --enable-ipv6                Build ipv6 native InspIRCd [no]
-  --enable-remote-ipv6         Build with ipv6 support for remote
-                               servers on the network [yes]
-  --disable-remote-ipv6        Do not allow remote ipv6 servers [not set]
-  --with-cc=[filename]         Use an alternative g++ binary to
-                               build InspIRCd [g++]
-  --with-ident-length=[n]      Specify max length of ident [12]
-  --with-quit-length=[n]       Specify max length of quit [200]
-  --with-topic-length=[n]      Specify max length of topic [350]
-  --with-kick-length=[n]       Specify max length of kick [200]
-  --with-gecos-length=[n]      Specify max length of gecos [150]
-  --with-away-length=[n]       Specify max length of away [150]
-  --with-max-modes=[n]         Specify max modes per line which
-                               have parameters [20]
+  --socketengine=[name]        Sets the socket engine to be used. Possible values are
+                               $SELIST.
   --prefix=[directory]         Base directory to install into (if defined,
-                               can automatically define config, module, bin
-                              and library dirs as subdirectories of prefix)
+                               can automatically define config, data, module,
+                               log and binary dirs as subdirectories of prefix)
                                [$PWD]
   --config-dir=[directory]     Config file directory for config and SSL certs
                                [$PWD/conf]
+  --log-dir=[directory]               Log file directory for logs
+                               [$PWD/logs]
+  --data-dir=[directory]       Data directory for variable data, such as the permchannel
+                               configuration and the XLine database
+                               [$PWD/data]
   --module-dir=[directory]     Modules directory for loadable modules
                                [$PWD/modules]
   --binary-dir=[directory]     Binaries directory for core binary
                                [$PWD/bin]
-  --library-dir=[directory]    Library directory for core libraries
-                               [$PWD/lib]
+  --list-extras                Show current status of extra modules
+  --enable-extras=[extras]     Enable the specified list of extras
+  --disable-extras=[extras]    Disable the specified list of extras
   --help                       Show this help text and exit
 
-";
+EOH
        exit(0);
 }