// Utils for configure script /* +----------------------------------------------------------------------+ | PHP Version 7 | +----------------------------------------------------------------------+ | Copyright (c) 1997-2008 The PHP Group | +----------------------------------------------------------------------+ | This source file is subject to version 3.01 of the PHP license, | | that is bundled with this package in the file LICENSE, and is | | available through the world-wide-web at the following url: | | http://www.php.net/license/3_01.txt | | If you did not receive a copy of the PHP license and are unable to | | obtain it through the world-wide-web, please send a note to | | license@php.net so we can mail you a copy immediately. | +----------------------------------------------------------------------+ | Author: Wez Furlong | +----------------------------------------------------------------------+ */ // $Id: confutils.js,v 1.60.2.1.2.8.2.33 2009-05-29 07:43:07 kalle Exp $ var STDOUT = WScript.StdOut; var STDERR = WScript.StdErr; var WshShell = WScript.CreateObject("WScript.Shell"); var FSO = WScript.CreateObject("Scripting.FileSystemObject"); var MFO = null; var SYSTEM_DRIVE = WshShell.Environment("Process").Item("SystemDrive"); var PROGRAM_FILES = WshShell.Environment("Process").Item("ProgramFiles"); var PROGRAM_FILESx86 = WshShell.Environment("Process").Item("ProgramFiles(x86)"); var VCINSTALLDIR = WshShell.Environment("Process").Item("VCINSTALLDIR"); var PHP_SRC_DIR=FSO.GetParentFolderName(WScript.ScriptFullName); var VS_TOOLSET = true; var CLANG_TOOLSET = false; var ICC_TOOLSET = false; var VCVERS = -1; var CLANGVERS = -1; var INTELVERS = -1; var COMPILER_NUMERIC_VERSION = -1; var COMPILER_NAME = "unknown"; var WINVER = "0x0600"; /* Vista */ // There's a minimum requirement for re2c.. var MINRE2C = "0.13.4"; /* Store the enabled extensions (summary + QA check) */ var extensions_enabled = new Array(); /* Store the SAPI enabled (summary + QA check) */ var sapi_enabled = new Array(); /* Store the headers to install */ var headers_install = new Array(); /* Mapping CL version > human readable name */ var VC_VERSIONS = new Array(); VC_VERSIONS[1200] = 'MSVC6 (Visual C++ 6.0)'; VC_VERSIONS[1300] = 'MSVC7 (Visual C++ 2002)'; VC_VERSIONS[1310] = 'MSVC7.1 (Visual C++ 2003)'; VC_VERSIONS[1400] = 'MSVC8 (Visual C++ 2005)'; VC_VERSIONS[1500] = 'MSVC9 (Visual C++ 2008)'; VC_VERSIONS[1600] = 'MSVC10 (Visual C++ 2010)'; VC_VERSIONS[1700] = 'MSVC11 (Visual C++ 2012)'; VC_VERSIONS[1800] = 'MSVC12 (Visual C++ 2013)'; VC_VERSIONS[1900] = 'MSVC14 (Visual C++ 2015)'; var VC_VERSIONS_SHORT = new Array(); VC_VERSIONS_SHORT[1200] = 'VC6'; VC_VERSIONS_SHORT[1300] = 'VC7'; VC_VERSIONS_SHORT[1310] = 'VC7.1'; VC_VERSIONS_SHORT[1400] = 'VC8'; VC_VERSIONS_SHORT[1500] = 'VC9'; VC_VERSIONS_SHORT[1600] = 'VC10'; VC_VERSIONS_SHORT[1700] = 'VC11'; VC_VERSIONS_SHORT[1800] = 'VC12'; VC_VERSIONS_SHORT[1900] = 'VC14'; if (PROGRAM_FILES == null) { PROGRAM_FILES = "C:\\Program Files"; } if (MODE_PHPIZE) { if (!FSO.FileExists("config.w32")) { STDERR.WriteLine("Must be run from the root of the extension source"); WScript.Quit(10); } } else { if (!FSO.FileExists("README.GIT-RULES")) { STDERR.WriteLine("Must be run from the root of the php source"); WScript.Quit(10); } } var CWD = WshShell.CurrentDirectory; if (typeof(CWD) == "undefined") { CWD = FSO.GetParentFolderName(FSO.GetAbsolutePathName("README.GIT-RULES")); } /* defaults; we pick up the precise versions from configure.in */ var PHP_VERSION = 5; var PHP_MINOR_VERSION = 0; var PHP_RELEASE_VERSION = 0; var PHP_EXTRA_VERSION = ""; var PHP_VERSION_STRING = "5.0.0"; function get_version_numbers() { var cin = file_get_contents("configure.in"); if (cin.match(new RegExp("PHP_MAJOR_VERSION=(\\d+)"))) { PHP_VERSION = RegExp.$1; } if (cin.match(new RegExp("PHP_MINOR_VERSION=(\\d+)"))) { PHP_MINOR_VERSION = RegExp.$1; } if (cin.match(new RegExp("PHP_RELEASE_VERSION=(\\d+)"))) { PHP_RELEASE_VERSION = RegExp.$1; } PHP_VERSION_STRING = PHP_VERSION + "." + PHP_MINOR_VERSION + "." + PHP_RELEASE_VERSION; if (cin.match(new RegExp("PHP_EXTRA_VERSION=\"([^\"]+)\""))) { PHP_EXTRA_VERSION = RegExp.$1; if (PHP_EXTRA_VERSION.length) { PHP_VERSION_STRING += PHP_EXTRA_VERSION; } } DEFINE('PHP_VERSION_STRING', PHP_VERSION_STRING); } configure_args = new Array(); configure_subst = WScript.CreateObject("Scripting.Dictionary"); configure_hdr = WScript.CreateObject("Scripting.Dictionary"); build_dirs = new Array(); extension_include_code = ""; extension_module_ptrs = ""; if (!MODE_PHPIZE) { get_version_numbers(); } /* execute a command and return the output as a string */ function execute(command_line) { var e = WshShell.Exec(command_line); var ret = ""; ret = e.StdOut.ReadAll(); //STDOUT.WriteLine("command " + command_line); //STDOUT.WriteLine(ret); return ret; } function probe_binary(EXE, what) { // tricky escapes to get stderr redirection to work var command = 'cmd /c ""' + EXE; if (what == "version") { command = command + '" -v"'; } var version = execute(command + '" 2>&1"'); if (what == "64") { if (version.match(/x64/)) { return 1; } } else { if (version.match(/(\d+\.\d+(\.\d+)?(\.\d+)?)/)) { return RegExp.$1; } } return 0; } function condense_path(path) { path = FSO.GetAbsolutePathName(path); if (path.substr(0, CWD.length).toLowerCase() == CWD.toLowerCase() && (path.charCodeAt(CWD.length) == 92 || path.charCodeAt(CWD.length) == 47)) { return path.substr(CWD.length + 1); } var a = CWD.split("\\"); var b = path.split("\\"); var i, j; for (i = 0; i < b.length; i++) { if (a[i].toLowerCase() == b[i].toLowerCase()) continue; if (i > 0) { /* first difference found */ path = ""; for (j = 0; j < a.length - i; j++) { path += "..\\"; } for (j = i; j < b.length; j++) { path += b[j]; if (j < b.length - 1) path += "\\"; } return path; } /* on a different drive */ break; } return path; } function ConfigureArg(type, optname, helptext, defval) { var opptype = type == "enable" ? "disable" : "without"; if (defval == "yes" || defval == "yes,shared") { this.arg = "--" + opptype + "-" + optname; this.imparg = "--" + type + "-" + optname; } else { this.arg = "--" + type + "-" + optname; this.imparg = "--" + opptype + "-" + optname; } this.optname = optname; this.helptext = helptext; this.defval = defval; this.symval = optname.toUpperCase().replace(new RegExp("-", "g"), "_"); this.seen = false; this.argval = defval; } function ARG_WITH(optname, helptext, defval) { configure_args[configure_args.length] = new ConfigureArg("with", optname, helptext, defval); } function ARG_ENABLE(optname, helptext, defval) { configure_args[configure_args.length] = new ConfigureArg("enable", optname, helptext, defval); } function analyze_arg(argval) { var ret = new Array(); var shared = false; if (argval == "shared") { shared = true; argval = "yes"; } else if (argval == null) { /* nothing */ } else if (arg_match = argval.match(new RegExp("^shared,(.*)"))) { shared = true; argval = arg_match[1]; } else if (arg_match = argval.match(new RegExp("^(.*),shared$"))) { shared = true; argval = arg_match[1]; } ret[0] = shared; ret[1] = argval; return ret; } function word_wrap_and_indent(indent, text, line_suffix, indent_char) { if (text == null) { return ""; } var words = text.split(new RegExp("\\s+", "g")); var i = 0; var ret_text = ""; var this_line = ""; var t; var space = ""; var lines = 0; if (line_suffix == null) { line_suffix = ""; } if (indent_char == null) { indent_char = " "; } for (i = 0; i < indent; i++) { space += indent_char; } for (i = 0; i < words.length; i++) { if (this_line.length) { t = this_line + " " + words[i]; } else { t = words[i]; } if (t.length + indent > 78) { if (lines++) { ret_text += space; } ret_text += this_line + line_suffix + "\r\n"; this_line = ""; } if (this_line.length) { this_line += " " + words[i]; } else { this_line = words[i]; } } if (this_line.length) { if (lines) ret_text += space; ret_text += this_line; } return ret_text; } function conf_process_args() { var i, j; var configure_help_mode = false; var analyzed = false; var nice = "cscript /nologo configure.js "; var disable_all = false; args = WScript.Arguments; for (i = 0; i < args.length; i++) { arg = args(i); nice += ' "' + arg + '"'; if (arg == "--help") { configure_help_mode = true; break; } if (arg == "--disable-all") { disable_all = true; continue; } // If it is --foo=bar, split on the equals sign arg = arg.split("=", 2); argname = arg[0]; if (arg.length > 1) { argval = arg[1]; } else { argval = null; } // Find the arg found = false; for (j = 0; j < configure_args.length; j++) { if (argname == configure_args[j].imparg || argname == configure_args[j].arg) { found = true; arg = configure_args[j]; arg.seen = true; analyzed = analyze_arg(argval); /* Force shared when called after phpize */ if (MODE_PHPIZE) { shared = "shared"; } else { shared = analyzed[0]; } argval = analyzed[1]; if (argname == arg.imparg) { /* we matched the implicit, or default arg */ if (argval == null) { argval = arg.defval; } } else { /* we matched the non-default arg */ if (argval == null) { argval = arg.defval == "no" ? "yes" : "no"; } } arg.argval = argval; eval("PHP_" + arg.symval + " = argval;"); eval("PHP_" + arg.symval + "_SHARED = shared;"); break; } } if (!found) { STDERR.WriteLine("Unknown option " + argname + "; please try configure.js --help for a list of valid options"); WScript.Quit(2); } } if (configure_help_mode) { STDOUT.WriteLine(word_wrap_and_indent(0, "Options that enable extensions and SAPI will accept \ 'yes' or 'no' as a parameter. They also accept 'shared' \ as a synonym for 'yes' and request a shared build of that \ module. Not all modules can be built as shared modules; \ configure will display [shared] after the module name if \ can be built that way. \ " )); STDOUT.WriteBlankLines(1); // Measure width to pretty-print the output max_width = 0; for (i = 0; i < configure_args.length; i++) { arg = configure_args[i]; if (arg.arg.length > max_width) max_width = arg.arg.length; } for (i = 0; i < configure_args.length; i++) { arg = configure_args[i]; n = max_width - arg.arg.length; pad = " "; for (j = 0; j < n; j++) { pad += " "; } STDOUT.WriteLine(" " + arg.arg + pad + word_wrap_and_indent(max_width + 5, arg.helptext)); } WScript.Quit(1); } var snapshot_build_exclusions = new Array( 'debug', 'crt-debug', 'lzf-better-compression', 'php-build', 'snapshot-template', 'ereg', 'pcre-regex', 'fastcgi', 'force-cgi-redirect', 'path-info-check', 'zts', 'ipv6', 'memory-limit', 'zend-multibyte', 'fd-setsize', 'memory-manager', 't1lib', 'pgi', 'pgo' ); var force; // Now set any defaults we might have missed out earlier for (i = 0; i < configure_args.length; i++) { arg = configure_args[i]; if (arg.seen) continue; analyzed = analyze_arg(arg.defval); shared = analyzed[0]; argval = analyzed[1]; // Don't trust a default "yes" answer for a non-core module // in a snapshot build if (PHP_SNAPSHOT_BUILD != "no" && argval == "yes" && !shared) { force = true; for (j = 0; j < snapshot_build_exclusions.length; j++) { if (snapshot_build_exclusions[j] == arg.optname) { force = false; break; } } if (force) { /* now check if it is a core module */ force = false; for (j = 0; j < core_module_list.length; j++) { if (core_module_list[j] == arg.optname) { force = true; break; } } if (!force) { STDOUT.WriteLine("snapshot: forcing " + arg.arg + " shared"); shared = true; } } } if (PHP_SNAPSHOT_BUILD != "no" && argval == "no") { force = true; for (j = 0; j < snapshot_build_exclusions.length; j++) { if (snapshot_build_exclusions[j] == arg.optname) { force = false; break; } } if (force) { STDOUT.WriteLine("snapshot: forcing " + arg.optname + " on"); argval = "yes"; shared = true; } } if (disable_all) { force = true; for (j = 0; j < snapshot_build_exclusions.length; j++) { if (snapshot_build_exclusions[j] == arg.optname) { force = false; break; } } if (force) { if (arg.defval == '') { argval = ''; } else { argval = "no"; } shared = false; } } eval("PHP_" + arg.symval + " = argval;"); eval("PHP_" + arg.symval + "_SHARED = shared;"); } MFO = FSO.CreateTextFile("Makefile.objects", true); STDOUT.WriteLine("Saving configure options to config.nice.bat"); var nicefile = FSO.CreateTextFile("config.nice.bat", true); nicefile.WriteLine(nice + " %*"); nicefile.Close(); AC_DEFINE('CONFIGURE_COMMAND', nice, "Configure line"); } function DEFINE(name, value) { if (configure_subst.Exists(name)) { configure_subst.Remove(name); } configure_subst.Add(name, value); } // Searches a set of paths for a file; // returns the dir in which the file was found, // true if it was found in the default env path, // or false if it was not found at all. // env_name is the optional name of an env var // specifying the default path to search function search_paths(thing_to_find, explicit_path, env_name) { var i, found = false, place = false, file, env; STDOUT.Write("Checking for " + thing_to_find + " ... "); thing_to_find = thing_to_find.replace(new RegExp("/", "g"), "\\"); if (explicit_path != null) { if (typeof(explicit_path) == "string") { explicit_path = explicit_path.split(";"); } for (i = 0; i < explicit_path.length; i++) { file = glob(explicit_path[i] + "\\" + thing_to_find); if (file) { found = true; place = file[0]; place = place.substr(0, place.length - thing_to_find.length - 1); break; } } } if (!found && env_name != null) { env = WshShell.Environment("Process").Item(env_name); env = env.split(";"); for (i = 0; i < env.length; i++) { file = glob(env[i] + "\\" + thing_to_find); if (file) { found = true; place = true; break; } } } if (found && place == true) { STDOUT.WriteLine(" "); } else if (found) { STDOUT.WriteLine(" " + place); } else { STDOUT.WriteLine(" "); } return place; } function PATH_PROG(progname, additional_paths, symbol) { var exe; var place; var cyg_path = PHP_CYGWIN + "\\bin;" + PHP_CYGWIN + "\\usr\\local\\bin"; var php_build_bin_path = PHP_PHP_BUILD + "\\bin" exe = progname + ".exe"; if (additional_paths == null) { additional_paths = cyg_path; } else { additional_paths += ";" + cyg_path; } additional_paths = additional_paths + ";" + php_build_bin_path; place = search_paths(exe, additional_paths, "PATH"); if (place == true) { place = exe; } else if (place != false) { place = place + "\\" + exe; } if (place) { if (symbol == null) { symbol = progname.toUpperCase(); } DEFINE(symbol, place); } return place; } function find_pattern_in_path(pattern, path) { if (path == null) { return false; } var dirs = path.split(';'); var i; var items; for (i = 0; i < dirs.length; i++) { items = glob(dirs[i] + "\\" + pattern); if (items) { return condense_path(items[0]); } } return false; } function CHECK_LIB(libnames, target, path_to_check, common_name) { STDOUT.Write("Checking for library " + libnames + " ... "); if (common_name == null && target != null) { common_name = target; } if (path_to_check == null) { path_to_check = ""; } // if they specified a common name for the package that contains // the library, tag some useful defaults on to the end of the // path to be searched if (common_name != null) { path_to_check += ";" + PHP_PHP_BUILD + "\\" + common_name + "*"; path_to_check += ";" + PHP_PHP_BUILD + "\\lib\\" + common_name + "*"; path_to_check += ";..\\" + common_name + "*"; } // Determine target for build flags if (target == null) { target = ""; } else { target = "_" + target.toUpperCase(); } // Expand path to include general dirs path_to_check += ";" + php_usual_lib_suspects; // It is common practice to put libs under one of these dir names var subdirs = new Array(PHP_DEBUG == "yes" ? "Debug" : (PHP_DEBUG_PACK == "yes"?"Release_Dbg":"Release"), "lib", "libs", "libexec"); // libnames can be ; separated list of accepted library names libnames = libnames.split(';'); // for debug builds, lib may have _debug appended, we want that first if (PHP_DEBUG == "yes") { var length = libnames.length; for (var i = 0; i < length; i++) { var name = new String(libnames[i]); rExp = /.lib$/i; name = name.replace(rExp,"_debug.lib"); libnames.unshift(name); } } var i, j, k, libname; var location = false; var path = path_to_check.split(';'); for (i = 0; i < libnames.length; i++) { libname = libnames[i]; for (k = 0; k < path.length; k++) { location = glob(path[k] + "\\" + libname); if (location) { location = location[0]; break; } for (j = 0; j < subdirs.length; j++) { location = glob(path[k] + "\\" + subdirs[j] + "\\" + libname); if (location) { location = location[0]; break; } } if (location) break; } if (location) { location = condense_path(location); var libdir = FSO.GetParentFolderName(location); libname = FSO.GetFileName(location); ADD_FLAG("LDFLAGS" + target, '/libpath:"' + libdir + '" '); ADD_FLAG("ARFLAGS" + target, '/libpath:"' + libdir + '" '); ADD_FLAG("LIBS" + target, libname); STDOUT.WriteLine(location); return location; } // Check in their standard lib path location = find_pattern_in_path(libname, WshShell.Environment("Process").Item("LIB")); if (location) { location = condense_path(location); libname = FSO.GetFileName(location); ADD_FLAG("LIBS" + target, libname); STDOUT.WriteLine(" " + libname); return location; } // Check in their general extra libs path location = find_pattern_in_path(libname, PHP_EXTRA_LIBS); if (location) { location = condense_path(location); libname = FSO.GetFileName(location); ADD_FLAG("LIBS" + target, libname); STDOUT.WriteLine(""); return location; } } STDOUT.WriteLine(""); return false; } function OLD_CHECK_LIB(libnames, target, path_to_check) { if (target == null) { target = ""; } else { target = "_" + target.toUpperCase(); } if (path_to_check == null) { path_to_check = php_usual_lib_suspects; } else { path_to_check += ";" + php_usual_lib_suspects; } var have = 0; var p; var i; var libname; var subdir = PHP_DEBUG == "yes" ? "Debug" : (PHP_DEBUG_PACK == "yes"?"Release_Dbg":"Release"); libnames = libnames.split(';'); for (i = 0; i < libnames.length; i++) { libname = libnames[i]; p = search_paths(libname, path_to_check, "LIB"); if (!p) { p = search_paths(subdir + "\\" + libname, path_to_check, "LIB"); if (p) { p += "\\" + subdir; } } if (typeof(p) == "string") { ADD_FLAG("LDFLAGS" + target, '/libpath:"' + p + '" '); ADD_FLAG("ARFLAGS" + target, '/libpath:"' + p + '" '); ADD_FLAG("LIBS" + target, libname); have = 1; } else if (p == true) { ADD_FLAG("LIBS" + target, libname); have = 1; } else { /* not found in the defaults or the explicit paths, * so check the general extra libs; if we find * it here, no need to add another /libpath: for it as we * already have it covered, but we need to add the lib * to LIBS_XXX */ if (false != search_paths(libname, PHP_EXTRA_LIBS, null)) { ADD_FLAG("LIBS" + target, libname); have = 1; } } if (have) { break; } } // AC_DEFINE("HAVE_" + header_name.toUpperCase().replace(new RegExp("/\\\\-\.", "g"), "_"), have); return have; } function CHECK_FUNC_IN_HEADER(header_name, func_name, path_to_check, add_to_flag) { var c = false; var sym; STDOUT.Write("Checking for " + func_name + " in " + header_name + " ... "); c = GREP_HEADER(header_name, func_name, path_to_check); sym = func_name.toUpperCase(); sym = sym.replace(new RegExp("[\\\\/\.-]", "g"), "_"); if (typeof(add_to_flag) == "undefined") { AC_DEFINE("HAVE_" + sym, c ? 1 : 0); } else { ADD_FLAG(add_to_flag, "/DHAVE_" + sym + "=" + (c ? "1" : "0")); } if (c) { STDOUT.WriteLine("OK"); return c; } STDOUT.WriteLine("No"); return false; } function GREP_HEADER(header_name, regex, path_to_check) { var c = false; if (FSO.FileExists(path_to_check + "\\" + header_name)) { c = file_get_contents(path_to_check + "\\" + header_name); } if (!c) { /* look in the include path */ var p = search_paths(header_name, path_to_check, "INCLUDE"); if (typeof(p) == "string") { c = file_get_contents(p); } else if (p == false) { p = search_paths(header_name, PHP_EXTRA_INCLUDES, null); if (typeof(p) == "string") { c = file_get_contents(p); } } if (!c) { return false; } } if (typeof(regex) == "string") { regex = new RegExp(regex); } if (c.match(regex)) { /* caller can now use RegExp.$1 etc. to get at patterns */ return true; } return false; } function CHECK_HEADER_ADD_INCLUDE(header_name, flag_name, path_to_check, use_env, add_dir_part, add_to_flag_only) { var dir_part_to_add = ""; if (use_env == null) { use_env = true; } // if true, add the dir part of the header_name to the include path if (add_dir_part == null) { add_dir_part = false; } else if (add_dir_part) { var basename = FSO.GetFileName(header_name); dir_part_to_add = "\\" + header_name.substr(0, header_name.length - basename.length - 1); } if (path_to_check == null) { path_to_check = php_usual_include_suspects; } else { path_to_check += ";" + php_usual_include_suspects; } var p = search_paths(header_name, path_to_check, use_env ? "INCLUDE" : null); var have = 0; var sym; if (typeof(p) == "string") { ADD_FLAG(flag_name, '/I "' + p + dir_part_to_add + '" '); } else if (p == false) { /* not found in the defaults or the explicit paths, * so check the general extra includes; if we find * it here, no need to add another /I for it as we * already have it covered, unless we are adding * the dir part.... */ p = search_paths(header_name, PHP_EXTRA_INCLUDES, null); if (typeof(p) == "string" && add_dir_part) { ADD_FLAG(flag_name, '/I "' + p + dir_part_to_add + '" '); } } have = p ? 1 : 0 sym = header_name.toUpperCase(); sym = sym.replace(new RegExp("[\\\\/\.-]", "g"), "_"); if (typeof(add_to_flag_only) == "undefined" && flag_name.match(new RegExp("^CFLAGS_(.*)$"))) { add_to_flag_only = true; } if (typeof(add_to_flag_only) != "undefined") { ADD_FLAG(flag_name, "/DHAVE_" + sym + "=" + have); } else { AC_DEFINE("HAVE_" + sym, have, "have the " + header_name + " header file"); } return p; } /* emits rule to generate version info for a SAPI * or extension. Returns the name of the .res file * that will be generated */ function generate_version_info_resource(makefiletarget, basename, creditspath, sapi) { var resname = makefiletarget + ".res"; var res_desc = makefiletarget; var res_prod_name = "PHP " + makefiletarget; var credits; var thanks = ""; var logo = ""; var debug = ""; var project_url = "http://www.php.net"; var project_header = creditspath + "/php_" + basename + ".h"; var versioning = ""; if (sapi) { var internal_name = basename.toUpperCase() + " SAPI"; } else { var internal_name = basename.toUpperCase() + " extension"; } if (FSO.FileExists(creditspath + '/CREDITS')) { credits = FSO.OpenTextFile(creditspath + '/CREDITS', 1); res_desc = credits.ReadLine(); try { thanks = credits.ReadLine(); } catch (e) { thanks = null; } if (thanks == null) { thanks = ""; } else { thanks = "Thanks to " + thanks; } credits.Close(); } if (creditspath.match(new RegExp("pecl"))) { /* PECL project url - this will eventually work correctly for all */ project_url = "http://pecl.php.net/" + basename; /* keep independent versioning PECL-specific for now */ if (FSO.FileExists(project_header)) { if (header = FSO.OpenTextFile(project_header, 1)) { contents = header.ReadAll(); /* allowed: x.x.x[a|b|-alpha|-beta][RCx][-dev] */ if (contents.match(new RegExp('PHP_' + basename.toUpperCase() + '_VERSION(\\s+)"((\\d+\.\\d+(\.\\d+)?)((a|b)(\\d)?|\-[a-z]{3,5})?(RC\\d+)?(\-dev)?)'))) { project_version = RegExp.$2; file_version = RegExp.$3.split('.'); if (!file_version[2]) { file_version[2] = 0; } versioning = '\\"" /d EXT_FILE_VERSION=' + file_version[0] + ',' + file_version[1] + ',' + file_version[2] + ' /d EXT_VERSION="\\"' + project_version; } header.Close(); } } } if (makefiletarget.match(new RegExp("\\.exe$"))) { logo = " /d WANT_LOGO "; } if (PHP_DEBUG != "no") { debug = " /d _DEBUG"; } /** * Use user supplied template.rc if it exists */ if (FSO.FileExists(creditspath + '\\template.rc')) { MFO.WriteLine("$(BUILD_DIR)\\" + resname + ": " + creditspath + "\\template.rc"); MFO.WriteLine("\t$(RC) /fo $(BUILD_DIR)\\" + resname + logo + debug + ' /d FILE_DESCRIPTION="\\"' + res_desc + '\\"" /d FILE_NAME="\\"' + makefiletarget + '\\"" /d PRODUCT_NAME="\\"' + res_prod_name + versioning + '\\"" /d THANKS_GUYS="\\"' + thanks + '\\"" ' + creditspath + '\\template.rc'); return resname; } if (MODE_PHPIZE) { MFO.WriteLine("$(BUILD_DIR)\\" + resname + ": $(PHP_DIR)\\build\\template.rc"); MFO.WriteLine("\t$(RC) /I $(PHP_DIR)/include /n /fo $(BUILD_DIR)\\" + resname + logo + debug + ' /d FILE_DESCRIPTION="\\"' + res_desc + '\\"" /d FILE_NAME="\\"' + makefiletarget + '\\"" /d URL="\\"' + project_url + '\\"" /d INTERNAL_NAME="\\"' + internal_name + versioning + '\\"" /d THANKS_GUYS="\\"' + thanks + '\\"" $(PHP_DIR)\\build\\template.rc'); } else { MFO.WriteLine("$(BUILD_DIR)\\" + resname + ": win32\\build\\template.rc"); MFO.WriteLine("\t$(RC) /n /fo $(BUILD_DIR)\\" + resname + logo + debug + ' /d FILE_DESCRIPTION="\\"' + res_desc + '\\"" /d FILE_NAME="\\"' + makefiletarget + '\\"" /d URL="\\"' + project_url + '\\"" /d INTERNAL_NAME="\\"' + internal_name + versioning + '\\"" /d THANKS_GUYS="\\"' + thanks + '\\"" win32\\build\\template.rc'); } MFO.WriteBlankLines(1); return resname; } /* Check if PGO is enabled for given module. To disable PGO for a particular module, define a global variable by the following name scheme before SAPI() or EXTENSION() call var PHP_MYMODULE_PGO = false; */ function is_pgo_desired(mod) { var varname = "PHP_" + mod.toUpperCase() + "_PGO"; /* XXX enable PGO in phpize mode */ if (MODE_PHPIZE) { return false; } /* don't disable if there's no mention of the varname */ if (eval("typeof " + varname + " == 'undefined'")) { return true; } return eval("!!" + varname); } function SAPI(sapiname, file_list, makefiletarget, cflags, obj_dir) { var SAPI = sapiname.toUpperCase(); var ldflags; var resname; var ld; var manifest; if (typeof(obj_dir) == "undefined") { sapiname_for_printing = configure_module_dirname; } else { sapiname_for_printing = configure_module_dirname + " (via " + obj_dir + ")"; } STDOUT.WriteLine("Enabling SAPI " + sapiname_for_printing); MFO.WriteBlankLines(1); MFO.WriteLine("# objects for SAPI " + sapiname); MFO.WriteBlankLines(1); if (cflags) { ADD_FLAG('CFLAGS_' + SAPI, cflags); } ADD_SOURCES(configure_module_dirname, file_list, sapiname, obj_dir); MFO.WriteBlankLines(1); MFO.WriteLine("# SAPI " + sapiname); MFO.WriteBlankLines(1); /* generate a .res file containing version information */ resname = generate_version_info_resource(makefiletarget, sapiname, configure_module_dirname, true); MFO.WriteLine(makefiletarget + ": $(BUILD_DIR)\\" + makefiletarget); MFO.WriteLine("\t@echo SAPI " + sapiname_for_printing + " build complete"); if (MODE_PHPIZE) { MFO.WriteLine("$(BUILD_DIR)\\" + makefiletarget + ": $(DEPS_" + SAPI + ") $(" + SAPI + "_GLOBAL_OBJS) $(PHPLIB) $(BUILD_DIR)\\" + resname); } else { MFO.WriteLine("$(BUILD_DIR)\\" + makefiletarget + ": $(DEPS_" + SAPI + ") $(" + SAPI + "_GLOBAL_OBJS) $(BUILD_DIR)\\$(PHPLIB) $(BUILD_DIR)\\" + resname); } if (makefiletarget.match(new RegExp("\\.dll$"))) { ldflags = "/dll $(LDFLAGS)"; manifest = "-@$(_VC_MANIFEST_EMBED_DLL)"; } else if (makefiletarget.match(new RegExp("\\.lib$"))) { ldflags = "$(ARFLAGS)"; ld = "$(MAKE_LIB)"; } else { ldflags = "$(LDFLAGS)"; manifest = "-@$(_VC_MANIFEST_EMBED_EXE)"; } if(is_pgo_desired(sapiname) && (PHP_PGI == "yes" || PHP_PGO != "no")) { // Add compiler and link flags if PGO options are selected if (PHP_DEBUG != "yes" && PHP_PGI == "yes") { ADD_FLAG('CFLAGS_' + SAPI, "/GL /O2"); ADD_FLAG('LDFLAGS_' + SAPI, "/LTCG:PGINSTRUMENT"); } else if (PHP_DEBUG != "yes" && PHP_PGO != "no") { ADD_FLAG('CFLAGS_' + SAPI, "/GL /O2"); ADD_FLAG('LDFLAGS_' + SAPI, "/LTCG:PGUPDATE"); } ldflags += " /PGD:$(PGOPGD_DIR)\\" + makefiletarget.substring(0, makefiletarget.indexOf(".")) + ".pgd"; } if (MODE_PHPIZE) { if (ld) { MFO.WriteLine("\t" + ld + " /nologo /out:$(BUILD_DIR)\\" + makefiletarget + " " + ldflags + " $(" + SAPI + "_GLOBAL_OBJS_RESP) $(PHPLIB) $(ARFLAGS_" + SAPI + ") $(LIBS_" + SAPI + ") $(BUILD_DIR)\\" + resname); } else { ld = '@"$(LINK)"'; MFO.WriteLine("\t" + ld + " /nologo " + " $(" + SAPI + "_GLOBAL_OBJS_RESP) $(PHPLIB) $(LIBS_" + SAPI + ") $(BUILD_DIR)\\" + resname + " /out:$(BUILD_DIR)\\" + makefiletarget + " " + ldflags + " $(LDFLAGS_" + SAPI + ")"); } } else { if (ld) { MFO.WriteLine("\t" + ld + " /nologo /out:$(BUILD_DIR)\\" + makefiletarget + " " + ldflags + " $(" + SAPI + "_GLOBAL_OBJS_RESP) $(BUILD_DIR)\\$(PHPLIB) $(ARFLAGS_" + SAPI + ") $(LIBS_" + SAPI + ") $(BUILD_DIR)\\" + resname); } else { ld = '@"$(LINK)"'; MFO.WriteLine("\t" + ld + " /nologo " + " $(" + SAPI + "_GLOBAL_OBJS_RESP) $(BUILD_DIR)\\$(PHPLIB) $(LIBS_" + SAPI + ") $(BUILD_DIR)\\" + resname + " /out:$(BUILD_DIR)\\" + makefiletarget + " " + ldflags + " $(LDFLAGS_" + SAPI + ")"); } } if (manifest) { MFO.WriteLine("\t" + manifest); } DEFINE('CFLAGS_' + SAPI + '_OBJ', '$(CFLAGS_' + SAPI + ')'); if (configure_module_dirname.match("pecl")) { ADD_FLAG("PECL_TARGETS", makefiletarget); } else { ADD_FLAG("SAPI_TARGETS", makefiletarget); } MFO.WriteBlankLines(1); sapi_enabled[sapi_enabled.length] = [sapiname]; } function ADD_DIST_FILE(filename) { if (configure_module_dirname.match("pecl")) { ADD_FLAG("PECL_EXTRA_DIST_FILES", filename); } else { ADD_FLAG("PHP_EXTRA_DIST_FILES", filename); } } function file_get_contents(filename) { var f, c; try { f = FSO.OpenTextFile(filename, 1); c = f.ReadAll(); f.Close(); return c; } catch (e) { STDOUT.WriteLine("Problem reading " + filename); return false; } } // Add a dependency on another extension, so that // the dependencies are built before extname function ADD_EXTENSION_DEP(extname, dependson, optional) { var EXT = extname.toUpperCase(); var DEP = dependson.toUpperCase(); var dep_present = false; var dep_shared = false; try { dep_present = eval("PHP_" + DEP); if (dep_present != "no") { try { dep_shared = eval("PHP_" + DEP + "_SHARED"); } catch (e) { dep_shared = false; } } } catch (e) { dep_present = "no"; } if (optional) { if (dep_present == "no") { MESSAGE("\t" + dependson + " not found: " + dependson + " support in " + extname + " disabled"); return false; } } var ext_shared = eval("PHP_" + EXT + "_SHARED"); if (dep_shared) { if (!ext_shared) { if (optional) { MESSAGE("\tstatic " + extname + " cannot depend on shared " + dependson + ": " + dependson + "support disabled"); return false; } ERROR("static " + extname + " cannot depend on shared " + dependson); } ADD_FLAG("LIBS_" + EXT, "php_" + dependson + ".lib"); if (MODE_PHPIZE) { ADD_FLAG("LDFLAGS_" + EXT, "/libpath:$(BUILD_DIR_DEV)\\lib"); ADD_FLAG("DEPS_" + EXT, "$(BUILD_DIR_DEV)\\lib\\php_" + dependson + ".lib"); } else { ADD_FLAG("LDFLAGS_" + EXT, "/libpath:$(BUILD_DIR)"); ADD_FLAG("DEPS_" + EXT, "$(BUILD_DIR)\\php_" + dependson + ".lib"); } } else { if (dep_present == "no") { if (ext_shared) { WARNING(extname + " cannot be built: missing dependency, " + dependson + " not found"); var dllname = ' php_' + extname + '.dll'; if (!REMOVE_TARGET(dllname, 'EXT_TARGETS')) { REMOVE_TARGET(dllname, 'PECL_TARGETS'); } return false; } ERROR("Cannot build " + extname + "; " + dependson + " not enabled"); return false; } } // dependency is statically built-in to PHP return true; } var static_pgo_enabled = false; function EXTENSION(extname, file_list, shared, cflags, dllname, obj_dir) { var objs = null; var EXT = extname.toUpperCase(); var extname_for_printing; var ldflags; if (shared == null) { eval("shared = PHP_" + EXT + "_SHARED;"); } else { eval("PHP_" + EXT + "_SHARED = shared;"); } if (cflags == null) { cflags = ""; } if (typeof(obj_dir) == "undefined") { extname_for_printing = configure_module_dirname; } else { extname_for_printing = configure_module_dirname + " (via " + obj_dir + ")"; } if (shared) { STDOUT.WriteLine("Enabling extension " + extname_for_printing + " [shared]"); cflags = "/D COMPILE_DL_" + EXT + " /D " + EXT + "_EXPORTS=1 " + cflags; ADD_FLAG("CFLAGS_PHP", "/D COMPILE_DL_" + EXT); } else { STDOUT.WriteLine("Enabling extension " + extname_for_printing); } MFO.WriteBlankLines(1); MFO.WriteLine("# objects for EXT " + extname); MFO.WriteBlankLines(1); ADD_SOURCES(configure_module_dirname, file_list, extname, obj_dir); MFO.WriteBlankLines(1); if (shared) { if (dllname == null) { dllname = "php_" + extname + ".dll"; } var libname = dllname.substring(0, dllname.length-4) + ".lib"; var resname = generate_version_info_resource(dllname, extname, configure_module_dirname, false); var ld = '@"$(LINK)"'; ldflags = ""; if (is_pgo_desired(extname) && (PHP_PGI == "yes" || PHP_PGO != "no")) { // Add compiler and link flags if PGO options are selected if (PHP_DEBUG != "yes" && PHP_PGI == "yes") { ADD_FLAG('LDFLAGS_' + EXT, "/LTCG:PGINSTRUMENT"); } else if (PHP_DEBUG != "yes" && PHP_PGO != "no") { ADD_FLAG('LDFLAGS_' + EXT, "/LTCG:PGUPDATE"); } ADD_FLAG('CFLAGS_' + EXT, "/GL /O2"); ldflags = " /PGD:$(PGOPGD_DIR)\\" + dllname.substring(0, dllname.indexOf(".")) + ".pgd"; } MFO.WriteLine("$(BUILD_DIR)\\" + libname + ": $(BUILD_DIR)\\" + dllname); MFO.WriteBlankLines(1); if (MODE_PHPIZE) { MFO.WriteLine("$(BUILD_DIR)\\" + dllname + ": $(DEPS_" + EXT + ") $(" + EXT + "_GLOBAL_OBJS) $(PHPLIB) $(BUILD_DIR)\\" + resname); MFO.WriteLine("\t" + ld + " $(" + EXT + "_GLOBAL_OBJS_RESP) $(PHPLIB) $(LIBS_" + EXT + ") $(LIBS) $(BUILD_DIR)\\" + resname + " /out:$(BUILD_DIR)\\" + dllname + " $(DLL_LDFLAGS) $(LDFLAGS) $(LDFLAGS_" + EXT + ")"); } else { MFO.WriteLine("$(BUILD_DIR)\\" + dllname + ": $(DEPS_" + EXT + ") $(" + EXT + "_GLOBAL_OBJS) $(BUILD_DIR)\\$(PHPLIB) $(BUILD_DIR)\\" + resname); MFO.WriteLine("\t" + ld + " $(" + EXT + "_GLOBAL_OBJS_RESP) $(BUILD_DIR)\\$(PHPLIB) $(LIBS_" + EXT + ") $(LIBS) $(BUILD_DIR)\\" + resname + " /out:$(BUILD_DIR)\\" + dllname + ldflags + " $(DLL_LDFLAGS) $(LDFLAGS) $(LDFLAGS_" + EXT + ")"); } MFO.WriteLine("\t-@$(_VC_MANIFEST_EMBED_DLL)"); MFO.WriteBlankLines(1); if (configure_module_dirname.match("pecl")) { ADD_FLAG("PECL_TARGETS", dllname); } else { ADD_FLAG("EXT_TARGETS", dllname); } MFO.WriteLine(dllname + ": $(BUILD_DIR)\\" + dllname); MFO.WriteLine("\t@echo EXT " + extname + " build complete"); MFO.WriteBlankLines(1); DEFINE('CFLAGS_' + EXT + '_OBJ', '$(CFLAGS_' + EXT + ')'); } else { ADD_FLAG("STATIC_EXT_OBJS", "$(" + EXT + "_GLOBAL_OBJS)"); ADD_FLAG("STATIC_EXT_OBJS_RESP", "$(" + EXT + "_GLOBAL_OBJS_RESP)"); ADD_FLAG("STATIC_EXT_LIBS", "$(LIBS_" + EXT + ")"); ADD_FLAG("STATIC_EXT_LDFLAGS", "$(LDFLAGS_" + EXT + ")"); ADD_FLAG("STATIC_EXT_CFLAGS", "$(CFLAGS_" + EXT + ")"); if (is_pgo_desired(extname) && (PHP_PGI == "yes" || PHP_PGO != "no")) { if (!static_pgo_enabled) { if (PHP_DEBUG != "yes" && PHP_PGI == "yes") { ADD_FLAG('STATIC_EXT_LDFLAGS', "/LTCG:PGINSTRUMENT"); } else if (PHP_DEBUG != "yes" && PHP_PGO != "no") { ADD_FLAG('STATIC_EXT_LDFLAGS', "/LTCG:PGUPDATE"); } ADD_FLAG("STATIC_EXT_CFLAGS", "/GL /O2"); static_pgo_enabled = true; } } /* find the header that declares the module pointer, * so we can include it in internal_functions.c */ var ext_dir = FSO.GetFolder(configure_module_dirname); var fc = new Enumerator(ext_dir.Files); var re = /\.h$/; var s, c; for (; !fc.atEnd(); fc.moveNext()) { s = fc.item() + ""; if (s.match(re)) { c = file_get_contents(s); if (c.match("phpext_")) { extension_include_code += '#include "' + configure_module_dirname + '/' + FSO.GetFileName(s) + '"\r\n'; } } } extension_module_ptrs += '\tphpext_' + extname + '_ptr,\r\n'; DEFINE('CFLAGS_' + EXT + '_OBJ', '$(CFLAGS_PHP) $(CFLAGS_' + EXT + ')'); } if (MODE_PHPIZE && FSO.FileExists(PHP_DIR + "/include/main/config.pickle.h")) { cflags = "/FI main/config.pickle.h " + cflags; } ADD_FLAG("CFLAGS_" + EXT, cflags); extensions_enabled[extensions_enabled.length] = [extname, shared ? 'shared' : 'static']; } function ADD_SOURCES(dir, file_list, target, obj_dir) { var i; var tv; var src, obj, sym, flags; if (target == null) { target = "php"; } sym = target.toUpperCase() + "_GLOBAL_OBJS"; flags = "CFLAGS_" + target.toUpperCase() + '_OBJ'; var bd = get_define('BUILD_DIR'); var respd = bd + '\\resp'; if (!FSO.FolderExists(respd)) { FSO.CreateFolder(respd); } var obj_lst_fn = respd + '\\' + sym + '.txt'; var resp = ""; if (configure_subst.Exists(sym)) { tv = configure_subst.Item(sym); } else { if (FSO.FileExists(obj_lst_fn)) { FSO.DeleteFile(obj_lst_fn, true); } tv = ""; } file_list = file_list.split(new RegExp("\\s+")); file_list.sort(); var re = new RegExp("\.[a-z0-9A-Z]+$"); dir = dir.replace(new RegExp("/", "g"), "\\"); var objs_line = ""; var srcs_line = ""; var sub_build = "$(BUILD_DIR)\\"; /* if module dir is not a child of the main source dir, * we need to tweak it; we should have detected such a * case in condense_path and rewritten the path to * be relative. * This probably breaks for non-sibling dirs, but that * is not a problem as buildconf only checks for pecl * as either a child or a sibling */ if (obj_dir == null) { var build_dir = dir.replace(new RegExp("^..\\\\"), ""); var mangle_dir = build_dir.replace(new RegExp("[\\\\/.-]", "g"), "_"); var bd_flags_name = "CFLAGS_BD_" + mangle_dir.toUpperCase(); } else { var build_dir = obj_dir.replace(new RegExp("^..\\\\"), ""); var mangle_dir = build_dir.replace(new RegExp("[\\\\/.-]", "g"), "_"); var bd_flags_name = "CFLAGS_BD_" + mangle_dir.toUpperCase(); } var dirs = build_dir.split("\\"); var i, d = ""; for (i = 0; i < dirs.length; i++) { d += dirs[i]; build_dirs[build_dirs.length] = d; d += "\\"; } sub_build += d; DEFINE(bd_flags_name, "/Fp" + sub_build + " /FR" + sub_build + " "); if (VS_TOOLSET) { ADD_FLAG(bd_flags_name, "/Fd" + sub_build); } for (i in file_list) { src = file_list[i]; obj = src.replace(re, ".obj"); tv += " " + sub_build + obj; resp += " " + sub_build.replace('$(BUILD_DIR)', bd) + obj; if (!PHP_MP_DISABLED) { if (i > 0) { objs_line += " " + sub_build + obj; srcs_line += " " + dir + "\\" + src; } else { objs_line = sub_build + obj; srcs_line = dir + "\\" + src; } } else { MFO.WriteLine(sub_build + obj + ": " + dir + "\\" + src); if (PHP_ANALYZER == "pvs") { MFO.WriteLine("\t@\"$(PVS_STUDIO)\" --cl-params $(" + flags + ") $(CFLAGS) $(" + bd_flags_name + ") /c " + dir + "\\" + src + " --source-file " + dir + "\\" + src + " --cfg PVS-Studio.conf --errors-off \"V122 V117 V111\" "); } MFO.WriteLine("\t@$(CC) $(" + flags + ") $(CFLAGS) $(" + bd_flags_name + ") /c " + dir + "\\" + src + " /Fo" + sub_build + obj); } } if (!PHP_MP_DISABLED) { MFO.WriteLine(objs_line + ": " + srcs_line); MFO.WriteLine("\t$(CC) $(" + flags + ") $(CFLAGS) /Fo" + sub_build + " $(" + bd_flags_name + ") /c " + srcs_line); } DEFINE(sym, tv); /* Generate the response file and define it to the Makefile. This can be useful when getting the "command line too long" linker errors. */ var obj_lst_fh = null; if (!FSO.FileExists(obj_lst_fn)) { obj_lst_fh = FSO.CreateTextFile(obj_lst_fn); //STDOUT.WriteLine("Creating " + obj_lst_fn); } else { //STDOUT.WriteLine("Appending to " + obj_lst_fn); obj_lst_fh = FSO.OpenTextFile(obj_lst_fn, 8); } obj_lst_fh.Write(" " + resp); obj_lst_fh.Close(); DEFINE(sym + "_RESP", '@"' + obj_lst_fn + '"'); } function REMOVE_TARGET(dllname, flag) { var dllname = dllname.replace(/\s/g, ""); var EXT = dllname.replace(/php_(\S+)\.dll/, "$1").toUpperCase(); var php_flags = configure_subst.Item("CFLAGS_PHP"); if (configure_subst.Exists(flag)) { var targets = configure_subst.Item(flag); if (targets.match(dllname)) { configure_subst.Remove(flag); targets = targets.replace(dllname, ""); targets = targets.replace(/\s+/, " "); targets = targets.replace(/\s$/, ""); configure_subst.Add(flag, targets); configure_hdr.Add("HAVE_" + EXT, new Array(0, "")); configure_subst.Item("CFLAGS_PHP") = php_flags.replace(" /D COMPILE_DL_" + EXT, ""); extensions_enabled.pop(); return true; } } return false; } function generate_internal_functions() { var infile, outfile; var indata; STDOUT.WriteLine("Generating main/internal_functions.c"); infile = FSO.OpenTextFile("main/internal_functions.c.in", 1); indata = infile.ReadAll(); infile.Close(); indata = indata.replace("@EXT_INCLUDE_CODE@", extension_include_code); indata = indata.replace("@EXT_MODULE_PTRS@", extension_module_ptrs); if (FSO.FileExists("main/internal_functions.c")) { var origdata = file_get_contents("main/internal_functions.c"); if (origdata == indata) { STDOUT.WriteLine("\t[content unchanged; skipping]"); return; } } outfile = FSO.CreateTextFile("main/internal_functions.c", true); outfile.Write(indata); outfile.Close(); } function output_as_table(header, ar_out) { var l = header.length; var cols = 80; var fixedlength = ""; var t = 0; var i,j,k,m; var out = "| "; var min = new Array(l); var max = new Array(l); if (!!ar_out[0] && l != ar_out[0].length) { STDOUT.WriteLine("Invalid header argument, can't output the table " + l + " " + ar_out[0].length ); return; } for (j=0; j < l; j++) { var tmax, tmin; /*Figure out the max length per column */ tmin = 0; tmax = 0; for (k = 0; k < ar_out.length; k++) { if(typeof ar_out[k][j] != 'undefined') { var t = ar_out[k][j].length; if (t > tmax) tmax = t; else if (t < tmin) tmin = t; } } if (tmax > header[j].length) { max[j] = tmax; } else { max[j] = header[j].length; } if (tmin < header[j].length) { min[j] = header[j].length; } } sep = ""; k = 0; for (i = 0; i < l; i++) { k += max[i] + 3; } k++; for (j=0; j < k; j++) { sep += "-"; } STDOUT.WriteLine(sep); out = "|"; for (j=0; j < l; j++) { out += " " + header[j]; for (var i = 0; i < (max[j] - header[j].length); i++){ out += " "; } out += " |"; } STDOUT.WriteLine(out); STDOUT.WriteLine(sep); out = "|"; for (i=0; i < ar_out.length; i++) { line = ar_out[i]; for (j=0; j < l; j++) { out += " " + line[j]; if(typeof line[j] != 'undefined') { for (var k = 0; k < (max[j] - line[j].length); k++){ out += " "; } } out += " |"; } STDOUT.WriteLine(out); out = "|"; } STDOUT.WriteLine(sep); } function write_summary() { var ar = new Array(); STDOUT.WriteBlankLines(2); STDOUT.WriteLine("Enabled extensions:"); output_as_table(["Extension", "Mode"], extensions_enabled.sort()); STDOUT.WriteBlankLines(2); if (!MODE_PHPIZE) { STDOUT.WriteLine("Enabled SAPI:"); output_as_table(["Sapi Name"], sapi_enabled); STDOUT.WriteBlankLines(2); } ar[0] = ['Build type', PHP_DEBUG == "yes" ? "Debug" : "Release"]; ar[1] = ['Thread Safety', PHP_ZTS == "yes" ? "Yes" : "No"]; ar[2] = ['Compiler', COMPILER_NAME]; ar[3] = ['Architecture', X64 ? 'x64' : 'x86']; if (PHP_PGO == "yes") { ar[4] = ['Optimization', "PGO"]; } else if (PHP_PGI == "yes") { ar[4] = ['Optimization', "PGI"]; } else { ar[4] = ['Optimization', PHP_DEBUG == "yes" ? "disabled" : "PGO disabled"]; } if (PHP_ANALYZER == "vs") { ar[5] = ['Static analyzer', 'Visual Studio']; } else if (PHP_ANALYZER == "pvs") { ar[5] = ['Static analyzer', 'PVS-Studio']; } else { ar[5] = ['Static analyzer', 'disabled']; } output_as_table(["",""], ar); STDOUT.WriteBlankLines(2); } function generate_files() { var i, dir, bd, last; STDOUT.WriteBlankLines(1); STDOUT.WriteLine("Creating build dirs..."); dir = get_define("BUILD_DIR"); build_dirs.sort(); last = null; if (!FSO.FolderExists(dir)) { FSO.CreateFolder(dir); } for (i = 0; i < build_dirs.length; i++) { bd = FSO.BuildPath(dir, build_dirs[i]); if (bd == last) { continue; } last = bd; build_dir = get_define('BUILD_DIR'); build_dir = build_dir.replace(new RegExp("\\\\", "g"), "\\\\"); if (build_dir.substr(build_dir.Length - 2, 2) != '\\\\') { build_dir += '\\\\'; } ADD_FLAG("BUILD_DIRS_SUB", bd.replace(new RegExp(build_dir), '')); if (!FSO.FolderExists(bd)) { FSO.CreateFolder(bd); } } STDOUT.WriteLine("Generating files..."); generate_makefile(); if (!MODE_PHPIZE) { generate_internal_functions(); generate_config_h(); generate_phpize(); } else { generate_config_pickle_h(); generate_ext_pickle(); } STDOUT.WriteLine("Done."); STDOUT.WriteBlankLines(1); write_summary(); if (PHP_SNAPSHOT_BUILD != "no") { STDOUT.WriteLine("Type 'nmake snap' to build a PHP snapshot"); } else { STDOUT.WriteLine("Type 'nmake' to build PHP"); } } function generate_ext_pickle() { var content; var DEPS = null; var dest; var deps_lines = new Array(); var build_var_name = function(name) { return "PHP_" + name.toUpperCase(); } STDOUT.WriteLine("Generating pickle deps"); dest = PHP_DIR + "/script/"; if (!FSO.FolderExists(dest)) { FSO.CreateFolder(dest); } if (FSO.FileExists(dest + "/ext_pickle.js")) { DEPS = FSO.OpenTextFile(dest + "/ext_pickle.js", 1); while (!DEPS.AtEndOfStream) { var ln = DEPS.ReadLine(); var found = false; for (var i in extensions_enabled) { var reg0 = new RegExp(build_var_name(extensions_enabled[i][0]) + "\s*=.+", "g"); var reg1 = new RegExp(build_var_name(extensions_enabled[i][0]) + "_SHARED" + "\s*=.+", "g"); if (ln.match(reg1) || ln.match(reg0)) { found = true; break; } } if (!found) { deps_lines.push(ln); } } } for (var i in extensions_enabled) { deps_lines.push(build_var_name(extensions_enabled[i][0]) + "=true;"); deps_lines.push(build_var_name(extensions_enabled[i][0]) + "_SHARED=" + (extensions_enabled[i][1] == 'shared' ? 'true' : 'false') + ";"); } if (!!DEPS) { DEPS.Close(); DEPS = null; } /* Replace the ext_pickle.js with the new content */ DEPS = FSO.CreateTextFile(dest + "/ext_pickle.js", true); for (var j in deps_lines) { DEPS.WriteLine(deps_lines[j]); } DEPS.Close(); } function generate_config_pickle_h() { var outfile = null; var lines = new Array(); var keys = (new VBArray(configure_hdr.Keys())).toArray(); dest = PHP_DIR + "/include/main"; var ignore_key = function(key) { var ignores = [ "CONFIGURE_COMMAND", "PHP_COMPILER_ID", "COMPILER", "ARCHITECTURE", "HAVE_STRNLEN", "PHP_DIR" ]; for (var k in ignores) { if (ignores[k] == key) { return true; } } return false; } STDOUT.WriteLine("Generating main/config.pickle.h"); if (FSO.FileExists(dest + "/config.pickle.h")) { outfile = FSO.OpenTextFile(dest + "/config.pickle.h", 1); while (!outfile.AtEndOfStream) { var found = false; var ln = outfile.ReadLine(); for (var i in keys) { var reg = new RegExp("#define[\s ]+" + keys[i] + "[\s ]*.*", "g"); if (ln.match(reg)) { found = true; break; } } if (!found) { lines.push(ln); } } } for (var i in keys) { var item = configure_hdr.Item(keys[i]); if (ignore_key(keys[i])) { continue; } /* XXX fix comment handling */ /*if (!lines[j].match(/^#define.+/g)) { continue; }*/ lines.push("#define " + keys[i] + " " + item[0]); } if (outfile) { outfile.Close(); outfile = null; } outfile = FSO.CreateTextFile(dest + "/config.pickle.h", true); for (var k in lines) { outfile.WriteLine(lines[k]); } outfile.Close(); } function generate_config_h() { var infile, outfile; var indata; var prefix; prefix = PHP_PREFIX.replace(new RegExp("\\\\", "g"), "\\\\"); STDOUT.WriteLine("Generating main/config.w32.h"); infile = FSO.OpenTextFile("win32/build/config.w32.h.in", 1); indata = infile.ReadAll(); infile.Close(); outfile = FSO.CreateTextFile("main/config.w32.h", true); indata = indata.replace(new RegExp("@PREFIX@", "g"), prefix); outfile.Write(indata); var keys = (new VBArray(configure_hdr.Keys())).toArray(); var i, j; var item; var pieces, stuff_to_crack, chunk; outfile.WriteBlankLines(1); outfile.WriteLine("/* values determined by configure.js */"); for (i in keys) { item = configure_hdr.Item(keys[i]); outfile.WriteBlankLines(1); pieces = item[0]; if (item[1] != undefined) { outfile.WriteLine("/* " + item[1] + " */"); } if (typeof(pieces) == "string" && pieces.charCodeAt(0) == 34) { /* quoted string have a maximal length of 2k under vc. * solution is to crack them and let the compiler concat * them implicitly */ stuff_to_crack = pieces; pieces = ""; while (stuff_to_crack.length) { j = 65; while (stuff_to_crack.charCodeAt(j) != 32 && j < stuff_to_crack.length) j++; chunk = stuff_to_crack.substr(0, j); pieces += chunk; stuff_to_crack = stuff_to_crack.substr(chunk.length); if (stuff_to_crack.length) pieces += '" "'; } } outfile.WriteLine("#define " + keys[i] + " " + pieces); } if (VS_TOOLSET && VCVERS >= 1800) { outfile.WriteLine(""); outfile.WriteLine("#define HAVE_ACOSH 1"); outfile.WriteLine("#define HAVE_ASINH 1"); outfile.WriteLine("#define HAVE_ATANH 1"); } outfile.Close(); } function generate_phpize() { STDOUT.WriteLine("Generating phpize"); dest = get_define("BUILD_DIR") + '/devel'; if (!FSO.FolderExists(dest)) { FSO.CreateFolder(dest); } var MF = FSO.CreateTextFile(dest + "/phpize.js", true); var DEPS = FSO.CreateTextFile(dest + "/ext_deps.js", true); prefix = get_define("PHP_PREFIX"); prefix = prefix.replace(new RegExp("/", "g"), "\\"); prefix = prefix.replace(new RegExp("\\\\", "g"), "\\\\"); MF.WriteLine("var PHP_PREFIX=" + '"' + prefix + '"'); MF.WriteLine("var PHP_ZTS=" + '"' + (PHP_ZTS.toLowerCase() == "yes" ? "Yes" : "No") + '"'); MF.WriteLine("var VC_VERSION=" + VCVERS); MF.WriteLine("var PHP_VERSION=" + PHP_VERSION); MF.WriteLine("var PHP_MINOR_VERSION=" + PHP_MINOR_VERSION); MF.WriteLine("var PHP_RELEASE_VERSION=" + PHP_RELEASE_VERSION); MF.WriteBlankLines(1); MF.WriteLine("/* Genereted extensions list with mode (static/shared) */"); var count = extensions_enabled.length; for (i in extensions_enabled) { out = "PHP_" + extensions_enabled[i][0].toUpperCase() + "_SHARED=" + (extensions_enabled[i][1] == 'shared' ? 'true' : 'false') + ";"; DEPS.WriteLine("PHP_" + extensions_enabled[i][0].toUpperCase() + "=true;"); DEPS.WriteLine(out); MF.WriteLine(out); } MF.WriteBlankLines(2); MF.WriteLine("/* Genereted win32/build/phpize.js.in */"); MF.WriteBlankLines(1); MF.Write(file_get_contents("win32/build/phpize.js.in")); MF.Close(); DEPS.Close(); /* Generate flags file */ /* spit out variable definitions */ CJ = FSO.CreateTextFile(dest + "/config.phpize.js"); CJ.WriteLine("var PHP_ZTS =" + '"' + PHP_ZTS + '"'); CJ.WriteLine("var PHP_DLL_LIB =" + '"' + get_define('PHPLIB') + '"'); CJ.WriteLine("var PHP_DLL =" + '"' + get_define('PHPDLL') + '"'); CJ.WriteBlankLines(1); CJ.Close(); } function generate_makefile() { STDOUT.WriteLine("Generating Makefile"); var MF = FSO.CreateTextFile("Makefile", true); MF.WriteLine("# Generated by configure.js"); /* spit out variable definitions */ var keys = (new VBArray(configure_subst.Keys())).toArray(); var i; MF.WriteLine("PHP_SRC_DIR =" + PHP_SRC_DIR); for (i in keys) { // The trailing space is needed to prevent the trailing backslash // that is part of the build dir flags (CFLAGS_BD_XXX) from being // seen as a line continuation character MF.WriteLine(keys[i] + "=" + /* \s+\/ eliminates extra whitespace caused when using \ for string continuation, whereby \/ is the start of the next compiler switch */ trim(configure_subst.Item(keys[i])).replace(/\s+\//gm, " /") + " " ); MF.WriteBlankLines(1); } MF.WriteBlankLines(1); if (MODE_PHPIZE) { var TF = FSO.OpenTextFile(PHP_DIR + "/script/Makefile.phpize", 1); } else { var TF = FSO.OpenTextFile("win32/build/Makefile", 1); } MF.Write(TF.ReadAll()); MF.WriteLine("build-headers:"); MF.WriteLine(" @if not exist $(BUILD_DIR_DEV)\\include mkdir $(BUILD_DIR_DEV)\\include >nul"); MF.WriteLine(" @for %D in ($(INSTALL_HEADERS_DIR)) do @if not exist $(BUILD_DIR_DEV)\\include\\%D mkdir $(BUILD_DIR_DEV)\\include\\%D >nul"); for (i in headers_install) { if (headers_install[i][2] != "") { MF.WriteLine(" @if not exist $(BUILD_DIR_DEV)\\include\\" + headers_install[i][2] + " mkdir $(BUILD_DIR_DEV)\\include\\" + headers_install[i][2] + ">nul"); MF.WriteLine(" @copy " + headers_install[i][0] + " " + "$(BUILD_DIR_DEV)\\include\\" + headers_install[i][2] + " /y >nul"); } } MF.WriteLine(" @for %D in ($(INSTALL_HEADERS_DIR)) do @copy %D*.h $(BUILD_DIR_DEV)\\include\\%D /y >nul"); if (MODE_PHPIZE) { MF.WriteBlankLines(1); MF.WriteLine("build-bins:"); for (var i in extensions_enabled) { var lib = "php_" + extensions_enabled[i][0] + ".lib"; var dll = "php_" + extensions_enabled[i][0] + ".dll"; MF.WriteLine(" @copy $(BUILD_DIR)\\" + lib + " $(BUILD_DIR_DEV)\\lib"); MF.WriteLine(" @copy $(BUILD_DIR)\\" + dll + " $(PHP_PREFIX)"); } } else { MF.WriteBlankLines(1); MF.WriteLine("build-ext-libs:"); MF.WriteLine(" @if not exist $(BUILD_DIR_DEV)\\lib mkdir $(BUILD_DIR_DEV)\\lib >nul"); for (var i in extensions_enabled) { var lib; lib = "php_" + extensions_enabled[i][0] + "*.lib"; if ('shared' == extensions_enabled[i][1]) { MF.WriteLine(" @if exist $(BUILD_DIR)\\" + lib + " copy $(BUILD_DIR)\\" + lib + " $(BUILD_DIR_DEV)\\lib"); } } } TF.Close(); MF.WriteBlankLines(2); MFO.Close(); TF = FSO.OpenTextFile("Makefile.objects", 1); if (!TF.AtEndOfStream) { MF.Write(TF.ReadAll()); } TF.Close(); MF.Close(); } function ADD_FLAG(name, flags, target) { if (target != null) { name = target.toUpperCase() + "_" + name; } flags = trim(flags); if (configure_subst.Exists(name)) { var curr_flags = configure_subst.Item(name); if (curr_flags.indexOf(flags) >= 0) { return; } flags = curr_flags + " " + flags; configure_subst.Remove(name); } configure_subst.Add(name, flags); } function get_define(name) { if (configure_subst.Exists(name)) { return configure_subst.Item(name); } return ""; } // Add a .def to the core to export symbols function ADD_DEF_FILE(name) { if (!configure_subst.Exists("PHPDEF")) { DEFINE("PHPDEF", "$(BUILD_DIR)\\$(PHPDLL).def"); ADD_FLAG("PHP_LDFLAGS", "/def:$(PHPDEF)"); } ADD_FLAG("PHP_DLL_DEF_SOURCES", name); } function AC_DEFINE(name, value, comment, quote) { if (quote == null) { quote = true; } if (quote && typeof(value) == "string") { value = '"' + value.replace(new RegExp('(["\\\\])', "g"), '\\$1') + '"'; } else if (typeof(value) != "undefined" && value.length == 0) { value = '""'; } var item = new Array(value, comment); if (configure_hdr.Exists(name)) { var orig_item = configure_hdr.Item(name); STDOUT.WriteLine("AC_DEFINE[" + name + "]=" + value + ": is already defined to " + orig_item[0]); } else { configure_hdr.Add(name, item); } } function MESSAGE(msg) { STDOUT.WriteLine("" + msg); } function ERROR(msg) { STDERR.WriteLine("ERROR: " + msg); WScript.Quit(3); } function WARNING(msg) { STDERR.WriteLine("WARNING: " + msg); STDERR.WriteBlankLines(1); } function copy_and_subst(srcname, destname, subst_array) { if (!FSO.FileExists(srcname)) { srcname = configure_module_dirname + "\\" + srcname; destname = configure_module_dirname + "\\" + destname; } var content = file_get_contents(srcname); var i; for (i = 0; i < subst_array.length; i+=2) { var re = subst_array[i]; var rep = subst_array[i+1]; content = content.replace(re, rep); } var f = FSO.CreateTextFile(destname, true); f.Write(content); f.Close(); } // glob using simple filename wildcards // returns an array of matches that are found // in the filesystem function glob(path_pattern) { var path_parts = path_pattern.replace(new RegExp("/", "g"), "\\").split("\\"); var p; var base = ""; var is_pat_re = /\*/; //STDOUT.WriteLine("glob: " + path_pattern); if (FSO.FileExists(path_pattern)) { return new Array(path_pattern); } // first, build as much as possible that doesn't have a pattern for (p = 0; p < path_parts.length; p++) { if (path_parts[p].match(is_pat_re)) break; if (p) base += "\\"; base += path_parts[p]; } return _inner_glob(base, p, path_parts); } function _inner_glob(base, p, parts) { var pat = parts[p]; var full_name = base + "\\" + pat; var re = null; var items = null; if (p == parts.length) { return false; } //STDOUT.WriteLine("inner: base=" + base + " p=" + p + " pat=" + pat); if (FSO.FileExists(full_name)) { if (p < parts.length - 1) { // we didn't reach the full extent of the pattern return false; } return new Array(full_name); } if (FSO.FolderExists(full_name) && p == parts.length - 1) { // we have reached the end of the pattern; no need to recurse return new Array(full_name); } // Convert the pattern into a regexp re = new RegExp("^" + pat.replace(/\./g, '\\.').replace(/\*/g, '.*').replace(/\?/g, '.') + "$", "i"); items = new Array(); if (!FSO.FolderExists(base)) { return false; } var folder = FSO.GetFolder(base); var fc = null; var subitems = null; var item_name = null; var j; fc = new Enumerator(folder.SubFolders); for (; !fc.atEnd(); fc.moveNext()) { item_name = FSO.GetFileName(fc.item()); if (item_name.match(re)) { // got a match; if we are at the end of the pattern, just add these // things to the items array if (p == parts.length - 1) { items[items.length] = fc.item(); } else { // we should recurse and do more matches subitems = _inner_glob(base + "\\" + item_name, p + 1, parts); if (subitems) { for (j = 0; j < subitems.length; j++) { items[items.length] = subitems[j]; } } } } } // if we are at the end of the pattern, we should match // files too if (p == parts.length - 1) { fc = new Enumerator(folder.Files); for (; !fc.atEnd(); fc.moveNext()) { item_name = FSO.GetFileName(fc.item()); if (item_name.match(re)) { items[items.length] = fc.item(); } } } if (items.length == 0) return false; return items; } function PHP_INSTALL_HEADERS(dir, headers_list) { headers_list = headers_list.split(new RegExp("\\s+")); headers_list.sort(); if (dir.length > 0 && dir.substr(dir.length - 1) != '/' && dir.substr(dir.length - 1) != '\\') { dir += '/'; } dir = dir.replace(new RegExp("/", "g"), "\\"); for (i in headers_list) { found = false; src = headers_list[i]; src = src.replace(new RegExp("/", "g"), "\\"); isdir = FSO.FolderExists(dir + src); isfile = FSO.FileExists(dir + src); if (isdir) { if (src.length > 0 && src.substr(src.length - 1) != '/' && src.substr(src.length - 1) != '\\') { src += '\\'; } headers_install[headers_install.length] = [dir + src, 'dir','']; ADD_FLAG("INSTALL_HEADERS_DIR", dir + src); found = true; } else if (isfile) { dirname = FSO.GetParentFolderName(dir + src); headers_install[headers_install.length] = [dir + src, 'file', dirname]; ADD_FLAG("INSTALL_HEADERS", dir + src); found = true; } else { path = configure_module_dirname + "\\"+ src; isdir = FSO.FolderExists(path); isfile = FSO.FileExists(path); if (isdir) { if (src.length > 0 && src.substr(src.length - 1) != '/' && src.substr(src.length - 1) != '\\') { src += '\\'; } headers_install[headers_install.length] = [path, 'dir','']; ADD_FLAG("INSTALL_HEADERS_DIR", path); } else if (isfile) { dirname = FSO.GetParentFolderName(path); headers_install[headers_install.length] = [path, 'file', dir]; ADD_FLAG("INSTALL_HEADERS", dir + src); found = true; } } if (found == false) { STDOUT.WriteLine(headers_list); ERROR("Cannot find header " + dir + src); } } } // for snapshot builders, this option will attempt to enable everything // and you can then build everything, ignoring fatal errors within a module // by running "nmake snap" PHP_SNAPSHOT_BUILD = "no"; if (!MODE_PHPIZE) { ARG_ENABLE('snapshot-build', 'Build a snapshot; turns on everything it can and ignores build errors', 'no'); // one-shot build optimizes build by asking compiler to build // several objects at once, reducing overhead of starting new // compiler processes. ARG_ENABLE('one-shot', 'Optimize for fast build - best for release and snapshot builders, not so hot for edit-and-rebuild hacking', 'no'); } function toolset_option_handle() { if ("clang" == PHP_TOOLSET) { VS_TOOLSET = false; CLANG_TOOLSET = true; ICC_TOOLSET = false; } else if ("icc" == PHP_TOOLSET) { VS_TOOLSET = false; CLANG_TOOLSET = false; ICC_TOOLSET = true; } else { /* Visual Studio is the default toolset. */ PHP_TOOLSET = "no" == PHP_TOOLSET ? "vs" : PHP_TOOLSET; if (!!PHP_TOOLSET && "vs" != PHP_TOOLSET) { ERROR("Unsupported toolset '" + PHP_TOOLSET + "'"); } VS_TOOLSET = true; CLANG_TOOLSET = false; ICC_TOOLSET = false; } } function toolset_setup_compiler() { PHP_CL = toolset_get_compiler(); if (!PHP_CL) { ERROR("Compiler not found"); } COMPILER_NUMERIC_VERSION = toolset_get_compiler_version(); COMPILER_NAME = toolset_get_compiler_name(); if (VS_TOOLSET) { /* For the record here: */ // 1200 is VC6 // 1300 is vs.net 2002 // 1310 is vs.net 2003 // 1400 is vs.net 2005 // 1500 is vs.net 2008 // 1600 is vs.net 2010 // Which version of the compiler do we have? VCVERS = COMPILER_NUMERIC_VERSION; if (VCVERS < 1500) { ERROR("Unsupported MS C++ Compiler, VC9 (2008) minimum is required"); } AC_DEFINE('COMPILER', COMPILER_NAME, "Detected compiler version"); DEFINE("PHP_COMPILER_SHORT", VC_VERSIONS_SHORT[VCVERS]); AC_DEFINE('PHP_COMPILER_ID', VC_VERSIONS_SHORT[VCVERS], "Compiler compatibility ID"); } else if (CLANG_TOOLSET) { CLANGVERS = COMPILER_NUMERIC_VERSION; AC_DEFINE('COMPILER', COMPILER_NAME, "Detected compiler version"); DEFINE("PHP_COMPILER_SHORT", "clang"); AC_DEFINE('PHP_COMPILER_ID', "clang"); /* XXX something better were to write here */ } else if (ICC_TOOLSET) { INTELVERS = COMPILER_NUMERIC_VERSION; AC_DEFINE('COMPILER', COMPILER_NAME, "Detected compiler version"); DEFINE("PHP_COMPILER_SHORT", "icc"); AC_DEFINE('PHP_COMPILER_ID', "icc"); /* XXX something better were to write here */ } STDOUT.WriteLine(" Detected compiler " + COMPILER_NAME); } function toolset_setup_project_tools() { PATH_PROG('nmake'); // we don't want to define LIB, as that will override the default library path // that is set in that env var PATH_PROG('lib', null, 'MAKE_LIB'); if (!PATH_PROG('bison')) { ERROR('bison is required') } RE2C = PATH_PROG('re2c'); if (RE2C) { var intvers, intmin; var pattern = /\./g; RE2CVERS = probe_binary(RE2C, "version"); STDOUT.WriteLine(' Detected re2c version ' + RE2CVERS); intvers = RE2CVERS.replace(pattern, '') - 0; intmin = MINRE2C.replace(pattern, '') - 0; if (intvers < intmin) { STDOUT.WriteLine('WARNING: The minimum RE2C version requirement is ' + MINRE2C); STDOUT.WriteLine('Parsers will not be generated. Upgrade your copy at http://sf.net/projects/re2c'); DEFINE('RE2C', ''); } else { DEFINE('RE2C_FLAGS', ''); } } else { STDOUT.WriteLine('Parsers will not be regenerated'); } PATH_PROG('zip'); PATH_PROG('lemon'); // avoid picking up midnight commander from cygwin PATH_PROG('mc', WshShell.Environment("Process").Item("PATH")); // Try locating manifest tool if (VS_TOOLSET && VCVERS > 1200) { PATH_PROG('mt', WshShell.Environment("Process").Item("PATH")); } } function toolset_get_compiler() { if (VS_TOOLSET) { return PATH_PROG('cl', null, 'PHP_CL') } else if (CLANG_TOOLSET) { return PATH_PROG('clang-cl', null, 'PHP_CL') } else if (ICC_TOOLSET) { return PATH_PROG('icl', null, 'PHP_CL') } ERROR("Unsupported toolset"); } function toolset_get_compiler_version() { var version; if (VS_TOOLSET) { version = probe_binary(PHP_CL).substr(0, 5).replace('.', ''); return version; } else if (CLANG_TOOLSET) { var command = 'cmd /c ""' + PHP_CL + '" -v"'; var full = execute(command + '" 2>&1"'); if (full.match(/clang version ([\d\.]+) \((.*)\)/)) { version = RegExp.$1; version = version.replace(/\./g, ''); version = version/100 < 1 ? version*10 : version; return version; } } else if (ICC_TOOLSET) { var command = 'cmd /c ""' + PHP_CL + '" -v"'; var full = execute(command + '" 2>&1"'); if (full.match(/Version (\d+\.\d+\.\d+)/)) { version = RegExp.$1; version = version.replace(/\./g, ''); version = version/100 < 1 ? version*10 : version; return version; } } ERROR("Failed to parse compiler version or unsupported toolset"); } function toolset_get_compiler_name() { var version; if (VS_TOOLSET) { version = probe_binary(PHP_CL).substr(0, 5).replace('.', ''); return VC_VERSIONS[version]; } else if (CLANG_TOOLSET || ICC_TOOLSET) { var command = 'cmd /c ""' + PHP_CL + '" -v"'; var full = execute(command + '" 2>&1"'); return full.split(/\n/)[0].replace(/\s/g, ' '); } WARNING("Unsupported toolset"); } function toolset_is_64() { if (VS_TOOLSET) { return probe_binary(PHP_CL, 64); } else if (CLANG_TOOLSET) { /*var command = 'cmd /c ""' + PHP_CL + '" -v"'; var full = execute(command + '" 2>&1"'); return null != full.match(/x86_64/);*/ /* Even executed within an environment setup with vcvars32.bat, clang-cl doesn't recognize the arch toolset. But as it needs the VS environment, checking the arch of cl.exe is correct. */ return probe_binary(PATH_PROG('cl', null), 64); } else if (ICC_TOOLSET) { var command = 'cmd /c ""' + PHP_CL + '" -v"'; var full = execute(command + '" 2>&1"'); return null != full.match(/Intel\(R\) 64/); } ERROR("Unsupported toolset"); } function toolset_setup_arch() { if (X64) { STDOUT.WriteLine(" Detected 64-bit compiler"); } else { STDOUT.WriteLine(" Detected 32-bit compiler"); } AC_DEFINE('ARCHITECTURE', X64 ? 'x64' : 'x86', "Detected compiler architecture"); DEFINE("PHP_ARCHITECTURE", X64 ? 'x64' : 'x86'); } function toolset_setup_linker() { if (VS_TOOLSET) { return PATH_PROG('link', WshShell.Environment("Process").Item("PATH")); } else if (CLANG_TOOLSET) { //return PATH_PROG('lld', WshShell.Environment("Process").Item("PATH"), "LINK"); return PATH_PROG('link', WshShell.Environment("Process").Item("PATH")); } else if (ICC_TOOLSET) { return PATH_PROG('xilink', WshShell.Environment("Process").Item("PATH"), "LINK"); } ERROR("Unsupported toolset"); } function toolset_setup_common_cflags() { // CFLAGS for building the PHP dll DEFINE("CFLAGS_PHP", "/D _USRDLL /D PHP7DLLTS_EXPORTS /D PHP_EXPORTS \ /D LIBZEND_EXPORTS /D TSRM_EXPORTS /D SAPI_EXPORTS /D WINVER=" + WINVER); DEFINE('CFLAGS_PHP_OBJ', '$(CFLAGS_PHP) $(STATIC_EXT_CFLAGS)'); // General CFLAGS for building objects DEFINE("CFLAGS", "/nologo $(BASE_INCLUDES) /D _WINDOWS \ /D ZEND_WIN32=1 /D PHP_WIN32=1 /D WIN32 /D _MBCS /W3 "); if (VS_TOOLSET) { ADD_FLAG("CFLAGS", " /FD "); if (VCVERS < 1400) { // Enable automatic precompiled headers ADD_FLAG('CFLAGS', ' /YX '); if (PHP_DEBUG == "yes") { // Set some debug/release specific options ADD_FLAG('CFLAGS', ' /GZ '); } } if (VCVERS >= 1400) { // fun stuff: MS deprecated ANSI stdio and similar functions // disable annoying warnings. In addition, time_t defaults // to 64-bit. Ask for 32-bit. if (X64) { ADD_FLAG('CFLAGS', ' /wd4996 '); } else { ADD_FLAG('CFLAGS', ' /wd4996 /D_USE_32BIT_TIME_T=1 '); } if (PHP_DEBUG == "yes") { // Set some debug/release specific options ADD_FLAG('CFLAGS', ' /RTC1 '); } } } else if (CLANG_TOOLSET) { if (X64) { ADD_FLAG('CFLAGS', '-m64'); } else { ADD_FLAG('CFLAGS', '-m32'); } ADD_FLAG("CFLAGS", " /fallback "); var vc_ver = probe_binary(PATH_PROG('cl', null)); ADD_FLAG("CFLAGS", "-fms-compatibility-version=" + vc_ver); } } function toolset_setup_common_ldlags() { // General DLL link flags DEFINE("DLL_LDFLAGS", "/dll "); // PHP DLL link flags DEFINE("PHP_LDFLAGS", "$(DLL_LDFLAGS)"); if (VS_TOOLSET) { if (VCVERS >= 1700) { DEFINE("LDFLAGS", "/nologo "); } else { DEFINE("LDFLAGS", "/nologo /version:" + PHP_VERSION + "." + PHP_MINOR_VERSION + "." + PHP_RELEASE_VERSION); } } else { DEFINE("LDFLAGS", "/nologo "); } // we want msvcrt in the PHP DLL ADD_FLAG("PHP_LDFLAGS", "/nodefaultlib:libcmt"); } function toolset_setup_common_libs() { // urlmon.lib ole32.lib oleaut32.lib uuid.lib gdi32.lib winspool.lib comdlg32.lib DEFINE("LIBS", "kernel32.lib ole32.lib user32.lib advapi32.lib shell32.lib ws2_32.lib Dnsapi.lib"); } function toolset_setup_build_mode() { if (PHP_DEBUG == "yes") { ADD_FLAG("CFLAGS", "/LDd /MDd /W3 /Gm /Od /D _DEBUG /D ZEND_DEBUG=1 " + (X64?"/Zi":"/ZI")); ADD_FLAG("LDFLAGS", "/debug"); // Avoid problems when linking to release libraries that use the release // version of the libc ADD_FLAG("PHP_LDFLAGS", "/nodefaultlib:msvcrt"); } else { // Generate external debug files when --enable-debug-pack is specified if (PHP_DEBUG_PACK == "yes") { ADD_FLAG("CFLAGS", "/Zi"); ADD_FLAG("LDFLAGS", "/incremental:no /debug /opt:ref,icf"); } // Equivalent to Release_TSInline build -> best optimization ADD_FLAG("CFLAGS", "/LD /MD /W3 /Ox /D NDebug /D NDEBUG /D ZEND_WIN32_FORCE_INLINE /GF /D ZEND_DEBUG=0"); // if you have VS.Net /GS hardens the binary against buffer overruns // ADD_FLAG("CFLAGS", "/GS"); } } function object_out_dir_option_handle() { if (PHP_OBJECT_OUT_DIR.length) { PHP_OBJECT_OUT_DIR = FSO.GetAbsolutePathName(PHP_OBJECT_OUT_DIR); if (!FSO.FolderExists(PHP_OBJECT_OUT_DIR)) { ERROR('you chosen output directory ' + PHP_OBJECT_OUT_DIR + ' does not exist'); } PHP_OBJECT_OUT_DIR += '\\'; } else { PHP_OBJECT_OUT_DIR = FSO.GetAbsolutePathName(".") + '\\'; if (X64) { PHP_OBJECT_OUT_DIR += 'x64\\'; if (!FSO.FolderExists(PHP_OBJECT_OUT_DIR)) { FSO.CreateFolder(PHP_OBJECT_OUT_DIR); } } } } function setup_zts_stuff() { if (PHP_ZTS == "yes") { ADD_FLAG("CFLAGS", "/D ZTS=1"); ADD_FLAG("ZTS", "1"); } else { ADD_FLAG("ZTS", "0"); } DEFINE("PHP_ZTS_ARCHIVE_POSTFIX", PHP_ZTS == "yes" ? '' : "-nts"); // set up the build dir and DLL name if (PHP_DEBUG == "yes" && PHP_ZTS == "yes") { DEFINE("BUILD_DIR", PHP_OBJECT_OUT_DIR + "Debug_TS"); if (!MODE_PHPIZE) { DEFINE("PHPDLL", "php" + PHP_VERSION + "ts_debug.dll"); DEFINE("PHPLIB", "php" + PHP_VERSION + "ts_debug.lib"); } } else if (PHP_DEBUG == "yes" && PHP_ZTS == "no") { DEFINE("BUILD_DIR", PHP_OBJECT_OUT_DIR + "Debug"); if (!MODE_PHPIZE) { DEFINE("PHPDLL", "php" + PHP_VERSION + "_debug.dll"); DEFINE("PHPLIB", "php" + PHP_VERSION + "_debug.lib"); } } else if (PHP_DEBUG == "no" && PHP_ZTS == "yes") { DEFINE("BUILD_DIR", PHP_OBJECT_OUT_DIR + "Release_TS"); if (!MODE_PHPIZE) { DEFINE("PHPDLL", "php" + PHP_VERSION + "ts.dll"); DEFINE("PHPLIB", "php" + PHP_VERSION + "ts.lib"); } } else if (PHP_DEBUG == "no" && PHP_ZTS == "no") { DEFINE("BUILD_DIR", PHP_OBJECT_OUT_DIR + "Release"); if (!MODE_PHPIZE) { DEFINE("PHPDLL", "php" + PHP_VERSION + ".dll"); DEFINE("PHPLIB", "php" + PHP_VERSION + ".lib"); } } if (!FSO.FolderExists(get_define('BUILD_DIR'))) { FSO.CreateFolder(get_define('BUILD_DIR')); } } function php_build_option_handle() { if (PHP_PHP_BUILD == 'no') { if (FSO.FolderExists("..\\deps")) { PHP_PHP_BUILD = "..\\deps"; } else { if (FSO.FolderExists("..\\php_build")) { PHP_PHP_BUILD = "..\\php_build"; } else { if (X64) { if (FSO.FolderExists("..\\win64build")) { PHP_PHP_BUILD = "..\\win64build"; } else if (FSO.FolderExists("..\\php-win64-dev\\php_build")) { PHP_PHP_BUILD = "..\\php-win64-dev\\php_build"; } } else { if (FSO.FolderExists("..\\win32build")) { PHP_PHP_BUILD = "..\\win32build"; } else if (FSO.FolderExists("..\\php-win32-dev\\php_build")) { PHP_PHP_BUILD = "..\\php-win32-dev\\php_build"; } } } } PHP_PHP_BUILD = FSO.GetAbsolutePathName(PHP_PHP_BUILD); } DEFINE("PHP_BUILD", PHP_PHP_BUILD); } // Poke around for some headers function probe_basic_headers() { var p; if (PHP_PHP_BUILD != "no") { php_usual_include_suspects += ";" + PHP_PHP_BUILD + "\\include"; php_usual_lib_suspects += ";" + PHP_PHP_BUILD + "\\lib"; } } function add_extra_dirs() { var path, i, f; if (PHP_EXTRA_INCLUDES.length) { path = PHP_EXTRA_INCLUDES.split(';'); for (i = 0; i < path.length; i++) { f = FSO.GetAbsolutePathName(path[i]); if (FSO.FolderExists(f)) { ADD_FLAG("CFLAGS", '/I "' + f + '" '); } } } if (PHP_EXTRA_LIBS.length) { path = PHP_EXTRA_LIBS.split(';'); for (i = 0; i < path.length; i++) { f = FSO.GetAbsolutePathName(path[i]); if (FSO.FolderExists(f)) { if (VS_TOOLSET && VCVERS <= 1200 && f.indexOf(" ") >= 0) { ADD_FLAG("LDFLAGS", '/libpath:"\\"' + f + '\\"" '); ADD_FLAG("ARFLAGS", '/libpath:"\\"' + f + '\\"" '); } else { ADD_FLAG("LDFLAGS", '/libpath:"' + f + '" '); ADD_FLAG("ARFLAGS", '/libpath:"' + f + '" '); } } } } } function trim(s) { return s.replace(/^\s+/, "").replace(/\s+$/, ""); }