diff options
Diffstat (limited to 'lib/test_server')
32 files changed, 5842 insertions, 1249 deletions
| diff --git a/lib/test_server/doc/src/Makefile b/lib/test_server/doc/src/Makefile index c7ba415e5b..8c7fa99886 100644 --- a/lib/test_server/doc/src/Makefile +++ b/lib/test_server/doc/src/Makefile @@ -1,7 +1,7 @@  #  # %CopyrightBegin%  # -# Copyright Ericsson AB 2002-2010. All Rights Reserved. +# Copyright Ericsson AB 2002-2012. All Rights Reserved.  #  # The contents of this file are subject to the Erlang Public License,  # Version 1.1, (the "License"); you may not use this file except in @@ -119,23 +119,17 @@ clean clean_docs:  include $(ERL_TOP)/make/otp_release_targets.mk  release_docs_spec: docs -	$(INSTALL_DIR) $(RELSYSDIR)/doc/pdf -	$(INSTALL_DATA) $(TOP_PDF_FILE) $(RELSYSDIR)/doc/pdf -	$(INSTALL_DIR) $(RELSYSDIR)/doc/html +	$(INSTALL_DIR) "$(RELSYSDIR)/doc/pdf" +	$(INSTALL_DATA) $(TOP_PDF_FILE) "$(RELSYSDIR)/doc/pdf" +	$(INSTALL_DIR) "$(RELSYSDIR)/doc/html"  	$(INSTALL_DATA) $(HTMLDIR)/* \ -		$(RELSYSDIR)/doc/html -	$(INSTALL_DATA) $(INFO_FILE) $(RELSYSDIR) -	$(INSTALL_DIR) $(RELEASE_PATH)/man/man3 -	$(INSTALL_DATA) $(MAN3_FILES) $(RELEASE_PATH)/man/man3 -	$(INSTALL_DIR) $(RELEASE_PATH)/man/man6 -	$(INSTALL_DATA) $(MAN6_FILES) $(RELEASE_PATH)/man/man6 +		"$(RELSYSDIR)/doc/html" +	$(INSTALL_DATA) $(INFO_FILE) "$(RELSYSDIR)" +	$(INSTALL_DIR) "$(RELEASE_PATH)/man/man3" +	$(INSTALL_DATA) $(MAN3_FILES) "$(RELEASE_PATH)/man/man3" +	$(INSTALL_DIR) "$(RELEASE_PATH)/man/man6" +	$(INSTALL_DATA) $(MAN6_FILES) "$(RELEASE_PATH)/man/man6"  release_spec:  release_tests_spec: - -# ---------------------------------------------------- -# Include dependency -# ---------------------------------------------------- - -include make.dep diff --git a/lib/test_server/doc/src/make.dep b/lib/test_server/doc/src/make.dep deleted file mode 100644 index ee9100bd08..0000000000 --- a/lib/test_server/doc/src/make.dep +++ /dev/null @@ -1,24 +0,0 @@ -# ---------------------------------------------------- -# >>>> Do not edit this file <<<< -# This file was automaticly generated by -# /home/otp/bin/docdepend -# ---------------------------------------------------- - - -# ---------------------------------------------------- -# TeX files that the DVI file depend on -# ---------------------------------------------------- - -book.dvi: basics_chapter.tex book.tex example_chapter.tex \ -		part.tex ref_man.tex run_test_chapter.tex \ -		test_server_app.tex test_server_ctrl.tex \ -		test_server.tex test_spec_chapter.tex \ -		write_framework_chapter.tex \ -		write_test_chapter.tex - -# ---------------------------------------------------- -# Source inlined when transforming from source to LaTeX -# ---------------------------------------------------- - -book.tex: ref_man.xml - diff --git a/lib/test_server/doc/src/notes.xml b/lib/test_server/doc/src/notes.xml index beeff55ffe..6a9add044a 100644 --- a/lib/test_server/doc/src/notes.xml +++ b/lib/test_server/doc/src/notes.xml @@ -4,7 +4,7 @@  <chapter>    <header>      <copyright> -      <year>2004</year><year>2011</year> +      <year>2004</year><year>2012</year>        <holder>Ericsson AB. All Rights Reserved.</holder>      </copyright>      <legalnotice> @@ -32,6 +32,311 @@      <file>notes.xml</file>    </header> +<section><title>Test_Server 3.5.3</title> + +    <section><title>Improvements and New Features</title> +      <list> +        <item> +          <p> +	    test_server_h will now recognize info_reports written by +	    ct connection handlers (according to the description in +	    cth_conn_log) and ignore them as they will be completely +	    handled by by ct_conn_log_h.</p> +          <p> +	    Earlier test_server_h would print a tag (testcase name) +	    before forwarding the report to error_logger_tty_h. This +	    would cause lots of tags in the log with no info report +	    following (since error_logger_tty_h did not handle them).</p> +          <p> +	    Own Id: OTP-10571</p> +        </item> +      </list> +    </section> + + +    <section><title>Known Bugs and Problems</title> +      <list> +        <item> +          <p> +	    Restore Config data if lost when test case fails.</p> +          <p> +	    Own Id: OTP-10070 Aux Id: kunagi-175 [86] </p> +        </item> +        <item> +          <p> +	    IO server error in test_server.</p> +          <p> +	    Own Id: OTP-10125 Aux Id: OTP-10101, kunagi-177 [88] </p> +        </item> +      </list> +    </section> + +</section> + +<section><title>Test_Server 3.5.2</title> + +    <section><title>Fixed Bugs and Malfunctions</title> +      <list> +        <item> +          <p> +	    The documentation has been updated with the latest +	    changes for the test_server_ctrl:report/2 function.</p> +          <p> +	    Own Id: OTP-10086 Aux Id: seq12066 </p> +        </item> +        <item> +          <p> +	    The ct:get_status/0 function failed to report status if a +	    parallel test case group was running at the time of the +	    call. This has been fixed and the return value for the +	    function has been updated. Please see the ct reference +	    manual for details.</p> +          <p> +	    Own Id: OTP-10172</p> +        </item> +      </list> +    </section> + + +    <section><title>Improvements and New Features</title> +      <list> +        <item> +          <p> +	    It is now possible to sort the HTML tables by clicking on +	    the header elements. In order to reset a sorted table, +	    the browser window should simply be refreshed. This +	    feature requires that the browser supports javascript, +	    and has javascript execution enabled. If the 'ct_run +	    -basic_html' flag is used, no javascript code is included +	    in the generated HTML code.</p> +          <p> +	    Own Id: OTP-9896 Aux Id: seq12034, OTP-9835 </p> +        </item> +        <item> +          <p> +	    Verbosity levels for log printouts has been added. This +	    makes it possible to specify preferred verbosity for +	    different categories of log printouts, as well as general +	    printouts (such as standard IO), to allow control over +	    which strings get printed and which get ignored. New +	    versions of the Common Test logging functions, ct:log, +	    ct:pal and ct:print, have been introduced, with a new +	    Importance argument added. The Importance value is +	    compared to the verbosity level at runtime. More +	    information can be found in the chapter about Logging in +	    the Common Test User's Guide.</p> +          <p> +	    Own Id: OTP-10067 Aux Id: seq12050 </p> +        </item> +        <item> +          <p> +	    The Erlang/OTP test runner ts has been extended to allow +	    cross compilation of test suites. To cross compile the +	    test suites first follow the normal cross compilation +	    procedures and release the tests on the build host. Then +	    install ts using an xcomp specification file and compile +	    test suites using ts:compile_testcases/0. For more +	    details see $ERL_TOP/xcomp/README.md.</p> +          <p> +	    Own Id: OTP-10074</p> +        </item> +      </list> +    </section> + +</section> + +<section><title>Test_Server 3.5.1</title> + +    <section><title>Fixed Bugs and Malfunctions</title> +      <list> +        <item> +          <p> +	    After a test case timeout or abortion, the +	    end_per_testcase function executes on a new dedicated +	    process. The group leader for this process should be set +	    to the IO server for the test case, which was not done +	    properly. The result of this error was that no warnings +	    about end_per_testcase failing or timing out were ever +	    printed in the test case log. Also, help functions such +	    as e.g. test_server:stop_node/1, attempting to +	    synchronize with the IO server, would hang. The fault has +	    been corrected.</p> +          <p> +	    Own Id: OTP-9666</p> +        </item> +        <item> +          <p> +	    A deadlock situation could occur if Common Test is +	    forwarding error_handler printouts to Test Server at the +	    same time a new test case is starting. This error has +	    been fixed.</p> +          <p> +	    Own Id: OTP-9894</p> +        </item> +        <item> +          <p> +	    When a test case was killed because of a timetrap +	    timeout, the current location (suite, case and line) was +	    not printed correctly in the log files. This has been +	    corrected.</p> +          <p> +	    Own Id: OTP-9930 Aux Id: seq12002 </p> +        </item> +        <item> +          <p> +	    Test Server and Common Test would add new error handlers +	    with each test run and fail to remove previously added +	    ones. In the case of Test Server, this would only happen +	    if SASL was not running on the test node. This has been +	    fixed.</p> +          <p> +	    Own Id: OTP-9941 Aux Id: seq12009 </p> +        </item> +        <item> +          <p> +	    If a test case process was terminated due to an exit +	    signal from a linked process, Test Server failed to +	    report the correct name of the suite and case to the +	    framework. This has been corrected.</p> +          <p> +	    Own Id: OTP-9958 Aux Id: OTP-9855 </p> +        </item> +      </list> +    </section> + + +    <section><title>Improvements and New Features</title> +      <list> +        <item> +          <p> +	    A new optional feature has been introduced that enables +	    Common Test to generate priv_dir directory names that are +	    unique for each test case or config function. The name of +	    the option/flag is 'create_priv_dir' and it can be set to +	    value 'auto_per_run' (which is the default, existing, +	    behaviour), or 'auto_per_tc' or 'manual_per_tc'. If +	    'auto_per_tc' is used, Test Server creates a dedicated +	    priv_dir automatically for each test case (which can be +	    very expensive in case of many and/or repeated cases). If +	    'manual_per_tc' is used, the user needs to create the +	    priv_dir explicitly by calling the new function +	    ct:make_priv_dir/0.</p> +          <p> +	    Own Id: OTP-9659 Aux Id: seq11930 </p> +        </item> +        <item> +          <p> +	    A column for test case group name has been added to the +	    suite overview HTML log file.</p> +          <p> +	    Own Id: OTP-9730 Aux Id: seq11952 </p> +        </item> +        <item> +          <p> +	    It is now possible to use the post_end_per_testcase CT +	    hook function to print a comment for a test case in the +	    overview log file, even if the test case gets killed by a +	    timetrap or unknown exit signal, or if the +	    end_per_testcase function times out.</p> +          <p> +	    Own Id: OTP-9855 Aux Id: seq11979 </p> +        </item> +        <item> +          <p> +	    Common Test will now print error information (with a time +	    stamp) in the test case log file immediately when a test +	    case fails. This makes it easier to see when, in time, +	    the fault actually occured, and aid the job of locating +	    relevant trace and debug printouts in the log.</p> +          <p> +	    Own Id: OTP-9904 Aux Id: seq11985, OTP-9900 </p> +        </item> +        <item> +          <p> +	    Test Server has been modified to check the SASL +	    errlog_type parameter when receiving an error logger +	    event, so that it doesn't print reports of type that the +	    user has disabled.</p> +          <p> +	    Own Id: OTP-9955 Aux Id: seq12013 </p> +        </item> +        <item> +          <p> +	    If an application cannot be found by ts it is +	    automatically skipped when testing.</p> +          <p> +	    Own Id: OTP-9971</p> +        </item> +        <item> +          <p> +	    By specifying a user defined function ({M,F,A} or fun) as +	    timetrap value, either by means of an info function or by +	    calling ct:timetrap/1, it is now possible to set a +	    timetrap that will be triggered when the user function +	    returns.</p> +          <p> +	    Own Id: OTP-9988 Aux Id: OTP-9501, seq11894 </p> +        </item> +        <item> +          <p> +	    If the optional configuration functions init_per_suite/1 +	    and end_per_suite/1 are not implemented in the test +	    suite, local Common Test versions of these functions are +	    called instead, and will be displayed in the overview log +	    file. Any printouts made by the pre- or +	    post_init_per_suite and pre- or post_end_per_suite hook +	    functions are saved in the log files for these functions.</p> +          <p> +	    Own Id: OTP-9992</p> +        </item> +      </list> +    </section> + +</section> + +<section><title>Test_Server 3.5</title> + +    <section><title>Improvements and New Features</title> +      <list> +        <item> +          <p> +	    The test case group info function has been implemented in +	    Common Test. Before execution of a test case group, a +	    call is now made to <c>TestSuite:group(GroupName)</c>. +	    The function returns a list of test properties, e.g. to +	    specify timetrap values, require configuration data, etc +	    (analogue to the test suite- and test case info +	    function). The scope of the properties set by +	    <c>group(GroupName)</c> is all test cases and sub-groups +	    of group <c>GroupName</c>.</p> +          <p> +	    Own Id: OTP-9235</p> +        </item> +        <item> +          <p> +	    The look of the HTML log files generated by Common Test +	    and Test Server has been improved (and made easier to +	    customize) by means of a CSS file.</p> +          <p> +	    Own Id: OTP-9706</p> +        </item> +      </list> +    </section> + + +    <section><title>Known Bugs and Problems</title> +      <list> +        <item> +          <p> +	    Fix problems in CT/TS due to line numbers in exceptions.</p> +          <p> +	    Own Id: OTP-9203</p> +        </item> +      </list> +    </section> + +</section> +  <section><title>Test_Server 3.4.5</title>      <section><title>Fixed Bugs and Malfunctions</title> diff --git a/lib/test_server/doc/src/test_server.xml b/lib/test_server/doc/src/test_server.xml index 78bb922cc5..5bfa42c36f 100644 --- a/lib/test_server/doc/src/test_server.xml +++ b/lib/test_server/doc/src/test_server.xml @@ -203,7 +203,7 @@      <func>        <name>format(Format) -> ok</name>        <name>format(Format, Args)</name> -      <name>format(Pri,Format)</name> +      <name>format(Pri, Format)</name>        <name>format(Pri, Format, Args)</name>        <fsummary></fsummary>        <type> diff --git a/lib/test_server/doc/src/test_server_ctrl.xml b/lib/test_server/doc/src/test_server_ctrl.xml index 9028a67ecb..41bc0bcc75 100644 --- a/lib/test_server/doc/src/test_server_ctrl.xml +++ b/lib/test_server/doc/src/test_server_ctrl.xml @@ -5,7 +5,7 @@    <header>      <copyright>        <year>2007</year> -      <year>2011</year> +      <year>2012</year>        <holder>Ericsson AB, All Rights Reserved</holder>      </copyright>      <legalnotice> @@ -769,11 +769,13 @@ Optional, if not given the test server controller node          constantly updated. The following can be reported:          </p>          <p><c>What = tests_start, Data = {Name,NumCases}</c><br></br> +	   <c>What = loginfo, Data = [{topdir,TestRootDir},{rundir,CurrLogDir}]</c><br></br>  	   <c>What = tests_done, Data = {Ok,Failed,{UserSkipped,AutoSkipped}}</c><br></br> -	   <c>What = tc_start, Data = {Mod,Func}</c><br></br> +	   <c>What = tc_start, Data = {{Mod,Func},TCLogFile}</c><br></br>  	   <c>What = tc_done, Data = {Mod,Func,Result}</c><br></br>  	   <c>What = tc_user_skip, Data = {Mod,Func,Comment}</c><br></br> -	   <c>What = tc_auto_skip, Data = {Mod,Func,Comment}</c></p> +	   <c>What = tc_auto_skip, Data = {Mod,Func,Comment}</c><br></br> +	   <c>What = framework_error, Data = {{FWMod,FWFunc},Error}</c></p>        </desc>      </func>      <func> diff --git a/lib/test_server/include/test_server.hrl b/lib/test_server/include/test_server.hrl index 4b96d84ace..36e7e1f83d 100644 --- a/lib/test_server/include/test_server.hrl +++ b/lib/test_server/include/test_server.hrl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %%  -%% Copyright Ericsson AB 1996-2009. All Rights Reserved. +%% Copyright Ericsson AB 1996-2011. All Rights Reserved.  %%   %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -20,11 +20,10 @@  -ifdef(line_trace).  -line_trace(true).  -define(line, -	put(test_server_loc,{?MODULE,?LINE}),  	io:format(lists:concat([?MODULE,",",integer_to_list(?LINE),": ~p"]),  		  [erlang:now()]),).  -else. --define(line,put(test_server_loc,{?MODULE,?LINE}),). +-define(line,).  -endif.  -define(t,test_server).  -define(config,test_server:lookup_config). diff --git a/lib/test_server/include/test_server_line.hrl b/lib/test_server/include/test_server_line.hrl index 60ef860883..3c309d3ee5 100644 --- a/lib/test_server/include/test_server_line.hrl +++ b/lib/test_server/include/test_server_line.hrl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %%  -%% Copyright Ericsson AB 2004-2009. All Rights Reserved. +%% Copyright Ericsson AB 2004-2011. All Rights Reserved.  %%   %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -16,5 +16,4 @@  %%   %% %CopyrightEnd%  %% --compile({parse_transform,test_server_line}). diff --git a/lib/test_server/src/Makefile b/lib/test_server/src/Makefile index 63a585d526..513720dc04 100644 --- a/lib/test_server/src/Makefile +++ b/lib/test_server/src/Makefile @@ -1,7 +1,7 @@  #  # %CopyrightBegin%  # -# Copyright Ericsson AB 1996-2011. All Rights Reserved. +# Copyright Ericsson AB 1996-2012. All Rights Reserved.  #  # The contents of this file are subject to the Erlang Public License,  # Version 1.1, (the "License"); you may not use this file except in @@ -43,7 +43,6 @@ MODULES= test_server_ctrl \  	 test_server_node \  	 test_server \  	 test_server_sup \ -	 test_server_line \  	 test_server_h \  	 erl2html2 \  	 vxworks_client @@ -125,22 +124,22 @@ $(APPUP_TARGET): $(APPUP_SRC) ../vsn.mk  include $(ERL_TOP)/make/otp_release_targets.mk  release_spec: opt -	$(INSTALL_DIR) $(RELSYSDIR)/src -	$(INSTALL_DATA) $(ERL_FILES) $(RELSYSDIR)/src -	$(INSTALL_DATA) $(INTERNAL_HRL_FILES) $(RELSYSDIR)/src -	$(INSTALL_DIR) $(RELSYSDIR)/include -	$(INSTALL_DATA) $(HRL_FILES) $(RELSYSDIR)/include -	$(INSTALL_DIR) $(RELSYSDIR)/ebin -	$(INSTALL_DATA) $(TARGET_FILES) $(RELSYSDIR)/ebin +	$(INSTALL_DIR) "$(RELSYSDIR)/src" +	$(INSTALL_DATA) $(ERL_FILES) "$(RELSYSDIR)/src" +	$(INSTALL_DATA) $(INTERNAL_HRL_FILES) "$(RELSYSDIR)/src" +	$(INSTALL_DIR) "$(RELSYSDIR)/include" +	$(INSTALL_DATA) $(HRL_FILES) "$(RELSYSDIR)/include" +	$(INSTALL_DIR) "$(RELSYSDIR)/ebin" +	$(INSTALL_DATA) $(TARGET_FILES) "$(RELSYSDIR)/ebin"  release_tests_spec: opt -	$(INSTALL_DIR) $(RELEASE_PATH)/test_server +	$(INSTALL_DIR) "$(RELEASE_PATH)/test_server"  	$(INSTALL_DATA) $(ERL_FILES) $(TS_ERL_FILES) \  		$(HRL_FILES) $(INTERNAL_HRL_FILES) $(TS_HRL_FILES) \  		$(TS_TARGET_FILES) \  		$(AUTOCONF_FILES) $(C_FILES) $(COVER_FILES) $(CONFIG) \ -		$(RELEASE_PATH)/test_server -	$(INSTALL_SCRIPT) $(PROGRAMS) $(RELEASE_PATH)/test_server +		"$(RELEASE_PATH)/test_server" +	$(INSTALL_SCRIPT) $(PROGRAMS) "$(RELEASE_PATH)/test_server"  release_docs_spec: diff --git a/lib/test_server/src/config.guess b/lib/test_server/src/config.guess index 6f1eeddfcc..38a833903b 120000..100755 --- a/lib/test_server/src/config.guess +++ b/lib/test_server/src/config.guess @@ -1 +1,1519 @@ -../../../erts/autoconf/config.guess
\ No newline at end of file +#! /bin/sh +# Attempt to guess a canonical system name. +#   Copyright (C) 1992, 1993, 1994, 1995, 1996, 1997, 1998, 1999, +#   2000, 2001, 2002, 2003, 2004, 2005, 2006 Free Software Foundation, +#   Inc. + +timestamp='2007-05-17' + +# This file 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; either version 2 of the License, or +# (at your option) any later version. +# +# 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, write to the Free Software +# Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA +# 02110-1301, USA. +# +# As a special exception to the GNU General Public License, if you +# distribute this file as part of a program that contains a +# configuration script generated by Autoconf, you may include it under +# the same distribution terms that you use for the rest of that program. + + +# Originally written by Per Bothner <[email protected]>. +# Please send patches to <[email protected]>.  Submit a context +# diff and a properly formatted ChangeLog entry. +# +# This script attempts to guess a canonical system name similar to +# config.sub.  If it succeeds, it prints the system name on stdout, and +# exits with 0.  Otherwise, it exits with 1. +# +# The plan is that this can be called by configure scripts if you +# don't specify an explicit build system type. + +me=`echo "$0" | sed -e 's,.*/,,'` + +usage="\ +Usage: $0 [OPTION] + +Output the configuration name of the system \`$me' is run on. + +Operation modes: +  -h, --help         print this help, then exit +  -t, --time-stamp   print date of last modification, then exit +  -v, --version      print version number, then exit + +Report bugs and patches to <[email protected]>." + +version="\ +GNU config.guess ($timestamp) + +Originally written by Per Bothner. +Copyright (C) 1992, 1993, 1994, 1995, 1996, 1997, 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005 +Free Software Foundation, Inc. + +This is free software; see the source for copying conditions.  There is NO +warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE." + +help=" +Try \`$me --help' for more information." + +# Parse command line +while test $# -gt 0 ; do +  case $1 in +    --time-stamp | --time* | -t ) +       echo "$timestamp" ; exit ;; +    --version | -v ) +       echo "$version" ; exit ;; +    --help | --h* | -h ) +       echo "$usage"; exit ;; +    -- )     # Stop option processing +       shift; break ;; +    - )	# Use stdin as input. +       break ;; +    -* ) +       echo "$me: invalid option $1$help" >&2 +       exit 1 ;; +    * ) +       break ;; +  esac +done + +if test $# != 0; then +  echo "$me: too many arguments$help" >&2 +  exit 1 +fi + +trap 'exit 1' 1 2 15 + +# CC_FOR_BUILD -- compiler used by this script. Note that the use of a +# compiler to aid in system detection is discouraged as it requires +# temporary files to be created and, as you can see below, it is a +# headache to deal with in a portable fashion. + +# Historically, `CC_FOR_BUILD' used to be named `HOST_CC'. We still +# use `HOST_CC' if defined, but it is deprecated. + +# Portable tmp directory creation inspired by the Autoconf team. + +set_cc_for_build=' +trap "exitcode=\$?; (rm -f \$tmpfiles 2>/dev/null; rmdir \$tmp 2>/dev/null) && exit \$exitcode" 0 ; +trap "rm -f \$tmpfiles 2>/dev/null; rmdir \$tmp 2>/dev/null; exit 1" 1 2 13 15 ; +: ${TMPDIR=/tmp} ; + { tmp=`(umask 077 && mktemp -d "$TMPDIR/cgXXXXXX") 2>/dev/null` && test -n "$tmp" && test -d "$tmp" ; } || + { test -n "$RANDOM" && tmp=$TMPDIR/cg$$-$RANDOM && (umask 077 && mkdir $tmp) ; } || + { tmp=$TMPDIR/cg-$$ && (umask 077 && mkdir $tmp) && echo "Warning: creating insecure temp directory" >&2 ; } || + { echo "$me: cannot create a temporary directory in $TMPDIR" >&2 ; exit 1 ; } ; +dummy=$tmp/dummy ; +tmpfiles="$dummy.c $dummy.o $dummy.rel $dummy" ; +case $CC_FOR_BUILD,$HOST_CC,$CC in + ,,)    echo "int x;" > $dummy.c ; +	for c in cc gcc c89 c99 ; do +	  if ($c -c -o $dummy.o $dummy.c) >/dev/null 2>&1 ; then +	     CC_FOR_BUILD="$c"; break ; +	  fi ; +	done ; +	if test x"$CC_FOR_BUILD" = x ; then +	  CC_FOR_BUILD=no_compiler_found ; +	fi +	;; + ,,*)   CC_FOR_BUILD=$CC ;; + ,*,*)  CC_FOR_BUILD=$HOST_CC ;; +esac ; set_cc_for_build= ;' + +# This is needed to find uname on a Pyramid OSx when run in the BSD universe. +# ([email protected] 1994-08-24) +if (test -f /.attbin/uname) >/dev/null 2>&1 ; then +	PATH=$PATH:/.attbin ; export PATH +fi + +UNAME_MACHINE=`(uname -m) 2>/dev/null` || UNAME_MACHINE=unknown +UNAME_RELEASE=`(uname -r) 2>/dev/null` || UNAME_RELEASE=unknown +UNAME_SYSTEM=`(uname -s) 2>/dev/null`  || UNAME_SYSTEM=unknown +UNAME_VERSION=`(uname -v) 2>/dev/null` || UNAME_VERSION=unknown + +# Note: order is significant - the case branches are not exclusive. + +case "${UNAME_MACHINE}:${UNAME_SYSTEM}:${UNAME_RELEASE}:${UNAME_VERSION}" in +    *:NetBSD:*:*) +	# NetBSD (nbsd) targets should (where applicable) match one or +	# more of the tupples: *-*-netbsdelf*, *-*-netbsdaout*, +	# *-*-netbsdecoff* and *-*-netbsd*.  For targets that recently +	# switched to ELF, *-*-netbsd* would select the old +	# object file format.  This provides both forward +	# compatibility and a consistent mechanism for selecting the +	# object file format. +	# +	# Note: NetBSD doesn't particularly care about the vendor +	# portion of the name.  We always set it to "unknown". +	sysctl="sysctl -n hw.machine_arch" +	UNAME_MACHINE_ARCH=`(/sbin/$sysctl 2>/dev/null || \ +	    /usr/sbin/$sysctl 2>/dev/null || echo unknown)` +	case "${UNAME_MACHINE_ARCH}" in +	    armeb) machine=armeb-unknown ;; +	    arm*) machine=arm-unknown ;; +	    sh3el) machine=shl-unknown ;; +	    sh3eb) machine=sh-unknown ;; +	    sh5el) machine=sh5le-unknown ;; +	    *) machine=${UNAME_MACHINE_ARCH}-unknown ;; +	esac +	# The Operating System including object format, if it has switched +	# to ELF recently, or will in the future. +	case "${UNAME_MACHINE_ARCH}" in +	    arm*|i386|m68k|ns32k|sh3*|sparc|vax) +		eval $set_cc_for_build +		if echo __ELF__ | $CC_FOR_BUILD -E - 2>/dev/null \ +			| grep __ELF__ >/dev/null +		then +		    # Once all utilities can be ECOFF (netbsdecoff) or a.out (netbsdaout). +		    # Return netbsd for either.  FIX? +		    os=netbsd +		else +		    os=netbsdelf +		fi +		;; +	    *) +	        os=netbsd +		;; +	esac +	# The OS release +	# Debian GNU/NetBSD machines have a different userland, and +	# thus, need a distinct triplet. However, they do not need +	# kernel version information, so it can be replaced with a +	# suitable tag, in the style of linux-gnu. +	case "${UNAME_VERSION}" in +	    Debian*) +		release='-gnu' +		;; +	    *) +		release=`echo ${UNAME_RELEASE}|sed -e 's/[-_].*/\./'` +		;; +	esac +	# Since CPU_TYPE-MANUFACTURER-KERNEL-OPERATING_SYSTEM: +	# contains redundant information, the shorter form: +	# CPU_TYPE-MANUFACTURER-OPERATING_SYSTEM is used. +	echo "${machine}-${os}${release}" +	exit ;; +    *:OpenBSD:*:*) +	UNAME_MACHINE_ARCH=`arch | sed 's/OpenBSD.//'` +	echo ${UNAME_MACHINE_ARCH}-unknown-openbsd${UNAME_RELEASE} +	exit ;; +    *:ekkoBSD:*:*) +	echo ${UNAME_MACHINE}-unknown-ekkobsd${UNAME_RELEASE} +	exit ;; +    *:SolidBSD:*:*) +	echo ${UNAME_MACHINE}-unknown-solidbsd${UNAME_RELEASE} +	exit ;; +    macppc:MirBSD:*:*) +	echo powerpc-unknown-mirbsd${UNAME_RELEASE} +	exit ;; +    *:MirBSD:*:*) +	echo ${UNAME_MACHINE}-unknown-mirbsd${UNAME_RELEASE} +	exit ;; +    alpha:OSF1:*:*) +	case $UNAME_RELEASE in +	*4.0) +		UNAME_RELEASE=`/usr/sbin/sizer -v | awk '{print $3}'` +		;; +	*5.*) +	        UNAME_RELEASE=`/usr/sbin/sizer -v | awk '{print $4}'` +		;; +	esac +	# According to Compaq, /usr/sbin/psrinfo has been available on +	# OSF/1 and Tru64 systems produced since 1995.  I hope that +	# covers most systems running today.  This code pipes the CPU +	# types through head -n 1, so we only detect the type of CPU 0. +	ALPHA_CPU_TYPE=`/usr/sbin/psrinfo -v | sed -n -e 's/^  The alpha \(.*\) processor.*$/\1/p' | head -n 1` +	case "$ALPHA_CPU_TYPE" in +	    "EV4 (21064)") +		UNAME_MACHINE="alpha" ;; +	    "EV4.5 (21064)") +		UNAME_MACHINE="alpha" ;; +	    "LCA4 (21066/21068)") +		UNAME_MACHINE="alpha" ;; +	    "EV5 (21164)") +		UNAME_MACHINE="alphaev5" ;; +	    "EV5.6 (21164A)") +		UNAME_MACHINE="alphaev56" ;; +	    "EV5.6 (21164PC)") +		UNAME_MACHINE="alphapca56" ;; +	    "EV5.7 (21164PC)") +		UNAME_MACHINE="alphapca57" ;; +	    "EV6 (21264)") +		UNAME_MACHINE="alphaev6" ;; +	    "EV6.7 (21264A)") +		UNAME_MACHINE="alphaev67" ;; +	    "EV6.8CB (21264C)") +		UNAME_MACHINE="alphaev68" ;; +	    "EV6.8AL (21264B)") +		UNAME_MACHINE="alphaev68" ;; +	    "EV6.8CX (21264D)") +		UNAME_MACHINE="alphaev68" ;; +	    "EV6.9A (21264/EV69A)") +		UNAME_MACHINE="alphaev69" ;; +	    "EV7 (21364)") +		UNAME_MACHINE="alphaev7" ;; +	    "EV7.9 (21364A)") +		UNAME_MACHINE="alphaev79" ;; +	esac +	# A Pn.n version is a patched version. +	# A Vn.n version is a released version. +	# A Tn.n version is a released field test version. +	# A Xn.n version is an unreleased experimental baselevel. +	# 1.2 uses "1.2" for uname -r. +	echo ${UNAME_MACHINE}-dec-osf`echo ${UNAME_RELEASE} | sed -e 's/^[PVTX]//' | tr 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' 'abcdefghijklmnopqrstuvwxyz'` +	exit ;; +    Alpha\ *:Windows_NT*:*) +	# How do we know it's Interix rather than the generic POSIX subsystem? +	# Should we change UNAME_MACHINE based on the output of uname instead +	# of the specific Alpha model? +	echo alpha-pc-interix +	exit ;; +    21064:Windows_NT:50:3) +	echo alpha-dec-winnt3.5 +	exit ;; +    Amiga*:UNIX_System_V:4.0:*) +	echo m68k-unknown-sysv4 +	exit ;; +    *:[Aa]miga[Oo][Ss]:*:*) +	echo ${UNAME_MACHINE}-unknown-amigaos +	exit ;; +    *:[Mm]orph[Oo][Ss]:*:*) +	echo ${UNAME_MACHINE}-unknown-morphos +	exit ;; +    *:OS/390:*:*) +	echo i370-ibm-openedition +	exit ;; +    *:z/VM:*:*) +	echo s390-ibm-zvmoe +	exit ;; +    *:OS400:*:*) +        echo powerpc-ibm-os400 +	exit ;; +    arm:RISC*:1.[012]*:*|arm:riscix:1.[012]*:*) +	echo arm-acorn-riscix${UNAME_RELEASE} +	exit ;; +    arm:riscos:*:*|arm:RISCOS:*:*) +	echo arm-unknown-riscos +	exit ;; +    SR2?01:HI-UX/MPP:*:* | SR8000:HI-UX/MPP:*:*) +	echo hppa1.1-hitachi-hiuxmpp +	exit ;; +    Pyramid*:OSx*:*:* | MIS*:OSx*:*:* | MIS*:SMP_DC-OSx*:*:*) +	# [email protected] (Earle F. Ake) contributed MIS and NILE. +	if test "`(/bin/universe) 2>/dev/null`" = att ; then +		echo pyramid-pyramid-sysv3 +	else +		echo pyramid-pyramid-bsd +	fi +	exit ;; +    NILE*:*:*:dcosx) +	echo pyramid-pyramid-svr4 +	exit ;; +    DRS?6000:unix:4.0:6*) +	echo sparc-icl-nx6 +	exit ;; +    DRS?6000:UNIX_SV:4.2*:7* | DRS?6000:isis:4.2*:7*) +	case `/usr/bin/uname -p` in +	    sparc) echo sparc-icl-nx7; exit ;; +	esac ;; +    sun4H:SunOS:5.*:*) +	echo sparc-hal-solaris2`echo ${UNAME_RELEASE}|sed -e 's/[^.]*//'` +	exit ;; +    sun4*:SunOS:5.*:* | tadpole*:SunOS:5.*:*) +	echo sparc-sun-solaris2`echo ${UNAME_RELEASE}|sed -e 's/[^.]*//'` +	exit ;; +    i86pc:SunOS:5.*:* | ix86xen:SunOS:5.*:*) +	echo i386-pc-solaris2`echo ${UNAME_RELEASE}|sed -e 's/[^.]*//'` +	exit ;; +    sun4*:SunOS:6*:*) +	# According to config.sub, this is the proper way to canonicalize +	# SunOS6.  Hard to guess exactly what SunOS6 will be like, but +	# it's likely to be more like Solaris than SunOS4. +	echo sparc-sun-solaris3`echo ${UNAME_RELEASE}|sed -e 's/[^.]*//'` +	exit ;; +    sun4*:SunOS:*:*) +	case "`/usr/bin/arch -k`" in +	    Series*|S4*) +		UNAME_RELEASE=`uname -v` +		;; +	esac +	# Japanese Language versions have a version number like `4.1.3-JL'. +	echo sparc-sun-sunos`echo ${UNAME_RELEASE}|sed -e 's/-/_/'` +	exit ;; +    sun3*:SunOS:*:*) +	echo m68k-sun-sunos${UNAME_RELEASE} +	exit ;; +    sun*:*:4.2BSD:*) +	UNAME_RELEASE=`(sed 1q /etc/motd | awk '{print substr($5,1,3)}') 2>/dev/null` +	test "x${UNAME_RELEASE}" = "x" && UNAME_RELEASE=3 +	case "`/bin/arch`" in +	    sun3) +		echo m68k-sun-sunos${UNAME_RELEASE} +		;; +	    sun4) +		echo sparc-sun-sunos${UNAME_RELEASE} +		;; +	esac +	exit ;; +    aushp:SunOS:*:*) +	echo sparc-auspex-sunos${UNAME_RELEASE} +	exit ;; +    # The situation for MiNT is a little confusing.  The machine name +    # can be virtually everything (everything which is not +    # "atarist" or "atariste" at least should have a processor +    # > m68000).  The system name ranges from "MiNT" over "FreeMiNT" +    # to the lowercase version "mint" (or "freemint").  Finally +    # the system name "TOS" denotes a system which is actually not +    # MiNT.  But MiNT is downward compatible to TOS, so this should +    # be no problem. +    atarist[e]:*MiNT:*:* | atarist[e]:*mint:*:* | atarist[e]:*TOS:*:*) +        echo m68k-atari-mint${UNAME_RELEASE} +	exit ;; +    atari*:*MiNT:*:* | atari*:*mint:*:* | atarist[e]:*TOS:*:*) +	echo m68k-atari-mint${UNAME_RELEASE} +        exit ;; +    *falcon*:*MiNT:*:* | *falcon*:*mint:*:* | *falcon*:*TOS:*:*) +        echo m68k-atari-mint${UNAME_RELEASE} +	exit ;; +    milan*:*MiNT:*:* | milan*:*mint:*:* | *milan*:*TOS:*:*) +        echo m68k-milan-mint${UNAME_RELEASE} +        exit ;; +    hades*:*MiNT:*:* | hades*:*mint:*:* | *hades*:*TOS:*:*) +        echo m68k-hades-mint${UNAME_RELEASE} +        exit ;; +    *:*MiNT:*:* | *:*mint:*:* | *:*TOS:*:*) +        echo m68k-unknown-mint${UNAME_RELEASE} +        exit ;; +    m68k:machten:*:*) +	echo m68k-apple-machten${UNAME_RELEASE} +	exit ;; +    powerpc:machten:*:*) +	echo powerpc-apple-machten${UNAME_RELEASE} +	exit ;; +    RISC*:Mach:*:*) +	echo mips-dec-mach_bsd4.3 +	exit ;; +    RISC*:ULTRIX:*:*) +	echo mips-dec-ultrix${UNAME_RELEASE} +	exit ;; +    VAX*:ULTRIX*:*:*) +	echo vax-dec-ultrix${UNAME_RELEASE} +	exit ;; +    2020:CLIX:*:* | 2430:CLIX:*:*) +	echo clipper-intergraph-clix${UNAME_RELEASE} +	exit ;; +    mips:*:*:UMIPS | mips:*:*:RISCos) +	eval $set_cc_for_build +	sed 's/^	//' << EOF >$dummy.c +#ifdef __cplusplus +#include <stdio.h>  /* for printf() prototype */ +	int main (int argc, char *argv[]) { +#else +	int main (argc, argv) int argc; char *argv[]; { +#endif +	#if defined (host_mips) && defined (MIPSEB) +	#if defined (SYSTYPE_SYSV) +	  printf ("mips-mips-riscos%ssysv\n", argv[1]); exit (0); +	#endif +	#if defined (SYSTYPE_SVR4) +	  printf ("mips-mips-riscos%ssvr4\n", argv[1]); exit (0); +	#endif +	#if defined (SYSTYPE_BSD43) || defined(SYSTYPE_BSD) +	  printf ("mips-mips-riscos%sbsd\n", argv[1]); exit (0); +	#endif +	#endif +	  exit (-1); +	} +EOF +	$CC_FOR_BUILD -o $dummy $dummy.c && +	  dummyarg=`echo "${UNAME_RELEASE}" | sed -n 's/\([0-9]*\).*/\1/p'` && +	  SYSTEM_NAME=`$dummy $dummyarg` && +	    { echo "$SYSTEM_NAME"; exit; } +	echo mips-mips-riscos${UNAME_RELEASE} +	exit ;; +    Motorola:PowerMAX_OS:*:*) +	echo powerpc-motorola-powermax +	exit ;; +    Motorola:*:4.3:PL8-*) +	echo powerpc-harris-powermax +	exit ;; +    Night_Hawk:*:*:PowerMAX_OS | Synergy:PowerMAX_OS:*:*) +	echo powerpc-harris-powermax +	exit ;; +    Night_Hawk:Power_UNIX:*:*) +	echo powerpc-harris-powerunix +	exit ;; +    m88k:CX/UX:7*:*) +	echo m88k-harris-cxux7 +	exit ;; +    m88k:*:4*:R4*) +	echo m88k-motorola-sysv4 +	exit ;; +    m88k:*:3*:R3*) +	echo m88k-motorola-sysv3 +	exit ;; +    AViiON:dgux:*:*) +        # DG/UX returns AViiON for all architectures +        UNAME_PROCESSOR=`/usr/bin/uname -p` +	if [ $UNAME_PROCESSOR = mc88100 ] || [ $UNAME_PROCESSOR = mc88110 ] +	then +	    if [ ${TARGET_BINARY_INTERFACE}x = m88kdguxelfx ] || \ +	       [ ${TARGET_BINARY_INTERFACE}x = x ] +	    then +		echo m88k-dg-dgux${UNAME_RELEASE} +	    else +		echo m88k-dg-dguxbcs${UNAME_RELEASE} +	    fi +	else +	    echo i586-dg-dgux${UNAME_RELEASE} +	fi + 	exit ;; +    M88*:DolphinOS:*:*)	# DolphinOS (SVR3) +	echo m88k-dolphin-sysv3 +	exit ;; +    M88*:*:R3*:*) +	# Delta 88k system running SVR3 +	echo m88k-motorola-sysv3 +	exit ;; +    XD88*:*:*:*) # Tektronix XD88 system running UTekV (SVR3) +	echo m88k-tektronix-sysv3 +	exit ;; +    Tek43[0-9][0-9]:UTek:*:*) # Tektronix 4300 system running UTek (BSD) +	echo m68k-tektronix-bsd +	exit ;; +    *:IRIX*:*:*) +	echo mips-sgi-irix`echo ${UNAME_RELEASE}|sed -e 's/-/_/g'` +	exit ;; +    ????????:AIX?:[12].1:2)   # AIX 2.2.1 or AIX 2.1.1 is RT/PC AIX. +	echo romp-ibm-aix     # uname -m gives an 8 hex-code CPU id +	exit ;;               # Note that: echo "'`uname -s`'" gives 'AIX ' +    i*86:AIX:*:*) +	echo i386-ibm-aix +	exit ;; +    ia64:AIX:*:*) +	if [ -x /usr/bin/oslevel ] ; then +		IBM_REV=`/usr/bin/oslevel` +	else +		IBM_REV=${UNAME_VERSION}.${UNAME_RELEASE} +	fi +	echo ${UNAME_MACHINE}-ibm-aix${IBM_REV} +	exit ;; +    *:AIX:2:3) +	if grep bos325 /usr/include/stdio.h >/dev/null 2>&1; then +		eval $set_cc_for_build +		sed 's/^		//' << EOF >$dummy.c +		#include <sys/systemcfg.h> + +		main() +			{ +			if (!__power_pc()) +				exit(1); +			puts("powerpc-ibm-aix3.2.5"); +			exit(0); +			} +EOF +		if $CC_FOR_BUILD -o $dummy $dummy.c && SYSTEM_NAME=`$dummy` +		then +			echo "$SYSTEM_NAME" +		else +			echo rs6000-ibm-aix3.2.5 +		fi +	elif grep bos324 /usr/include/stdio.h >/dev/null 2>&1; then +		echo rs6000-ibm-aix3.2.4 +	else +		echo rs6000-ibm-aix3.2 +	fi +	exit ;; +    *:AIX:*:[45]) +	IBM_CPU_ID=`/usr/sbin/lsdev -C -c processor -S available | sed 1q | awk '{ print $1 }'` +	if /usr/sbin/lsattr -El ${IBM_CPU_ID} | grep ' POWER' >/dev/null 2>&1; then +		IBM_ARCH=rs6000 +	else +		IBM_ARCH=powerpc +	fi +	if [ -x /usr/bin/oslevel ] ; then +		IBM_REV=`/usr/bin/oslevel` +	else +		IBM_REV=${UNAME_VERSION}.${UNAME_RELEASE} +	fi +	echo ${IBM_ARCH}-ibm-aix${IBM_REV} +	exit ;; +    *:AIX:*:*) +	echo rs6000-ibm-aix +	exit ;; +    ibmrt:4.4BSD:*|romp-ibm:BSD:*) +	echo romp-ibm-bsd4.4 +	exit ;; +    ibmrt:*BSD:*|romp-ibm:BSD:*)            # covers RT/PC BSD and +	echo romp-ibm-bsd${UNAME_RELEASE}   # 4.3 with uname added to +	exit ;;                             # report: romp-ibm BSD 4.3 +    *:BOSX:*:*) +	echo rs6000-bull-bosx +	exit ;; +    DPX/2?00:B.O.S.:*:*) +	echo m68k-bull-sysv3 +	exit ;; +    9000/[34]??:4.3bsd:1.*:*) +	echo m68k-hp-bsd +	exit ;; +    hp300:4.4BSD:*:* | 9000/[34]??:4.3bsd:2.*:*) +	echo m68k-hp-bsd4.4 +	exit ;; +    9000/[34678]??:HP-UX:*:*) +	HPUX_REV=`echo ${UNAME_RELEASE}|sed -e 's/[^.]*.[0B]*//'` +	case "${UNAME_MACHINE}" in +	    9000/31? )            HP_ARCH=m68000 ;; +	    9000/[34]?? )         HP_ARCH=m68k ;; +	    9000/[678][0-9][0-9]) +		if [ -x /usr/bin/getconf ]; then +		    sc_cpu_version=`/usr/bin/getconf SC_CPU_VERSION 2>/dev/null` +                    sc_kernel_bits=`/usr/bin/getconf SC_KERNEL_BITS 2>/dev/null` +                    case "${sc_cpu_version}" in +                      523) HP_ARCH="hppa1.0" ;; # CPU_PA_RISC1_0 +                      528) HP_ARCH="hppa1.1" ;; # CPU_PA_RISC1_1 +                      532)                      # CPU_PA_RISC2_0 +                        case "${sc_kernel_bits}" in +                          32) HP_ARCH="hppa2.0n" ;; +                          64) HP_ARCH="hppa2.0w" ;; +			  '') HP_ARCH="hppa2.0" ;;   # HP-UX 10.20 +                        esac ;; +                    esac +		fi +		if [ "${HP_ARCH}" = "" ]; then +		    eval $set_cc_for_build +		    sed 's/^              //' << EOF >$dummy.c + +              #define _HPUX_SOURCE +              #include <stdlib.h> +              #include <unistd.h> + +              int main () +              { +              #if defined(_SC_KERNEL_BITS) +                  long bits = sysconf(_SC_KERNEL_BITS); +              #endif +                  long cpu  = sysconf (_SC_CPU_VERSION); + +                  switch (cpu) +              	{ +              	case CPU_PA_RISC1_0: puts ("hppa1.0"); break; +              	case CPU_PA_RISC1_1: puts ("hppa1.1"); break; +              	case CPU_PA_RISC2_0: +              #if defined(_SC_KERNEL_BITS) +              	    switch (bits) +              		{ +              		case 64: puts ("hppa2.0w"); break; +              		case 32: puts ("hppa2.0n"); break; +              		default: puts ("hppa2.0"); break; +              		} break; +              #else  /* !defined(_SC_KERNEL_BITS) */ +              	    puts ("hppa2.0"); break; +              #endif +              	default: puts ("hppa1.0"); break; +              	} +                  exit (0); +              } +EOF +		    (CCOPTS= $CC_FOR_BUILD -o $dummy $dummy.c 2>/dev/null) && HP_ARCH=`$dummy` +		    test -z "$HP_ARCH" && HP_ARCH=hppa +		fi ;; +	esac +	if [ ${HP_ARCH} = "hppa2.0w" ] +	then +	    eval $set_cc_for_build + +	    # hppa2.0w-hp-hpux* has a 64-bit kernel and a compiler generating +	    # 32-bit code.  hppa64-hp-hpux* has the same kernel and a compiler +	    # generating 64-bit code.  GNU and HP use different nomenclature: +	    # +	    # $ CC_FOR_BUILD=cc ./config.guess +	    # => hppa2.0w-hp-hpux11.23 +	    # $ CC_FOR_BUILD="cc +DA2.0w" ./config.guess +	    # => hppa64-hp-hpux11.23 + +	    if echo __LP64__ | (CCOPTS= $CC_FOR_BUILD -E - 2>/dev/null) | +		grep __LP64__ >/dev/null +	    then +		HP_ARCH="hppa2.0w" +	    else +		HP_ARCH="hppa64" +	    fi +	fi +	echo ${HP_ARCH}-hp-hpux${HPUX_REV} +	exit ;; +    ia64:HP-UX:*:*) +	HPUX_REV=`echo ${UNAME_RELEASE}|sed -e 's/[^.]*.[0B]*//'` +	echo ia64-hp-hpux${HPUX_REV} +	exit ;; +    3050*:HI-UX:*:*) +	eval $set_cc_for_build +	sed 's/^	//' << EOF >$dummy.c +	#include <unistd.h> +	int +	main () +	{ +	  long cpu = sysconf (_SC_CPU_VERSION); +	  /* The order matters, because CPU_IS_HP_MC68K erroneously returns +	     true for CPU_PA_RISC1_0.  CPU_IS_PA_RISC returns correct +	     results, however.  */ +	  if (CPU_IS_PA_RISC (cpu)) +	    { +	      switch (cpu) +		{ +		  case CPU_PA_RISC1_0: puts ("hppa1.0-hitachi-hiuxwe2"); break; +		  case CPU_PA_RISC1_1: puts ("hppa1.1-hitachi-hiuxwe2"); break; +		  case CPU_PA_RISC2_0: puts ("hppa2.0-hitachi-hiuxwe2"); break; +		  default: puts ("hppa-hitachi-hiuxwe2"); break; +		} +	    } +	  else if (CPU_IS_HP_MC68K (cpu)) +	    puts ("m68k-hitachi-hiuxwe2"); +	  else puts ("unknown-hitachi-hiuxwe2"); +	  exit (0); +	} +EOF +	$CC_FOR_BUILD -o $dummy $dummy.c && SYSTEM_NAME=`$dummy` && +		{ echo "$SYSTEM_NAME"; exit; } +	echo unknown-hitachi-hiuxwe2 +	exit ;; +    9000/7??:4.3bsd:*:* | 9000/8?[79]:4.3bsd:*:* ) +	echo hppa1.1-hp-bsd +	exit ;; +    9000/8??:4.3bsd:*:*) +	echo hppa1.0-hp-bsd +	exit ;; +    *9??*:MPE/iX:*:* | *3000*:MPE/iX:*:*) +	echo hppa1.0-hp-mpeix +	exit ;; +    hp7??:OSF1:*:* | hp8?[79]:OSF1:*:* ) +	echo hppa1.1-hp-osf +	exit ;; +    hp8??:OSF1:*:*) +	echo hppa1.0-hp-osf +	exit ;; +    i*86:OSF1:*:*) +	if [ -x /usr/sbin/sysversion ] ; then +	    echo ${UNAME_MACHINE}-unknown-osf1mk +	else +	    echo ${UNAME_MACHINE}-unknown-osf1 +	fi +	exit ;; +    parisc*:Lites*:*:*) +	echo hppa1.1-hp-lites +	exit ;; +    C1*:ConvexOS:*:* | convex:ConvexOS:C1*:*) +	echo c1-convex-bsd +        exit ;; +    C2*:ConvexOS:*:* | convex:ConvexOS:C2*:*) +	if getsysinfo -f scalar_acc +	then echo c32-convex-bsd +	else echo c2-convex-bsd +	fi +        exit ;; +    C34*:ConvexOS:*:* | convex:ConvexOS:C34*:*) +	echo c34-convex-bsd +        exit ;; +    C38*:ConvexOS:*:* | convex:ConvexOS:C38*:*) +	echo c38-convex-bsd +        exit ;; +    C4*:ConvexOS:*:* | convex:ConvexOS:C4*:*) +	echo c4-convex-bsd +        exit ;; +    CRAY*Y-MP:*:*:*) +	echo ymp-cray-unicos${UNAME_RELEASE} | sed -e 's/\.[^.]*$/.X/' +	exit ;; +    CRAY*[A-Z]90:*:*:*) +	echo ${UNAME_MACHINE}-cray-unicos${UNAME_RELEASE} \ +	| sed -e 's/CRAY.*\([A-Z]90\)/\1/' \ +	      -e y/ABCDEFGHIJKLMNOPQRSTUVWXYZ/abcdefghijklmnopqrstuvwxyz/ \ +	      -e 's/\.[^.]*$/.X/' +	exit ;; +    CRAY*TS:*:*:*) +	echo t90-cray-unicos${UNAME_RELEASE} | sed -e 's/\.[^.]*$/.X/' +	exit ;; +    CRAY*T3E:*:*:*) +	echo alphaev5-cray-unicosmk${UNAME_RELEASE} | sed -e 's/\.[^.]*$/.X/' +	exit ;; +    CRAY*SV1:*:*:*) +	echo sv1-cray-unicos${UNAME_RELEASE} | sed -e 's/\.[^.]*$/.X/' +	exit ;; +    *:UNICOS/mp:*:*) +	echo craynv-cray-unicosmp${UNAME_RELEASE} | sed -e 's/\.[^.]*$/.X/' +	exit ;; +    F30[01]:UNIX_System_V:*:* | F700:UNIX_System_V:*:*) +	FUJITSU_PROC=`uname -m | tr 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' 'abcdefghijklmnopqrstuvwxyz'` +        FUJITSU_SYS=`uname -p | tr 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' 'abcdefghijklmnopqrstuvwxyz' | sed -e 's/\///'` +        FUJITSU_REL=`echo ${UNAME_RELEASE} | sed -e 's/ /_/'` +        echo "${FUJITSU_PROC}-fujitsu-${FUJITSU_SYS}${FUJITSU_REL}" +        exit ;; +    5000:UNIX_System_V:4.*:*) +        FUJITSU_SYS=`uname -p | tr 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' 'abcdefghijklmnopqrstuvwxyz' | sed -e 's/\///'` +        FUJITSU_REL=`echo ${UNAME_RELEASE} | tr 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' 'abcdefghijklmnopqrstuvwxyz' | sed -e 's/ /_/'` +        echo "sparc-fujitsu-${FUJITSU_SYS}${FUJITSU_REL}" +	exit ;; +    i*86:BSD/386:*:* | i*86:BSD/OS:*:* | *:Ascend\ Embedded/OS:*:*) +	echo ${UNAME_MACHINE}-pc-bsdi${UNAME_RELEASE} +	exit ;; +    sparc*:BSD/OS:*:*) +	echo sparc-unknown-bsdi${UNAME_RELEASE} +	exit ;; +    *:BSD/OS:*:*) +	echo ${UNAME_MACHINE}-unknown-bsdi${UNAME_RELEASE} +	exit ;; +    *:FreeBSD:*:*) +	case ${UNAME_MACHINE} in +	    pc98) +		echo i386-unknown-freebsd`echo ${UNAME_RELEASE}|sed -e 's/[-(].*//'` ;; +	    amd64) +		echo x86_64-unknown-freebsd`echo ${UNAME_RELEASE}|sed -e 's/[-(].*//'` ;; +	    *) +		echo ${UNAME_MACHINE}-unknown-freebsd`echo ${UNAME_RELEASE}|sed -e 's/[-(].*//'` ;; +	esac +	exit ;; +    i*:CYGWIN*:*) +	echo ${UNAME_MACHINE}-pc-cygwin +	exit ;; +    *:MINGW*:*) +	echo ${UNAME_MACHINE}-pc-mingw32 +	exit ;; +    i*:windows32*:*) +    	# uname -m includes "-pc" on this system. +    	echo ${UNAME_MACHINE}-mingw32 +	exit ;; +    i*:PW*:*) +	echo ${UNAME_MACHINE}-pc-pw32 +	exit ;; +    *:Interix*:[3456]*) +    	case ${UNAME_MACHINE} in +	    x86)  +		echo i586-pc-interix${UNAME_RELEASE} +		exit ;; +	    EM64T | authenticamd) +		echo x86_64-unknown-interix${UNAME_RELEASE} +		exit ;; +	esac ;; +    [345]86:Windows_95:* | [345]86:Windows_98:* | [345]86:Windows_NT:*) +	echo i${UNAME_MACHINE}-pc-mks +	exit ;; +    i*:Windows_NT*:* | Pentium*:Windows_NT*:*) +	# How do we know it's Interix rather than the generic POSIX subsystem? +	# It also conflicts with pre-2.0 versions of AT&T UWIN. Should we +	# UNAME_MACHINE based on the output of uname instead of i386? +	echo i586-pc-interix +	exit ;; +    i*:UWIN*:*) +	echo ${UNAME_MACHINE}-pc-uwin +	exit ;; +    amd64:CYGWIN*:*:* | x86_64:CYGWIN*:*:*) +	echo x86_64-unknown-cygwin +	exit ;; +    p*:CYGWIN*:*) +	echo powerpcle-unknown-cygwin +	exit ;; +    prep*:SunOS:5.*:*) +	echo powerpcle-unknown-solaris2`echo ${UNAME_RELEASE}|sed -e 's/[^.]*//'` +	exit ;; +    *:GNU:*:*) +	# the GNU system +	echo `echo ${UNAME_MACHINE}|sed -e 's,[-/].*$,,'`-unknown-gnu`echo ${UNAME_RELEASE}|sed -e 's,/.*$,,'` +	exit ;; +    *:GNU/*:*:*) +	# other systems with GNU libc and userland +	echo ${UNAME_MACHINE}-unknown-`echo ${UNAME_SYSTEM} | sed 's,^[^/]*/,,' | tr '[A-Z]' '[a-z]'``echo ${UNAME_RELEASE}|sed -e 's/[-(].*//'`-gnu +	exit ;; +    i*86:Minix:*:*) +	echo ${UNAME_MACHINE}-pc-minix +	exit ;; +    arm*:Linux:*:*) +	echo ${UNAME_MACHINE}-unknown-linux-gnu +	exit ;; +    avr32*:Linux:*:*) +	echo ${UNAME_MACHINE}-unknown-linux-gnu +	exit ;; +    cris:Linux:*:*) +	echo cris-axis-linux-gnu +	exit ;; +    crisv32:Linux:*:*) +	echo crisv32-axis-linux-gnu +	exit ;; +    frv:Linux:*:*) +    	echo frv-unknown-linux-gnu +	exit ;; +    ia64:Linux:*:*) +	echo ${UNAME_MACHINE}-unknown-linux-gnu +	exit ;; +    m32r*:Linux:*:*) +	echo ${UNAME_MACHINE}-unknown-linux-gnu +	exit ;; +    m68*:Linux:*:*) +	echo ${UNAME_MACHINE}-unknown-linux-gnu +	exit ;; +    mips:Linux:*:*) +	eval $set_cc_for_build +	sed 's/^	//' << EOF >$dummy.c +	#undef CPU +	#undef mips +	#undef mipsel +	#if defined(__MIPSEL__) || defined(__MIPSEL) || defined(_MIPSEL) || defined(MIPSEL) +	CPU=mipsel +	#else +	#if defined(__MIPSEB__) || defined(__MIPSEB) || defined(_MIPSEB) || defined(MIPSEB) +	CPU=mips +	#else +	CPU= +	#endif +	#endif +EOF +	eval "`$CC_FOR_BUILD -E $dummy.c 2>/dev/null | sed -n ' +	    /^CPU/{ +		s: ::g +		p +	    }'`" +	test x"${CPU}" != x && { echo "${CPU}-unknown-linux-gnu"; exit; } +	;; +    mips64:Linux:*:*) +	eval $set_cc_for_build +	sed 's/^	//' << EOF >$dummy.c +	#undef CPU +	#undef mips64 +	#undef mips64el +	#if defined(__MIPSEL__) || defined(__MIPSEL) || defined(_MIPSEL) || defined(MIPSEL) +	CPU=mips64el +	#else +	#if defined(__MIPSEB__) || defined(__MIPSEB) || defined(_MIPSEB) || defined(MIPSEB) +	CPU=mips64 +	#else +	CPU= +	#endif +	#endif +EOF +	eval "`$CC_FOR_BUILD -E $dummy.c 2>/dev/null | sed -n ' +	    /^CPU/{ +		s: ::g +		p +	    }'`" +	test x"${CPU}" != x && { echo "${CPU}-unknown-linux-gnu"; exit; } +	;; +    or32:Linux:*:*) +	echo or32-unknown-linux-gnu +	exit ;; +    ppc:Linux:*:*) +	echo powerpc-unknown-linux-gnu +	exit ;; +    ppc64:Linux:*:*) +	echo powerpc64-unknown-linux-gnu +	exit ;; +    alpha:Linux:*:*) +	case `sed -n '/^cpu model/s/^.*: \(.*\)/\1/p' < /proc/cpuinfo` in +	  EV5)   UNAME_MACHINE=alphaev5 ;; +	  EV56)  UNAME_MACHINE=alphaev56 ;; +	  PCA56) UNAME_MACHINE=alphapca56 ;; +	  PCA57) UNAME_MACHINE=alphapca56 ;; +	  EV6)   UNAME_MACHINE=alphaev6 ;; +	  EV67)  UNAME_MACHINE=alphaev67 ;; +	  EV68*) UNAME_MACHINE=alphaev68 ;; +        esac +	objdump --private-headers /bin/sh | grep ld.so.1 >/dev/null +	if test "$?" = 0 ; then LIBC="libc1" ; else LIBC="" ; fi +	echo ${UNAME_MACHINE}-unknown-linux-gnu${LIBC} +	exit ;; +    parisc:Linux:*:* | hppa:Linux:*:*) +	# Look for CPU level +	case `grep '^cpu[^a-z]*:' /proc/cpuinfo 2>/dev/null | cut -d' ' -f2` in +	  PA7*) echo hppa1.1-unknown-linux-gnu ;; +	  PA8*) echo hppa2.0-unknown-linux-gnu ;; +	  *)    echo hppa-unknown-linux-gnu ;; +	esac +	exit ;; +    parisc64:Linux:*:* | hppa64:Linux:*:*) +	echo hppa64-unknown-linux-gnu +	exit ;; +    s390:Linux:*:* | s390x:Linux:*:*) +	echo ${UNAME_MACHINE}-ibm-linux +	exit ;; +    sh64*:Linux:*:*) +    	echo ${UNAME_MACHINE}-unknown-linux-gnu +	exit ;; +    sh*:Linux:*:*) +	echo ${UNAME_MACHINE}-unknown-linux-gnu +	exit ;; +    sparc:Linux:*:* | sparc64:Linux:*:*) +	echo ${UNAME_MACHINE}-unknown-linux-gnu +	exit ;; +    tile:Linux:*:*) +	echo tile-unknown-linux-gnu +	exit ;; +    vax:Linux:*:*) +	echo ${UNAME_MACHINE}-dec-linux-gnu +	exit ;; +    x86_64:Linux:*:*) +	echo x86_64-unknown-linux-gnu +	exit ;; +    xtensa:Linux:*:*) +    	echo xtensa-unknown-linux-gnu +	exit ;; +    i*86:Linux:*:*) +	# The BFD linker knows what the default object file format is, so +	# first see if it will tell us. cd to the root directory to prevent +	# problems with other programs or directories called `ld' in the path. +	# Set LC_ALL=C to ensure ld outputs messages in English. +	ld_supported_targets=`cd /; LC_ALL=C ld --help 2>&1 \ +			 | sed -ne '/supported targets:/!d +				    s/[ 	][ 	]*/ /g +				    s/.*supported targets: *// +				    s/ .*// +				    p'` +        case "$ld_supported_targets" in +	  elf32-i386) +		TENTATIVE="${UNAME_MACHINE}-pc-linux-gnu" +		;; +	  a.out-i386-linux) +		echo "${UNAME_MACHINE}-pc-linux-gnuaout" +		exit ;; +	  coff-i386) +		echo "${UNAME_MACHINE}-pc-linux-gnucoff" +		exit ;; +	  "") +		# Either a pre-BFD a.out linker (linux-gnuoldld) or +		# one that does not give us useful --help. +		echo "${UNAME_MACHINE}-pc-linux-gnuoldld" +		exit ;; +	esac +	# Determine whether the default compiler is a.out or elf +	eval $set_cc_for_build +	sed 's/^	//' << EOF >$dummy.c +	#include <features.h> +	#ifdef __ELF__ +	# ifdef __GLIBC__ +	#  if __GLIBC__ >= 2 +	LIBC=gnu +	#  else +	LIBC=gnulibc1 +	#  endif +	# else +	LIBC=gnulibc1 +	# endif +	#else +	#if defined(__INTEL_COMPILER) || defined(__PGI) || defined(__SUNPRO_C) || defined(__SUNPRO_CC) +	LIBC=gnu +	#else +	LIBC=gnuaout +	#endif +	#endif +	#ifdef __dietlibc__ +	LIBC=dietlibc +	#endif +EOF +	eval "`$CC_FOR_BUILD -E $dummy.c 2>/dev/null | sed -n ' +	    /^LIBC/{ +		s: ::g +		p +	    }'`" +	test x"${LIBC}" != x && { +		echo "${UNAME_MACHINE}-pc-linux-${LIBC}" +		exit +	} +	test x"${TENTATIVE}" != x && { echo "${TENTATIVE}"; exit; } +	;; +    i*86:DYNIX/ptx:4*:*) +	# ptx 4.0 does uname -s correctly, with DYNIX/ptx in there. +	# earlier versions are messed up and put the nodename in both +	# sysname and nodename. +	echo i386-sequent-sysv4 +	exit ;; +    i*86:UNIX_SV:4.2MP:2.*) +        # Unixware is an offshoot of SVR4, but it has its own version +        # number series starting with 2... +        # I am not positive that other SVR4 systems won't match this, +	# I just have to hope.  -- rms. +        # Use sysv4.2uw... so that sysv4* matches it. +	echo ${UNAME_MACHINE}-pc-sysv4.2uw${UNAME_VERSION} +	exit ;; +    i*86:OS/2:*:*) +	# If we were able to find `uname', then EMX Unix compatibility +	# is probably installed. +	echo ${UNAME_MACHINE}-pc-os2-emx +	exit ;; +    i*86:XTS-300:*:STOP) +	echo ${UNAME_MACHINE}-unknown-stop +	exit ;; +    i*86:atheos:*:*) +	echo ${UNAME_MACHINE}-unknown-atheos +	exit ;; +    i*86:syllable:*:*) +	echo ${UNAME_MACHINE}-pc-syllable +	exit ;; +    i*86:LynxOS:2.*:* | i*86:LynxOS:3.[01]*:* | i*86:LynxOS:4.0*:*) +	echo i386-unknown-lynxos${UNAME_RELEASE} +	exit ;; +    i*86:*DOS:*:*) +	echo ${UNAME_MACHINE}-pc-msdosdjgpp +	exit ;; +    i*86:*:4.*:* | i*86:SYSTEM_V:4.*:*) +	UNAME_REL=`echo ${UNAME_RELEASE} | sed 's/\/MP$//'` +	if grep Novell /usr/include/link.h >/dev/null 2>/dev/null; then +		echo ${UNAME_MACHINE}-univel-sysv${UNAME_REL} +	else +		echo ${UNAME_MACHINE}-pc-sysv${UNAME_REL} +	fi +	exit ;; +    i*86:*:5:[678]*) +    	# UnixWare 7.x, OpenUNIX and OpenServer 6. +	case `/bin/uname -X | grep "^Machine"` in +	    *486*)	     UNAME_MACHINE=i486 ;; +	    *Pentium)	     UNAME_MACHINE=i586 ;; +	    *Pent*|*Celeron) UNAME_MACHINE=i686 ;; +	esac +	echo ${UNAME_MACHINE}-unknown-sysv${UNAME_RELEASE}${UNAME_SYSTEM}${UNAME_VERSION} +	exit ;; +    i*86:*:3.2:*) +	if test -f /usr/options/cb.name; then +		UNAME_REL=`sed -n 's/.*Version //p' </usr/options/cb.name` +		echo ${UNAME_MACHINE}-pc-isc$UNAME_REL +	elif /bin/uname -X 2>/dev/null >/dev/null ; then +		UNAME_REL=`(/bin/uname -X|grep Release|sed -e 's/.*= //')` +		(/bin/uname -X|grep i80486 >/dev/null) && UNAME_MACHINE=i486 +		(/bin/uname -X|grep '^Machine.*Pentium' >/dev/null) \ +			&& UNAME_MACHINE=i586 +		(/bin/uname -X|grep '^Machine.*Pent *II' >/dev/null) \ +			&& UNAME_MACHINE=i686 +		(/bin/uname -X|grep '^Machine.*Pentium Pro' >/dev/null) \ +			&& UNAME_MACHINE=i686 +		echo ${UNAME_MACHINE}-pc-sco$UNAME_REL +	else +		echo ${UNAME_MACHINE}-pc-sysv32 +	fi +	exit ;; +    pc:*:*:*) +	# Left here for compatibility: +        # uname -m prints for DJGPP always 'pc', but it prints nothing about +        # the processor, so we play safe by assuming i386. +	echo i386-pc-msdosdjgpp +        exit ;; +    Intel:Mach:3*:*) +	echo i386-pc-mach3 +	exit ;; +    paragon:*:*:*) +	echo i860-intel-osf1 +	exit ;; +    i860:*:4.*:*) # i860-SVR4 +	if grep Stardent /usr/include/sys/uadmin.h >/dev/null 2>&1 ; then +	  echo i860-stardent-sysv${UNAME_RELEASE} # Stardent Vistra i860-SVR4 +	else # Add other i860-SVR4 vendors below as they are discovered. +	  echo i860-unknown-sysv${UNAME_RELEASE}  # Unknown i860-SVR4 +	fi +	exit ;; +    mini*:CTIX:SYS*5:*) +	# "miniframe" +	echo m68010-convergent-sysv +	exit ;; +    mc68k:UNIX:SYSTEM5:3.51m) +	echo m68k-convergent-sysv +	exit ;; +    M680?0:D-NIX:5.3:*) +	echo m68k-diab-dnix +	exit ;; +    M68*:*:R3V[5678]*:*) +	test -r /sysV68 && { echo 'm68k-motorola-sysv'; exit; } ;; +    3[345]??:*:4.0:3.0 | 3[34]??A:*:4.0:3.0 | 3[34]??,*:*:4.0:3.0 | 3[34]??/*:*:4.0:3.0 | 4400:*:4.0:3.0 | 4850:*:4.0:3.0 | SKA40:*:4.0:3.0 | SDS2:*:4.0:3.0 | SHG2:*:4.0:3.0 | S7501*:*:4.0:3.0) +	OS_REL='' +	test -r /etc/.relid \ +	&& OS_REL=.`sed -n 's/[^ ]* [^ ]* \([0-9][0-9]\).*/\1/p' < /etc/.relid` +	/bin/uname -p 2>/dev/null | grep 86 >/dev/null \ +	  && { echo i486-ncr-sysv4.3${OS_REL}; exit; } +	/bin/uname -p 2>/dev/null | /bin/grep entium >/dev/null \ +	  && { echo i586-ncr-sysv4.3${OS_REL}; exit; } ;; +    3[34]??:*:4.0:* | 3[34]??,*:*:4.0:*) +        /bin/uname -p 2>/dev/null | grep 86 >/dev/null \ +          && { echo i486-ncr-sysv4; exit; } ;; +    m68*:LynxOS:2.*:* | m68*:LynxOS:3.0*:*) +	echo m68k-unknown-lynxos${UNAME_RELEASE} +	exit ;; +    mc68030:UNIX_System_V:4.*:*) +	echo m68k-atari-sysv4 +	exit ;; +    TSUNAMI:LynxOS:2.*:*) +	echo sparc-unknown-lynxos${UNAME_RELEASE} +	exit ;; +    rs6000:LynxOS:2.*:*) +	echo rs6000-unknown-lynxos${UNAME_RELEASE} +	exit ;; +    PowerPC:LynxOS:2.*:* | PowerPC:LynxOS:3.[01]*:* | PowerPC:LynxOS:4.0*:*) +	echo powerpc-unknown-lynxos${UNAME_RELEASE} +	exit ;; +    SM[BE]S:UNIX_SV:*:*) +	echo mips-dde-sysv${UNAME_RELEASE} +	exit ;; +    RM*:ReliantUNIX-*:*:*) +	echo mips-sni-sysv4 +	exit ;; +    RM*:SINIX-*:*:*) +	echo mips-sni-sysv4 +	exit ;; +    *:SINIX-*:*:*) +	if uname -p 2>/dev/null >/dev/null ; then +		UNAME_MACHINE=`(uname -p) 2>/dev/null` +		echo ${UNAME_MACHINE}-sni-sysv4 +	else +		echo ns32k-sni-sysv +	fi +	exit ;; +    PENTIUM:*:4.0*:*) # Unisys `ClearPath HMP IX 4000' SVR4/MP effort +                      # says <[email protected]> +        echo i586-unisys-sysv4 +        exit ;; +    *:UNIX_System_V:4*:FTX*) +	# From Gerald Hewes <[email protected]>. +	# How about differentiating between stratus architectures? -djm +	echo hppa1.1-stratus-sysv4 +	exit ;; +    *:*:*:FTX*) +	# From [email protected]. +	echo i860-stratus-sysv4 +	exit ;; +    i*86:VOS:*:*) +	# From [email protected]. +	echo ${UNAME_MACHINE}-stratus-vos +	exit ;; +    *:VOS:*:*) +	# From [email protected]. +	echo hppa1.1-stratus-vos +	exit ;; +    mc68*:A/UX:*:*) +	echo m68k-apple-aux${UNAME_RELEASE} +	exit ;; +    news*:NEWS-OS:6*:*) +	echo mips-sony-newsos6 +	exit ;; +    R[34]000:*System_V*:*:* | R4000:UNIX_SYSV:*:* | R*000:UNIX_SV:*:*) +	if [ -d /usr/nec ]; then +	        echo mips-nec-sysv${UNAME_RELEASE} +	else +	        echo mips-unknown-sysv${UNAME_RELEASE} +	fi +        exit ;; +    BeBox:BeOS:*:*)	# BeOS running on hardware made by Be, PPC only. +	echo powerpc-be-beos +	exit ;; +    BeMac:BeOS:*:*)	# BeOS running on Mac or Mac clone, PPC only. +	echo powerpc-apple-beos +	exit ;; +    BePC:BeOS:*:*)	# BeOS running on Intel PC compatible. +	echo i586-pc-beos +	exit ;; +    SX-4:SUPER-UX:*:*) +	echo sx4-nec-superux${UNAME_RELEASE} +	exit ;; +    SX-5:SUPER-UX:*:*) +	echo sx5-nec-superux${UNAME_RELEASE} +	exit ;; +    SX-6:SUPER-UX:*:*) +	echo sx6-nec-superux${UNAME_RELEASE} +	exit ;; +    SX-7:SUPER-UX:*:*) +	echo sx7-nec-superux${UNAME_RELEASE} +	exit ;; +    SX-8:SUPER-UX:*:*) +	echo sx8-nec-superux${UNAME_RELEASE} +	exit ;; +    SX-8R:SUPER-UX:*:*) +	echo sx8r-nec-superux${UNAME_RELEASE} +	exit ;; +    Power*:Rhapsody:*:*) +	echo powerpc-apple-rhapsody${UNAME_RELEASE} +	exit ;; +    *:Rhapsody:*:*) +	echo ${UNAME_MACHINE}-apple-rhapsody${UNAME_RELEASE} +	exit ;; +    *:Darwin:*:*) +	UNAME_PROCESSOR=`uname -p` || UNAME_PROCESSOR=unknown +	case $UNAME_PROCESSOR in +	    unknown) UNAME_PROCESSOR=powerpc ;; +	esac +	echo ${UNAME_PROCESSOR}-apple-darwin${UNAME_RELEASE} +	exit ;; +    *:procnto*:*:* | *:QNX:[0123456789]*:*) +	UNAME_PROCESSOR=`uname -p` +	if test "$UNAME_PROCESSOR" = "x86"; then +		UNAME_PROCESSOR=i386 +		UNAME_MACHINE=pc +	fi +	echo ${UNAME_PROCESSOR}-${UNAME_MACHINE}-nto-qnx${UNAME_RELEASE} +	exit ;; +    *:QNX:*:4*) +	echo i386-pc-qnx +	exit ;; +    NSE-?:NONSTOP_KERNEL:*:*) +	echo nse-tandem-nsk${UNAME_RELEASE} +	exit ;; +    NSR-?:NONSTOP_KERNEL:*:*) +	echo nsr-tandem-nsk${UNAME_RELEASE} +	exit ;; +    *:NonStop-UX:*:*) +	echo mips-compaq-nonstopux +	exit ;; +    BS2000:POSIX*:*:*) +	echo bs2000-siemens-sysv +	exit ;; +    DS/*:UNIX_System_V:*:*) +	echo ${UNAME_MACHINE}-${UNAME_SYSTEM}-${UNAME_RELEASE} +	exit ;; +    *:Plan9:*:*) +	# "uname -m" is not consistent, so use $cputype instead. 386 +	# is converted to i386 for consistency with other x86 +	# operating systems. +	if test "$cputype" = "386"; then +	    UNAME_MACHINE=i386 +	else +	    UNAME_MACHINE="$cputype" +	fi +	echo ${UNAME_MACHINE}-unknown-plan9 +	exit ;; +    *:TOPS-10:*:*) +	echo pdp10-unknown-tops10 +	exit ;; +    *:TENEX:*:*) +	echo pdp10-unknown-tenex +	exit ;; +    KS10:TOPS-20:*:* | KL10:TOPS-20:*:* | TYPE4:TOPS-20:*:*) +	echo pdp10-dec-tops20 +	exit ;; +    XKL-1:TOPS-20:*:* | TYPE5:TOPS-20:*:*) +	echo pdp10-xkl-tops20 +	exit ;; +    *:TOPS-20:*:*) +	echo pdp10-unknown-tops20 +	exit ;; +    *:ITS:*:*) +	echo pdp10-unknown-its +	exit ;; +    SEI:*:*:SEIUX) +        echo mips-sei-seiux${UNAME_RELEASE} +	exit ;; +    *:DragonFly:*:*) +	echo ${UNAME_MACHINE}-unknown-dragonfly`echo ${UNAME_RELEASE}|sed -e 's/[-(].*//'` +	exit ;; +    *:*VMS:*:*) +    	UNAME_MACHINE=`(uname -p) 2>/dev/null` +	case "${UNAME_MACHINE}" in +	    A*) echo alpha-dec-vms ; exit ;; +	    I*) echo ia64-dec-vms ; exit ;; +	    V*) echo vax-dec-vms ; exit ;; +	esac ;; +    *:XENIX:*:SysV) +	echo i386-pc-xenix +	exit ;; +    i*86:skyos:*:*) +	echo ${UNAME_MACHINE}-pc-skyos`echo ${UNAME_RELEASE}` | sed -e 's/ .*$//' +	exit ;; +    i*86:rdos:*:*) +	echo ${UNAME_MACHINE}-pc-rdos +	exit ;; +esac + +#echo '(No uname command or uname output not recognized.)' 1>&2 +#echo "${UNAME_MACHINE}:${UNAME_SYSTEM}:${UNAME_RELEASE}:${UNAME_VERSION}" 1>&2 + +eval $set_cc_for_build +cat >$dummy.c <<EOF +#ifdef _SEQUENT_ +# include <sys/types.h> +# include <sys/utsname.h> +#endif +main () +{ +#if defined (sony) +#if defined (MIPSEB) +  /* BFD wants "bsd" instead of "newsos".  Perhaps BFD should be changed, +     I don't know....  */ +  printf ("mips-sony-bsd\n"); exit (0); +#else +#include <sys/param.h> +  printf ("m68k-sony-newsos%s\n", +#ifdef NEWSOS4 +          "4" +#else +	  "" +#endif +         ); exit (0); +#endif +#endif + +#if defined (__arm) && defined (__acorn) && defined (__unix) +  printf ("arm-acorn-riscix\n"); exit (0); +#endif + +#if defined (hp300) && !defined (hpux) +  printf ("m68k-hp-bsd\n"); exit (0); +#endif + +#if defined (NeXT) +#if !defined (__ARCHITECTURE__) +#define __ARCHITECTURE__ "m68k" +#endif +  int version; +  version=`(hostinfo | sed -n 's/.*NeXT Mach \([0-9]*\).*/\1/p') 2>/dev/null`; +  if (version < 4) +    printf ("%s-next-nextstep%d\n", __ARCHITECTURE__, version); +  else +    printf ("%s-next-openstep%d\n", __ARCHITECTURE__, version); +  exit (0); +#endif + +#if defined (MULTIMAX) || defined (n16) +#if defined (UMAXV) +  printf ("ns32k-encore-sysv\n"); exit (0); +#else +#if defined (CMU) +  printf ("ns32k-encore-mach\n"); exit (0); +#else +  printf ("ns32k-encore-bsd\n"); exit (0); +#endif +#endif +#endif + +#if defined (__386BSD__) +  printf ("i386-pc-bsd\n"); exit (0); +#endif + +#if defined (sequent) +#if defined (i386) +  printf ("i386-sequent-dynix\n"); exit (0); +#endif +#if defined (ns32000) +  printf ("ns32k-sequent-dynix\n"); exit (0); +#endif +#endif + +#if defined (_SEQUENT_) +    struct utsname un; + +    uname(&un); + +    if (strncmp(un.version, "V2", 2) == 0) { +	printf ("i386-sequent-ptx2\n"); exit (0); +    } +    if (strncmp(un.version, "V1", 2) == 0) { /* XXX is V1 correct? */ +	printf ("i386-sequent-ptx1\n"); exit (0); +    } +    printf ("i386-sequent-ptx\n"); exit (0); + +#endif + +#if defined (vax) +# if !defined (ultrix) +#  include <sys/param.h> +#  if defined (BSD) +#   if BSD == 43 +      printf ("vax-dec-bsd4.3\n"); exit (0); +#   else +#    if BSD == 199006 +      printf ("vax-dec-bsd4.3reno\n"); exit (0); +#    else +      printf ("vax-dec-bsd\n"); exit (0); +#    endif +#   endif +#  else +    printf ("vax-dec-bsd\n"); exit (0); +#  endif +# else +    printf ("vax-dec-ultrix\n"); exit (0); +# endif +#endif + +#if defined (alliant) && defined (i860) +  printf ("i860-alliant-bsd\n"); exit (0); +#endif + +  exit (1); +} +EOF + +$CC_FOR_BUILD -o $dummy $dummy.c 2>/dev/null && SYSTEM_NAME=`$dummy` && +	{ echo "$SYSTEM_NAME"; exit; } + +# Apollos put the system type in the environment. + +test -d /usr/apollo && { echo ${ISP}-apollo-${SYSTYPE}; exit; } + +# Convex versions that predate uname can use getsysinfo(1) + +if [ -x /usr/convex/getsysinfo ] +then +    case `getsysinfo -f cpu_type` in +    c1*) +	echo c1-convex-bsd +	exit ;; +    c2*) +	if getsysinfo -f scalar_acc +	then echo c32-convex-bsd +	else echo c2-convex-bsd +	fi +	exit ;; +    c34*) +	echo c34-convex-bsd +	exit ;; +    c38*) +	echo c38-convex-bsd +	exit ;; +    c4*) +	echo c4-convex-bsd +	exit ;; +    esac +fi + +cat >&2 <<EOF +$0: unable to guess system type + +This script, last modified $timestamp, has failed to recognize +the operating system you are using. It is advised that you +download the most up to date version of the config scripts from + +  http://savannah.gnu.org/cgi-bin/viewcvs/*checkout*/config/config/config.guess +and +  http://savannah.gnu.org/cgi-bin/viewcvs/*checkout*/config/config/config.sub + +If the version you run ($0) is already up to date, please +send the following data and any information you think might be +pertinent to <[email protected]> in order to provide the needed +information to handle your system. + +config.guess timestamp = $timestamp + +uname -m = `(uname -m) 2>/dev/null || echo unknown` +uname -r = `(uname -r) 2>/dev/null || echo unknown` +uname -s = `(uname -s) 2>/dev/null || echo unknown` +uname -v = `(uname -v) 2>/dev/null || echo unknown` + +/usr/bin/uname -p = `(/usr/bin/uname -p) 2>/dev/null` +/bin/uname -X     = `(/bin/uname -X) 2>/dev/null` + +hostinfo               = `(hostinfo) 2>/dev/null` +/bin/universe          = `(/bin/universe) 2>/dev/null` +/usr/bin/arch -k       = `(/usr/bin/arch -k) 2>/dev/null` +/bin/arch              = `(/bin/arch) 2>/dev/null` +/usr/bin/oslevel       = `(/usr/bin/oslevel) 2>/dev/null` +/usr/convex/getsysinfo = `(/usr/convex/getsysinfo) 2>/dev/null` + +UNAME_MACHINE = ${UNAME_MACHINE} +UNAME_RELEASE = ${UNAME_RELEASE} +UNAME_SYSTEM  = ${UNAME_SYSTEM} +UNAME_VERSION = ${UNAME_VERSION} +EOF + +exit 1 + +# Local variables: +# eval: (add-hook 'write-file-hooks 'time-stamp) +# time-stamp-start: "timestamp='" +# time-stamp-format: "%:y-%02m-%02d" +# time-stamp-end: "'" +# End: diff --git a/lib/test_server/src/config.sub b/lib/test_server/src/config.sub index 47a0f10138..f43233b104 120000..100755 --- a/lib/test_server/src/config.sub +++ b/lib/test_server/src/config.sub @@ -1 +1,1630 @@ -../../../erts/autoconf/config.sub
\ No newline at end of file +#! /bin/sh +# Configuration validation subroutine script. +#   Copyright (C) 1992, 1993, 1994, 1995, 1996, 1997, 1998, 1999, +#   2000, 2001, 2002, 2003, 2004, 2005, 2006 Free Software Foundation, +#   Inc. + +timestamp='2007-04-29' + +# This file is (in principle) common to ALL GNU software. +# The presence of a machine in this file suggests that SOME GNU software +# can handle that machine.  It does not imply ALL GNU software can. +# +# This file 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; either version 2 of the License, or +# (at your option) any later version. +# +# 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, write to the Free Software +# Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA +# 02110-1301, USA. +# +# As a special exception to the GNU General Public License, if you +# distribute this file as part of a program that contains a +# configuration script generated by Autoconf, you may include it under +# the same distribution terms that you use for the rest of that program. + + +# Please send patches to <[email protected]>.  Submit a context +# diff and a properly formatted ChangeLog entry. +# +# Configuration subroutine to validate and canonicalize a configuration type. +# Supply the specified configuration type as an argument. +# If it is invalid, we print an error message on stderr and exit with code 1. +# Otherwise, we print the canonical config type on stdout and succeed. + +# This file is supposed to be the same for all GNU packages +# and recognize all the CPU types, system types and aliases +# that are meaningful with *any* GNU software. +# Each package is responsible for reporting which valid configurations +# it does not support.  The user should be able to distinguish +# a failure to support a valid configuration from a meaningless +# configuration. + +# The goal of this file is to map all the various variations of a given +# machine specification into a single specification in the form: +#	CPU_TYPE-MANUFACTURER-OPERATING_SYSTEM +# or in some cases, the newer four-part form: +#	CPU_TYPE-MANUFACTURER-KERNEL-OPERATING_SYSTEM +# It is wrong to echo any other type of specification. + +me=`echo "$0" | sed -e 's,.*/,,'` + +usage="\ +Usage: $0 [OPTION] CPU-MFR-OPSYS +       $0 [OPTION] ALIAS + +Canonicalize a configuration name. + +Operation modes: +  -h, --help         print this help, then exit +  -t, --time-stamp   print date of last modification, then exit +  -v, --version      print version number, then exit + +Report bugs and patches to <[email protected]>." + +version="\ +GNU config.sub ($timestamp) + +Copyright (C) 1992, 1993, 1994, 1995, 1996, 1997, 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005 +Free Software Foundation, Inc. + +This is free software; see the source for copying conditions.  There is NO +warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE." + +help=" +Try \`$me --help' for more information." + +# Parse command line +while test $# -gt 0 ; do +  case $1 in +    --time-stamp | --time* | -t ) +       echo "$timestamp" ; exit ;; +    --version | -v ) +       echo "$version" ; exit ;; +    --help | --h* | -h ) +       echo "$usage"; exit ;; +    -- )     # Stop option processing +       shift; break ;; +    - )	# Use stdin as input. +       break ;; +    -* ) +       echo "$me: invalid option $1$help" +       exit 1 ;; + +    *local*) +       # First pass through any local machine types. +       echo $1 +       exit ;; + +    * ) +       break ;; +  esac +done + +case $# in + 0) echo "$me: missing argument$help" >&2 +    exit 1;; + 1) ;; + *) echo "$me: too many arguments$help" >&2 +    exit 1;; +esac + +# Separate what the user gave into CPU-COMPANY and OS or KERNEL-OS (if any). +# Here we must recognize all the valid KERNEL-OS combinations. +maybe_os=`echo $1 | sed 's/^\(.*\)-\([^-]*-[^-]*\)$/\2/'` +case $maybe_os in +  nto-qnx* | linux-gnu* | linux-dietlibc | linux-newlib* | linux-uclibc* | \ +  uclinux-uclibc* | uclinux-gnu* | kfreebsd*-gnu* | knetbsd*-gnu* | netbsd*-gnu* | \ +  storm-chaos* | os2-emx* | rtmk-nova*) +    os=-$maybe_os +    basic_machine=`echo $1 | sed 's/^\(.*\)-\([^-]*-[^-]*\)$/\1/'` +    ;; +  *) +    basic_machine=`echo $1 | sed 's/-[^-]*$//'` +    if [ $basic_machine != $1 ] +    then os=`echo $1 | sed 's/.*-/-/'` +    else os=; fi +    ;; +esac + +### Let's recognize common machines as not being operating systems so +### that things like config.sub decstation-3100 work.  We also +### recognize some manufacturers as not being operating systems, so we +### can provide default operating systems below. +case $os in +	-sun*os*) +		# Prevent following clause from handling this invalid input. +		;; +	-dec* | -mips* | -sequent* | -encore* | -pc532* | -sgi* | -sony* | \ +	-att* | -7300* | -3300* | -delta* | -motorola* | -sun[234]* | \ +	-unicom* | -ibm* | -next | -hp | -isi* | -apollo | -altos* | \ +	-convergent* | -ncr* | -news | -32* | -3600* | -3100* | -hitachi* |\ +	-c[123]* | -convex* | -sun | -crds | -omron* | -dg | -ultra | -tti* | \ +	-harris | -dolphin | -highlevel | -gould | -cbm | -ns | -masscomp | \ +	-apple | -axis | -knuth | -cray) +		os= +		basic_machine=$1 +		;; +	-sim | -cisco | -oki | -wec | -winbond) +		os= +		basic_machine=$1 +		;; +	-scout) +		;; +	-wrs) +		os=-vxworks +		basic_machine=$1 +		;; +	-chorusos*) +		os=-chorusos +		basic_machine=$1 +		;; + 	-chorusrdb) + 		os=-chorusrdb +		basic_machine=$1 + 		;; +	-hiux*) +		os=-hiuxwe2 +		;; +	-sco6) +		os=-sco5v6 +		basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'` +		;; +	-sco5) +		os=-sco3.2v5 +		basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'` +		;; +	-sco4) +		os=-sco3.2v4 +		basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'` +		;; +	-sco3.2.[4-9]*) +		os=`echo $os | sed -e 's/sco3.2./sco3.2v/'` +		basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'` +		;; +	-sco3.2v[4-9]*) +		# Don't forget version if it is 3.2v4 or newer. +		basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'` +		;; +	-sco5v6*) +		# Don't forget version if it is 3.2v4 or newer. +		basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'` +		;; +	-sco*) +		os=-sco3.2v2 +		basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'` +		;; +	-udk*) +		basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'` +		;; +	-isc) +		os=-isc2.2 +		basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'` +		;; +	-clix*) +		basic_machine=clipper-intergraph +		;; +	-isc*) +		basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'` +		;; +	-lynx*) +		os=-lynxos +		;; +	-ptx*) +		basic_machine=`echo $1 | sed -e 's/86-.*/86-sequent/'` +		;; +	-windowsnt*) +		os=`echo $os | sed -e 's/windowsnt/winnt/'` +		;; +	-psos*) +		os=-psos +		;; +	-mint | -mint[0-9]*) +		basic_machine=m68k-atari +		os=-mint +		;; +esac + +# Decode aliases for certain CPU-COMPANY combinations. +case $basic_machine in +	# Recognize the basic CPU types without company name. +	# Some are omitted here because they have special meanings below. +	1750a | 580 \ +	| a29k \ +	| alpha | alphaev[4-8] | alphaev56 | alphaev6[78] | alphapca5[67] \ +	| alpha64 | alpha64ev[4-8] | alpha64ev56 | alpha64ev6[78] | alpha64pca5[67] \ +	| am33_2.0 \ +	| arc | arm | arm[bl]e | arme[lb] | armv[2345] | armv[345][lb] | avr | avr32 \ +	| bfin \ +	| c4x | clipper \ +	| d10v | d30v | dlx | dsp16xx \ +	| fido | fr30 | frv \ +	| h8300 | h8500 | hppa | hppa1.[01] | hppa2.0 | hppa2.0[nw] | hppa64 \ +	| i370 | i860 | i960 | ia64 \ +	| ip2k | iq2000 \ +	| m32c | m32r | m32rle | m68000 | m68k | m88k \ +	| maxq | mb | microblaze | mcore | mep \ +	| mips | mipsbe | mipseb | mipsel | mipsle \ +	| mips16 \ +	| mips64 | mips64el \ +	| mips64vr | mips64vrel \ +	| mips64orion | mips64orionel \ +	| mips64vr4100 | mips64vr4100el \ +	| mips64vr4300 | mips64vr4300el \ +	| mips64vr5000 | mips64vr5000el \ +	| mips64vr5900 | mips64vr5900el \ +	| mipsisa32 | mipsisa32el \ +	| mipsisa32r2 | mipsisa32r2el \ +	| mipsisa64 | mipsisa64el \ +	| mipsisa64r2 | mipsisa64r2el \ +	| mipsisa64sb1 | mipsisa64sb1el \ +	| mipsisa64sr71k | mipsisa64sr71kel \ +	| mipstx39 | mipstx39el \ +	| mn10200 | mn10300 \ +	| mt \ +	| msp430 \ +	| nios | nios2 \ +	| ns16k | ns32k \ +	| or32 \ +	| pdp10 | pdp11 | pj | pjl \ +	| powerpc | powerpc64 | powerpc64le | powerpcle | ppcbe \ +	| pyramid \ +	| score \ +	| sh | sh[1234] | sh[24]a | sh[23]e | sh[34]eb | sheb | shbe | shle | sh[1234]le | sh3ele \ +	| sh64 | sh64le \ +	| sparc | sparc64 | sparc64b | sparc64v | sparc86x | sparclet | sparclite \ +	| sparcv8 | sparcv9 | sparcv9b | sparcv9v \ +	| spu | strongarm \ +	| tahoe | thumb | tic4x | tic80 | tron \ +	| v850 | v850e \ +	| we32k \ +	| x86 | xc16x | xscale | xscalee[bl] | xstormy16 | xtensa \ +	| z8k) +		basic_machine=$basic_machine-unknown +		;; +	m6811 | m68hc11 | m6812 | m68hc12) +		# Motorola 68HC11/12. +		basic_machine=$basic_machine-unknown +		os=-none +		;; +	m88110 | m680[12346]0 | m683?2 | m68360 | m5200 | v70 | w65 | z8k) +		;; +	ms1) +		basic_machine=mt-unknown +		;; + +	# We use `pc' rather than `unknown' +	# because (1) that's what they normally are, and +	# (2) the word "unknown" tends to confuse beginning users. +	i*86 | x86_64) +	  basic_machine=$basic_machine-pc +	  ;; +	# Object if more than one company name word. +	*-*-*) +		echo Invalid configuration \`$1\': machine \`$basic_machine\' not recognized 1>&2 +		exit 1 +		;; +	# Recognize the basic CPU types with company name. +	580-* \ +	| a29k-* \ +	| alpha-* | alphaev[4-8]-* | alphaev56-* | alphaev6[78]-* \ +	| alpha64-* | alpha64ev[4-8]-* | alpha64ev56-* | alpha64ev6[78]-* \ +	| alphapca5[67]-* | alpha64pca5[67]-* | arc-* \ +	| arm-*  | armbe-* | armle-* | armeb-* | armv*-* \ +	| avr-* | avr32-* \ +	| bfin-* | bs2000-* \ +	| c[123]* | c30-* | [cjt]90-* | c4x-* | c54x-* | c55x-* | c6x-* \ +	| clipper-* | craynv-* | cydra-* \ +	| d10v-* | d30v-* | dlx-* \ +	| elxsi-* \ +	| f30[01]-* | f700-* | fido-* | fr30-* | frv-* | fx80-* \ +	| h8300-* | h8500-* \ +	| hppa-* | hppa1.[01]-* | hppa2.0-* | hppa2.0[nw]-* | hppa64-* \ +	| i*86-* | i860-* | i960-* | ia64-* \ +	| ip2k-* | iq2000-* \ +	| m32c-* | m32r-* | m32rle-* \ +	| m68000-* | m680[012346]0-* | m68360-* | m683?2-* | m68k-* \ +	| m88110-* | m88k-* | maxq-* | mcore-* \ +	| mips-* | mipsbe-* | mipseb-* | mipsel-* | mipsle-* \ +	| mips16-* \ +	| mips64-* | mips64el-* \ +	| mips64vr-* | mips64vrel-* \ +	| mips64orion-* | mips64orionel-* \ +	| mips64vr4100-* | mips64vr4100el-* \ +	| mips64vr4300-* | mips64vr4300el-* \ +	| mips64vr5000-* | mips64vr5000el-* \ +	| mips64vr5900-* | mips64vr5900el-* \ +	| mipsisa32-* | mipsisa32el-* \ +	| mipsisa32r2-* | mipsisa32r2el-* \ +	| mipsisa64-* | mipsisa64el-* \ +	| mipsisa64r2-* | mipsisa64r2el-* \ +	| mipsisa64sb1-* | mipsisa64sb1el-* \ +	| mipsisa64sr71k-* | mipsisa64sr71kel-* \ +	| mipstx39-* | mipstx39el-* \ +	| mmix-* \ +	| mt-* \ +	| msp430-* \ +	| nios-* | nios2-* \ +	| none-* | np1-* | ns16k-* | ns32k-* \ +	| orion-* \ +	| pdp10-* | pdp11-* | pj-* | pjl-* | pn-* | power-* \ +	| powerpc-* | powerpc64-* | powerpc64le-* | powerpcle-* | ppcbe-* \ +	| pyramid-* \ +	| romp-* | rs6000-* \ +	| sh-* | sh[1234]-* | sh[24]a-* | sh[23]e-* | sh[34]eb-* | sheb-* | shbe-* \ +	| shle-* | sh[1234]le-* | sh3ele-* | sh64-* | sh64le-* \ +	| sparc-* | sparc64-* | sparc64b-* | sparc64v-* | sparc86x-* | sparclet-* \ +	| sparclite-* \ +	| sparcv8-* | sparcv9-* | sparcv9b-* | sparcv9v-* | strongarm-* | sv1-* | sx?-* \ +	| tahoe-* | thumb-* \ +	| tic30-* | tic4x-* | tic54x-* | tic55x-* | tic6x-* | tic80-* \ +	| tron-* \ +	| v850-* | v850e-* | vax-* \ +	| we32k-* \ +	| x86-* | x86_64-* | xc16x-* | xps100-* | xscale-* | xscalee[bl]-* \ +	| xstormy16-* | xtensa-* \ +	| ymp-* \ +	| z8k-*) +		;; +	# Recognize the various machine names and aliases which stand +	# for a CPU type and a company and sometimes even an OS. +	386bsd) +		basic_machine=i386-unknown +		os=-bsd +		;; +	3b1 | 7300 | 7300-att | att-7300 | pc7300 | safari | unixpc) +		basic_machine=m68000-att +		;; +	3b*) +		basic_machine=we32k-att +		;; +	a29khif) +		basic_machine=a29k-amd +		os=-udi +		;; +    	abacus) +		basic_machine=abacus-unknown +		;; +	adobe68k) +		basic_machine=m68010-adobe +		os=-scout +		;; +	alliant | fx80) +		basic_machine=fx80-alliant +		;; +	altos | altos3068) +		basic_machine=m68k-altos +		;; +	am29k) +		basic_machine=a29k-none +		os=-bsd +		;; +	amd64) +		basic_machine=x86_64-pc +		;; +	amd64-*) +		basic_machine=x86_64-`echo $basic_machine | sed 's/^[^-]*-//'` +		;; +	amdahl) +		basic_machine=580-amdahl +		os=-sysv +		;; +	amiga | amiga-*) +		basic_machine=m68k-unknown +		;; +	amigaos | amigados) +		basic_machine=m68k-unknown +		os=-amigaos +		;; +	amigaunix | amix) +		basic_machine=m68k-unknown +		os=-sysv4 +		;; +	apollo68) +		basic_machine=m68k-apollo +		os=-sysv +		;; +	apollo68bsd) +		basic_machine=m68k-apollo +		os=-bsd +		;; +	aux) +		basic_machine=m68k-apple +		os=-aux +		;; +	balance) +		basic_machine=ns32k-sequent +		os=-dynix +		;; +	c90) +		basic_machine=c90-cray +		os=-unicos +		;; +	convex-c1) +		basic_machine=c1-convex +		os=-bsd +		;; +	convex-c2) +		basic_machine=c2-convex +		os=-bsd +		;; +	convex-c32) +		basic_machine=c32-convex +		os=-bsd +		;; +	convex-c34) +		basic_machine=c34-convex +		os=-bsd +		;; +	convex-c38) +		basic_machine=c38-convex +		os=-bsd +		;; +	cray | j90) +		basic_machine=j90-cray +		os=-unicos +		;; +	craynv) +		basic_machine=craynv-cray +		os=-unicosmp +		;; +	cr16c) +		basic_machine=cr16c-unknown +		os=-elf +		;; +	crds | unos) +		basic_machine=m68k-crds +		;; +	crisv32 | crisv32-* | etraxfs*) +		basic_machine=crisv32-axis +		;; +	cris | cris-* | etrax*) +		basic_machine=cris-axis +		;; +	crx) +		basic_machine=crx-unknown +		os=-elf +		;; +	da30 | da30-*) +		basic_machine=m68k-da30 +		;; +	decstation | decstation-3100 | pmax | pmax-* | pmin | dec3100 | decstatn) +		basic_machine=mips-dec +		;; +	decsystem10* | dec10*) +		basic_machine=pdp10-dec +		os=-tops10 +		;; +	decsystem20* | dec20*) +		basic_machine=pdp10-dec +		os=-tops20 +		;; +	delta | 3300 | motorola-3300 | motorola-delta \ +	      | 3300-motorola | delta-motorola) +		basic_machine=m68k-motorola +		;; +	delta88) +		basic_machine=m88k-motorola +		os=-sysv3 +		;; +	djgpp) +		basic_machine=i586-pc +		os=-msdosdjgpp +		;; +	dpx20 | dpx20-*) +		basic_machine=rs6000-bull +		os=-bosx +		;; +	dpx2* | dpx2*-bull) +		basic_machine=m68k-bull +		os=-sysv3 +		;; +	ebmon29k) +		basic_machine=a29k-amd +		os=-ebmon +		;; +	elxsi) +		basic_machine=elxsi-elxsi +		os=-bsd +		;; +	encore | umax | mmax) +		basic_machine=ns32k-encore +		;; +	es1800 | OSE68k | ose68k | ose | OSE) +		basic_machine=m68k-ericsson +		os=-ose +		;; +	fx2800) +		basic_machine=i860-alliant +		;; +	genix) +		basic_machine=ns32k-ns +		;; +	gmicro) +		basic_machine=tron-gmicro +		os=-sysv +		;; +	go32) +		basic_machine=i386-pc +		os=-go32 +		;; +	h3050r* | hiux*) +		basic_machine=hppa1.1-hitachi +		os=-hiuxwe2 +		;; +	h8300hms) +		basic_machine=h8300-hitachi +		os=-hms +		;; +	h8300xray) +		basic_machine=h8300-hitachi +		os=-xray +		;; +	h8500hms) +		basic_machine=h8500-hitachi +		os=-hms +		;; +	harris) +		basic_machine=m88k-harris +		os=-sysv3 +		;; +	hp300-*) +		basic_machine=m68k-hp +		;; +	hp300bsd) +		basic_machine=m68k-hp +		os=-bsd +		;; +	hp300hpux) +		basic_machine=m68k-hp +		os=-hpux +		;; +	hp3k9[0-9][0-9] | hp9[0-9][0-9]) +		basic_machine=hppa1.0-hp +		;; +	hp9k2[0-9][0-9] | hp9k31[0-9]) +		basic_machine=m68000-hp +		;; +	hp9k3[2-9][0-9]) +		basic_machine=m68k-hp +		;; +	hp9k6[0-9][0-9] | hp6[0-9][0-9]) +		basic_machine=hppa1.0-hp +		;; +	hp9k7[0-79][0-9] | hp7[0-79][0-9]) +		basic_machine=hppa1.1-hp +		;; +	hp9k78[0-9] | hp78[0-9]) +		# FIXME: really hppa2.0-hp +		basic_machine=hppa1.1-hp +		;; +	hp9k8[67]1 | hp8[67]1 | hp9k80[24] | hp80[24] | hp9k8[78]9 | hp8[78]9 | hp9k893 | hp893) +		# FIXME: really hppa2.0-hp +		basic_machine=hppa1.1-hp +		;; +	hp9k8[0-9][13679] | hp8[0-9][13679]) +		basic_machine=hppa1.1-hp +		;; +	hp9k8[0-9][0-9] | hp8[0-9][0-9]) +		basic_machine=hppa1.0-hp +		;; +	hppa-next) +		os=-nextstep3 +		;; +	hppaosf) +		basic_machine=hppa1.1-hp +		os=-osf +		;; +	hppro) +		basic_machine=hppa1.1-hp +		os=-proelf +		;; +	i370-ibm* | ibm*) +		basic_machine=i370-ibm +		;; +# I'm not sure what "Sysv32" means.  Should this be sysv3.2? +	i*86v32) +		basic_machine=`echo $1 | sed -e 's/86.*/86-pc/'` +		os=-sysv32 +		;; +	i*86v4*) +		basic_machine=`echo $1 | sed -e 's/86.*/86-pc/'` +		os=-sysv4 +		;; +	i*86v) +		basic_machine=`echo $1 | sed -e 's/86.*/86-pc/'` +		os=-sysv +		;; +	i*86sol2) +		basic_machine=`echo $1 | sed -e 's/86.*/86-pc/'` +		os=-solaris2 +		;; +	i386mach) +		basic_machine=i386-mach +		os=-mach +		;; +	i386-vsta | vsta) +		basic_machine=i386-unknown +		os=-vsta +		;; +	iris | iris4d) +		basic_machine=mips-sgi +		case $os in +		    -irix*) +			;; +		    *) +			os=-irix4 +			;; +		esac +		;; +	isi68 | isi) +		basic_machine=m68k-isi +		os=-sysv +		;; +	m88k-omron*) +		basic_machine=m88k-omron +		;; +	magnum | m3230) +		basic_machine=mips-mips +		os=-sysv +		;; +	merlin) +		basic_machine=ns32k-utek +		os=-sysv +		;; +	mingw32) +		basic_machine=i386-pc +		os=-mingw32 +		;; +	mingw32ce) +		basic_machine=arm-unknown +		os=-mingw32ce +		;; +	miniframe) +		basic_machine=m68000-convergent +		;; +	*mint | -mint[0-9]* | *MiNT | *MiNT[0-9]*) +		basic_machine=m68k-atari +		os=-mint +		;; +	mips3*-*) +		basic_machine=`echo $basic_machine | sed -e 's/mips3/mips64/'` +		;; +	mips3*) +		basic_machine=`echo $basic_machine | sed -e 's/mips3/mips64/'`-unknown +		;; +	monitor) +		basic_machine=m68k-rom68k +		os=-coff +		;; +	morphos) +		basic_machine=powerpc-unknown +		os=-morphos +		;; +	msdos) +		basic_machine=i386-pc +		os=-msdos +		;; +	ms1-*) +		basic_machine=`echo $basic_machine | sed -e 's/ms1-/mt-/'` +		;; +	mvs) +		basic_machine=i370-ibm +		os=-mvs +		;; +	ncr3000) +		basic_machine=i486-ncr +		os=-sysv4 +		;; +	netbsd386) +		basic_machine=i386-unknown +		os=-netbsd +		;; +	netwinder) +		basic_machine=armv4l-rebel +		os=-linux +		;; +	news | news700 | news800 | news900) +		basic_machine=m68k-sony +		os=-newsos +		;; +	news1000) +		basic_machine=m68030-sony +		os=-newsos +		;; +	news-3600 | risc-news) +		basic_machine=mips-sony +		os=-newsos +		;; +	necv70) +		basic_machine=v70-nec +		os=-sysv +		;; +	next | m*-next ) +		basic_machine=m68k-next +		case $os in +		    -nextstep* ) +			;; +		    -ns2*) +		      os=-nextstep2 +			;; +		    *) +		      os=-nextstep3 +			;; +		esac +		;; +	nh3000) +		basic_machine=m68k-harris +		os=-cxux +		;; +	nh[45]000) +		basic_machine=m88k-harris +		os=-cxux +		;; +	nindy960) +		basic_machine=i960-intel +		os=-nindy +		;; +	mon960) +		basic_machine=i960-intel +		os=-mon960 +		;; +	nonstopux) +		basic_machine=mips-compaq +		os=-nonstopux +		;; +	np1) +		basic_machine=np1-gould +		;; +	nsr-tandem) +		basic_machine=nsr-tandem +		;; +	op50n-* | op60c-*) +		basic_machine=hppa1.1-oki +		os=-proelf +		;; +	openrisc | openrisc-*) +		basic_machine=or32-unknown +		;; +	os400) +		basic_machine=powerpc-ibm +		os=-os400 +		;; +	OSE68000 | ose68000) +		basic_machine=m68000-ericsson +		os=-ose +		;; +	os68k) +		basic_machine=m68k-none +		os=-os68k +		;; +	pa-hitachi) +		basic_machine=hppa1.1-hitachi +		os=-hiuxwe2 +		;; +	paragon) +		basic_machine=i860-intel +		os=-osf +		;; +	pbd) +		basic_machine=sparc-tti +		;; +	pbb) +		basic_machine=m68k-tti +		;; +	pc532 | pc532-*) +		basic_machine=ns32k-pc532 +		;; +	pc98) +		basic_machine=i386-pc +		;; +	pc98-*) +		basic_machine=i386-`echo $basic_machine | sed 's/^[^-]*-//'` +		;; +	pentium | p5 | k5 | k6 | nexgen | viac3) +		basic_machine=i586-pc +		;; +	pentiumpro | p6 | 6x86 | athlon | athlon_*) +		basic_machine=i686-pc +		;; +	pentiumii | pentium2 | pentiumiii | pentium3) +		basic_machine=i686-pc +		;; +	pentium4) +		basic_machine=i786-pc +		;; +	pentium-* | p5-* | k5-* | k6-* | nexgen-* | viac3-*) +		basic_machine=i586-`echo $basic_machine | sed 's/^[^-]*-//'` +		;; +	pentiumpro-* | p6-* | 6x86-* | athlon-*) +		basic_machine=i686-`echo $basic_machine | sed 's/^[^-]*-//'` +		;; +	pentiumii-* | pentium2-* | pentiumiii-* | pentium3-*) +		basic_machine=i686-`echo $basic_machine | sed 's/^[^-]*-//'` +		;; +	pentium4-*) +		basic_machine=i786-`echo $basic_machine | sed 's/^[^-]*-//'` +		;; +	pn) +		basic_machine=pn-gould +		;; +	power)	basic_machine=power-ibm +		;; +	ppc)	basic_machine=powerpc-unknown +		;; +	ppc-*)	basic_machine=powerpc-`echo $basic_machine | sed 's/^[^-]*-//'` +		;; +	ppcle | powerpclittle | ppc-le | powerpc-little) +		basic_machine=powerpcle-unknown +		;; +	ppcle-* | powerpclittle-*) +		basic_machine=powerpcle-`echo $basic_machine | sed 's/^[^-]*-//'` +		;; +	ppc64)	basic_machine=powerpc64-unknown +		;; +	ppc64-*) basic_machine=powerpc64-`echo $basic_machine | sed 's/^[^-]*-//'` +		;; +	ppc64le | powerpc64little | ppc64-le | powerpc64-little) +		basic_machine=powerpc64le-unknown +		;; +	ppc64le-* | powerpc64little-*) +		basic_machine=powerpc64le-`echo $basic_machine | sed 's/^[^-]*-//'` +		;; +	ps2) +		basic_machine=i386-ibm +		;; +	pw32) +		basic_machine=i586-unknown +		os=-pw32 +		;; +	rdos) +		basic_machine=i386-pc +		os=-rdos +		;; +	rom68k) +		basic_machine=m68k-rom68k +		os=-coff +		;; +	rm[46]00) +		basic_machine=mips-siemens +		;; +	rtpc | rtpc-*) +		basic_machine=romp-ibm +		;; +	s390 | s390-*) +		basic_machine=s390-ibm +		;; +	s390x | s390x-*) +		basic_machine=s390x-ibm +		;; +	sa29200) +		basic_machine=a29k-amd +		os=-udi +		;; +	sb1) +		basic_machine=mipsisa64sb1-unknown +		;; +	sb1el) +		basic_machine=mipsisa64sb1el-unknown +		;; +	sde) +		basic_machine=mipsisa32-sde +		os=-elf +		;; +	sei) +		basic_machine=mips-sei +		os=-seiux +		;; +	sequent) +		basic_machine=i386-sequent +		;; +	sh) +		basic_machine=sh-hitachi +		os=-hms +		;; +	sh5el) +		basic_machine=sh5le-unknown +		;; +	sh64) +		basic_machine=sh64-unknown +		;; +	sparclite-wrs | simso-wrs) +		basic_machine=sparclite-wrs +		os=-vxworks +		;; +	sps7) +		basic_machine=m68k-bull +		os=-sysv2 +		;; +	spur) +		basic_machine=spur-unknown +		;; +	st2000) +		basic_machine=m68k-tandem +		;; +	stratus) +		basic_machine=i860-stratus +		os=-sysv4 +		;; +	sun2) +		basic_machine=m68000-sun +		;; +	sun2os3) +		basic_machine=m68000-sun +		os=-sunos3 +		;; +	sun2os4) +		basic_machine=m68000-sun +		os=-sunos4 +		;; +	sun3os3) +		basic_machine=m68k-sun +		os=-sunos3 +		;; +	sun3os4) +		basic_machine=m68k-sun +		os=-sunos4 +		;; +	sun4os3) +		basic_machine=sparc-sun +		os=-sunos3 +		;; +	sun4os4) +		basic_machine=sparc-sun +		os=-sunos4 +		;; +	sun4sol2) +		basic_machine=sparc-sun +		os=-solaris2 +		;; +	sun3 | sun3-*) +		basic_machine=m68k-sun +		;; +	sun4) +		basic_machine=sparc-sun +		;; +	sun386 | sun386i | roadrunner) +		basic_machine=i386-sun +		;; +	sv1) +		basic_machine=sv1-cray +		os=-unicos +		;; +	symmetry) +		basic_machine=i386-sequent +		os=-dynix +		;; +	t3e) +		basic_machine=alphaev5-cray +		os=-unicos +		;; +	t90) +		basic_machine=t90-cray +		os=-unicos +		;; +	tic54x | c54x*) +		basic_machine=tic54x-unknown +		os=-coff +		;; +	tic55x | c55x*) +		basic_machine=tic55x-unknown +		os=-coff +		;; +	tic6x | c6x*) +		basic_machine=tic6x-unknown +		os=-coff +		;; +	tx39) +		basic_machine=mipstx39-unknown +		;; +	tx39el) +		basic_machine=mipstx39el-unknown +		;; +	tile*) +		basic_machine=tile-tilera +		os=-linux-gnu +		;; +	toad1) +		basic_machine=pdp10-xkl +		os=-tops20 +		;; +	tower | tower-32) +		basic_machine=m68k-ncr +		;; +	tpf) +		basic_machine=s390x-ibm +		os=-tpf +		;; +	udi29k) +		basic_machine=a29k-amd +		os=-udi +		;; +	ultra3) +		basic_machine=a29k-nyu +		os=-sym1 +		;; +	v810 | necv810) +		basic_machine=v810-nec +		os=-none +		;; +	vaxv) +		basic_machine=vax-dec +		os=-sysv +		;; +	vms) +		basic_machine=vax-dec +		os=-vms +		;; +	vpp*|vx|vx-*) +		basic_machine=f301-fujitsu +		;; +	vxworks960) +		basic_machine=i960-wrs +		os=-vxworks +		;; +	vxworks68) +		basic_machine=m68k-wrs +		os=-vxworks +		;; +	vxworks29k) +		basic_machine=a29k-wrs +		os=-vxworks +		;; +	w65*) +		basic_machine=w65-wdc +		os=-none +		;; +	w89k-*) +		basic_machine=hppa1.1-winbond +		os=-proelf +		;; +	xbox) +		basic_machine=i686-pc +		os=-mingw32 +		;; +	xps | xps100) +		basic_machine=xps100-honeywell +		;; +	ymp) +		basic_machine=ymp-cray +		os=-unicos +		;; +	z8k-*-coff) +		basic_machine=z8k-unknown +		os=-sim +		;; +	none) +		basic_machine=none-none +		os=-none +		;; + +# Here we handle the default manufacturer of certain CPU types.  It is in +# some cases the only manufacturer, in others, it is the most popular. +	w89k) +		basic_machine=hppa1.1-winbond +		;; +	op50n) +		basic_machine=hppa1.1-oki +		;; +	op60c) +		basic_machine=hppa1.1-oki +		;; +	romp) +		basic_machine=romp-ibm +		;; +	mmix) +		basic_machine=mmix-knuth +		;; +	rs6000) +		basic_machine=rs6000-ibm +		;; +	vax) +		basic_machine=vax-dec +		;; +	pdp10) +		# there are many clones, so DEC is not a safe bet +		basic_machine=pdp10-unknown +		;; +	pdp11) +		basic_machine=pdp11-dec +		;; +	we32k) +		basic_machine=we32k-att +		;; +	sh[1234] | sh[24]a | sh[34]eb | sh[1234]le | sh[23]ele) +		basic_machine=sh-unknown +		;; +	sparc | sparcv8 | sparcv9 | sparcv9b | sparcv9v) +		basic_machine=sparc-sun +		;; +	cydra) +		basic_machine=cydra-cydrome +		;; +	orion) +		basic_machine=orion-highlevel +		;; +	orion105) +		basic_machine=clipper-highlevel +		;; +	mac | mpw | mac-mpw) +		basic_machine=m68k-apple +		;; +	pmac | pmac-mpw) +		basic_machine=powerpc-apple +		;; +	*-unknown) +		# Make sure to match an already-canonicalized machine name. +		;; +	*) +		echo Invalid configuration \`$1\': machine \`$basic_machine\' not recognized 1>&2 +		exit 1 +		;; +esac + +# Here we canonicalize certain aliases for manufacturers. +case $basic_machine in +	*-digital*) +		basic_machine=`echo $basic_machine | sed 's/digital.*/dec/'` +		;; +	*-commodore*) +		basic_machine=`echo $basic_machine | sed 's/commodore.*/cbm/'` +		;; +	*) +		;; +esac + +# Decode manufacturer-specific aliases for certain operating systems. + +if [ x"$os" != x"" ] +then +case $os in +        # First match some system type aliases +        # that might get confused with valid system types. +	# -solaris* is a basic system type, with this one exception. +	-solaris1 | -solaris1.*) +		os=`echo $os | sed -e 's|solaris1|sunos4|'` +		;; +	-solaris) +		os=-solaris2 +		;; +	-svr4*) +		os=-sysv4 +		;; +	-unixware*) +		os=-sysv4.2uw +		;; +	-gnu/linux*) +		os=`echo $os | sed -e 's|gnu/linux|linux-gnu|'` +		;; +	# First accept the basic system types. +	# The portable systems comes first. +	# Each alternative MUST END IN A *, to match a version number. +	# -sysv* is not here because it comes later, after sysvr4. +	-gnu* | -bsd* | -mach* | -minix* | -genix* | -ultrix* | -irix* \ +	      | -*vms* | -sco* | -esix* | -isc* | -aix* | -sunos | -sunos[34]*\ +	      | -hpux* | -unos* | -osf* | -luna* | -dgux* | -solaris* | -sym* \ +	      | -amigaos* | -amigados* | -msdos* | -newsos* | -unicos* | -aof* \ +	      | -aos* \ +	      | -nindy* | -vxsim* | -vxworks* | -ebmon* | -hms* | -mvs* \ +	      | -clix* | -riscos* | -uniplus* | -iris* | -rtu* | -xenix* \ +	      | -hiux* | -386bsd* | -knetbsd* | -mirbsd* | -netbsd* \ +	      | -openbsd* | -solidbsd* \ +	      | -ekkobsd* | -kfreebsd* | -freebsd* | -riscix* | -lynxos* \ +	      | -bosx* | -nextstep* | -cxux* | -aout* | -elf* | -oabi* \ +	      | -ptx* | -coff* | -ecoff* | -winnt* | -domain* | -vsta* \ +	      | -udi* | -eabi* | -lites* | -ieee* | -go32* | -aux* \ +	      | -chorusos* | -chorusrdb* \ +	      | -cygwin* | -pe* | -psos* | -moss* | -proelf* | -rtems* \ +	      | -mingw32* | -linux-gnu* | -linux-newlib* | -linux-uclibc* \ +	      | -uxpv* | -beos* | -mpeix* | -udk* \ +	      | -interix* | -uwin* | -mks* | -rhapsody* | -darwin* | -opened* \ +	      | -openstep* | -oskit* | -conix* | -pw32* | -nonstopux* \ +	      | -storm-chaos* | -tops10* | -tenex* | -tops20* | -its* \ +	      | -os2* | -vos* | -palmos* | -uclinux* | -nucleus* \ +	      | -morphos* | -superux* | -rtmk* | -rtmk-nova* | -windiss* \ +	      | -powermax* | -dnix* | -nx6 | -nx7 | -sei* | -dragonfly* \ +	      | -skyos* | -haiku* | -rdos* | -toppers* | -drops*) +	# Remember, each alternative MUST END IN *, to match a version number. +		;; +	-qnx*) +		case $basic_machine in +		    x86-* | i*86-*) +			;; +		    *) +			os=-nto$os +			;; +		esac +		;; +	-nto-qnx*) +		;; +	-nto*) +		os=`echo $os | sed -e 's|nto|nto-qnx|'` +		;; +	-sim | -es1800* | -hms* | -xray | -os68k* | -none* | -v88r* \ +	      | -windows* | -osx | -abug | -netware* | -os9* | -beos* | -haiku* \ +	      | -macos* | -mpw* | -magic* | -mmixware* | -mon960* | -lnews*) +		;; +	-mac*) +		os=`echo $os | sed -e 's|mac|macos|'` +		;; +	-linux-dietlibc) +		os=-linux-dietlibc +		;; +	-linux*) +		os=`echo $os | sed -e 's|linux|linux-gnu|'` +		;; +	-sunos5*) +		os=`echo $os | sed -e 's|sunos5|solaris2|'` +		;; +	-sunos6*) +		os=`echo $os | sed -e 's|sunos6|solaris3|'` +		;; +	-opened*) +		os=-openedition +		;; +        -os400*) +		os=-os400 +		;; +	-wince*) +		os=-wince +		;; +	-osfrose*) +		os=-osfrose +		;; +	-osf*) +		os=-osf +		;; +	-utek*) +		os=-bsd +		;; +	-dynix*) +		os=-bsd +		;; +	-acis*) +		os=-aos +		;; +	-atheos*) +		os=-atheos +		;; +	-syllable*) +		os=-syllable +		;; +	-386bsd) +		os=-bsd +		;; +	-ctix* | -uts*) +		os=-sysv +		;; +	-nova*) +		os=-rtmk-nova +		;; +	-ns2 ) +		os=-nextstep2 +		;; +	-nsk*) +		os=-nsk +		;; +	# Preserve the version number of sinix5. +	-sinix5.*) +		os=`echo $os | sed -e 's|sinix|sysv|'` +		;; +	-sinix*) +		os=-sysv4 +		;; +        -tpf*) +		os=-tpf +		;; +	-triton*) +		os=-sysv3 +		;; +	-oss*) +		os=-sysv3 +		;; +	-svr4) +		os=-sysv4 +		;; +	-svr3) +		os=-sysv3 +		;; +	-sysvr4) +		os=-sysv4 +		;; +	# This must come after -sysvr4. +	-sysv*) +		;; +	-ose*) +		os=-ose +		;; +	-es1800*) +		os=-ose +		;; +	-xenix) +		os=-xenix +		;; +	-*mint | -mint[0-9]* | -*MiNT | -MiNT[0-9]*) +		os=-mint +		;; +	-aros*) +		os=-aros +		;; +	-kaos*) +		os=-kaos +		;; +	-zvmoe) +		os=-zvmoe +		;; +	-none) +		;; +	*) +		# Get rid of the `-' at the beginning of $os. +		os=`echo $os | sed 's/[^-]*-//'` +		echo Invalid configuration \`$1\': system \`$os\' not recognized 1>&2 +		exit 1 +		;; +esac +else + +# Here we handle the default operating systems that come with various machines. +# The value should be what the vendor currently ships out the door with their +# machine or put another way, the most popular os provided with the machine. + +# Note that if you're going to try to match "-MANUFACTURER" here (say, +# "-sun"), then you have to tell the case statement up towards the top +# that MANUFACTURER isn't an operating system.  Otherwise, code above +# will signal an error saying that MANUFACTURER isn't an operating +# system, and we'll never get to this point. + +case $basic_machine in +        score-*) +		os=-elf +		;; +        spu-*) +		os=-elf +		;; +	*-acorn) +		os=-riscix1.2 +		;; +	arm*-rebel) +		os=-linux +		;; +	arm*-semi) +		os=-aout +		;; +        c4x-* | tic4x-*) +        	os=-coff +		;; +	# This must come before the *-dec entry. +	pdp10-*) +		os=-tops20 +		;; +	pdp11-*) +		os=-none +		;; +	*-dec | vax-*) +		os=-ultrix4.2 +		;; +	m68*-apollo) +		os=-domain +		;; +	i386-sun) +		os=-sunos4.0.2 +		;; +	m68000-sun) +		os=-sunos3 +		# This also exists in the configure program, but was not the +		# default. +		# os=-sunos4 +		;; +	m68*-cisco) +		os=-aout +		;; +        mep-*) +		os=-elf +		;; +	mips*-cisco) +		os=-elf +		;; +	mips*-*) +		os=-elf +		;; +	or32-*) +		os=-coff +		;; +	*-tti)	# must be before sparc entry or we get the wrong os. +		os=-sysv3 +		;; +	sparc-* | *-sun) +		os=-sunos4.1.1 +		;; +	*-be) +		os=-beos +		;; +	*-haiku) +		os=-haiku +		;; +	*-ibm) +		os=-aix +		;; +    	*-knuth) +		os=-mmixware +		;; +	*-wec) +		os=-proelf +		;; +	*-winbond) +		os=-proelf +		;; +	*-oki) +		os=-proelf +		;; +	*-hp) +		os=-hpux +		;; +	*-hitachi) +		os=-hiux +		;; +	i860-* | *-att | *-ncr | *-altos | *-motorola | *-convergent) +		os=-sysv +		;; +	*-cbm) +		os=-amigaos +		;; +	*-dg) +		os=-dgux +		;; +	*-dolphin) +		os=-sysv3 +		;; +	m68k-ccur) +		os=-rtu +		;; +	m88k-omron*) +		os=-luna +		;; +	*-next ) +		os=-nextstep +		;; +	*-sequent) +		os=-ptx +		;; +	*-crds) +		os=-unos +		;; +	*-ns) +		os=-genix +		;; +	i370-*) +		os=-mvs +		;; +	*-next) +		os=-nextstep3 +		;; +	*-gould) +		os=-sysv +		;; +	*-highlevel) +		os=-bsd +		;; +	*-encore) +		os=-bsd +		;; +	*-sgi) +		os=-irix +		;; +	*-siemens) +		os=-sysv4 +		;; +	*-masscomp) +		os=-rtu +		;; +	f30[01]-fujitsu | f700-fujitsu) +		os=-uxpv +		;; +	*-rom68k) +		os=-coff +		;; +	*-*bug) +		os=-coff +		;; +	*-apple) +		os=-macos +		;; +	*-atari*) +		os=-mint +		;; +	*) +		os=-none +		;; +esac +fi + +# Here we handle the case where we know the os, and the CPU type, but not the +# manufacturer.  We pick the logical manufacturer. +vendor=unknown +case $basic_machine in +	*-unknown) +		case $os in +			-riscix*) +				vendor=acorn +				;; +			-sunos*) +				vendor=sun +				;; +			-aix*) +				vendor=ibm +				;; +			-beos*) +				vendor=be +				;; +			-hpux*) +				vendor=hp +				;; +			-mpeix*) +				vendor=hp +				;; +			-hiux*) +				vendor=hitachi +				;; +			-unos*) +				vendor=crds +				;; +			-dgux*) +				vendor=dg +				;; +			-luna*) +				vendor=omron +				;; +			-genix*) +				vendor=ns +				;; +			-mvs* | -opened*) +				vendor=ibm +				;; +			-os400*) +				vendor=ibm +				;; +			-ptx*) +				vendor=sequent +				;; +			-tpf*) +				vendor=ibm +				;; +			-vxsim* | -vxworks* | -windiss*) +				vendor=wrs +				;; +			-aux*) +				vendor=apple +				;; +			-hms*) +				vendor=hitachi +				;; +			-mpw* | -macos*) +				vendor=apple +				;; +			-*mint | -mint[0-9]* | -*MiNT | -MiNT[0-9]*) +				vendor=atari +				;; +			-vos*) +				vendor=stratus +				;; +		esac +		basic_machine=`echo $basic_machine | sed "s/unknown/$vendor/"` +		;; +esac + +echo $basic_machine$os +exit + +# Local variables: +# eval: (add-hook 'write-file-hooks 'time-stamp) +# time-stamp-start: "timestamp='" +# time-stamp-format: "%:y-%02m-%02d" +# time-stamp-end: "'" +# End: diff --git a/lib/test_server/src/configure.in b/lib/test_server/src/configure.in index 097853bcfc..785bab395c 100644 --- a/lib/test_server/src/configure.in +++ b/lib/test_server/src/configure.in @@ -2,7 +2,7 @@ dnl Process this file with autoconf to produce a configure script for Erlang.  dnl  dnl %CopyrightBegin%  dnl  -dnl Copyright Ericsson AB 1997-2011. All Rights Reserved. +dnl Copyright Ericsson AB 1997-2012. All Rights Reserved.  dnl   dnl The contents of this file are subject to the Erlang Public License,  dnl Version 1.1, (the "License"); you may not use this file except in @@ -108,7 +108,7 @@ AC_CHECK_HEADER(poll.h, AC_DEFINE(HAVE_POLL_H))  # for the system.  AC_MSG_CHECKING([system version (for dynamic loading)]) -system=`uname -s`-`uname -r` +system=`./config.sub $host`  AC_MSG_RESULT($system)  # Step 2: check for existence of -ldl library.  This is needed because @@ -119,10 +119,9 @@ AC_CHECK_LIB(dl, dlopen, have_dl=yes, have_dl=no)  # Step 3: set configuration options based on system name and version.  SHLIB_LDLIBS= -  fullSrcDir=`cd $srcdir; pwd`  case $system in -    Linux*) +    *-linux-*)  	SHLIB_CFLAGS="-fPIC"  	SHLIB_SUFFIX=".so"  	if test "$have_dl" = yes; then @@ -136,7 +135,7 @@ case $system in  	fi  	SHLIB_EXTRACT_ALL=""  	;; -    NetBSD-*|FreeBSD-*|OpenBSD-*|DragonFly*) +    *-netbsd*|*-freebsd*|*-openbsd*|*-dragonfly*)  	# Not available on all versions:  check for include file.  	AC_CHECK_HEADER(dlfcn.h, [  	    SHLIB_CFLAGS="-fpic" @@ -153,28 +152,21 @@ case $system in  	])  	SHLIB_EXTRACT_ALL=""  	;; -    SunOS-4*) -	SHLIB_CFLAGS="-PIC" -	SHLIB_LD="ld" -	SHLIB_LDFLAGS="$LDFLAGS" -	SHLIB_SUFFIX=".so" -	SHLIB_EXTRACT_ALL="" -	;; -    SunOS-5*|UNIX_SV-4.2*) +    *-solaris2*|*-sysv4*)  	SHLIB_CFLAGS="-KPIC"  	SHLIB_LD="/usr/ccs/bin/ld"  	SHLIB_LDFLAGS="$LDFLAGS -G -z text"  	SHLIB_SUFFIX=".so"  	SHLIB_EXTRACT_ALL="-z allextract"  	;; -    Darwin*) +    *darwin*)  	SHLIB_CFLAGS="-fno-common"  	SHLIB_LD="cc"  	SHLIB_LDFLAGS="$LDFLAGS -bundle -flat_namespace -undefined suppress"  	SHLIB_SUFFIX=".so"  	SHLIB_EXTRACT_ALL=""  	;; -    OSF1*) +    *osf1*)  	SHLIB_CFLAGS="-fPIC"  	SHLIB_LD="ld"  	SHLIB_LDFLAGS="$LDFLAGS -shared" @@ -206,19 +198,19 @@ esac  if test "$CC" = "gcc" -o `$CC -v 2>&1 | grep -c gcc` != "0" ; then  	case $system in -	    AIX-*) +	    *-aix)  		;; -	    BSD/OS*) +	    *-bsd*)  		;; -	    IRIX*) +	    *-irix)  		;; -	    NetBSD-*|FreeBSD-*|OpenBSD-*) +	    *-netbsd|*-freebsd|*-openbsd)  		;; -	    RISCos-*) +	    *-riscos)  		;; -	    ULTRIX-4.*) +	    *ultrix4.*)  		;; -	    Darwin*) +	    *darwin*)  		;;  	    *)  		SHLIB_CFLAGS="-fPIC" diff --git a/lib/test_server/src/erl2html2.erl b/lib/test_server/src/erl2html2.erl index c94d4627f9..6891e87e48 100644 --- a/lib/test_server/src/erl2html2.erl +++ b/lib/test_server/src/erl2html2.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %%  -%% Copyright Ericsson AB 1997-2009. All Rights Reserved. +%% Copyright Ericsson AB 1997-2011. All Rights Reserved.  %%   %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -32,32 +32,34 @@  %--------------------------------------------------------------------  -module(erl2html2). --export([convert/2]). +-export([convert/2, convert/3]).  convert([], _Dest) ->   % Fake clause.      ok;  convert(File, Dest) -> +    %% The generated code uses the BGCOLOR attribute in the +    %% BODY tag, which wasn't valid until HTML 3.2.  Also, +    %% good HTML should either override all colour attributes +    %% or none of them -- *never* just a few. +    %% +    %% FIXME: The colours should *really* be set with +    %% stylesheets... +    Header = ["<!DOCTYPE HTML PUBLIC " +	      "\"-//W3C//DTD HTML 3.2 Final//EN\">\n" +	      "<!-- autogenerated by '"++atom_to_list(?MODULE)++"'. -->\n" +	      "<html>\n" +	      "<head><title>", File, "</title></head>\n\n" +	      "<body bgcolor=\"white\" text=\"black\"" +	      " link=\"blue\" vlink=\"purple\" alink=\"red\">\n"], +    convert(File, Dest, Header). +     +convert(File, Dest, Header) ->      case file:read_file(File) of  	{ok, Bin} ->  	    Code=binary_to_list(Bin),  	    statistics(runtime), -	    %% The generated code uses the BGCOLOR attribute in the -	    %% BODY tag, which wasn't valid until HTML 3.2.  Also, -	    %% good HTML should either override all colour attributes -	    %% or none of them -- *never* just a few. -	    %% -	    %% FIXME: The colours should *really* be set with -	    %% stylesheets... -	    Html0  -		= ["<!DOCTYPE HTML PUBLIC " -		   "\"-//W3C//DTD HTML 3.2 Final//EN\">\n" -		   "<!-- autogenerated by '"++atom_to_list(?MODULE)++"'. -->\n" -		   "<html>\n" -		   "<head><title>", File, "</title></head>\n\n" -		   "<body bgcolor=\"white\" text=\"black\"" -		   " link=\"blue\" vlink=\"purple\" alink=\"red\">\n"],  	    {Html1, Lines} = root(Code, [], 1), -	    Html = [Html0,  +	    Html = [Header,  		    "<pre>\n", Html1, "</pre>\n",   		    footer(Lines),"</body>\n</html>\n"],  	    file:write_file(Dest, Html); @@ -173,10 +175,11 @@ linenum(Line) ->  	end,       [A,Pred,integer_to_list(Line),":"]. -footer(Lines) -> -    {_, Time} = statistics(runtime), -%    io:format("Converted ~p lines in ~.2f Seconds.~n", -%	      [Lines, Time/1000]), -    S = "<i>The transformation of this file (~p lines) took ~.2f seconds</i>", -    F = lists:flatten(io_lib:format(S, [Lines, Time/1000])), -    ["<hr size=1>",F,"<br>\n"]. +footer(_Lines) -> +    "". +%%    {_, Time} = statistics(runtime), +%%    io:format("Converted ~p lines in ~.2f Seconds.~n", +%%	      [Lines, Time/1000]), +%%    S = "<i>The transformation of this file (~p lines) took ~.2f seconds</i>", +%%    F = lists:flatten(io_lib:format(S, [Lines, Time/1000])), +%%    ["<hr size=1>",F,"<br>\n"]. diff --git a/lib/test_server/src/install-sh b/lib/test_server/src/install-sh index a859cade7f..a5897de6ea 120000..100755 --- a/lib/test_server/src/install-sh +++ b/lib/test_server/src/install-sh @@ -1 +1,519 @@ -../../../erts/autoconf/install-sh
\ No newline at end of file +#!/bin/sh +# install - install a program, script, or datafile + +scriptversion=2006-12-25.00 + +# This originates from X11R5 (mit/util/scripts/install.sh), which was +# later released in X11R6 (xc/config/util/install.sh) with the +# following copyright and license. +# +# Copyright (C) 1994 X Consortium +# +# Permission is hereby granted, free of charge, to any person obtaining a copy +# of this software and associated documentation files (the "Software"), to +# deal in the Software without restriction, including without limitation the +# rights to use, copy, modify, merge, publish, distribute, sublicense, and/or +# sell copies of the Software, and to permit persons to whom the Software is +# furnished to do so, subject to the following conditions: +# +# The above copyright notice and this permission notice shall be included in +# all copies or substantial portions of the Software. +# +# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL THE +# X CONSORTIUM BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN +# AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNEC- +# TION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. +# +# Except as contained in this notice, the name of the X Consortium shall not +# be used in advertising or otherwise to promote the sale, use or other deal- +# ings in this Software without prior written authorization from the X Consor- +# tium. +# +# +# FSF changes to this file are in the public domain. +# +# Calling this script install-sh is preferred over install.sh, to prevent +# `make' implicit rules from creating a file called install from it +# when there is no Makefile. +# +# This script is compatible with the BSD install script, but was written +# from scratch. + +nl=' +' +IFS=" ""	$nl" + +# set DOITPROG to echo to test this script + +# Don't use :- since 4.3BSD and earlier shells don't like it. +doit=${DOITPROG-} +if test -z "$doit"; then +  doit_exec=exec +else +  doit_exec=$doit +fi + +# Put in absolute file names if you don't have them in your path; +# or use environment vars. + +chgrpprog=${CHGRPPROG-chgrp} +chmodprog=${CHMODPROG-chmod} +chownprog=${CHOWNPROG-chown} +cmpprog=${CMPPROG-cmp} +cpprog=${CPPROG-cp} +mkdirprog=${MKDIRPROG-mkdir} +mvprog=${MVPROG-mv} +rmprog=${RMPROG-rm} +stripprog=${STRIPPROG-strip} + +posix_glob='?' +initialize_posix_glob=' +  test "$posix_glob" != "?" || { +    if (set -f) 2>/dev/null; then +      posix_glob= +    else +      posix_glob=: +    fi +  } +' + +posix_mkdir= + +# Desired mode of installed file. +mode=0755 + +chgrpcmd= +chmodcmd=$chmodprog +chowncmd= +mvcmd=$mvprog +rmcmd="$rmprog -f" +stripcmd= + +src= +dst= +dir_arg= +dst_arg= + +copy_on_change=false +no_target_directory= + +usage="\ +Usage: $0 [OPTION]... [-T] SRCFILE DSTFILE +   or: $0 [OPTION]... SRCFILES... DIRECTORY +   or: $0 [OPTION]... -t DIRECTORY SRCFILES... +   or: $0 [OPTION]... -d DIRECTORIES... + +In the 1st form, copy SRCFILE to DSTFILE. +In the 2nd and 3rd, copy all SRCFILES to DIRECTORY. +In the 4th, create DIRECTORIES. + +Options: +     --help     display this help and exit. +     --version  display version info and exit. + +  -c            (ignored) +  -C            install only if different (preserve the last data modification time) +  -d            create directories instead of installing files. +  -g GROUP      $chgrpprog installed files to GROUP. +  -m MODE       $chmodprog installed files to MODE. +  -o USER       $chownprog installed files to USER. +  -s            $stripprog installed files. +  -t DIRECTORY  install into DIRECTORY. +  -T            report an error if DSTFILE is a directory. + +Environment variables override the default commands: +  CHGRPPROG CHMODPROG CHOWNPROG CMPPROG CPPROG MKDIRPROG MVPROG +  RMPROG STRIPPROG +" + +while test $# -ne 0; do +  case $1 in +    -c) ;; + +    -C) copy_on_change=true;; + +    -d) dir_arg=true;; + +    -g) chgrpcmd="$chgrpprog $2" +	shift;; + +    --help) echo "$usage"; exit $?;; + +    -m) mode=$2 +	case $mode in +	  *' '* | *'	'* | *' +'*	  | *'*'* | *'?'* | *'['*) +	    echo "$0: invalid mode: $mode" >&2 +	    exit 1;; +	esac +	shift;; + +    -o) chowncmd="$chownprog $2" +	shift;; + +    -s) stripcmd=$stripprog;; + +    -t) dst_arg=$2 +	shift;; + +    -T) no_target_directory=true;; + +    --version) echo "$0 $scriptversion"; exit $?;; + +    --)	shift +	break;; + +    -*)	echo "$0: invalid option: $1" >&2 +	exit 1;; + +    *)  break;; +  esac +  shift +done + +if test $# -ne 0 && test -z "$dir_arg$dst_arg"; then +  # When -d is used, all remaining arguments are directories to create. +  # When -t is used, the destination is already specified. +  # Otherwise, the last argument is the destination.  Remove it from $@. +  for arg +  do +    if test -n "$dst_arg"; then +      # $@ is not empty: it contains at least $arg. +      set fnord "$@" "$dst_arg" +      shift # fnord +    fi +    shift # arg +    dst_arg=$arg +  done +fi + +if test $# -eq 0; then +  if test -z "$dir_arg"; then +    echo "$0: no input file specified." >&2 +    exit 1 +  fi +  # It's OK to call `install-sh -d' without argument. +  # This can happen when creating conditional directories. +  exit 0 +fi + +if test -z "$dir_arg"; then +  trap '(exit $?); exit' 1 2 13 15 + +  # Set umask so as not to create temps with too-generous modes. +  # However, 'strip' requires both read and write access to temps. +  case $mode in +    # Optimize common cases. +    *644) cp_umask=133;; +    *755) cp_umask=22;; + +    *[0-7]) +      if test -z "$stripcmd"; then +	u_plus_rw= +      else +	u_plus_rw='% 200' +      fi +      cp_umask=`expr '(' 777 - $mode % 1000 ')' $u_plus_rw`;; +    *) +      if test -z "$stripcmd"; then +	u_plus_rw= +      else +	u_plus_rw=,u+rw +      fi +      cp_umask=$mode$u_plus_rw;; +  esac +fi + +for src +do +  # Protect names starting with `-'. +  case $src in +    -*) src=./$src;; +  esac + +  if test -n "$dir_arg"; then +    dst=$src +    dstdir=$dst +    test -d "$dstdir" +    dstdir_status=$? +  else + +    # Waiting for this to be detected by the "$cpprog $src $dsttmp" command +    # might cause directories to be created, which would be especially bad +    # if $src (and thus $dsttmp) contains '*'. +    if test ! -f "$src" && test ! -d "$src"; then +      echo "$0: $src does not exist." >&2 +      exit 1 +    fi + +    if test -z "$dst_arg"; then +      echo "$0: no destination specified." >&2 +      exit 1 +    fi + +    dst=$dst_arg +    # Protect names starting with `-'. +    case $dst in +      -*) dst=./$dst;; +    esac + +    # If destination is a directory, append the input filename; won't work +    # if double slashes aren't ignored. +    if test -d "$dst"; then +      if test -n "$no_target_directory"; then +	echo "$0: $dst_arg: Is a directory" >&2 +	exit 1 +      fi +      dstdir=$dst +      dst=$dstdir/`basename "$src"` +      dstdir_status=0 +    else +      # Prefer dirname, but fall back on a substitute if dirname fails. +      dstdir=` +	(dirname "$dst") 2>/dev/null || +	expr X"$dst" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ +	     X"$dst" : 'X\(//\)[^/]' \| \ +	     X"$dst" : 'X\(//\)$' \| \ +	     X"$dst" : 'X\(/\)' \| . 2>/dev/null || +	echo X"$dst" | +	    sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ +		   s//\1/ +		   q +		 } +		 /^X\(\/\/\)[^/].*/{ +		   s//\1/ +		   q +		 } +		 /^X\(\/\/\)$/{ +		   s//\1/ +		   q +		 } +		 /^X\(\/\).*/{ +		   s//\1/ +		   q +		 } +		 s/.*/./; q' +      ` + +      test -d "$dstdir" +      dstdir_status=$? +    fi +  fi + +  obsolete_mkdir_used=false + +  if test $dstdir_status != 0; then +    case $posix_mkdir in +      '') +	# Create intermediate dirs using mode 755 as modified by the umask. +	# This is like FreeBSD 'install' as of 1997-10-28. +	umask=`umask` +	case $stripcmd.$umask in +	  # Optimize common cases. +	  *[2367][2367]) mkdir_umask=$umask;; +	  .*0[02][02] | .[02][02] | .[02]) mkdir_umask=22;; + +	  *[0-7]) +	    mkdir_umask=`expr $umask + 22 \ +	      - $umask % 100 % 40 + $umask % 20 \ +	      - $umask % 10 % 4 + $umask % 2 +	    `;; +	  *) mkdir_umask=$umask,go-w;; +	esac + +	# With -d, create the new directory with the user-specified mode. +	# Otherwise, rely on $mkdir_umask. +	if test -n "$dir_arg"; then +	  mkdir_mode=-m$mode +	else +	  mkdir_mode= +	fi + +	posix_mkdir=false +	case $umask in +	  *[123567][0-7][0-7]) +	    # POSIX mkdir -p sets u+wx bits regardless of umask, which +	    # is incompatible with FreeBSD 'install' when (umask & 300) != 0. +	    ;; +	  *) +	    tmpdir=${TMPDIR-/tmp}/ins$RANDOM-$$ +	    trap 'ret=$?; rmdir "$tmpdir/d" "$tmpdir" 2>/dev/null; exit $ret' 0 + +	    if (umask $mkdir_umask && +		exec $mkdirprog $mkdir_mode -p -- "$tmpdir/d") >/dev/null 2>&1 +	    then +	      if test -z "$dir_arg" || { +		   # Check for POSIX incompatibilities with -m. +		   # HP-UX 11.23 and IRIX 6.5 mkdir -m -p sets group- or +		   # other-writeable bit of parent directory when it shouldn't. +		   # FreeBSD 6.1 mkdir -m -p sets mode of existing directory. +		   ls_ld_tmpdir=`ls -ld "$tmpdir"` +		   case $ls_ld_tmpdir in +		     d????-?r-*) different_mode=700;; +		     d????-?--*) different_mode=755;; +		     *) false;; +		   esac && +		   $mkdirprog -m$different_mode -p -- "$tmpdir" && { +		     ls_ld_tmpdir_1=`ls -ld "$tmpdir"` +		     test "$ls_ld_tmpdir" = "$ls_ld_tmpdir_1" +		   } +		 } +	      then posix_mkdir=: +	      fi +	      rmdir "$tmpdir/d" "$tmpdir" +	    else +	      # Remove any dirs left behind by ancient mkdir implementations. +	      rmdir ./$mkdir_mode ./-p ./-- 2>/dev/null +	    fi +	    trap '' 0;; +	esac;; +    esac + +    if +      $posix_mkdir && ( +	umask $mkdir_umask && +	$doit_exec $mkdirprog $mkdir_mode -p -- "$dstdir" +      ) +    then : +    else + +      # The umask is ridiculous, or mkdir does not conform to POSIX, +      # or it failed possibly due to a race condition.  Create the +      # directory the slow way, step by step, checking for races as we go. + +      case $dstdir in +	/*) prefix='/';; +	-*) prefix='./';; +	*)  prefix='';; +      esac + +      eval "$initialize_posix_glob" + +      oIFS=$IFS +      IFS=/ +      $posix_glob set -f +      set fnord $dstdir +      shift +      $posix_glob set +f +      IFS=$oIFS + +      prefixes= + +      for d +      do +	test -z "$d" && continue + +	prefix=$prefix$d +	if test -d "$prefix"; then +	  prefixes= +	else +	  if $posix_mkdir; then +	    (umask=$mkdir_umask && +	     $doit_exec $mkdirprog $mkdir_mode -p -- "$dstdir") && break +	    # Don't fail if two instances are running concurrently. +	    test -d "$prefix" || exit 1 +	  else +	    case $prefix in +	      *\'*) qprefix=`echo "$prefix" | sed "s/'/'\\\\\\\\''/g"`;; +	      *) qprefix=$prefix;; +	    esac +	    prefixes="$prefixes '$qprefix'" +	  fi +	fi +	prefix=$prefix/ +      done + +      if test -n "$prefixes"; then +	# Don't fail if two instances are running concurrently. +	(umask $mkdir_umask && +	 eval "\$doit_exec \$mkdirprog $prefixes") || +	  test -d "$dstdir" || exit 1 +	obsolete_mkdir_used=true +      fi +    fi +  fi + +  if test -n "$dir_arg"; then +    { test -z "$chowncmd" || $doit $chowncmd "$dst"; } && +    { test -z "$chgrpcmd" || $doit $chgrpcmd "$dst"; } && +    { test "$obsolete_mkdir_used$chowncmd$chgrpcmd" = false || +      test -z "$chmodcmd" || $doit $chmodcmd $mode "$dst"; } || exit 1 +  else + +    # Make a couple of temp file names in the proper directory. +    dsttmp=$dstdir/_inst.$$_ +    rmtmp=$dstdir/_rm.$$_ + +    # Trap to clean up those temp files at exit. +    trap 'ret=$?; rm -f "$dsttmp" "$rmtmp" && exit $ret' 0 + +    # Copy the file name to the temp name. +    (umask $cp_umask && $doit_exec $cpprog "$src" "$dsttmp") && + +    # and set any options; do chmod last to preserve setuid bits. +    # +    # If any of these fail, we abort the whole thing.  If we want to +    # ignore errors from any of these, just make sure not to ignore +    # errors from the above "$doit $cpprog $src $dsttmp" command. +    # +    { test -z "$chowncmd" || $doit $chowncmd "$dsttmp"; } && +    { test -z "$chgrpcmd" || $doit $chgrpcmd "$dsttmp"; } && +    { test -z "$stripcmd" || $doit $stripcmd "$dsttmp"; } && +    { test -z "$chmodcmd" || $doit $chmodcmd $mode "$dsttmp"; } && + +    # If -C, don't bother to copy if it wouldn't change the file. +    if $copy_on_change && +       old=`LC_ALL=C ls -dlL "$dst"	2>/dev/null` && +       new=`LC_ALL=C ls -dlL "$dsttmp"	2>/dev/null` && + +       eval "$initialize_posix_glob" && +       $posix_glob set -f && +       set X $old && old=:$2:$4:$5:$6 && +       set X $new && new=:$2:$4:$5:$6 && +       $posix_glob set +f && + +       test "$old" = "$new" && +       $cmpprog "$dst" "$dsttmp" >/dev/null 2>&1 +    then +      rm -f "$dsttmp" +    else +      # Rename the file to the real destination. +      $doit $mvcmd -f "$dsttmp" "$dst" 2>/dev/null || + +      # The rename failed, perhaps because mv can't rename something else +      # to itself, or perhaps because mv is so ancient that it does not +      # support -f. +      { +	# Now remove or move aside any old file at destination location. +	# We try this two ways since rm can't unlink itself on some +	# systems and the destination file might be busy for other +	# reasons.  In this case, the final cleanup might fail but the new +	# file should still install successfully. +	{ +	  test ! -f "$dst" || +	  $doit $rmcmd -f "$dst" 2>/dev/null || +	  { $doit $mvcmd -f "$dst" "$rmtmp" 2>/dev/null && +	    { $doit $rmcmd -f "$rmtmp" 2>/dev/null; :; } +	  } || +	  { echo "$0: cannot unlink or rename $dst" >&2 +	    (exit 1); exit 1 +	  } +	} && + +	# Now rename the file to the real destination. +	$doit $mvcmd "$dsttmp" "$dst" +      } +    fi || exit 1 + +    trap '' 0 +  fi +done + +# Local variables: +# eval: (add-hook 'write-file-hooks 'time-stamp) +# time-stamp-start: "scriptversion=" +# time-stamp-format: "%:y-%02m-%02d.%02H" +# time-stamp-end: "$" +# End: diff --git a/lib/test_server/src/test_server.app.src b/lib/test_server/src/test_server.app.src index af2d4dc2cb..faf7db835e 100644 --- a/lib/test_server/src/test_server.app.src +++ b/lib/test_server/src/test_server.app.src @@ -1,7 +1,7 @@  % This is an -*- erlang -*- file.  %% %CopyrightBegin%  %%  -%% Copyright Ericsson AB 2009. All Rights Reserved. +%% Copyright Ericsson AB 2009-2011. All Rights Reserved.  %%   %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -24,7 +24,6 @@  	     test_server_ctrl,  	     test_server,  	     test_server_h, -	     test_server_line,  	     test_server_node,  	     test_server_sup  	    ]}, diff --git a/lib/test_server/src/test_server.erl b/lib/test_server/src/test_server.erl index 244207e140..17c5f5b253 100644 --- a/lib/test_server/src/test_server.erl +++ b/lib/test_server/src/test_server.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 1996-2011. All Rights Reserved. +%% Copyright Ericsson AB 1996-2012. All Rights Reserved.  %%  %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -35,20 +35,22 @@  -export([fail/0,fail/1,format/1,format/2,format/3]).  -export([capture_start/0,capture_stop/0,capture_get/0]).  -export([messages_get/0]). +-export([permit_io/2]).  -export([hours/1,minutes/1,seconds/1,sleep/1,adjusted_sleep/1,timecall/3]). --export([timetrap_scale_factor/0,timetrap/1,timetrap_cancel/1,timetrap_cancel/0]). +-export([timetrap_scale_factor/0,timetrap/1,get_timetrap_info/0, +	 timetrap_cancel/1,timetrap_cancel/0]).  -export([m_out_of_n/3,do_times/4,do_times/2]).  -export([call_crash/3,call_crash/4,call_crash/5]).  -export([temp_name/1]).  -export([start_node/3, stop_node/1, wait_for_node/1, is_release_available/1]).  -export([app_test/1, app_test/2]).  -export([is_native/1]). --export([comment/1]). +-export([comment/1, make_priv_dir/0]).  -export([os_type/0]).  -export([run_on_shielded_node/2]).  -export([is_cover/0,is_debug/0,is_commercial/0]). --export([break/1,continue/0]). +-export([break/1,break/2,break/3,continue/0,continue/1]).  %%% DEBUGGER INTERFACE %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%  -export([purify_new_leaks/0, purify_format/2, purify_new_fds_inuse/0, @@ -522,7 +524,7 @@ stick_all_sticky(Node,Sticky) ->  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% -%% run_test_case_apply(Mod,Func,Args,Name,RunInit,TimetrapData) -> +%% run_test_case_apply(Mod,Func,Args,Name,RunInit,TimetrapData,RejectIoReqs) ->  %%               {Time,Value,Loc,Opts,Comment} | {died,Reason,unknown,Comment}  %%  %% Time = float()   (seconds) @@ -557,8 +559,12 @@ stick_all_sticky(Node,Sticky) ->  %% ScaleTimetrap indicates if test_server should attemp to automatically  %% compensate timetraps for runtime delays introduced by e.g. tools like  %% cover. +%%  +%% RejectIoReqs (bool) is information about whether printouts to stdout +%% should be visible in the minor log file or not. -run_test_case_apply({CaseNum,Mod,Func,Args,Name,RunInit,TimetrapData}) -> +run_test_case_apply({CaseNum,Mod,Func,Args,Name, +		     RunInit,TimetrapData,RejectIoReqs}) ->      purify_format("Test case #~w ~w:~w/1", [CaseNum, Mod, Func]),      case os:getenv("TS_RUN_VALGRIND") of  	false -> @@ -569,17 +575,19 @@ run_test_case_apply({CaseNum,Mod,Func,Args,Name,RunInit,TimetrapData}) ->      end,      test_server_h:testcase({Mod,Func,1}),      ProcBef = erlang:system_info(process_count), -    Result = run_test_case_apply(Mod, Func, Args, Name, RunInit, TimetrapData), +    Result = run_test_case_apply(Mod, Func, Args, Name, RunInit, +				 TimetrapData, RejectIoReqs),      ProcAft = erlang:system_info(process_count),      purify_new_leaks(),      DetFail = get(test_server_detected_fail),      {Result,DetFail,ProcBef,ProcAft}. -run_test_case_apply(Mod, Func, Args, Name, RunInit, TimetrapData) -> +run_test_case_apply(Mod, Func, Args, Name, RunInit, TimetrapData, RejectIoReqs) ->      case get(test_server_job_dir) of  	undefined ->  	    %% i'm a local target -	    do_run_test_case_apply(Mod, Func, Args, Name, RunInit, TimetrapData); +	    do_run_test_case_apply(Mod, Func, Args, Name, RunInit, +				   TimetrapData, RejectIoReqs);  	JobDir ->  	    %% i'm a remote target  	    case Args of @@ -594,13 +602,14 @@ run_test_case_apply(Mod, Func, Args, Name, RunInit, TimetrapData) ->  		    Config2 = lists:keyreplace(priv_dir, 1, Config1,  					       {priv_dir,TargetPrivDir}),  		    do_run_test_case_apply(Mod, Func, [Config2], Name, RunInit, -					   TimetrapData); +					   TimetrapData, RejectIoReqs);  		_other ->  		    do_run_test_case_apply(Mod, Func, Args, Name, RunInit, -					   TimetrapData) +					   TimetrapData, RejectIoReqs)  	    end      end. -do_run_test_case_apply(Mod, Func, Args, Name, RunInit, TimetrapData) -> +do_run_test_case_apply(Mod, Func, Args, Name, RunInit, +		       TimetrapData, RejectIoReqs) ->      {ok,Cwd} = file:get_cwd(),      Args2Print = case Args of  		     [Args1] when is_list(Args1) -> @@ -611,6 +620,7 @@ do_run_test_case_apply(Mod, Func, Args, Name, RunInit, TimetrapData) ->      print(minor, "Test case started with:\n~s:~s(~p)\n", [Mod,Func,Args2Print]),      print(minor, "Current directory is ~p\n", [Cwd]),      print_timestamp(minor,"Started at "), +    print(minor, "", [], internal_raw),      TCCallback = get(test_server_testcase_callback),      LogOpts = get(test_server_logopts),      Ref = make_ref(), @@ -626,7 +636,8 @@ do_run_test_case_apply(Mod, Func, Args, Name, RunInit, TimetrapData) ->  	  end),      group_leader(OldGLeader, self()),      put(test_server_detected_fail, []), -    run_test_case_msgloop(Ref, Pid, false, false, "", undefined). +    run_test_case_msgloop(Ref, Pid, false, RejectIoReqs, false, "", +			  undefined, starting).  %% Ugly bug (pre R5A):  %% If this process (group leader of the test case) terminates before @@ -637,19 +648,37 @@ do_run_test_case_apply(Mod, Func, Args, Name, RunInit, TimetrapData) ->  %% A test case is known to have failed if it returns {'EXIT', _} tuple,  %% or sends a message {failed, File, Line} to it's group_leader  %% -run_test_case_msgloop(Ref, Pid, CaptureStdout, Terminate, Comment, CurrConf) -> +run_test_case_msgloop(Ref, Pid, CaptureStdout, RejectIoReqs, Terminate, +		      Comment, CurrConf, Status) ->      %% NOTE: Keep job_proxy_msgloop/0 up to date when changes      %%       are made in this function!      {Timeout,ReturnValue} =  	case Terminate of  	    {true, ReturnVal} -> +		%% stop any timetrap timers for the test case +		%% that have been started by this process +		timetrap_cancel_all(Pid, false),  		{20, ReturnVal};  	    false ->  		{infinity, should_never_appear}  	end,      receive +	{test_case_initialized,Pid} -> +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  Comment,CurrConf,running); +	Abort = {abort_current_testcase,_,_} when Status == starting -> +	    %% we're in init phase, must must postpone this operation +	    %% until test case execution is in progress (or FW:init_tc +	    %% gets killed) +	    self() ! Abort, +	    erlang:yield(), +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  Comment,CurrConf,Status);  	{abort_current_testcase,Reason,From} -> -	    Line = get_loc(Pid), +	    Line = case is_process_alive(Pid) of +		       true -> get_loc(Pid); +		       false -> unknown +		   end,  	    Mon = erlang:monitor(process, Pid),  	    exit(Pid,{testcase_aborted,Reason,Line}),  	    erlang:yield(), @@ -663,76 +692,104 @@ run_test_case_msgloop(Ref, Pid, CaptureStdout, Terminate, Comment, CurrConf) ->  			    exit(Pid, kill),  			    %% here's the only place we know Reason, so we save  			    %% it as a comment, potentially replacing user data -			    Error = lists:flatten(io_lib:format("Aborted: ~p",[Reason])), +			    Error = lists:flatten(io_lib:format("Aborted: ~p", +								[Reason])),  			    Error1 = lists:flatten([string:strip(S,left) || -						    S <- string:tokens(Error,[$\n])]), +						    S <- string:tokens(Error, +								       [$\n])]),  			    if length(Error1) > 63 ->  				    string:substr(Error1,1,60) ++ "...";  			       true ->  				    Error1  			    end  		    end, -	    run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,NewComment,CurrConf); +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  NewComment,CurrConf,Status); +	{permit_io,FromPid} -> +	    put({permit_io,FromPid},true), +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  Comment,CurrConf,Status);          {io_request,From,ReplyAs,{put_chars,io_lib,Func,[Format,Args]}}  	when is_list(Format) ->  	    Msg = (catch io_lib:Func(Format,Args)), -	    run_test_case_msgloop_io(ReplyAs,CaptureStdout,Msg,From,Func), -            run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf); +	    run_test_case_msgloop_io(From,ReplyAs,CaptureStdout,RejectIoReqs, +				     Msg,From,Func), +            run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  Comment,CurrConf,Status);          {io_request,From,ReplyAs,{put_chars,io_lib,Func,[Format,Args]}}  	when is_atom(Format) ->  	    Msg = (catch io_lib:Func(Format,Args)), -	    run_test_case_msgloop_io(ReplyAs,CaptureStdout,Msg,From,Func), -            run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf); +	    run_test_case_msgloop_io(From,ReplyAs,CaptureStdout,RejectIoReqs, +				     Msg,From,Func), +            run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  Comment,CurrConf,Status);          {io_request,From,ReplyAs,{put_chars,Bytes}} -> -	    run_test_case_msgloop_io( -	      ReplyAs,CaptureStdout,Bytes,From,put_chars), -            run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf); +	    run_test_case_msgloop_io(From,ReplyAs,CaptureStdout,RejectIoReqs, +				     Bytes,From,put_chars), +            run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  Comment,CurrConf,Status);          {io_request,From,ReplyAs,{put_chars,unicode,io_lib,Func,[Format,Args]}}  	when is_list(Format) ->  	    Msg = unicode_to_latin1(catch io_lib:Func(Format,Args)), -	    run_test_case_msgloop_io(ReplyAs,CaptureStdout,Msg,From,Func), -            run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf); +	    run_test_case_msgloop_io(From,ReplyAs,CaptureStdout,RejectIoReqs, +				     Msg,From,Func), +            run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  Comment,CurrConf,Status);          {io_request,From,ReplyAs,{put_chars,latin1,io_lib,Func,[Format,Args]}}  	when is_list(Format) ->  	    Msg = (catch io_lib:Func(Format,Args)), -	    run_test_case_msgloop_io(ReplyAs,CaptureStdout,Msg,From,Func), -            run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf); +	    run_test_case_msgloop_io(From,ReplyAs,CaptureStdout,RejectIoReqs, +				     Msg,From,Func), +            run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  Comment,CurrConf,Status);          {io_request,From,ReplyAs,{put_chars,unicode,io_lib,Func,[Format,Args]}}  	when is_atom(Format) ->  	    Msg = unicode_to_latin1(catch io_lib:Func(Format,Args)), -	    run_test_case_msgloop_io(ReplyAs,CaptureStdout,Msg,From,Func), -            run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf); +	    run_test_case_msgloop_io(From,ReplyAs,CaptureStdout,RejectIoReqs, +				     Msg,From,Func), +            run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  Comment,CurrConf,Status);          {io_request,From,ReplyAs,{put_chars,latin1,io_lib,Func,[Format,Args]}}  	when is_atom(Format) ->  	    Msg = (catch io_lib:Func(Format,Args)), -	    run_test_case_msgloop_io(ReplyAs,CaptureStdout,Msg,From,Func), -            run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf); +	    run_test_case_msgloop_io(From,ReplyAs,CaptureStdout,RejectIoReqs, +				     Msg,From,Func), +            run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  Comment,CurrConf,Status);          {io_request,From,ReplyAs,{put_chars,unicode,Bytes}} -> -	    run_test_case_msgloop_io( -	      ReplyAs,CaptureStdout,unicode_to_latin1(Bytes),From,put_chars), -            run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf); +	    run_test_case_msgloop_io(From,ReplyAs,CaptureStdout,RejectIoReqs, +				     unicode_to_latin1(Bytes),From,put_chars), +            run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  Comment,CurrConf,Status);          {io_request,From,ReplyAs,{put_chars,latin1,Bytes}} -> -	    run_test_case_msgloop_io( -	      ReplyAs,CaptureStdout,Bytes,From,put_chars), -            run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf); +	    run_test_case_msgloop_io(From,ReplyAs,CaptureStdout,RejectIoReqs, +				     Bytes,From,put_chars), +            run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  Comment,CurrConf,Status);          IoReq when element(1, IoReq) == io_request ->  	    %% something else, just pass it on              group_leader() ! IoReq, -            run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf); +            run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  Comment,CurrConf,Status);  	{structured_io,ClientPid,Msg} ->  	    output(Msg, ClientPid), -            run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf); +            run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  Comment,CurrConf,Status);  	{capture,NewCapture} -> -            run_test_case_msgloop(Ref,Pid,NewCapture,Terminate,Comment,CurrConf); +            run_test_case_msgloop(Ref,Pid,NewCapture,RejectIoReqs,Terminate, +				  Comment,CurrConf,Status);  	{sync_apply,From,MFA} ->  	    sync_local_or_remote_apply(false,From,MFA), -	    run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf); +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  Comment,CurrConf,Status);  	{sync_apply_proxy,Proxy,From,MFA} ->  	    sync_local_or_remote_apply(Proxy,From,MFA), -	    run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf); +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  Comment,CurrConf,Status);  	{printout,Detail,Format,Args} ->  	    print(Detail,Format,Args), -	    run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf); +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  Comment,CurrConf,Status);  	{comment,NewComment} ->  	    NewComment1 = test_server_ctrl:to_string(NewComment),  	    NewComment2 = test_server_sup:framework_call(format_comment, @@ -745,16 +802,42 @@ run_test_case_msgloop(Ref, Pid, CaptureStdout, Terminate, Comment, CurrConf) ->  		    Other ->  			Other  		end, -	    run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate1,NewComment2,CurrConf); +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate1, +				  NewComment2,CurrConf,Status);  	{read_comment,From} ->  	    From ! {self(),read_comment,Comment}, -	    run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf); +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  Comment,CurrConf,Status);  	{set_curr_conf,From,NewCurrConf} ->  	    From ! {self(),set_curr_conf,ok}, -	    run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,NewCurrConf); +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  Comment,NewCurrConf,Status); +	{make_priv_dir,From} when CurrConf == undefined -> +	    From ! {self(),make_priv_dir,{error,no_priv_dir_in_config}}, +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  Comment,CurrConf,Status); +	{make_priv_dir,From} -> +	    Result = +		case proplists:get_value(priv_dir, element(2, CurrConf)) of +		    undefined -> +			{error,no_priv_dir_in_config}; +		    PrivDir -> +			case file:make_dir(PrivDir) of +			    ok -> +				ok; +			    {error, eexist} -> +				ok; +			    MkDirError -> +				{error,{MkDirError,PrivDir}} +			end +		end, +	    From ! {self(),make_priv_dir,Result}, +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs,Terminate, +				  Comment,CurrConf,Status);  	{'EXIT',Pid,{Ref,Time,Value,Loc,Opts}} ->  	    RetVal = {Time/1000000,Value,mod_loc(Loc),Opts,Comment}, -	    run_test_case_msgloop(Ref,Pid,CaptureStdout,{true,RetVal},Comment,undefined); +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs, +				  {true,RetVal},Comment,undefined,Status);  	{'EXIT',Pid,Reason} ->  	    case Reason of  		{timetrap_timeout,TVal,Loc} -> @@ -764,37 +847,45 @@ run_test_case_msgloop(Ref, Pid, CaptureStdout, Terminate, Comment, CurrConf) ->  			    %% timout during framework call  			    spawn_fw_call(FwMod,FwFunc,CurrConf,Pid,  					  {framework_error,{timetrap,TVal}}, -					  unknown,self(),Comment), -			    run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate, -						  Comment,undefined); +					  unknown,self()), +			    run_test_case_msgloop(Ref,Pid, +						  CaptureStdout,RejectIoReqs, +						  Terminate,Comment, +						  undefined,Status);  			Loc1 -> -			    {Mod,Func} = get_mf(Loc1),  			    %% call end_per_testcase on a separate process, -			    %% only so that the user has a chance to clean up -			    %% after init_per_testcase, even after a timetrap timeout +			    %% only so that the user has a chance to +			    %% clean up after init_per_testcase, even after +			    %% a timetrap timeout  			    NewCurrConf =  				case CurrConf of  				    {{Mod,Func},Conf} ->  					EndConfPid = -					    call_end_conf(Mod,Func,Pid, -							  {timetrap_timeout,TVal}, -							  Loc1,[{tc_status, -								 {failed, -								  timetrap_timeout}}|Conf], -							  TVal), +					    call_end_conf( +					      Mod,Func,Pid, +					      {timetrap_timeout,TVal}, +					      Loc1,[{tc_status, +						     {failed, +						      timetrap_timeout}}|Conf], +					      TVal),  					{EndConfPid,{Mod,Func},Conf};  				    _ -> -					%% The framework functions mustn't execute on this -					%% group leader process or io will cause deadlock, -					%% so we spawn a dedicated process for the operation -					%% and let the group leader go back to handle io. +					{Mod,Func} = get_mf(Loc1), +					%% The framework functions mustn't +					%% execute on this group leader process +					%% or io will cause deadlock, so we +					%% spawn a dedicated process for the +					%% operation and let the group leader +					%% go back to handle io.  					spawn_fw_call(Mod,Func,CurrConf,Pid,  						      {timetrap_timeout,TVal}, -						      Loc1,self(),Comment), +						      Loc1,self()),  					undefined  				end, -			    run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate, -						  Comment,NewCurrConf) +			    run_test_case_msgloop(Ref,Pid, +						  CaptureStdout,RejectIoReqs, +						  Terminate,Comment, +						  NewCurrConf,Status)  		    end;  		{timetrap_timeout,TVal,Loc,InitOrEnd} ->  		    case mod_loc(Loc) of @@ -802,14 +893,25 @@ run_test_case_msgloop(Ref, Pid, CaptureStdout, Terminate, Comment, CurrConf) ->  			    %% timout during framework call  			    spawn_fw_call(FwMod,FwFunc,CurrConf,Pid,  					  {framework_error,{timetrap,TVal}}, -					  unknown,self(),Comment); +					  unknown,self());  			Loc1 ->  			    {Mod,_Func} = get_mf(Loc1),  			    spawn_fw_call(Mod,InitOrEnd,CurrConf,Pid,  					  {timetrap_timeout,TVal}, -					  Loc1,self(),Comment) +					  Loc1,self())  		    end, -		    run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf); +		    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs, +					  Terminate,Comment,CurrConf,Status); +		{testcase_aborted,ErrorMsg={user_timetrap_error,_},AbortLoc} -> +		    %% user timetrap function caused exit +		    %% during start of test case +		    {Mod,Func} = get_mf(mod_loc(AbortLoc)), +		    spawn_fw_call(Mod,Func,CurrConf,Pid, +				  ErrorMsg,unknown,self()), +		    run_test_case_msgloop(Ref,Pid, +					  CaptureStdout,RejectIoReqs, +					  Terminate,Comment, +					  undefined,Status);		  		{testcase_aborted,AbortReason,AbortLoc} ->  		    ErrorMsg = {testcase_aborted,AbortReason},  		    case mod_loc(AbortLoc) of @@ -817,66 +919,108 @@ run_test_case_msgloop(Ref, Pid, CaptureStdout, Terminate, Comment, CurrConf) ->  			    %% abort during framework call  			    spawn_fw_call(FwMod,FwFunc,CurrConf,Pid,  					  {framework_error,ErrorMsg}, -					  unknown,self(),Comment), -			    run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate, -						  Comment,undefined); +					  unknown,self()), +			    run_test_case_msgloop(Ref,Pid, +						  CaptureStdout,RejectIoReqs, +						  Terminate,Comment, +						  undefined,Status);  			Loc1 -> -			    {Mod,Func} = get_mf(Loc1), -			    %% call end_per_testcase on a separate process, only so -			    %% that the user has a chance to clean up after init_per_testcase, -			    %% even after abortion +			    %% call end_per_testcase on a separate process, +			    %% only so that the user has a chance to clean up +			    %% after init_per_testcase, even after abortion  			    NewCurrConf =  				case CurrConf of  				    {{Mod,Func},Conf} -> -					TVal = case lists:keysearch(default_timeout,1,Conf) of -						   {value,{default_timeout,Tmo}} -> Tmo; -						   _ -> ?DEFAULT_TIMETRAP_SECS*1000 -					       end, +					TVal = +					    case lists:keysearch(default_timeout, +								 1, +								 Conf) of +						{value,{default_timeout,Tmo}} -> +						    Tmo; +						_ -> +						    ?DEFAULT_TIMETRAP_SECS*1000 +					    end,  					EndConfPid = -					    call_end_conf(Mod,Func,Pid,ErrorMsg, -							  Loc1, -							  [{tc_status,{failed,ErrorMsg}}|Conf], -							  TVal), +					    call_end_conf( +					      Mod,Func,Pid, +					      ErrorMsg,Loc1, +					      [{tc_status, +						{failed,ErrorMsg}}|Conf],TVal),  					{EndConfPid,{Mod,Func},Conf};  				    _ -> -					spawn_fw_call(Mod,Func,CurrConf,Pid,ErrorMsg, -						      Loc1,self(),Comment), +					{Mod,Func} = get_mf(Loc1), +					spawn_fw_call(Mod,Func,CurrConf,Pid, +						      ErrorMsg,Loc1,self()),  					undefined  				end, -			    run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate, -						  Comment,NewCurrConf) +			    run_test_case_msgloop(Ref,Pid, +						  CaptureStdout,RejectIoReqs, +						  Terminate,Comment, +						  NewCurrConf,Status)  		    end;  		killed ->  		    %% result of an exit(TestCase,kill) call, which is the  		    %% only way to abort a testcase process that traps exits  		    %% (see abort_current_testcase) -		    spawn_fw_call(undefined,undefined,CurrConf,Pid, +		    {Mod,Func} = case CurrConf of +				     {MF,_} -> MF; +				     _      -> {undefined,undefined} +				 end, +		    spawn_fw_call(Mod,Func,CurrConf,Pid,  				  testcase_aborted_or_killed, -				  unknown,self(),Comment), -		    run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf); +				  unknown,self()), +		    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs, +					  Terminate,Comment,CurrConf,Status);  		{fw_error,{FwMod,FwFunc,FwError}} -> -		    spawn_fw_call(FwMod,FwFunc,CurrConf,Pid,{framework_error,FwError}, -				  unknown,self(),Comment), -		    run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf); +		    spawn_fw_call(FwMod,FwFunc,CurrConf,Pid, +				  {framework_error,FwError}, +				  unknown,self()), +		    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs, +					  Terminate,Comment,CurrConf,Status);  		_Other ->  		    %% the testcase has terminated because of Reason (e.g. an exit  		    %% because a linked process failed) -		    spawn_fw_call(undefined,undefined,CurrConf,Pid,Reason, -				  unknown,self(),Comment), -		    run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf) +		    {Mod,Func} = case CurrConf of +				     {MF,_} -> MF; +				     _      -> {undefined,undefined} +				 end, +		    spawn_fw_call(Mod,Func,CurrConf,Pid, +				  Reason,unknown,self()), +		    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs, +					  Terminate,Comment,CurrConf,Status)  	    end;  	{EndConfPid,{call_end_conf,Data,_Result}} ->  	    case CurrConf of  		{EndConfPid,{Mod,Func},_Conf} ->  		    {_Mod,_Func,TCPid,TCExitReason,Loc} = Data, -		    spawn_fw_call(Mod,Func,CurrConf,TCPid,TCExitReason,Loc,self(),Comment), -		    run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,undefined); +		    spawn_fw_call(Mod,Func,CurrConf,TCPid, +				  TCExitReason,Loc,self()), +		    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs, +					  Terminate,Comment,undefined,Status);  		_ -> -		    run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf) +		    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs, +					  Terminate,Comment,CurrConf,Status)  	    end; -	{_FwCallPid,fw_notify_done,RetVal} -> +	{_FwCallPid,fw_notify_done,{T,Value,Loc,Opts,AddToComment}} ->  	    %% the framework has been notified, we're finished -	    run_test_case_msgloop(Ref,Pid,CaptureStdout,{true,RetVal},Comment,undefined); +	    RetVal = +		case AddToComment of +		    undefined -> +			{T,Value,Loc,Opts,Comment}; +		    _ -> +			Comment1 = +			    if Comment == "" ->  +				    AddToComment; +			       true ->  +				    Comment ++ +				    test_server_ctrl:xhtml("<br>", +							   "<br />") ++ +				    AddToComment +			    end, +			{T,Value,Loc,Opts,Comment1} +		end, +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs, +				  {true,RetVal},Comment,undefined,Status);   	{'EXIT',_FwCallPid,{fw_notify_done,Func,Error}} ->  	    %% a framework function failed  	    CB = os:getenv("TEST_SERVER_FRAMEWORK"), @@ -887,53 +1031,108 @@ run_test_case_msgloop(Ref, Pid, CaptureStdout, Terminate, Comment, CurrConf) ->  			  {list_to_atom(CB),Func}  		  end,  	    RetVal = {died,{framework_error,Loc,Error},Loc,"Framework error"}, -	    run_test_case_msgloop(Ref,Pid,CaptureStdout,{true,RetVal},Comment,undefined); +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs, +				  {true,RetVal},Comment,undefined,Status);  	{failed,File,Line} ->  	    put(test_server_detected_fail,  		[{File, Line}| get(test_server_detected_fail)]), -	    run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf); +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs, +				  Terminate,Comment,CurrConf,Status); + +	{user_timetrap,Pid,_TrapTime,StartTime,E={user_timetrap_error,_},_} -> +	    case update_user_timetraps(Pid, StartTime) of +		proceed -> +		    self() ! {abort_current_testcase,E,Pid}; +		ignore -> +		    ok +	    end, +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs, +				  Terminate,Comment,CurrConf,Status); +	{user_timetrap,Pid,TrapTime,StartTime,ElapsedTime,Scale} -> +	    %% a user timetrap is triggered, ignore it if new +	    %% timetrap has been started since +	    case update_user_timetraps(Pid, StartTime) of +		proceed -> +		    TotalTime = if is_integer(TrapTime) ->  +					TrapTime + ElapsedTime; +				   true ->  +					TrapTime +				end, +		    timetrap(TrapTime, TotalTime, Pid, Scale); +		ignore -> +		    ok +	    end, +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs, +				  Terminate,Comment,CurrConf,Status); +	{timetrap_cancel_one,Handle,_From} -> +	    timetrap_cancel_one(Handle, false), +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs, +				  Terminate,Comment,CurrConf,Status); +	{timetrap_cancel_all,TCPid,_From} -> +	    timetrap_cancel_all(TCPid, false), +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs, +				  Terminate,Comment,CurrConf,Status); +	{get_timetrap_info,TCPid,From} -> +	    Info = get_timetrap_info(TCPid, false), +	    From ! {self(),get_timetrap_info,Info}, +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs, +				  Terminate,Comment,CurrConf,Status);  	_Other when not is_tuple(_Other) ->  	    %% ignore anything not generated by test server -	    run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf); +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs, +				  Terminate,Comment,CurrConf,Status);  	_Other when element(1, _Other) /= 'EXIT',  		    element(1, _Other) /= started,  		    element(1, _Other) /= finished,  		    element(1, _Other) /= print ->  	    %% ignore anything not generated by test server -	    run_test_case_msgloop(Ref,Pid,CaptureStdout,Terminate,Comment,CurrConf) +	    run_test_case_msgloop(Ref,Pid,CaptureStdout,RejectIoReqs, +				  Terminate,Comment,CurrConf,Status)      after Timeout ->  	    ReturnValue      end. -run_test_case_msgloop_io(ReplyAs,CaptureStdout,Msg,From,Func) -> +run_test_case_msgloop_io(From,ReplyAs,CaptureStdout,RejectIoReqs, +			 Msg,From,Func) ->      case Msg of  	{'EXIT',_} ->  	    From ! {io_reply,ReplyAs,{error,Func}};  	_ ->  	    From ! {io_reply,ReplyAs,ok}      end, -    if  CaptureStdout /= false -> -	    CaptureStdout ! {captured,Msg}; -	true -> +    Proceed = if RejectIoReqs -> get({permit_io,From}); +		 true        -> true +	      end, +    if Proceed -> +	    if CaptureStdout /= false -> +		    CaptureStdout ! {captured,Msg}; +	       true -> +		    ok +	    end, +	    output({minor,Msg},From); +       true ->  	    ok -    end, -    output({minor,Msg},From). +    end.  output(Msg,Sender) ->      local_or_remote_apply({test_server_ctrl,output,[Msg,Sender]}).  call_end_conf(Mod,Func,TCPid,TCExitReason,Loc,Conf,TVal) -> +    %% Starter is also the group leader process      Starter = self(),      Data = {Mod,Func,TCPid,TCExitReason,Loc},      EndConfProc =  	fun() -> +		group_leader(Starter, self()),  		Supervisor = self(),  		EndConfApply =  		    fun() ->  			    case catch apply(Mod,end_per_testcase,[Func,Conf]) of  				{'EXIT',Why} -> +				    timer:sleep(1),  				    group_leader() ! {printout,12, -						      "ERROR! ~p:end_per_testcase(~p, ~p)" +						      "WARNING! " +						      "~p:end_per_testcase(~p, ~p)"  						      " crashed!\n\tReason: ~p\n",  						      [Mod,Func,Conf,Why]};  				_ -> @@ -948,15 +1147,23 @@ call_end_conf(Mod,Func,TCPid,TCExitReason,Loc,Conf,TVal) ->  		    {'EXIT',Pid,Reason} ->  			Starter ! {self(),{call_end_conf,Data,{error,Reason}}}  		after TVal -> +			exit(Pid, kill), +			group_leader() ! {printout,12, +					  "WARNING! ~p:end_per_testcase(~p, ~p)" +					  " failed!\n\tReason: timetrap timeout" +					  " after ~w ms!\n", [Mod,Func,Conf,TVal]},  			Starter ! {self(),{call_end_conf,Data,{error,timeout}}}  		end  	end,      spawn_link(EndConfProc).  spawn_fw_call(Mod,{init_per_testcase,Func},_,Pid,{timetrap_timeout,TVal}=Why, -	      Loc,SendTo,Comment) -> +	      Loc,SendTo) ->      FwCall =  	fun() -> +		%% set group leader so that printouts/comments +		%% from the framework get printed in the logs +		group_leader(SendTo, self()),  		Skip = {skip,{failed,{Mod,init_per_testcase,Why}}},  		%% if init_per_testcase fails, the test case  		%% should be skipped @@ -968,12 +1175,12 @@ spawn_fw_call(Mod,{init_per_testcase,Func},_,Pid,{timetrap_timeout,TVal}=Why,  		end,  		%% finished, report back  		SendTo ! {self(),fw_notify_done, -			  {TVal/1000,Skip,Loc,[],Comment}} +			  {TVal/1000,Skip,Loc,[],undefined}}  	end,      spawn_link(FwCall);  spawn_fw_call(Mod,{end_per_testcase,Func},EndConf,Pid, -	      {timetrap_timeout,TVal}=Why,_Loc,SendTo,Comment) -> +	      {timetrap_timeout,TVal}=Why,_Loc,SendTo) ->      %%! This is a temporary fix that keeps Test Server alive during      %%! execution of a parallel test case group, when sometimes      %%! this clause gets called with EndConf == undefined. See OTP-9594 @@ -985,6 +1192,9 @@ spawn_fw_call(Mod,{end_per_testcase,Func},EndConf,Pid,  	       end,      FwCall =  	fun() -> +		%% set group leader so that printouts/comments +		%% from the framework get printed in the logs +		group_leader(SendTo, self()),  		{RetVal,Report} =  		    case proplists:get_value(tc_status, EndConf1) of  			undefined -> @@ -996,6 +1206,10 @@ spawn_fw_call(Mod,{end_per_testcase,Func},EndConf,Pid,  			    E = {failed,{Mod,end_per_testcase,Why}},  			    {Result,E}  		    end, +		group_leader() ! {printout,12, +				  "WARNING! ~p:end_per_testcase(~p, ~p)" +				  " failed!\n\tReason: timetrap timeout" +				  " after ~w ms!\n", [Mod,Func,EndConf,TVal]},  		FailLoc = proplists:get_value(tc_fail_loc, EndConf1),  		case catch do_end_tc_call(Mod,Func, FailLoc,  					  {Pid,Report,[EndConf1]}, Why) of @@ -1004,39 +1218,48 @@ spawn_fw_call(Mod,{end_per_testcase,Func},EndConf,Pid,  		    _ ->  			ok  		end, -		%% if end_per_testcase fails a warning should be -		%% printed as comment -		Comment1 = if Comment == "" -> ""; -			      true -> Comment ++ "<br>" -			   end, -		%% finished, report back +		Warn = "<font color=\"red\">" +		       "WARNING: end_per_testcase timed out!</font>", +		%% finished, report back (if end_per_testcase fails, a warning +		%% should be printed as part of the comment)  		SendTo ! {self(),fw_notify_done, -			  {TVal/1000,RetVal,FailLoc,[], -			   [Comment1,"<font color=\"red\">" -			    "WARNING: end_per_testcase timed out!" -			    "</font>"]}} +			  {TVal/1000,RetVal,FailLoc,[],Warn}}  	end,      spawn_link(FwCall); -spawn_fw_call(FwMod,FwFunc,_,_Pid,{framework_error,FwError},_,SendTo,_Comment) -> +spawn_fw_call(FwMod,FwFunc,_,_Pid,{framework_error,FwError},_,SendTo) ->      FwCall =  	fun() -> +		%% set group leader so that printouts/comments +		%% from the framework get printed in the logs +		group_leader(SendTo, self()),  		test_server_sup:framework_call(report, [framework_error, -							{{FwMod,FwFunc},FwError}]), +							{{FwMod,FwFunc}, +							 FwError}]),  		Comment =  		    lists:flatten(  		      io_lib:format("<font color=\"red\">" -				    "WARNING! ~w:~w failed!</font>", [FwMod,FwFunc])), +				    "WARNING! ~w:~w failed!</font>", +				    [FwMod,FwFunc])),  	    %% finished, report back  	    SendTo ! {self(),fw_notify_done, -		      {died,{error,{FwMod,FwFunc,FwError}},{FwMod,FwFunc},[],Comment}} +		      {died,{error,{FwMod,FwFunc,FwError}}, +		       {FwMod,FwFunc},[],Comment}}  	end,      spawn_link(FwCall); -spawn_fw_call(Mod,Func,_,Pid,Error,Loc,SendTo,Comment) -> +spawn_fw_call(Mod,Func,CurrConf,Pid,Error,Loc,SendTo) -> +    {Mod1,Func1} = +	case {Mod,Func,CurrConf} of +	    {undefined,undefined,{{M,F},_}} -> {M,F}; +	    _ -> {Mod,Func} +	end,	          FwCall =  	fun() -> -		case catch fw_error_notify(Mod,Func,[], +		%% set group leader so that printouts/comments +		%% from the framework get printed in the logs +		group_leader(SendTo, self()), +		case catch fw_error_notify(Mod1,Func1,[],  					   Error,Loc) of  		    {'EXIT',FwErrorNotifyErr} ->  			exit({fw_notify_done,error_notification, @@ -1045,7 +1268,7 @@ spawn_fw_call(Mod,Func,_,Pid,Error,Loc,SendTo,Comment) ->  			ok  		end,  		Conf = [{tc_status,{failed,timetrap_timeout}}], -		case catch do_end_tc_call(Mod,Func, Loc, +		case catch do_end_tc_call(Mod1,Func1, Loc,  					  {Pid,Error,[Conf]},Error) of  		    {'EXIT',FwEndTCErr} ->  			exit({fw_notify_done,end_tc,FwEndTCErr}); @@ -1053,7 +1276,7 @@ spawn_fw_call(Mod,Func,_,Pid,Error,Loc,SendTo,Comment) ->  			ok  		end,  		%% finished, report back -		SendTo ! {self(),fw_notify_done,{died,Error,Loc,Comment}} +		SendTo ! {self(),fw_notify_done,{died,Error,Loc,[],undefined}}  	end,      spawn_link(FwCall). @@ -1110,10 +1333,11 @@ run_test_case_eval(Mod, Func, Args0, Name, Ref, RunInit,  		   TimetrapData, LogOpts, TCCallback) ->      put(test_server_multiply_timetraps, TimetrapData),      put(test_server_logopts, LogOpts), - +    FWInitResult = test_server_sup:framework_call(init_tc,[?pl2a(Mod),Func,Args0], +						  {ok,Args0}), +    group_leader() ! {test_case_initialized,self()},      {{Time,Value},Loc,Opts} = -	case test_server_sup:framework_call(init_tc,[?pl2a(Mod),Func,Args0], -					    {ok,Args0}) of +	case FWInitResult of  	    {ok,Args} ->  		run_test_case_eval1(Mod, Func, Args, Name, RunInit, TCCallback);  	    Error = {error,_Reason} -> @@ -1135,12 +1359,15 @@ run_test_case_eval(Mod, Func, Args0, Name, Ref, RunInit,  	    {auto_skip,Reason} ->  		Where = {Mod,Func},  		NewResult = do_end_tc_call(Mod,Func, Where, {{skip,Reason},Args0}, -					   {skip,{fw_auto_skip,Reason}}), +					   {skip,Reason}),  		{{0,NewResult},Where,[]}  	end,      exit({Ref,Time,Value,Loc,Opts}).  run_test_case_eval1(Mod, Func, Args, Name, RunInit, TCCallback) -> +    %% save current state in controller loop +    sync_send(group_leader(),set_curr_conf,{{Mod,Func},hd(Args)}, +	      5000, fun() -> exit(no_answer_from_group_leader) end),      case RunInit of  	run_init ->  	    put(test_server_init_or_end_conf,{init_per_testcase,Func}), @@ -1199,8 +1426,8 @@ run_test_case_eval1(Mod, Func, Args, Name, RunInit, TCCallback) ->  		    %% call user callback function if defined  		    EndConf1 = user_callback(TCCallback, Mod, Func, 'end', EndConf),  		    %% update current state in controller loop -		    sync_send(group_leader(),set_curr_conf,EndConf1, -			      5000, fun() -> exit(no_answer_from_group_leader) end), +		    sync_send(group_leader(),set_curr_conf,EndConf1, 5000, +			      fun() -> exit(no_answer_from_group_leader) end),  		    {FWReturn1,TSReturn1,EndConf2} =  			case end_per_testcase(Mod, Func, EndConf1) of  			    SaveCfg1={save_config,_} -> @@ -1253,11 +1480,15 @@ run_test_case_eval1(Mod, Func, Args, Name, RunInit, TCCallback) ->      end.  do_end_tc_call(M,F, Loc, Res, Return) -> +    IsSuite = case lists:reverse(atom_to_list(M)) of +		  [$E,$T,$I,$U,$S,$_|_]  -> true; +		  _ -> false +	      end,      FwMod = os:getenv("TEST_SERVER_FRAMEWORK"),      {Mod,Func} =  	if FwMod == M ; FwMod == "undefined"; FwMod == false ->  		{M,F}; -	   is_list(Loc) and (length(Loc)>1) -> +	   (not IsSuite) and is_list(Loc) and (length(Loc)>1) ->  		%% If failure in other module (M) than suite, try locate  		%% suite name in Loc list and call end_tc with Suite:TestCase  		%% instead of M:F. @@ -1388,57 +1619,62 @@ init_per_testcase(Mod, Func, Args) ->  	false -> code:load_file(Mod);  	_ -> ok      end, -    %% init_per_testcase defined, returns new configuration -    case erlang:function_exported(Mod,init_per_testcase,2) of +    case erlang:function_exported(Mod, init_per_testcase, 2) of  	true -> -	    case catch my_apply(Mod, init_per_testcase, [Func|Args]) of -		{'$test_server_ok',{Skip,Reason}} when Skip==skip; -						       Skip==skipped -> -		    {skip,Reason}; -		{'$test_server_ok',Res={skip_and_save,_,_}} -> -		    Res; -		{'$test_server_ok',NewConf} when is_list(NewConf) -> -		    case lists:filter(fun(T) when is_tuple(T) -> false; -					 (_) -> true end, NewConf) of -			[] -> -			    {ok,NewConf}; -			Bad -> -			    group_leader() ! {printout,12, -					      "ERROR! init_per_testcase has returned " -					      "bad elements in Config: ~p\n",[Bad]}, -			    {skip,{failed,{Mod,init_per_testcase,bad_return}}} -		    end; -		{'$test_server_ok',Res={fail,_Reason}} -> -		    Res; -		{'$test_server_ok',_Other} -> -		    group_leader() ! {printout,12, -				      "ERROR! init_per_testcase did not return " -				      "a Config list.\n",[]}, -		    {skip,{failed,{Mod,init_per_testcase,bad_return}}}; -		{'EXIT',Reason} -> -		    Line = get_loc(), -		    FormattedLoc = test_server_sup:format_loc(mod_loc(Line)), -		    group_leader() ! {printout,12, -				      "ERROR! init_per_testcase crashed!\n" -				      "\tLocation: ~s\n\tReason: ~p\n", -				      [FormattedLoc,Reason]}, -		    {skip,{failed,{Mod,init_per_testcase,Reason}}}; -		Other -> -		    Line = get_loc(), -		    FormattedLoc = test_server_sup:format_loc(mod_loc(Line)), -		    group_leader() ! {printout,12, -				      "ERROR! init_per_testcase thrown!\n" -				      "\tLocation: ~s\n\tReason: ~p\n", -				      [FormattedLoc, Other]}, -		    {skip,{failed,{Mod,init_per_testcase,Other}}} -	    end; +	    do_init_per_testcase(Mod, [Func|Args]);  	false -> -%% Optional init_per_testcase not defined -%% keep quiet. +	    %% Optional init_per_testcase is not defined -- keep quiet.  	    [Config] = Args,  	    {ok, Config}      end. +do_init_per_testcase(Mod, Args) -> +    try	apply(Mod, init_per_testcase, Args) of +	{Skip,Reason} when Skip =:= skip; Skip =:= skipped -> +	    {skip,Reason}; +	{skip_and_save,_,_}=Res -> +	    Res; +	NewConf when is_list(NewConf) -> +	    case lists:filter(fun(T) when is_tuple(T) -> false; +				 (_) -> true end, NewConf) of +		[] -> +		    {ok,NewConf}; +		Bad -> +		    group_leader() ! {printout,12, +				      "ERROR! init_per_testcase has returned " +				      "bad elements in Config: ~p\n",[Bad]}, +		    {skip,{failed,{Mod,init_per_testcase,bad_return}}} +	    end; +	{fail,_Reason}=Res -> +	    Res; +	_Other -> +	    group_leader() ! {printout,12, +			      "ERROR! init_per_testcase did not return " +			      "a Config list.\n",[]}, +	    {skip,{failed,{Mod,init_per_testcase,bad_return}}} +    catch +	throw:Other -> +	    set_loc(erlang:get_stacktrace()), +	    Line = get_loc(), +	    FormattedLoc = test_server_sup:format_loc(mod_loc(Line)), +	    group_leader() ! {printout,12, +			      "ERROR! init_per_testcase thrown!\n" +			      "\tLocation: ~s\n\tReason: ~p\n", +			      [FormattedLoc, Other]}, +	    {skip,{failed,{Mod,init_per_testcase,Other}}}; +	_:Reason0 -> +	    Stk = erlang:get_stacktrace(), +	    Reason = {Reason0,Stk}, +	    set_loc(Stk), +	    Line = get_loc(), +	    FormattedLoc = test_server_sup:format_loc(mod_loc(Line)), +	    group_leader() ! {printout,12, +			      "ERROR! init_per_testcase crashed!\n" +			      "\tLocation: ~s\n\tReason: ~p\n", +			      [FormattedLoc,Reason]}, +	    {skip,{failed,{Mod,init_per_testcase,Reason}}} +    end. +  end_per_testcase(Mod, Func, Conf) ->      case erlang:function_exported(Mod,end_per_testcase,2) of  	true -> @@ -1456,77 +1692,121 @@ end_per_testcase(Mod, Func, Conf) ->  do_end_per_testcase(Mod,EndFunc,Func,Conf) ->      put(test_server_init_or_end_conf,{EndFunc,Func}),      put(test_server_loc, {Mod,{EndFunc,Func}}), -    case catch my_apply(Mod, EndFunc, [Func,Conf]) of -	{'$test_server_ok',SaveCfg={save_config,_}} -> +    try Mod:EndFunc(Func, Conf) of +	{save_config,_}=SaveCfg ->  	    SaveCfg; -	{'$test_server_ok',{fail,_}=Fail} -> +	{fail,_}=Fail ->  	    Fail; -	{'$test_server_ok',_} -> -	    ok; -	{'EXIT',Reason} = Why -> +	_ -> +	    ok +    catch +	throw:Other ->  	    Comment0 = case read_comment() of  			   ""  -> ""; -			   Cmt -> Cmt ++ "<br>" +			   Cmt -> Cmt ++ test_server_ctrl:xhtml("<br>", +								"<br />")  		       end, +	    set_loc(erlang:get_stacktrace()),  	    comment(io_lib:format("~s<font color=\"red\">" -				  "WARNING: ~w crashed!" +				  "WARNING: ~w thrown!"  				  "</font>\n",[Comment0,EndFunc])),  	    group_leader() ! {printout,12, -			      "WARNING: ~w crashed!\n" +			      "WARNING: ~w thrown!\n"  			      "Reason: ~p\n"  			      "Line: ~s\n", -			      [EndFunc, Reason, +			      [EndFunc, Other,  			       test_server_sup:format_loc(  				 mod_loc(get_loc()))]}, -	    {failed,{Mod,end_per_testcase,Why}}; -	Other -> +	    {failed,{Mod,end_per_testcase,Other}}; +	  Class:Reason -> +	    Stk = erlang:get_stacktrace(), +	    set_loc(Stk), +	    Why = case Class of +		      exit -> {'EXIT',Reason}; +		      error -> {'EXIT',{Reason,Stk}} +		  end,  	    Comment0 = case read_comment() of  			   ""  -> ""; -			   Cmt -> Cmt ++ "<br>" +			   Cmt -> Cmt ++ test_server_ctrl:xhtml("<br>", +								"<br />")  		       end,  	    comment(io_lib:format("~s<font color=\"red\">" -				  "WARNING: ~w thrown!" +				  "WARNING: ~w crashed!"  				  "</font>\n",[Comment0,EndFunc])),  	    group_leader() ! {printout,12, -			      "WARNING: ~w thrown!\n" +			      "WARNING: ~w crashed!\n"  			      "Reason: ~p\n"  			      "Line: ~s\n", -			      [EndFunc, Other, +			      [EndFunc, Reason,  			       test_server_sup:format_loc(  				 mod_loc(get_loc()))]}, -	    {failed,{Mod,end_per_testcase,Other}} +	    {failed,{Mod,end_per_testcase,Why}}      end.  get_loc() -> -    case catch test_server_line:get_lines() of -	[] -> -	    get(test_server_loc); -	{'EXIT',_} -> -	    get(test_server_loc); -	Loc -> -	    Loc -    end. +    get(test_server_loc).  get_loc(Pid) -> -    {dictionary,Dict} = process_info(Pid, dictionary), -    lists:foreach(fun({Key,Val}) -> put(Key,Val) end,Dict), +    [{current_stacktrace,Stk0},{dictionary,Dict}] = +	process_info(Pid, [current_stacktrace,dictionary]), +    lists:foreach(fun({Key,Val}) -> put(Key, Val) end, Dict), +    Stk = [rewrite_loc_item(Loc) || Loc <- Stk0], +    case get(test_server_loc) of +	undefined -> +	    put(test_server_loc, Stk); +	{Suite,Case} -> +	    %% location info unknown, check if {Suite,Case,Line} +	    %% is available in stacktrace. and if so, use stacktrace +	    %% instead of currect test_server_loc +	    case [match || {S,C,_L} <- Stk, S == Suite, C == Case] of +		[match|_] -> put(test_server_loc, Stk); +		_         -> ok +	    end; +	_ -> +	    ok +    end,      get_loc(). -get_mf([{M,F,_}|_]) -> {M,F}; -get_mf([{M,F}|_])   -> {M,F}; -get_mf(_)           -> {undefined,undefined}. +%% find the latest known Suite:Testcase +get_mf(MFs) -> +    get_mf(MFs, {undefined,undefined}). + +get_mf([MF|MFs], _Found) when is_tuple(MF) -> +    ModFunc = {Mod,_} = case MF of +			    {M,F,_} -> {M,F}; +			    MF -> MF +			end, +    case is_suite(Mod) of +	true -> ModFunc; +	false -> get_mf(MFs, ModFunc) +    end; +get_mf(_, Found) -> +    Found. + +is_suite(Mod) -> +    case lists:reverse(atom_to_list(Mod)) of +	"ETIUS" ++ _ -> true; +	_ -> false +    end.  mod_loc(Loc) ->      %% handle diff line num versions      case Loc of  	[{{_M,_F},_L}|_] -> -	    [{?pl2a(M),F,L} || {{M,F},L} <- Loc]; +	    [begin if L /= 0 -> {?pl2a(M),F,L}; +		      true   -> {?pl2a(M),F} end end || {{M,F},L} <- Loc];  	[{_M,_F}|_] ->  	    [{?pl2a(M),F} || {M,F} <- Loc]; +	{{M,F},0} -> +	    [{?pl2a(M),F}];  	{{M,F},L} ->  	    [{?pl2a(M),F,L}];  	{M,ForL} ->  	    [{?pl2a(M),ForL}]; +	{M,F,0} -> +	    [{M,F}]; +	[{M,F,0}|Stack] -> +	    [{M,F}|Stack];  	_ ->  	    Loc      end. @@ -1544,7 +1824,7 @@ fw_error_notify(Mod, Func, Args, Error, Loc) ->  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% -%% print(Detail,Format,Args) -> ok +%% print(Detail,Format,Args,Printer) -> ok  %% Detail = integer()  %% Format = string()  %% Args = [term()] @@ -1555,6 +1835,9 @@ fw_error_notify(Mod, Func, Args, Error, Loc) ->  print(Detail,Format,Args) ->      local_or_remote_apply({test_server_ctrl,print,[Detail,Format,Args]}). +print(Detail,Format,Args,Printer) -> +    local_or_remote_apply({test_server_ctrl,print,[Detail,Format,Args,Printer]}). +  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%  %% print_timsteamp(Detail,Leader) -> ok  %% @@ -1587,16 +1870,22 @@ lookup_config(Key,Config) ->  %% timer:tc/3  ts_tc(M, F, A) ->      Before = erlang:now(), -    Val = (catch my_apply(M, F, A)), +    Result = try +		 apply(M, F, A) +	     catch +		 Type:Reason -> +		     Stk = erlang:get_stacktrace(), +		     set_loc(Stk), +		     case Type of +			 throw -> +			     {failed,{thrown,Reason}}; +			 error -> +			     {'EXIT',{Reason,Stk}}; +			 exit -> +			     {'EXIT',Reason} +		     end +	     end,      After = erlang:now(), -    Result = case Val of -		 {'$test_server_ok', R} -> -		     R; % test case ok -		 {'EXIT',_Reason} = R -> -		     R; % test case crashed -		 Other -> -		     {failed, {thrown,Other}} % test case was thrown -	  end,      Elapsed =  	(element(1,After)*1000000000000  	 +element(2,After)*1000000+element(3,After)) - @@ -1604,8 +1893,12 @@ ts_tc(M, F, A) ->  	 +element(2,Before)*1000000+element(3,Before)),      {Elapsed, Result}. -my_apply(M, F, A) -> -    {'$test_server_ok',apply(M, F, A)}. +set_loc(Stk) -> +    Loc = [rewrite_loc_item(I) || {_,_,_,_}=I <- Stk], +    put(test_server_loc, Loc). + +rewrite_loc_item({M,F,_,Loc}) -> +    {M,F,proplists:get_value(line, Loc, 0)}.  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% @@ -1706,6 +1999,13 @@ messages_get() ->      test_server_sup:messages_get([]).  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%% permit_io(GroupLeader, FromPid) -> ok +%% +%% Make sure proceeding IO from FromPid won't get rejected +permit_io(GroupLeader, FromPid) -> +    GroupLeader ! {permit_io,FromPid}. + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%  %% sleep(Time) -> ok  %% Time = integer() | float() | infinity  %% @@ -1768,7 +2068,16 @@ adjusted_sleep(MSecs) ->  %% to read when using this function, rather than exit directly.  fail(Reason) ->      comment(cast_to_list(Reason)), -    exit({suite_failed,Reason}). +    try +	exit({suite_failed,Reason}) +    catch +	Class:R -> +	    case erlang:get_stacktrace() of +		[{?MODULE,fail,1,_}|Stk] -> ok; +		Stk -> ok +	    end, +	    erlang:raise(Class, R, Stk) +    end.  cast_to_list(X) when is_list(X) -> X;  cast_to_list(X) when is_atom(X) -> atom_to_list(X); @@ -1782,7 +2091,16 @@ cast_to_list(X) -> lists:flatten(io_lib:format("~p", [X])).  %% Immediately calls exit. Included because test suites are easier  %% to read when using this function, rather than exit directly.  fail() -> -    exit(suite_failed). +    try +	exit(suite_failed) +    catch +	Class:R -> +	    case erlang:get_stacktrace() of +		[{?MODULE,fail,0,_}|Stk] -> ok; +		Stk -> ok +	    end, +	    erlang:raise(Class, R, Stk) +    end.  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%  %% break(Comment) -> ok @@ -1790,29 +2108,40 @@ fail() ->  %% Break a test case so part of the test can be done manually.  %% Use continue/0 to continue.  break(Comment) -> -    case erase(test_server_timetraps) of -	undefined -> ok; -	List -> lists:foreach(fun({Ref,_}) -> timetrap_cancel(Ref) end, List) -    end, +    break(?MODULE, Comment). + +break(CBM, Comment) -> +    break(CBM, '', Comment). + +break(CBM, TestCase, Comment) -> +    timetrap_cancel(), +    {TCName,CntArg,PName} = +	if TestCase == '' ->  +		{"", "", test_server_break_process}; +	   true -> +		Str = atom_to_list(TestCase), +		{[32 | Str], Str, +		 list_to_atom("test_server_break_process_" ++ Str)} +	end,      io:format(user,  	      "\n\n\n--- SEMIAUTOMATIC TESTING ---" -	      "\nThe test case executes on process ~w" +	      "\nThe test case~s executes on process ~w"  	      "\n\n\n~s"  	      "\n\n\n-----------------------------\n\n" -	      "Continue with --> test_server:continue().\n", -	      [self(),Comment]), -    case whereis(test_server_break_process) of +	      "Continue with --> ~w:continue(~s).\n", +	      [TCName,self(),Comment,CBM,CntArg]), +    case whereis(PName) of  	undefined -> -	    spawn_break_process(self()); +	    spawn_break_process(self(), PName);  	OldBreakProcess ->  	    OldBreakProcess ! cancel, -	    spawn_break_process(self()) +	    spawn_break_process(self(), PName)      end,      receive continue -> ok end. -spawn_break_process(Pid) -> +spawn_break_process(Pid, PName) ->      spawn(fun() -> -		  register(test_server_break_process,self()), +		  register(PName, self()),  		  receive  		      continue -> continue(Pid);  		      cancel -> ok @@ -1821,13 +2150,19 @@ spawn_break_process(Pid) ->  continue() ->      case whereis(test_server_break_process) of -	undefined -> -	     ok; -	BreakProcess -> -	    BreakProcess ! continue +	undefined    -> ok; +	BreakProcess -> BreakProcess ! continue      end. -continue(Pid) -> +continue(TestCase) when is_atom(TestCase) -> +    PName = list_to_atom("test_server_break_process_" ++ +			 atom_to_list(TestCase)), +    case whereis(PName) of +	undefined    -> ok; +	BreakProcess -> BreakProcess ! continue +    end;	 + +continue(Pid) when is_pid(Pid) ->      Pid ! continue.  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% @@ -1865,26 +2200,56 @@ timetrap_scale_factor() ->  %%  %% Creates a time trap, that will kill the calling process if the  %% trap is not cancelled with timetrap_cancel/1, within Timeout milliseconds. -timetrap(Timeout0) -> -    Timeout = time_ms(Timeout0), -    cancel_default_timetrap(), -    case get(test_server_multiply_timetraps) of -	undefined -> timetrap1(Timeout, true); -	{undefined,false} -> timetrap1(Timeout, false); -	{undefined,_} -> timetrap1(Timeout, true); -	{infinity,_} -> infinity; -	{_Int,_Scale} when Timeout == infinity -> infinity; -	{Int,Scale} -> timetrap1(Timeout*Int, Scale) -    end. +timetrap(Timeout) -> +    MultAndScale = +	case get(test_server_multiply_timetraps) of +	    undefined -> {fun(T) -> T end, true}; +	    {undefined,false} -> {fun(T) -> T end, false}; +	    {undefined,_} -> {fun(T) -> T end, true}; +	    {infinity,_} -> {fun(_) -> infinity end, false}; +	    {Int,Scale} -> {fun(infinity) -> infinity; +			       (T) -> T*Int end, Scale} +	end,	     +    timetrap(Timeout, Timeout, self(), MultAndScale). + +%% when the function is called from different process than +%% the test case, the test_server_multiply_timetraps data +%% is unknown and must be passed as argument +timetrap(Timeout, TCPid, MultAndScale) -> +    timetrap(Timeout, Timeout, TCPid, MultAndScale). + +timetrap(Timeout0, TimeToReport0, TCPid, MultAndScale = {Multiplier,Scale}) -> +    %% the time_ms call will either convert Timeout to ms or spawn a +    %% user timetrap which sends the result to the IO server process +    Timeout = time_ms(Timeout0, TCPid, MultAndScale), +    Timeout1 = Multiplier(Timeout), +    TimeToReport = if Timeout0 == TimeToReport0 -> +			   Timeout1; +		      true -> +			   %% only convert to ms, don't start a +			   %% user timetrap +			   time_ms_check(TimeToReport0) +		   end, +    cancel_default_timetrap(self() == TCPid), +    Handle = case Timeout1 of +		 infinity -> +		     infinity; +		 _ -> +		     spawn_link(test_server_sup,timetrap,[Timeout1,TimeToReport, +							  Scale,TCPid]) +	     end, + +    %% ERROR! This sets dict on IO process instead of testcase process +    %% if Timeout is return value from previous user timetrap!! -timetrap1(Timeout, Scale) -> -    TCPid = self(), -    Ref = spawn_link(test_server_sup,timetrap,[Timeout,Scale,TCPid]),      case get(test_server_timetraps) of -	undefined -> put(test_server_timetraps,[{Ref,TCPid}]); -	List -> put(test_server_timetraps,[{Ref,TCPid}|List]) +	undefined -> +	    put(test_server_timetraps,[{Handle,TCPid,{TimeToReport,Scale}}]); +	List -> +	    List1 = lists:delete({infinity,TCPid,{infinity,false}}, List), +	    put(test_server_timetraps,[{Handle,TCPid,{TimeToReport,Scale}}|List1])      end, -    Ref. +    Handle.  ensure_timetrap(Config) ->      case get(test_server_timetraps) of @@ -1909,7 +2274,10 @@ ensure_timetrap(Config) ->  	    put(test_server_default_timetrap, timetrap(seconds(DTmo)))      end. -cancel_default_timetrap() -> +%% executing on IO process, no default timetrap ever set here +cancel_default_timetrap(false) -> +    ok; +cancel_default_timetrap(true) ->      case get(test_server_default_timetrap) of  	undefined ->  	    ok; @@ -1927,75 +2295,175 @@ cancel_default_timetrap() ->  	    error      end. - -time_ms({hours,N}) -> hours(N); -time_ms({minutes,N}) -> minutes(N); -time_ms({seconds,N}) -> seconds(N); -time_ms({Other,_N}) -> +time_ms({hours,N}, _, _) -> hours(N); +time_ms({minutes,N}, _, _) -> minutes(N); +time_ms({seconds,N}, _, _) -> seconds(N); +time_ms({Other,_N}, _, _) ->      format("=== ERROR: Invalid time specification: ~p. "  	   "Should be seconds, minutes, or hours.~n", [Other]),      exit({invalid_time_format,Other}); -time_ms(Ms) when is_integer(Ms) -> Ms; -time_ms(infinity) -> infinity; -time_ms(Fun) when is_function(Fun) -> -    time_ms_apply(Fun); -time_ms({M,F,A}=MFA) when is_atom(M), is_atom(F), is_list(A) -> -    time_ms_apply(MFA); -time_ms(Other) -> exit({invalid_time_format,Other}). - -time_ms_apply(Func) -> -    time_ms_apply(Func, [5000,30000,60000,infinity]). - -time_ms_apply(Func, TOs) -> -    Apply = fun() -> -		    case Func of -			{M,F,A} -> -			    exit({self(),apply(M, F, A)}); -			Fun -> -			    exit({self(),Fun()}) -		    end -	    end, -    Pid = spawn(Apply), -    Ref = monitor(process, Pid), -    time_ms_wait(Func, Pid, Ref, TOs). - -time_ms_wait(Func, Pid, Ref, [TO|TOs]) -> -    receive -	{'DOWN',Ref,process,Pid,{Pid,Result}} -> -	    time_ms_check(Result); -	{'DOWN',Ref,process,Pid,Error} -> -	    exit({timetrap_error,Error}) -    after -	TO -> -	    format("=== WARNING: No return from timetrap function ~p~n", [Func]), -	    time_ms_wait(Func, Pid, Ref, TOs) -    end; -%% this clause will never execute if 'infinity' is in TOs list, that's ok! -time_ms_wait(Func, Pid, Ref, []) -> -    demonitor(Ref), -    exit(Pid, kill), -    exit({timetrap_error,{no_return_from_timetrap_function,Func}}). +time_ms(Ms, _, _) when is_integer(Ms) -> Ms; +time_ms(infinity, _, _) -> infinity; +time_ms(Fun, TCPid, MultAndScale) when is_function(Fun) -> +    time_ms_apply(Fun, TCPid, MultAndScale); +time_ms({M,F,A}=MFA, TCPid, MultAndScale) when is_atom(M), is_atom(F), is_list(A) -> +    time_ms_apply(MFA, TCPid, MultAndScale); +time_ms(Other, _, _) -> exit({invalid_time_format,Other}).  time_ms_check(MFA = {M,F,A}) when is_atom(M), is_atom(F), is_list(A) -> -    exit({invalid_time_format,MFA}); +    MFA;  time_ms_check(Fun) when is_function(Fun) -> -    exit({invalid_time_format,Fun}); +    Fun;  time_ms_check(Other) -> -    time_ms(Other). +    time_ms(Other, undefined, undefined). + +time_ms_apply(Func, TCPid, MultAndScale) -> +    {_,GL} = process_info(TCPid, group_leader), +    WhoAmI = self(),				% either TC or IO server +    T0 = now(), +    UserTTSup =  +	spawn(fun() ->  +		      user_timetrap_supervisor(Func, WhoAmI, TCPid, +					       GL, T0, MultAndScale) +	      end), +    receive +	{UserTTSup,infinity} -> +	    %% remember the user timetrap so that it can be cancelled +	    save_user_timetrap(TCPid, UserTTSup, T0), +	    %% we need to make sure the user timetrap function +	    %% gets time to execute and return +	    timetrap(infinity, TCPid, MultAndScale) +    after 5000 -> +	    exit(UserTTSup, kill), +	    if WhoAmI /= GL -> +		    exit({user_timetrap_error,time_ms_apply}); +	       true -> +		    format("=== ERROR: User timetrap execution failed!", []), +		    ignore +	    end +    end. + +user_timetrap_supervisor(Func, Spawner, TCPid, GL, T0, MultAndScale) -> +    process_flag(trap_exit, true), +    Spawner ! {self(),infinity}, +    MonRef = monitor(process, TCPid), +    UserTTSup = self(), +    group_leader(GL, UserTTSup), +    UserTT = spawn_link(fun() -> call_user_timetrap(Func, UserTTSup) end), +    receive +	{UserTT,Result} -> +	    demonitor(MonRef, [flush]), +	    Elapsed = trunc(timer:now_diff(now(), T0) / 1000), +	    try time_ms_check(Result) of +		TimeVal -> +		    %% this is the new timetrap value to set (return value +		    %% from a fun or an MFA) +		    GL ! {user_timetrap,TCPid,TimeVal,T0,Elapsed,MultAndScale} +	    catch _:_ ->		     +		    %% when other than a legal timetrap value is returned +		    %% which will be the normal case for user timetraps +		    GL ! {user_timetrap,TCPid,0,T0,Elapsed,MultAndScale} +	    end; +	{'EXIT',UserTT,Error} when Error /= normal -> +	    demonitor(MonRef, [flush]), +	    GL ! {user_timetrap,TCPid,0,T0,{user_timetrap_error,Error}, +		  MultAndScale}; +	{'DOWN',MonRef,_,_,_} -> +	    demonitor(MonRef, [flush]), +	    exit(UserTT, kill) +    end. + +call_user_timetrap(Func, Sup) when is_function(Func) -> +    try Func() of +	Result ->  +	    Sup ! {self(),Result} +    catch _:Error -> +	    exit({Error,erlang:get_stacktrace()}) +    end; +call_user_timetrap({M,F,A}, Sup) -> +    try apply(M,F,A) of +	Result ->  +	    Sup ! {self(),Result} +    catch _:Error -> +	    exit({Error,erlang:get_stacktrace()}) +    end. + +save_user_timetrap(TCPid, UserTTSup, StartTime) -> +    %% save pid of user timetrap supervisor process so that +    %% it may be stopped even before the timetrap func has returned +    NewUserTT = {TCPid,{UserTTSup,StartTime}}, +    case get(test_server_user_timetrap) of +	undefined -> +	    put(test_server_user_timetrap, [NewUserTT]); +	UserTTSups -> +	    case proplists:get_value(TCPid, UserTTSups) of +		undefined -> +		    put(test_server_user_timetrap, +			[NewUserTT | UserTTSups]); +		PrevTTSup -> +		    %% remove prev user timetrap +		    remove_user_timetrap(PrevTTSup), +		    put(test_server_user_timetrap, +			[NewUserTT | proplists:delete(TCPid, +						      UserTTSups)]) +	    end +    end. +     +update_user_timetraps(TCPid, StartTime) -> +    %% called when a user timetrap is triggered +    case get(test_server_user_timetrap) of +	undefined -> +	    proceed; +	UserTTs -> +	    case proplists:get_value(TCPid, UserTTs) of +		{_UserTTSup,StartTime} ->	% same timetrap +		    put(test_server_user_timetrap, +			proplists:delete(TCPid, UserTTs)), +		    proceed; +		{OtherUserTTSup,OtherStartTime} -> +		    case timer:now_diff(OtherStartTime, StartTime) of +			Diff when Diff >= 0 -> +			    ignore; +			_ -> +			    exit(OtherUserTTSup, kill), +			    put(test_server_user_timetrap, +				proplists:delete(TCPid, UserTTs)), +			    proceed +		    end; +		undefined -> +		    proceed +	    end +    end. + +remove_user_timetrap(TTSup) -> +    exit(TTSup, kill).  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%  %% timetrap_cancel(Handle) -> ok  %% Handle = term()  %%  %% Cancels a time trap. -timetrap_cancel(infinity) -> -    ok;  timetrap_cancel(Handle) -> +    timetrap_cancel_one(Handle, true). + +timetrap_cancel_one(infinity, _SendToServer) -> +    ok; +timetrap_cancel_one(Handle, SendToServer) ->      case get(test_server_timetraps) of -	undefined -> ok; -	[{Handle,_}] -> erase(test_server_timetraps); -	Timers -> put(test_server_timetraps, -		      lists:keydelete(Handle, 1, Timers)) +	undefined -> +	    ok; +	[{Handle,_,_}] -> +	    erase(test_server_timetraps); +	Timers -> +	    case lists:keysearch(Handle, 1, Timers) of +		{value,_} -> +		    put(test_server_timetraps, +			lists:keydelete(Handle, 1, Timers)); +		false when SendToServer == true -> +		    group_leader() ! {timetrap_cancel_one,Handle,self()}; +		false -> +		    ok +	    end      end,      test_server_sup:timetrap_cancel(Handle). @@ -2004,16 +2472,61 @@ timetrap_cancel(Handle) ->  %%  %% Cancels timetrap for current test case.  timetrap_cancel() -> +    timetrap_cancel_all(self(), true). + +timetrap_cancel_all(TCPid, SendToServer) ->      case get(test_server_timetraps) of  	undefined ->  	    ok;  	Timers -> -	    case lists:keysearch(self(), 2, Timers) of -		{value,{Handle,_}} -> -		    timetrap_cancel(Handle); -		_ -> +	    [timetrap_cancel_one(Handle, false) || +		{Handle,Pid,_} <- Timers, Pid == TCPid] +    end, +    case get(test_server_user_timetrap) of +	undefined -> +	    ok; +	UserTTs -> +	    case proplists:get_value(TCPid, UserTTs) of +		{UserTTSup,_StartTime} -> +		    remove_user_timetrap(UserTTSup), +		    put(test_server_user_timetrap, +			proplists:delete(TCPid, UserTTs)); +		undefined ->  		    ok  	    end +    end, +    if SendToServer == true -> +	    group_leader() ! {timetrap_cancel_all,TCPid,self()}; +       true -> +	    ok +    end, +    ok. + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%% get_timetrap_info() -> {Timeout,Scale} | undefined +%% +%% Read timetrap info for current test case +get_timetrap_info() -> +    get_timetrap_info(self(), true). + +get_timetrap_info(TCPid, SendToServer) -> +    case get(test_server_timetraps) of +	undefined -> +	    undefined; +	Timers -> +	    case [Info || {Handle,Pid,Info} <- Timers,  +			  Pid == TCPid, Handle /= infinity] of +		[I|_] -> +		    I; +		[] when SendToServer == true -> +		    MsgLooper = group_leader(), +		    MsgLooper ! {get_timetrap_info,TCPid,self()}, +		    receive +			{MsgLooper,get_timetrap_info,I} -> I +		    end; +		[] -> +		    undefined +	    end      end.  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% @@ -2437,11 +2950,23 @@ read_comment() ->      MsgLooper = group_leader(),      MsgLooper ! {read_comment,self()},      receive -	{MsgLooper,read_comment,Comment} -> -	    Comment +	{MsgLooper,read_comment,Comment} -> Comment +    after +	5000 -> "" +    end. + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%% make_priv_dir() -> ok +%% +%% Order test server to create the private directory +%% for the current test case. +make_priv_dir() -> +    MsgLooper = group_leader(), +    group_leader() ! {make_priv_dir,self()}, +    receive +	{MsgLooper,make_priv_dir,Result} -> Result      after -	5000 -> -	    "" +	5000 -> error      end.  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% diff --git a/lib/test_server/src/test_server_ctrl.erl b/lib/test_server/src/test_server_ctrl.erl index 4fad86d16d..df2187bc04 100644 --- a/lib/test_server/src/test_server_ctrl.erl +++ b/lib/test_server/src/test_server_ctrl.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2002-2011. All Rights Reserved. +%% Copyright Ericsson AB 2002-2012. All Rights Reserved.  %%  %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -162,8 +162,9 @@  -export([jobs/0, run_test/1, wait_finish/0, idle_notify/1,  	 abort_current_testcase/1, abort/0]).  -export([start_get_totals/1, stop_get_totals/0]). --export([get_levels/0, set_levels/3]). +-export([reject_io_reqs/1, get_levels/0, set_levels/3]).  -export([multiply_timetraps/1, scale_timetraps/1, get_timetrap_parameters/0]). +-export([create_priv_dir/1]).  -export([cover/2, cover/3, cover/7,  	 cross_cover_analyse/1, cross_cover_analyse/2, trc/1, stop_trace/0]).  -export([testcase_callback/1]). @@ -171,7 +172,7 @@  -export([kill_slavenodes/0]).  %%% TEST_SERVER INTERFACE %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% --export([output/2, print/2, print/3, print_timestamp/2]). +-export([output/2, print/2, print/3, print/4, print_timestamp/2]).  -export([start_node/3, stop_node/1, wait_for_node/1, is_release_available/1]).  -export([format/1, format/2, format/3, to_string/1]).  -export([get_target_info/0]). @@ -187,6 +188,7 @@  -export([handle_call/3, handle_cast/2, handle_info/2]).  -export([do_test_cases/4]).  -export([do_spec/2, do_spec_list/2]). +-export([xhtml/2]).  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% @@ -214,9 +216,13 @@  			  X == auto_skip -> skipped;  			  true -> X end). --record(state,{jobs=[],levels={1,19,10}, -	       multiply_timetraps=1,scale_timetraps=true, -	       finish=false, +-define(auto_skip_color, "#FFA64D"). +-define(user_skip_color, "#FF8000"). +-define(sortable_table_name, "SortableTable"). + +-record(state,{jobs=[], levels={1,19,10}, reject_io_reqs=false, +	       multiply_timetraps=1, scale_timetraps=true, +	       create_priv_dir=auto_per_run, finish=false,  	       target_info, trc=false, cover=false, wait_for_node=[],  	       testcase_callback=undefined, idle_notify=[],  	       get_totals=false, random_seed=undefined}). @@ -493,6 +499,9 @@ get_levels() ->  set_levels(Show, Major, Minor) ->      controller_call({set_levels,Show,Major,Minor}). +reject_io_reqs(Bool) -> +    controller_call({reject_io_reqs,Bool}). +  multiply_timetraps(N) ->      controller_call({multiply_timetraps,N}). @@ -502,6 +511,9 @@ scale_timetraps(Bool) ->  get_timetrap_parameters() ->      controller_call(get_timetrap_parameters). +create_priv_dir(Value) -> +    controller_call({create_priv_dir,Value}). +  trc(TraceFile) ->      controller_call({trace,TraceFile}, 2*?ACCEPT_TIMEOUT). @@ -642,8 +654,8 @@ init([Param]) ->  contact_main_target(local) ->      %% When used by a general framework, global registration of      %% test_server should not be required. -    case os:getenv("TEST_SERVER_FRAMEWORK") of -	FW when FW =:= false; FW =:= "undefined" -> +    case get_fw_mod(undefined) of +	undefined ->  	    %% Local target! The global test_server process implemented by  	    %% test_server.erl will not be started, so we simulate it by  	    %% globally registering this process instead. @@ -807,6 +819,8 @@ handle_call({add_job,Dir,Name,TopCase,Skip}, _From, State) ->  			    [SpecName,{State#state.multiply_timetraps,  				       State#state.scale_timetraps}],  			    LogDir, Name, State#state.levels, +			    State#state.reject_io_reqs, +			    State#state.create_priv_dir,  			    State#state.testcase_callback, ExtraTools1),  		    NewJobs = [{Name,Pid}|State#state.jobs],  		    {reply, ok, State#state{jobs=NewJobs}}; @@ -816,6 +830,8 @@ handle_call({add_job,Dir,Name,TopCase,Skip}, _From, State) ->  			    [SpecList,{State#state.multiply_timetraps,  				       State#state.scale_timetraps}],  			    LogDir, Name, State#state.levels, +			    State#state.reject_io_reqs, +			    State#state.create_priv_dir,  			    State#state.testcase_callback, ExtraTools1),  		    NewJobs = [{Name,Pid}|State#state.jobs],  		    {reply, ok, State#state{jobs=NewJobs}}; @@ -833,6 +849,8 @@ handle_call({add_job,Dir,Name,TopCase,Skip}, _From, State) ->  				     {State#state.multiply_timetraps,  				      State#state.scale_timetraps}],  				    LogDir, Name, State#state.levels, +				    State#state.reject_io_reqs, +				    State#state.create_priv_dir,  				    State#state.testcase_callback, ExtraTools1),  			    NewJobs = [{Name,Pid}|State#state.jobs],  			    {reply, ok, State#state{jobs=NewJobs}} @@ -964,6 +982,15 @@ handle_call({set_levels,Show,Major,Minor}, _From, State) ->      {reply,ok,State#state{levels={Show,Major,Minor}}};  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%% handle_call({reject_io_reqs,Bool}, _, State) -> ok +%% Bool = bool() +%% +%% May be used to switch off stdout printouts to the minor log file + +handle_call({reject_io_reqs,Bool}, _From, State) -> +    {reply,ok,State#state{reject_io_reqs=Bool}}; + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%  %% handle_call({multiply_timetraps,N}, _, State) -> ok  %% N = integer() | infinity  %% @@ -1041,6 +1068,18 @@ handle_call({cover,App,Analyse}, _From, State) ->      {reply,ok,State#state{cover={App,Analyse}}};  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%% handle_call({create_priv_dir,Value}, _, State) -> ok | {error,Reason} +%% +%% Set create_priv_dir to either auto_per_run (create common priv dir once +%% per test run), manual_per_tc (the priv dir name will be unique for each +%% test case, but the user has to call test_server:make_priv_dir/0 to create +%% it), or auto_per_tc (unique priv dir created automatically for each test +%% case). + +handle_call({create_priv_dir,Value}, _From, State) -> +    {reply,ok,State#state{create_priv_dir=Value}}; + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%  %% handle_call({testcase_callback,{Mod,Func}}, _, State) -> ok | {error,Reason}  %%  %% Add a callback function that will be called before and after every @@ -1297,7 +1336,12 @@ terminate(_Reason, State) ->      end,      kill_all_jobs(State#state.jobs),      test_server_node:stop(State#state.target_info), -    test_server_h:restore(), +    case lists:keysearch(sasl, 1, application:which_applications()) of +	{value,_} -> +	    test_server_h:restore(); +	_ -> +	    ok +    end,      ok.  kill_all_jobs([{_Name,JobPid}|Jobs]) -> @@ -1312,14 +1356,16 @@ kill_all_jobs([]) ->  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% -%% spawn_tester(Mod, Func, Args, Dir, Name, Levels, -%%              TestCaseCallback, ExtraTools) -> Pid +%% spawn_tester(Mod, Func, Args, Dir, Name, Levels, RejectIoReqs, +%%              CreatePrivDir, TestCaseCallback, ExtraTools) -> Pid  %% Mod = atom()  %% Func = atom()  %% Args = [term(),...]  %% Dir = string()  %% Name = string()  %% Levels = {integer(),integer(),integer()} +%% RejectIoReqs = bool() +%% CreatePrivDir = auto_per_run | manual_per_tc | auto_per_tc  %% TestCaseCallback = {CBMod,CBFunc} | undefined  %% ExtraTools = [ExtraTool,...]  %% ExtraTool = CoverInfo | TraceInfo | RandomSeed @@ -1330,14 +1376,15 @@ kill_all_jobs([]) ->  %% When the named function is done executing, a summary of the results  %% is printed to the log files. -spawn_tester(Mod, Func, Args, Dir, Name, Levels, TCCallback, ExtraTools) -> +spawn_tester(Mod, Func, Args, Dir, Name, Levels, RejectIoReqs, +	     CreatePrivDir, TCCallback, ExtraTools) ->      spawn_link( -      fun() -> init_tester(Mod, Func, Args, Dir, Name, Levels, -			   TCCallback, ExtraTools) +      fun() -> init_tester(Mod, Func, Args, Dir, Name, Levels, RejectIoReqs, +			   CreatePrivDir, TCCallback, ExtraTools)        end). -init_tester(Mod, Func, Args, Dir, Name, {SumLev,MajLev,MinLev}, -	    TCCallback, ExtraTools) -> +init_tester(Mod, Func, Args, Dir, Name, {SumLev,MajLev,MinLev}, RejectIoReqs, +	    CreatePrivDir, TCCallback, ExtraTools) ->      process_flag(trap_exit, true),      put(test_server_name, Name),      put(test_server_dir, Dir), @@ -1348,8 +1395,22 @@ init_tester(Mod, Func, Args, Dir, Name, {SumLev,MajLev,MinLev},      put(test_server_summary_level, SumLev),      put(test_server_major_level, MajLev),      put(test_server_minor_level, MinLev), +    put(test_server_reject_io_reqs, RejectIoReqs), +    put(test_server_create_priv_dir, CreatePrivDir),      put(test_server_random_seed, proplists:get_value(random_seed, ExtraTools)),      put(test_server_testcase_callback, TCCallback), +    case os:getenv("TEST_SERVER_FRAMEWORK") of +	FW when FW =:= false; FW =:= "undefined" -> +	    put(test_server_framework, '$none');	 +	FW -> +	    put(test_server_framework_name, list_to_atom(FW)), +	    case os:getenv("TEST_SERVER_FRAMEWORK_NAME") of +		FWName when FWName =:= false; FWName =:= "undefined" -> +		    put(test_server_framework_name, '$none');	 +		FWName -> +		    put(test_server_framework_name, list_to_atom(FWName)) +	    end +    end,      %% before first print, read and set logging options      LogOpts = test_server_sup:framework_call(get_logopts, [], []),      put(test_server_logopts, LogOpts), @@ -1381,8 +1442,10 @@ init_tester(Mod, Func, Args, Dir, Name, {SumLev,MajLev,MinLev},  	end,      OkN = get(test_server_ok),      FailedN = get(test_server_failed), -    print(html,"<tr><td></td><td><b>TOTAL</b></td><td></td><td></td>" -	  "<td>~.3fs</td><td><b>~s</b></td><td>~p Ok, ~p Failed~s of ~p</td></tr>\n", +    print(html,"\n</tbody>\n<tfoot>\n" +	  "<tr><td></td><td><b>TOTAL</b></td><td></td><td></td><td></td>" +	  "<td>~.3fs</td><td><b>~s</b></td><td>~p Ok, ~p Failed~s of ~p</td></tr>\n" +	  "</tfoot>\n",  	  [Time,SuccessStr,OkN,FailedN,SkipStr,OkN+FailedN+SkippedN]).  %% timer:tc/3 @@ -1443,7 +1506,7 @@ stop_extra_tools([], _) ->  %% Reads the named test suite specification file, and executes it.  %%  %% This function is meant to be called by a process created by -%% spawn_tester/7, which sets up some necessary dictionary values. +%% spawn_tester/10, which sets up some necessary dictionary values.  do_spec(SpecName, TimetrapSpec) when is_list(SpecName) ->      case file:consult(SpecName) of @@ -1492,7 +1555,7 @@ do_spec(SpecName, TimetrapSpec) when is_list(SpecName) ->  %% should not be used. Use a configuration test case instead.  %%  %% This function is meant to be called by a process created by -%% spawn_tester/7, which sets up some necessary dictionary values. +%% spawn_tester/10, which sets up some necessary dictionary values.  do_spec_list(TermList0, TimetrapSpec) ->      Nodes = [], @@ -1659,7 +1722,7 @@ add_mod(Mod, Mods) ->  %% configuration information into the log files.  %%  %% This function is meant to be called by a process created by -%% spawn_tester/7, which sets up some necessary dictionary values. +%% spawn_tester/10, which sets up some necessary dictionary values.  do_test_cases(TopCases, SkipCases,  	      Config, MultiplyTimetrap) when is_integer(MultiplyTimetrap);  					     MultiplyTimetrap == infinity -> @@ -1668,12 +1731,8 @@ do_test_cases(TopCases, SkipCases,  do_test_cases(TopCases, SkipCases,  	      Config, TimetrapData) when is_list(TopCases),  					 is_tuple(TimetrapData) -> -    start_log_file(), -    FwMod = -	case os:getenv("TEST_SERVER_FRAMEWORK") of -	    FW when FW =:= false; FW =:= "undefined" -> ?MODULE; -	    FW -> list_to_atom(FW) -	end, +    {ok,TestDir} = start_log_file(), +    FwMod = get_fw_mod(?MODULE),      case collect_all_cases(TopCases, SkipCases) of  	{error,Why} ->  	    print(1, "Error starting: ~p", [Why]), @@ -1692,60 +1751,89 @@ do_test_cases(TopCases, SkipCases,  		  [print_if_known(N, {", ~w test cases",[N]},  				  {" (with repeated test cases)",[]})]),  	    Test = get(test_server_name), -	    test_server_sup:framework_call(report, [tests_start,{Test,N}]), +	    TestName = 	if is_list(Test) ->  +				lists:flatten(io_lib:format("~s", [Test])); +			   true -> +				lists:flatten(io_lib:format("~p", [Test])) +			end,			   +	    TestDescr = "Test " ++ TestName ++ " results", -	    Header = -		case test_server_sup:framework_call(overview_html_header, [Test], "") of -		    ""  -> -			TestName = lists:flatten(io_lib:format("~p", [Test])), -			["<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 3.2 Final//EN\">\n", -			 "<!-- autogenerated by '", atom_to_list(?MODULE), "'. -->\n", -			 "<html>\n", -			 "<head><title>Test ", TestName, " results</title>\n", -			 "<meta http-equiv=\"cache-control\" content=\"no-cache\">\n", -			 "</head>\n", -			 "<body bgcolor=\"white\" text=\"black\" ", -			 "link=\"blue\" vlink=\"purple\" alink=\"red\">", -			 "<h2>Results from test ", TestName, "</h2>\n"]; -		    Html -> -			["<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 3.2 Final//EN\">\n", -			 "<!-- autogenerated by '", atom_to_list(?MODULE), "'. -->\n" | Html] +	    test_server_sup:framework_call(report, [tests_start,{Test,N}]), +	    {Header,Footer} = +		case test_server_sup:framework_call(get_html_wrapper,  +						    [TestDescr,true,TestDir, +						    {[],[2,3,4,7,8],[1,6]}], "") of +		    Empty when (Empty == "") ; (element(2,Empty) == "")  -> +			put(basic_html, true), +			{["<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 3.2 Final//EN\">\n", +			  "<!-- autogenerated by '", atom_to_list(?MODULE), "'. -->\n", +			  "<html>\n", +			  "<head><title>", TestDescr, "</title>\n", +			  "<meta http-equiv=\"cache-control\" content=\"no-cache\">\n", +			  "</head>\n", +			  "<body bgcolor=\"white\" text=\"black\" ", +			  "link=\"blue\" vlink=\"purple\" alink=\"red\">", +			  "<h2>Results for test ", TestName, "</h2>\n"], +			 "\n</body>\n</html>\n"}; +		    {basic_html,Html0,Html1} -> +			put(basic_html, true), +			{Html0++["<h1>Results for <i>",TestName,"</i></h1>\n"], +			 Html1}; +		    {xhtml,Html0,Html1} -> +			put(basic_html, false), +			{Html0++["<h1>Results for <i>",TestName,"</i></h1>\n"], +			 Html1}  		end, -	    print(html, Header, []), -	    print_timestamp(html, "Test started at "), -	    print(html, "<p>Host:<br>\n"), +	    print(html, Header), + +	    print(html, xhtml("<p>", "<h4>")), +	    print_timestamp(html, "Test started at "), +	    print(html, xhtml("</p>", "</h4>")), +	     +	    print(html, xhtml("\n<p><b>Host info:</b><br>\n", +			      "\n<p><b>Host info:</b><br />\n")),  	    print_who(test_server_sup:hoststr(), test_server_sup:get_username()), -	    print(html, "<br>Used Erlang ~s in <tt>~s</tt>.\n", +	    print(html, xhtml("<br>Used Erlang v~s in <tt>~s</tt></p>\n", +			      "<br />Used Erlang v~s in \"~s\"</p>\n"),  		  [erlang:system_info(version), code:root_dir()]), - +	      	    if FwMod == ?MODULE -> -		    print(html, "<p>Target:<br>\n"), +		    print(html, xhtml("\n<p><b>Target Info:</b><br>\n", +				      "\n<p><b>Target Info:</b><br />\n")),  		    print_who(TI#target_info.host, TI#target_info.username), -		    print(html, "<br>Used Erlang ~s in <tt>~s</tt>.\n", +		    print(html, xhtml("<br>Used Erlang v~s in <tt>~s</tt></p>\n", +				      "<br />Used Erlang v~s in \"~s\"</p>\n"),  			  [TI#target_info.version, TI#target_info.root_dir]); -		true -> +	       true ->  		    case test_server_sup:framework_call(target_info, []) of  			TargetInfo when is_list(TargetInfo),  			                length(TargetInfo) > 0 -> -			    print(html, "<p>Target:<br>\n"), -			    print(html, "~s\n", [TargetInfo]); +			    print(html, xhtml("\n<p><b>Target info:</b><br>\n", +					      "\n<p><b>Target info:</b><br />\n")), +			    print(html, "~s</p>\n", [TargetInfo]);  			_ ->  			    ok  		    end  	    end, - +	      	    print(html, -		  "<p><a href=\"~s\">Full textual log</a>\n" -		  "<br><a href=\"~s\">Coverage log</a>\n", +		  "<p><ul>\n" +		  "<li><a href=\"~s\">Full textual log</a></li>\n" +		  "<li><a href=\"~s\">Coverage log</a></li>\n</ul></p>\n",  		  [?suitelog_name,?coverlog_name]), -	    print(html,"<p>~s" -		  "<p>\n" -		  "<table bgcolor=\"white\" border=\"3\" cellpadding=\"5\">" -		  "<tr><th>Num</th><th>Module</th><th>Case</th><th>Log</th>" -		  "<th>Time</th><th>Result</th><th>Comment</th></tr>\n", -		  [print_if_known(N, {"Suite contains ~p test cases.\n",[N]}, +	    print(html, +		  "<p>~s</p>\n" ++ +		  xhtml("<table bgcolor=\"white\" border=\"3\" cellpadding=\"5\">", +			["<table id=\"",?sortable_table_name,"\">\n", +			 "<thead>\n"]) ++ +		      "<tr><th>Num</th><th>Module</th><th>Group</th>" ++ +		      "<th>Case</th><th>Log</th><th>Time</th><th>Result</th>" ++ +		      "<th>Comment</th></tr>\n</thead>\n<tbody>\n", +		  [print_if_known(N, {"<i>Executing <b>~p</b> test cases...</i>" ++ +				      xhtml("\n<br>\n", "\n<br />\n"),[N]},  				  {"",[]})]), +  	    print(major, "=cases         ~p", [get(test_server_cases)]),  	    print(major, "=user          ~s", [TI#target_info.username]),  	    print(major, "=host          ~s", [TI#target_info.host]), @@ -1764,6 +1852,9 @@ do_test_cases(TopCases, SkipCases,  	    print(major, "=otp_release   ~s", [TI#target_info.otp_release]),  	    print(major, "=started       ~s",  		   [lists:flatten(timestamp_get(""))]), + +	    put(test_server_html_footer, Footer), +  	    run_test_cases(TestSpec, Config, TimetrapData)      end; @@ -1773,13 +1864,13 @@ do_test_cases(TopCase, SkipCases, Config, TimetrapSpec) ->  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% -%% start_log_file() -> ok | exit({Error,Reason}) +%% start_log_file() -> {ok,TestDirName} | exit({Error,Reason})  %% Stem = string()  %%  %% Creates the log directories, the major log file and the html log file.  %% The log files are initialized with some header information.  %% -%% The name of the log directory will be <Name>.LOGS/run.<Date>/ where +%% The name of the log directory will be <Name>.logs/run.<Date>/ where  %% Name is the test suite name and Date is the current date and time.  start_log_file() -> @@ -1793,18 +1884,27 @@ start_log_file() ->  	    exit({cant_create_log_dir,{MkDirError,Dir}})      end,      TestDir = timestamp_filename_get(filename:join(Dir, "run.")), -    case file:make_dir(TestDir) of -	ok -> -	    ok; -	MkDirError2 -> -	    exit({cant_create_log_dir,{MkDirError2,TestDir}}) -    end, - -    ok = file:write_file(filename:join(Dir, ?last_file), TestDir ++ "\n"), -    ok = file:write_file(?last_file, TestDir ++ "\n"), - -    put(test_server_log_dir_base,TestDir), -    MajorName = filename:join(TestDir, ?suitelog_name), +    TestDir1 = +	case file:make_dir(TestDir) of +	    ok -> +		TestDir; +	    {error,eexist} -> +		timer:sleep(1000), +		%% we need min 1 second between timestamps unfortunately +		TestDirX = timestamp_filename_get(filename:join(Dir, "run.")), +		case file:make_dir(TestDirX) of +		    ok -> +			TestDirX; +		    MkDirError2 -> +			exit({cant_create_log_dir,{MkDirError2,TestDirX}}) +		end; +	    MkDirError2 -> +		exit({cant_create_log_dir,{MkDirError2,TestDir}}) +	end, +    ok = file:write_file(filename:join(Dir, ?last_file), TestDir1 ++ "\n"), +    ok = file:write_file(?last_file, TestDir1 ++ "\n"), +    put(test_server_log_dir_base,TestDir1), +    MajorName = filename:join(TestDir1, ?suitelog_name),      HtmlName = MajorName ++ ?html_ext,      {ok,Major} = file:open(MajorName, [write]),      {ok,Html}  = file:open(HtmlName,  [write]), @@ -1817,14 +1917,14 @@ start_log_file() ->      make_html_link(LinkName ++ ?html_ext, HtmlName,  		   filename:basename(Dir)), -    PrivDir = filename:join(TestDir, ?priv_dir), +    PrivDir = filename:join(TestDir1, ?priv_dir),      ok = file:make_dir(PrivDir),      put(test_server_priv_dir,PrivDir++"/"),      print_timestamp(13,"Suite started at "), -    LogInfo = [{topdir,Dir},{rundir,lists:flatten(TestDir)}], +    LogInfo = [{topdir,Dir},{rundir,lists:flatten(TestDir1)}],      test_server_sup:framework_call(report, [loginfo,LogInfo]), -    ok. +    {ok,TestDir1}.  make_html_link(LinkName, Target, Explanation) ->      %% if possible use a relative reference to Target. @@ -1881,16 +1981,33 @@ start_minor_log_file1(Mod, Func, LogDir, AbsName) ->      {ok,Fd} = file:open(AbsName, [write]),      Lev = get(test_server_minor_level)+1000, %% far down in the minor levels      put(test_server_minor_fd, Fd), -    io:fwrite(Fd, -	      "<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 3.2 Final//EN\">\n" -	      "<!-- autogenerated by '"++atom_to_list(?MODULE)++"'. -->\n" -	      "<html>\n" -	      "<head><title>"++cast_to_list(Mod)++"</title>\n" -	      "<meta http-equiv=\"cache-control\" content=\"no-cache\">\n" -	      "</head>\n" -	      "<body bgcolor=\"white\" text=\"black\"" -	      " link=\"blue\" vlink=\"purple\" alink=\"red\">\n", -	      []), +     +    TestDescr = io_lib:format("Test ~p:~p result", [Mod,Func]), +    {Header,Footer} = +	case test_server_sup:framework_call(get_html_wrapper,  +					    [TestDescr,false, +					     filename:dirname(AbsName), +					     undefined], "") of +	    Empty when (Empty == "") ; (element(2,Empty) == "")  -> +		put(basic_html, true), +		{["<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 3.2 Final//EN\">\n", +		 "<!-- autogenerated by '", atom_to_list(?MODULE), "'. -->\n", +		 "<html>\n", +		 "<head><title>", TestDescr, "</title>\n", +		 "<meta http-equiv=\"cache-control\" content=\"no-cache\">\n", +		 "</head>\n", +		 "<body bgcolor=\"white\" text=\"black\" ", +		 "link=\"blue\" vlink=\"purple\" alink=\"red\">\n"], +		 "\n</body>\n</html>\n"}; +	    {basic_html,Html0,Html1} -> +		put(basic_html, true), +		{Html0,Html1}; +	    {xhtml,Html0,Html1} -> +		put(basic_html, false), +		{Html0,Html1} +	end, +    put(test_server_minor_footer, Footer), +    io:fwrite(Fd, Header, []),      SrcListing = downcase(cast_to_list(Mod)) ++ ?src_listing_ext,      case {filelib:is_file(filename:join(LogDir, SrcListing)), @@ -1913,7 +2030,8 @@ start_minor_log_file1(Mod, Func, LogDir, AbsName) ->  stop_minor_log_file() ->      Fd = get(test_server_minor_fd), -    io:fwrite(Fd, "</pre>\n</body>\n</html>\n", []), +    Footer = get(test_server_minor_footer), +    io:fwrite(Fd, "</pre>\n" ++ Footer, []),      file:close(Fd),      put(test_server_minor_fd, undefined). @@ -1992,12 +2110,29 @@ html_convert_modules([]) -> ok.  %% Convert source code to HTML if possible and needed.  html_possibly_convert(Src, SrcInfo, Dest) ->      case file:read_file_info(Dest) of -	{error,_Reason} ->			% no dest file -	    erl2html2:convert(Src, Dest); -	{ok,DestInfo} when DestInfo#file_info.mtime < SrcInfo#file_info.mtime -> -	    erl2html2:convert(Src, Dest); -	{ok,_DestInfo} -> -	    ok					% dest file up to date +	{ok,DestInfo} when DestInfo#file_info.mtime >= SrcInfo#file_info.mtime -> +	    ok;					% dest file up to date +	_ -> +	    OutDir = get(test_server_log_dir_base), +	    Header = +		case test_server_sup:framework_call(get_html_wrapper,  +						    ["Module "++Src,false, +						     OutDir,undefined], "") of +		    Empty when (Empty == "") ; (element(2,Empty) == "")  -> +			["<!DOCTYPE HTML PUBLIC", +			 "\"-//W3C//DTD HTML 3.2 Final//EN\">\n", +			 "<!-- autogenerated by 'erl2html2' -->\n", +			 "<html>\n", +			 "<head><title>Module ", Src, "</title>\n", +			 "<meta http-equiv=\"cache-control\" ", +			 "content=\"no-cache\">\n", +			 "</head>\n", +			 "<body bgcolor=\"white\" text=\"black\" ", +			 "link=\"blue\" vlink=\"purple\" alink=\"red\">\n"]; +		    {_,Html,_} -> +			Html +		end, +	    erl2html2:convert(Src, Dest, Header)      end.  %% Copy all HTML files in InDir to OutDir. @@ -2025,17 +2160,17 @@ add_init_and_end_per_suite([{make,_,_}=Case|Cases], LastMod, LastRef, FwMod) ->  add_init_and_end_per_suite([{skip_case,{{Mod,all},_}}=Case|Cases], LastMod,  			   LastRef, FwMod) when Mod =/= LastMod ->      {PreCases, NextMod, NextRef} = -	do_add_end_per_suite_and_skip(LastMod, LastRef, Mod), +	do_add_end_per_suite_and_skip(LastMod, LastRef, Mod, FwMod),      PreCases ++ [Case|add_init_and_end_per_suite(Cases, NextMod, NextRef, FwMod)];  add_init_and_end_per_suite([{skip_case,{{Mod,_},_}}=Case|Cases], LastMod,  			   LastRef, FwMod) when Mod =/= LastMod ->      {PreCases, NextMod, NextRef} = -	do_add_init_and_end_per_suite(LastMod, LastRef, Mod), +	do_add_init_and_end_per_suite(LastMod, LastRef, Mod, FwMod),      PreCases ++ [Case|add_init_and_end_per_suite(Cases, NextMod, NextRef, FwMod)];  add_init_and_end_per_suite([{skip_case,{conf,_,{Mod,_},_}}=Case|Cases], LastMod,  			   LastRef, FwMod) when Mod =/= LastMod ->      {PreCases, NextMod, NextRef} = -	do_add_init_and_end_per_suite(LastMod, LastRef, Mod), +	do_add_init_and_end_per_suite(LastMod, LastRef, Mod, FwMod),      PreCases ++ [Case|add_init_and_end_per_suite(Cases, NextMod, NextRef, FwMod)];  add_init_and_end_per_suite([{skip_case,_}=Case|Cases], LastMod, LastRef, FwMod) ->      [Case|add_init_and_end_per_suite(Cases, LastMod, LastRef, FwMod)]; @@ -2047,7 +2182,7 @@ add_init_and_end_per_suite([{conf,Ref,Props,{FwMod,Func}}=Case|Cases], LastMod,      case proplists:get_value(suite, Props) of  	Suite when Suite =/= undefined, Suite =/= LastMod ->  	    {PreCases, NextMod, NextRef} = -		do_add_init_and_end_per_suite(LastMod, LastRef, Suite), +		do_add_init_and_end_per_suite(LastMod, LastRef, Suite, FwMod),  	    Case1 = {conf,Ref,proplists:delete(suite,Props),{FwMod,Func}},  	    PreCases ++ [Case1|add_init_and_end_per_suite(Cases, NextMod,  							  NextRef, FwMod)]; @@ -2057,19 +2192,19 @@ add_init_and_end_per_suite([{conf,Ref,Props,{FwMod,Func}}=Case|Cases], LastMod,  add_init_and_end_per_suite([{conf,_,_,{Mod,_}}=Case|Cases], LastMod,  			   LastRef, FwMod) when Mod =/= LastMod, Mod =/= FwMod ->      {PreCases, NextMod, NextRef} = -	do_add_init_and_end_per_suite(LastMod, LastRef, Mod), +	do_add_init_and_end_per_suite(LastMod, LastRef, Mod, FwMod),      PreCases ++ [Case|add_init_and_end_per_suite(Cases, NextMod, NextRef, FwMod)];  add_init_and_end_per_suite([{conf,_,_,_}=Case|Cases], LastMod, LastRef, FwMod) ->      [Case|add_init_and_end_per_suite(Cases, LastMod, LastRef, FwMod)];  add_init_and_end_per_suite([{Mod,_}=Case|Cases], LastMod, LastRef, FwMod)    when Mod =/= LastMod, Mod =/= FwMod ->      {PreCases, NextMod, NextRef} = -	do_add_init_and_end_per_suite(LastMod, LastRef, Mod), +	do_add_init_and_end_per_suite(LastMod, LastRef, Mod, FwMod),      PreCases ++ [Case|add_init_and_end_per_suite(Cases, NextMod, NextRef, FwMod)];  add_init_and_end_per_suite([{Mod,_,_}=Case|Cases], LastMod, LastRef, FwMod)    when Mod =/= LastMod, Mod =/= FwMod ->      {PreCases, NextMod, NextRef} = -	do_add_init_and_end_per_suite(LastMod, LastRef, Mod), +	do_add_init_and_end_per_suite(LastMod, LastRef, Mod, FwMod),      PreCases ++ [Case|add_init_and_end_per_suite(Cases, NextMod, NextRef, FwMod)];  add_init_and_end_per_suite([Case|Cases], LastMod, LastRef, FwMod)->      [Case|add_init_and_end_per_suite(Cases, LastMod, LastRef, FwMod)]; @@ -2077,10 +2212,23 @@ add_init_and_end_per_suite([], _LastMod, undefined, _FwMod) ->      [];  add_init_and_end_per_suite([], _LastMod, skipped_suite, _FwMod) ->      []; -add_init_and_end_per_suite([], LastMod, LastRef, _FwMod) -> -    [{conf,LastRef,[],{LastMod,end_per_suite}}]. +add_init_and_end_per_suite([], LastMod, LastRef, FwMod) -> +    %% we'll add end_per_suite here even if it's not exported +    %% (and simply let the call fail if it's missing) +    case erlang:function_exported(LastMod, end_per_suite, 1) of +	true -> +	    [{conf,LastRef,[],{LastMod,end_per_suite}}]; +	false -> +	    %% let's call a "fake" end_per_suite if it exists			 +	    case erlang:function_exported(FwMod, end_per_suite, 1) of +		true ->					 +		    [{conf,LastRef,[{suite,LastMod}],{FwMod,end_per_suite}}]; +		false ->		 +		    [{conf,LastRef,[],{LastMod,end_per_suite}}] +	    end +    end.     -do_add_init_and_end_per_suite(LastMod, LastRef, Mod) -> +do_add_init_and_end_per_suite(LastMod, LastRef, Mod, FwMod) ->      case code:is_loaded(Mod) of  	false -> code:load_file(Mod);  	_ -> ok @@ -2091,7 +2239,16 @@ do_add_init_and_end_per_suite(LastMod, LastRef, Mod) ->  		Ref = make_ref(),  		{[{conf,Ref,[],{Mod,init_per_suite}}],Mod,Ref};  	    false -> -		{[],Mod,undefined} +		%% let's call a "fake" init_per_suite if it exists +		case erlang:function_exported(FwMod, init_per_suite, 1) of +		    true -> +			Ref = make_ref(), +			{[{conf,Ref,[{suite,Mod}], +			   {FwMod,init_per_suite}}],Mod,Ref}; +		    false -> +			{[],Mod,undefined} +		end +  	end,      Cases =  	if LastRef==undefined -> @@ -2099,20 +2256,44 @@ do_add_init_and_end_per_suite(LastMod, LastRef, Mod) ->  	   LastRef==skipped_suite ->  		Init;  	   true -> -		%% Adding end_per_suite here without checking if the -		%% function is actually exported. This is because a -		%% conf case must have an end case - so if it doesn't -		%% exist, it will only fail... -		[{conf,LastRef,[],{LastMod,end_per_suite}}|Init] +		%% we'll add end_per_suite here even if it's not exported +		%% (and simply let the call fail if it's missing) +		case erlang:function_exported(LastMod, end_per_suite, 1) of +		    true -> +			[{conf,LastRef,[],{LastMod,end_per_suite}}|Init]; +		    false -> +			%% let's call a "fake" end_per_suite if it exists +			case erlang:function_exported(FwMod, end_per_suite, 1) of +			    true ->				 +				[{conf,LastRef,[{suite,Mod}], +				  {FwMod,end_per_suite}}|Init]; +			    false -> +				[{conf,LastRef,[],{LastMod,end_per_suite}}|Init] +			end +		end  	end,      {Cases,NextMod,NextRef}. -do_add_end_per_suite_and_skip(LastMod, LastRef, Mod) -> +do_add_end_per_suite_and_skip(LastMod, LastRef, Mod, FwMod) ->      case LastRef of  	No when No==undefined ; No==skipped_suite ->  	    {[],Mod,skipped_suite};  	_Ref -> -	    {[{conf,LastRef,[],{LastMod,end_per_suite}}],Mod,skipped_suite} +	    case erlang:function_exported(LastMod, end_per_suite, 1) of +		true -> +		    {[{conf,LastRef,[],{LastMod,end_per_suite}}], +		     Mod,skipped_suite}; +		false -> +		    case erlang:function_exported(FwMod, end_per_suite, 1) of +			true ->				 +			    %% let's call "fake" end_per_suite if it exists +			    {[{conf,LastRef,[],{FwMod,end_per_suite}}], +			     Mod,skipped_suite}; +			false -> +			    {[{conf,LastRef,[],{LastMod,end_per_suite}}], +			     Mod,skipped_suite} +		    end +	    end    	          end. @@ -2658,23 +2839,42 @@ run_test_cases_loop([{conf,Ref,Props,{Mod,Func}}|_Cases]=Cs0,  				      "(configuration case ~w)", [What]);  			   (_) -> ok  			end, -      CfgProps = if StartConf ->  		       if Shuffle == undefined ->  			       [{tc_group_properties,Props}];  			  true -> -			       [{tc_group_properties,[Shuffle|delete_shuffle(Props)]}] +			       [{tc_group_properties, +				 [Shuffle|delete_shuffle(Props)]}]  		       end;  		  not StartConf ->  		       {TcOk,TcSkip,TcFail} = get_tc_results(Status1),  		       [{tc_group_properties,get_props(Mode0)}, -			{tc_group_result,[{ok,TcOk},{skipped,TcSkip},{failed,TcFail}]}] +			{tc_group_result,[{ok,TcOk}, +					  {skipped,TcSkip}, +					  {failed,TcFail}]}]  	       end, -    ActualCfg = -	update_config(hd(Config), [{priv_dir,get(test_server_priv_dir)}, -				   {data_dir,get_data_dir(Mod)}] ++ CfgProps), -    CurrMode = curr_mode(Ref, Mode0, Mode), +    SuiteName = proplists:get_value(suite, Props), +    case get(test_server_create_priv_dir) of +	auto_per_run ->				% use common priv_dir +	    TSDirs = [{priv_dir,get(test_server_priv_dir)}, +		      {data_dir,get_data_dir(Mod, SuiteName)}];     +	_ -> +	    TSDirs = [{data_dir,get_data_dir(Mod, SuiteName)}] +    end, + +    ActualCfg =  +	if not StartConf -> +		update_config(hd(Config), TSDirs ++ CfgProps); +	   true -> +		GroupPath = lists:flatmap(fun({_Ref,[],_T}) -> []; +					     ({_Ref,GrProps,_T}) -> [GrProps] +					  end, Mode0), +		update_config(hd(Config),  +			      TSDirs ++ [{tc_group_path,GroupPath} | CfgProps]) +	end, + +    CurrMode = curr_mode(Ref, Mode0, Mode),      ConfCaseResult = run_test_case(Ref, 0, Mod, Func, [ActualCfg], skip_init, target,  				   TimetrapData, CurrMode), @@ -2825,8 +3025,13 @@ run_test_cases_loop([{conf,_Ref,_Props,_X}=Conf|_Cases0],  run_test_cases_loop([{Mod,Case}|Cases], Config, TimetrapData, Mode, Status) ->      ActualCfg = -	update_config(hd(Config), [{priv_dir,get(test_server_priv_dir)}, -				   {data_dir,get_data_dir(Mod)}]), +	case get(test_server_create_priv_dir) of +	    auto_per_run -> +		update_config(hd(Config), [{priv_dir,get(test_server_priv_dir)}, +					   {data_dir,get_data_dir(Mod)}]); +	    _ -> +		update_config(hd(Config), [{data_dir,get_data_dir(Mod)}]) +	end,      run_test_cases_loop([{Mod,Case,[ActualCfg]}|Cases], Config,  			TimetrapData, Mode, Status); @@ -2991,13 +3196,20 @@ conf_start(Ref, Mode) ->  	false -> 0      end. +  get_data_dir(Mod) -> -    case code:which(Mod) of +    get_data_dir(Mod, undefined). + +get_data_dir(Mod, Suite) -> +    UseMod = if Suite == undefined -> Mod; +		true               -> Suite +	     end, +    case code:which(UseMod) of  	non_existing ->  	    print(12, "The module ~p is not loaded", [Mod]),  	    [];  	FullPath -> -	    filename:dirname(FullPath) ++ "/" ++ cast_to_list(Mod) ++ +	    filename:dirname(FullPath) ++ "/" ++ cast_to_list(UseMod) ++  		?data_dir_suffix      end. @@ -3153,8 +3365,8 @@ skip_case(Type, Ref, CaseNum, Case, Comment, SendSync, Mode) ->  skip_case1(Type, CaseNum, Mod, Func, Comment, Mode) ->      {{Col0,Col1},_} = get_font_style((CaseNum > 0), Mode), -    ResultCol = if Type == auto -> "#ffcc99"; -		   Type == user -> "#ff9933" +    ResultCol = if Type == auto -> ?auto_skip_color; +		   Type == user -> ?user_skip_color  		end,      Comment1 = reason_to_string(Comment), @@ -3163,16 +3375,21 @@ skip_case1(Type, CaseNum, Mod, Func, Comment, Mode) ->      print(major, "=started         ~s", [lists:flatten(timestamp_get(""))]),      print(major, "=result          skipped: ~s", [Comment1]),      print(2,"*** Skipping test case #~w ~p ***", [CaseNum,{Mod,Func}]), +    TR = xhtml("<tr valign=\"top\">", ["<tr class=\"",odd_or_even(),"\">"]),	        +    GroupName =	case get_name(Mode) of +		    undefined -> ""; +		    Name      -> cast_to_list(Name) +		end,      print(html, -	  "<tr valign=top>" -	  "<td>" ++ Col0 ++ "~s" ++ Col1 ++ "</td>" +	  TR ++ "<td>" ++ Col0 ++ "~s" ++ Col1 ++ "</td>"  	  "<td>" ++ Col0 ++ "~p" ++ Col1 ++ "</td>" +	  "<td>" ++ Col0 ++ "~s" ++ Col1 ++ "</td>"  	  "<td>" ++ Col0 ++ "~p" ++ Col1 ++ "</td>"  	  "<td>" ++ Col0 ++ "< >" ++ Col1 ++ "</td>"  	  "<td>" ++ Col0 ++ "0.000s" ++ Col1 ++ "</td>"  	  "<td><font color=\"~s\">SKIPPED</font></td>"  	  "<td>~s</td></tr>\n", -	  [num2str(CaseNum),Mod,Func,ResultCol,Comment1]), +	  [num2str(CaseNum),fw_name(Mod),GroupName,Func,ResultCol,Comment1]),      if CaseNum > 0 ->  	    {US,AS} = get(test_server_skipped),  	    case Type of @@ -3542,9 +3759,14 @@ run_test_case1(Ref, Num, Mod, Func, Args, RunInit, Where,      %% if this runs on a parallel test case process,      %% copy the dictionary from the main process      do_if_parallel(Main, fun() -> process_flag(trap_exit, true) end, ok), -    CopyDict = fun() -> lists:foreach(fun({Key,Val}) -> put(Key, Val) end, State) end, +    CopyDict = fun() -> lists:foreach(fun({Key,Val}) -> +					      put(Key, Val) +				      end, State) +	       end,      do_if_parallel(Main, CopyDict, ok), -    do_if_parallel(Main, fun() -> put(test_server_common_io_handler, {tc,Main}) end, ok), +    do_if_parallel(Main, fun() ->  +				 put(test_server_common_io_handler, {tc,Main}) +			 end, ok),      %% if io is being buffered, send start io session message      %% (no matter if case runs on parallel or main process)      case get(test_server_common_io_handler) of @@ -3558,33 +3780,72 @@ run_test_case1(Ref, Num, Mod, Func, Args, RunInit, Where,  	host ->  	    ok      end, -    test_server_sup:framework_call(report, [tc_start,{?pl2a(Mod),Func}]),      print(major, "=case          ~p:~p", [Mod, Func]),      MinorName = start_minor_log_file(Mod, Func), -    print(minor, "<a name=top></a>", []), +    print(minor, "<a name=\"top\"></a>", [], internal_raw),      MinorBase = filename:basename(MinorName),      print(major, "=logfile       ~s", [filename:basename(MinorName)]), + +    UpdatedArgs = +	%% maybe create unique private directory for test case or config func +	case get(test_server_create_priv_dir) of +	    auto_per_run -> +		update_config(hd(Args), [{tc_logfile,MinorName}]); +	    PrivDirMode -> +		RunDir = filename:dirname(MinorName), +		Ext = +		    if Num == 0 -> +			    {_,S,Us} = now(), +			    lists:flatten(io_lib:format(".~w.~w", [S,Us])); +		       true -> +			    %% create unique private directory for test case +			    RunDir = filename:dirname(MinorName), +			    lists:flatten(io_lib:format(".~w", [Num])) +		    end, +		PrivDir = filename:join(RunDir, ?priv_dir) ++ Ext, +		if PrivDirMode == auto_per_tc -> +			ok = file:make_dir(PrivDir); +		   PrivDirMode == manual_per_tc -> +			ok +		end, +		update_config(hd(Args), [{priv_dir,PrivDir++"/"}, +					 {tc_logfile,MinorName}]) +	end, + +    test_server_sup:framework_call(report, +				   [tc_start,{{?pl2a(Mod),Func},MinorName}]), +      print_props((RunInit==skip_init), get_props(Mode)), +    GroupName =	case get_name(Mode) of +		    undefined -> ""; +		    Name      -> cast_to_list(Name) +		end,      print(major, "=started       ~s", [lists:flatten(timestamp_get(""))]),      {{Col0,Col1},Style} = get_font_style((RunInit==run_init), Mode), -    print(html,	  "<tr valign=top><td>" ++ Col0 ++ "~s" ++ Col1 ++ "</td>" -		  "<td>" ++ Col0 ++ "~p" ++ Col1 ++ "</td>" -		  "<td><a href=\"~s\">~p</a></td>" -		  "<td><a href=\"~s#top\"><</a> <a href=\"~s#end\">></a></td>", -		  [num2str(Num),Mod,MinorBase,Func,MinorBase,MinorBase]), +    TR = xhtml("<tr valign=\"top\">", ["<tr class=\"",odd_or_even(),"\">"]), +    print(html,	TR ++ "<td>" ++ Col0 ++ "~s" ++ Col1 ++ "</td>" +	  "<td>" ++ Col0 ++ "~p" ++ Col1 ++ "</td>" +	  "<td>" ++ Col0 ++ "~s" ++ Col1 ++ "</td>" +	  "<td><a href=\"~s\">~p</a></td>" +	  "<td><a href=\"~s#top\"><</a> <a href=\"~s#end\">></a></td>", +	  [num2str(Num),fw_name(Mod),GroupName,MinorBase,Func, +	   MinorBase,MinorBase]),      do_if_parallel(Main, ok, fun erlang:yield/0), + +    RejectIoReqs = get(test_server_reject_io_reqs),      %% run the test case      {Result,DetectedFail,ProcsBefore,ProcsAfter} = -	run_test_case_apply(Num, Mod, Func, Args, get_name(Mode), -			    RunInit, Where, TimetrapData), +	run_test_case_apply(Num, Mod, Func, [UpdatedArgs], get_name(Mode), +			    RunInit, Where, TimetrapData, RejectIoReqs),      {Time,RetVal,Loc,Opts,Comment} =  	case Result of  	    Normal={_Time,_RetVal,_Loc,_Opts,_Comment} -> Normal;  	    {died,DReason,DLoc,DCmt} -> {died,DReason,DLoc,[],DCmt}  	end, -    print(minor, "<a name=end></a>", []), +    print(minor, "<a name=\"end\"></a>", [], internal_raw), +    print(minor, "\n", [], internal_raw),      print_timestamp(minor, "Ended at "),      print(major, "=ended         ~s", [lists:flatten(timestamp_get(""))]), @@ -3838,9 +4099,10 @@ check_new_crash_dumps(Where) ->  progress(skip, CaseNum, Mod, Func, Loc, Reason, Time,  	 Comment, {St0,St1}) -> -    {Reason1,{Color,Ret}} = if_auto_skip(Reason, -					 fun() -> {"#ffcc99",auto_skip} end, -					 fun() -> {"#ff9933",skip} end), +    {Reason1,{Color,Ret}} =  +	if_auto_skip(Reason, +		     fun() -> {?auto_skip_color,auto_skip} end, +		     fun() -> {?user_skip_color,skip} end),      print(major, "=result        skipped", []),      print(1, "*** SKIPPED *** ~s",  	  [get_info_str(Func, CaseNum, get(test_server_cases))]), @@ -3857,7 +4119,7 @@ progress(skip, CaseNum, Mod, Func, Loc, Reason, Time,  	end,      Comment1 = case Comment of  		   "" -> ""; -		   _ -> "<br>(" ++ to_string(Comment) ++ ")" +		   _ -> xhtml("<br>(","<br />(") ++ to_string(Comment) ++ ")"  	       end,      print(html,  	  "<td>" ++ St0 ++ "~.3fs" ++ St1 ++ "</td>" @@ -3882,8 +4144,8 @@ progress(failed, CaseNum, Mod, Func, Loc, timetrap_timeout, T,      Comment =  	case Comment0 of  	    "" -> "<font color=\"red\">" ++ ErrorReason ++ "</font>"; -	    _ -> "<font color=\"red\">" ++ ErrorReason ++ "</font><br>" ++ -                 to_string(Comment0) +	    _ -> "<font color=\"red\">" ++ ErrorReason ++  +		 xhtml("</font><br>","</font><br />") ++ to_string(Comment0)  	end,      print(html,  	  "<td>" ++ St0 ++ "~.3fs" ++ St1 ++ "</td>" @@ -3908,8 +4170,8 @@ progress(failed, CaseNum, Mod, Func, Loc, {testcase_aborted,Reason}, _T,      Comment =  	case Comment0 of  	    "" -> "<font color=\"red\">" ++ ErrorReason ++ "</font>"; -	    _ -> "<font color=\"red\">" ++ ErrorReason ++ "</font><br>" ++ -                 to_string(Comment0) +	    _ -> "<font color=\"red\">" ++ ErrorReason ++  +		 xhtml("</font><br>","</font><br />") ++ to_string(Comment0)  	end,      print(html,  	  "<td>" ++ St0 ++ "died" ++ St1 ++ "</td>" @@ -3943,7 +4205,8 @@ progress(failed, CaseNum, Mod, Func, unknown, Reason, Time,      Comment =  	case Comment0 of  	    "" -> "<font color=\"red\">" ++ ErrorReason2 ++ "</font>"; -	    _ -> "<font color=\"red\">" ++ ErrorReason2 ++ "</font><br>" ++ +	    _ -> "<font color=\"red\">" ++ ErrorReason2 ++  +		 xhtml("</font><br>","</font><br />") ++  		 to_string(Comment0)  	end,      print(html, @@ -3953,7 +4216,7 @@ progress(failed, CaseNum, Mod, Func, unknown, Reason, Time,  	  [TimeStr,Comment]),      print(minor, "=== location ~s", [unknown]),      {FStr,FormattedReason} = format_exception(Reason), -    print(minor, "=== reason = "++FStr, [FormattedReason]), +    print(minor, "=== reason = " ++ FStr, [FormattedReason]),      failed;  progress(failed, CaseNum, Mod, Func, Loc, Reason, Time, @@ -3969,7 +4232,7 @@ progress(failed, CaseNum, Mod, Func, Loc, Reason, Time,      Comment =  	case Comment0 of  	    "" -> ""; -	    _ -> "<br>" ++ to_string(Comment0) +	    _ -> xhtml("<br>","<br />") ++ to_string(Comment0)  	end,      FormatLastLoc = test_server_sup:format_loc(get_last_loc(Loc)),      print(html, @@ -3980,7 +4243,7 @@ progress(failed, CaseNum, Mod, Func, Loc, Reason, Time,      FormatLoc = test_server_sup:format_loc(Loc),      print(minor, "=== location ~s", [FormatLoc]),      {FStr,FormattedReason} = format_exception(Reason), -    print(minor, "=== reason = "++FStr, [FormattedReason]), +    print(minor, "=== reason = " ++ FStr, [FormattedReason]),      failed;  progress(ok, _CaseNum, Mod, Func, _Loc, RetVal, Time, @@ -3999,7 +4262,7 @@ progress(ok, _CaseNum, Mod, Func, _Loc, RetVal, Time,  	    _ ->  		print(major, "=result        ok", []),  		case Comment0 of -		    "" -> ""; +		    "" -> "<td></td>";  		    _ -> "<td>" ++ to_string(Comment0) ++ "</td>"  		end  	end, @@ -4015,6 +4278,46 @@ progress(ok, _CaseNum, Mod, Func, _Loc, RetVal, Time,  %%--------------------------------------------------------------------  %% various help functions +get_fw_mod(Mod) -> +    case get(test_server_framework) of +	undefined -> +	    case os:getenv("TEST_SERVER_FRAMEWORK") of +		FW when FW =:= false; FW =:= "undefined" -> +		    Mod; +		FW -> +		    list_to_atom(FW) +	    end; +	'$none' -> Mod; +	FW      -> FW +    end. + +fw_name(?MODULE) -> +    test_server; +fw_name(Mod) -> +    case get(test_server_framework_name) of +	undefined -> +	    case get_fw_mod(undefined) of +		undefined -> +		    Mod; +		Mod -> +		    case os:getenv("TEST_SERVER_FRAMEWORK_NAME") of +			FWName when FWName =:= false; FWName =:= "undefined" -> +			    Mod; +			FWName -> +			    list_to_atom(FWName) +		    end; +		_ -> +		    Mod +	    end; +	'$none' -> +	    Mod; +	FWName -> +	    case get_fw_mod(Mod) of +		Mod -> FWName; +		_ -> Mod +	    end	 +    end. +  if_auto_skip(Reason={failed,{_,init_per_testcase,_}}, True, _False) ->      {Reason,True()};  if_auto_skip({_T,{skip,Reason={failed,{_,init_per_testcase,_}}},_Opts}, True, _False) -> @@ -4118,8 +4421,8 @@ get_font_style1(default) ->  %% set to false.  format_exception(Reason={_Error,Stack}) when is_list(Stack) -> -    case os:getenv("TEST_SERVER_FRAMEWORK") of -	FW when FW =:= false; FW =:= "undefined" -> +    case get_fw_mod(undefined) of +	undefined ->  	    case application:get_env(test_server, format_exception) of  		{ok,false} ->  		    {"~p",Reason}; @@ -4127,7 +4430,7 @@ format_exception(Reason={_Error,Stack}) when is_list(Stack) ->  		    do_format_exception(Reason)  	    end;  	FW -> -	    case application:get_env(list_to_atom(FW), format_exception) of +	    case application:get_env(FW, format_exception) of  		{ok,false} ->  		    {"~p",Reason};  		_ -> @@ -4153,7 +4456,7 @@ do_format_exception(Reason={Error,Stack}) ->  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%  %% run_test_case_apply(CaseNum, Mod, Func, Args, Name, RunInit, -%%                     Where, TimetrapData) -> +%%                     Where, TimetrapData, RejectIoReqs) ->  %%  {{Time,RetVal,Loc,Opts,Comment},DetectedFail,ProcessesBefore,ProcessesAfter} |  %%  {{died,Reason,unknown,Comment},DetectedFail,ProcessesBefore,ProcessesAfter}  %% Name = atom() @@ -4172,19 +4475,21 @@ do_format_exception(Reason={Error,Stack}) ->  %% sent over socket to target, and test_server runs the case and sends the  %% result back over the socket. Else test_server runs the case directly on host. -run_test_case_apply(CaseNum, Mod, Func, Args, Name, RunInit, host, TimetrapData) -> +run_test_case_apply(CaseNum, Mod, Func, Args, Name, RunInit, host, +		    TimetrapData, RejectIoReqs) ->      test_server:run_test_case_apply({CaseNum,Mod,Func,Args,Name,RunInit, -				     TimetrapData}); -run_test_case_apply(CaseNum, Mod, Func, Args, Name, RunInit, target, TimetrapData) -> +				     TimetrapData,RejectIoReqs}); +run_test_case_apply(CaseNum, Mod, Func, Args, Name, RunInit, target, +		    TimetrapData, RejectIoReqs) ->      case get(test_server_ctrl_job_sock) of  	undefined ->  	    %% local target  	    test_server:run_test_case_apply({CaseNum,Mod,Func,Args,Name,RunInit, -					     TimetrapData}); +					     TimetrapData,RejectIoReqs});  	JobSock ->  	    %% remote target  	    request(JobSock, {test_case,{CaseNum,Mod,Func,Args,Name,RunInit, -					 TimetrapData}}), +					 TimetrapData,RejectIoReqs}}),  	    read_job_sock_loop(JobSock)      end. @@ -4349,7 +4654,13 @@ output({html,Msg}, _Sender) ->  		    %% We are writing to a seekable file.  Finalise so  		    %% we get complete valid (and viewable) HTML code.  		    %% Then rewind to overwrite the finalising code. -		    io:put_chars(Fd, "\n</table>\n</body>\n</html>\n"), +		    io:put_chars(Fd, "\n</table>\n"), +		    case get(test_server_html_footer) of +			undefined -> +			    io:put_chars(Fd, "</body>\n</html>\n"); +			Footer -> +			    io:put_chars(Fd, Footer) +		    end,  		    file:position(Fd, Pos);  		{error, epipe} ->  		    %% The file is not seekable.  We cannot erase what @@ -4394,6 +4705,28 @@ output_to_fd(Fd, Msg, _Sender) ->      end.  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%% xhtml(BasicHtml, XHtml) -> BasicHtml | XHtml +%% +xhtml(HTML, XHTML) -> +    case get(basic_html) of +	true -> HTML; +	_ -> XHTML +    end. + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%% odd_or_even() -> "odd" | "even" +%% +odd_or_even() -> +    case get(odd_or_even) of +	even -> +	    put(odd_or_even, odd), +	    "even"; +	_ -> +	    put(odd_or_even, even), +	    "odd" +    end. + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%  %% timestamp_filename_get(Leader) -> string()  %% Leader = string()  %% @@ -4695,8 +5028,8 @@ collect_case([Case | Cases], St, Acc) ->      collect_case(Cases, NewSt, Acc ++ FlatCases).  collect_case_invoke(Mod, Case, MFA, St) -> -    case os:getenv("TEST_SERVER_FRAMEWORK") of -	FW when FW =:= false; FW =:= "undefined" -> +    case get_fw_mod(undefined) of +	undefined ->  	    case catch apply(Mod, Case, [suite]) of  		{'EXIT',_} ->  		    {ok,[MFA],St}; @@ -4704,7 +5037,9 @@ collect_case_invoke(Mod, Case, MFA, St) ->  		    collect_subcases(Mod, Case, MFA, St, Suite)  	    end;  	_ -> -	    Suite = test_server_sup:framework_call(get_suite, [?pl2a(Mod),Case], []), +	    Suite = test_server_sup:framework_call(get_suite, +						   [?pl2a(Mod),Case], +						   []),  	    collect_subcases(Mod, Case, MFA, St, Suite)      end. @@ -4858,7 +5193,9 @@ init_props(Props) ->      end.  keep_name(Props) -> -    lists:filter(fun({name,_}) -> true; (_) -> false end, Props). +    lists:filter(fun({name,_}) -> true; +		    ({suite,_}) -> true; +		    (_) -> false end, Props).  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%  %%                 Target node handling functions                   %% @@ -5499,9 +5836,10 @@ write_default_cross_coverlog(TestDir) ->  	file:open(filename:join(TestDir,?cross_coverlog_name), [write]),      write_coverlog_header(CrossCoverLog),      io:fwrite(CrossCoverLog, -	      "No cross cover modules exist for this application,<br>" -	      "or cross cover analysis is not completed.\n" -	      "</body></html>\n", []), +	      ["No cross cover modules exist for this application,", +	       xhtml("<br>","<br />"), +	       "or cross cover analysis is not completed.\n" +	       "</body></html>\n"], []),      file:close(CrossCoverLog).  write_cover_result_table(CoverLog,Coverage) -> diff --git a/lib/test_server/src/test_server_h.erl b/lib/test_server/src/test_server_h.erl index e423863b99..78daba855d 100644 --- a/lib/test_server/src/test_server_h.erl +++ b/lib/test_server/src/test_server_h.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %%  -%% Copyright Ericsson AB 2005-2009. All Rights Reserved. +%% Copyright Ericsson AB 2005-2012. All Rights Reserved.  %%   %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -79,10 +79,21 @@ set_group_leader() ->  handle_event({_Type, GL, _Msg}, State) when node(GL)/=node() ->      {ok, State};  handle_event({Tag, _GL, {_Pid, Type, _Report}} = Event, State) -> -    case report(Tag, Type) of -	sasl -> -	    tag(State#state.testcase), -	    sasl_report_tty_h:handle_event(Event, State#state.sasl); +    SASL = lists:keyfind(sasl, 1, application:which_applications()), +    case report_receiver(Tag, Type) of +	sasl when SASL /= false -> +	    {ok,ErrLogType} = application:get_env(sasl, errlog_type), +	    SReport = sasl_report:format_report(group_leader(), ErrLogType, +						tag_event(Event)), +	    if is_list(SReport) -> +		    tag(State#state.testcase), +		    sasl_report_tty_h:handle_event(Event, +						   State#state.sasl); +	       true -> %% Report is an atom if no logging is to be done +		    ignore +	    end; +	sasl -> %% SASL not running +	    ignore;  	kernel ->  	    tag(State#state.testcase),  	    error_logger_tty_h:handle_event(Event, State#state.kernel); @@ -111,19 +122,27 @@ terminate(_Reason, _State) ->  code_change(_OldVsn, State, _Extra) ->      {ok, State}. -report(error_report, supervisor_report) -> sasl; -report(error_report, crash_report) -> sasl; -report(info_report, progress) -> sasl; -report(error, _) -> kernel; -report(error_report, _) -> kernel; -report(warning_msg, _) -> kernel; -report(warning_report, _) -> kernel; -report(info, _) -> kernel; -report(info_msg, _) -> kernel; -report(info_report, _) -> kernel; -report(_, _) -> none. +report_receiver(error_report, supervisor_report) -> sasl; +report_receiver(error_report, crash_report) -> sasl; +report_receiver(info_report, progress) -> sasl; +report_receiver(error, _) -> kernel; +report_receiver(error_report, _) -> kernel; +report_receiver(warning_msg, _) -> kernel; +report_receiver(warning_report, _) -> kernel; +report_receiver(info, _) -> kernel; +report_receiver(info_msg, _) -> kernel; +report_receiver(info_report,Tuple) +  when is_tuple(Tuple) andalso +       (element(1,Tuple)==ct_connection orelse +	element(1,Tuple)==conn_log) -> +    none; +report_receiver(info_report, _) -> kernel; +report_receiver(_, _) -> none.  tag({M,F,A}) when is_atom(M), is_atom(F), is_integer(A) ->      io:format(user, "~n=TESTCASE: ~p:~p/~p", [M,F,A]);  tag(Testcase) ->      io:format(user, "~n=TESTCASE: ~p", [Testcase]). + +tag_event(Event) -> +    {calendar:local_time(), Event}. diff --git a/lib/test_server/src/test_server_line.erl b/lib/test_server/src/test_server_line.erl deleted file mode 100644 index 848a9c23dd..0000000000 --- a/lib/test_server/src/test_server_line.erl +++ /dev/null @@ -1,387 +0,0 @@ -%% -%% %CopyrightBegin% -%% -%% Copyright Ericsson AB 2004-2010. All Rights Reserved. -%% -%% The contents of this file are subject to the Erlang Public License, -%% Version 1.1, (the "License"); you may not use this file except in -%% compliance with the License. You should have received a copy of the -%% Erlang Public License along with this software. If not, it can be -%% retrieved online at http://www.erlang.org/. -%% -%% Software distributed under the License is distributed on an "AS IS" -%% basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See -%% the License for the specific language governing rights and limitations -%% under the License. -%% -%% %CopyrightEnd% -%% --module(test_server_line). - -%% User interface --export([get_lines/0]). --export([clear/0]). - -%% Parse transform functions --export([parse_transform/2]). --export(['$test_server_line'/3]). --export(['$test_server_lineQ'/3]). --export([trace_line/3]). - --define(TEST_SERVER_LINE_SIZE, 10). -%-define(STORAGE_FUNCTION, '$test_server_line'). --define(STORAGE_FUNCTION, '$test_server_lineQ'). - --include("test_server.hrl"). - --record(vars, {module,                      % atom() Module name -	       function,                    % atom() Function name -	       arity,                       % int()  Function arity -	       lines,                       % [int()]  seen lines -	       is_guard=false,              % boolean() -	       no_lines=[],                 % [{atom(),integer()}]  -					    %    Functions to exclude -	       line_trace=false -	      }). - - - - -%% Process dictionary littering variant -%% - -'$test_server_line'(Mod, Func, Line) -> -    {Prev,Next} =  -	case get('$test_server_line') of -	    I when is_integer(I) -> -		if  1 =< I, I < ?TEST_SERVER_LINE_SIZE -> {I,I+1}; -		    true -> {?TEST_SERVER_LINE_SIZE,1} -		end; -	    _ -> {?TEST_SERVER_LINE_SIZE,1} -	end, -    PrevTag = {'$test_server_line',Prev}, -    case get(PrevTag) of -	{Mod,Func,_} -> put(PrevTag, {Mod,Func,Line}); -	_ ->  -	    put({'$test_server_line',Next}, {Mod,Func,Line}), -	    put('$test_server_line', Next) -    end, ok. - -test_server_line_get() -> -    case get('$test_server_line') of -	I when is_integer(I), 1 =< I, I =< ?TEST_SERVER_LINE_SIZE -> -	    test_server_line_get_1(?TEST_SERVER_LINE_SIZE, I, []); -	_ -> [] -    end. - -test_server_line_get_1(0, _I, R) -> -    R; -test_server_line_get_1(Cnt, I, R) -> -    J = if  I < ?TEST_SERVER_LINE_SIZE -> I+1; -	    true -> 1 end, -    case get({'$test_server_line',J}) of -	undefined ->  -	    %% Less than ?TEST_SERVER_LINE_SIZE number of lines stored -	    %% Start from line 1 and stop at actutual number of lines -	    case get({'$test_server_line',1}) of  -		undefined -> R; % no lines at all stored -		E -> test_server_line_get_1(I-1,1,[E|R]) -	    end; -	E ->  -	    test_server_line_get_1(Cnt-1, J, [E|R]) -    end. - -test_server_line_clear() -> -    Is = lists:seq(1,?TEST_SERVER_LINE_SIZE), -    lists:foreach(fun (I) -> erase({'$test_server_line',I}) end, Is), -    erase('$test_server_line'), -    ok. - - -%% Queue variant, uses just one process dictionary entry -%% - -'$test_server_lineQ'(Mod, Func, Line) ->		 -    case get('$test_server_lineQ') of -	{I,Q} when is_integer(I), 1 =< I, I =< ?TEST_SERVER_LINE_SIZE ->  -	    case queue:head(Q) of -		{Mod,Func,_} -> -		    %% Replace queue head -		    put('$test_server_lineQ', -			{I,queue:cons({Mod,Func,Line}, queue:tail(Q))}); -		_ when I < ?TEST_SERVER_LINE_SIZE -> -		    put('$test_server_lineQ', -			{I+1,queue:cons({Mod,Func,Line}, Q)}); -		_ ->  -		    %% Waste last in queue -		    put('$test_server_lineQ', -			{I,queue:cons({Mod,Func,Line}, queue:lait(Q))}) -	    end; -	_ ->  -	    Q = queue:new(), -	    put('$test_server_lineQ', {1,queue:cons({Mod,Func,Line}, Q)}) -    end, ok. - -%test_server_lineQ_get() -> -%    case get('$test_server_lineQ') of -%	{I,Q} when integer(I), 1 =< I, I =< ?TEST_SERVER_LINE_SIZE ->  -%	    queue:to_list(Q); -%	_ -> [] -%    end. - -test_server_lineQ_clear() -> -    erase('$test_server_lineQ'), -    ok. - - -%% Get line - check if queue or dictionary is used, then get the lines -%% - -get_lines() -> -    case get('$test_server_lineQ') of -	{I,Q} when is_integer(I), 1 =< I, I =< ?TEST_SERVER_LINE_SIZE ->  -	    queue:to_list(Q); -	_ ->  -	    test_server_line_get() -    end. -	 -%% Clear all dictionary entries -%% -clear() -> -    test_server_line_clear(), -    test_server_lineQ_clear(). - - -trace_line(Mod,Func,Line) -> -    io:format(lists:concat([Mod,":",Func,",",integer_to_list(Line),": ~p"]), -	      [erlang:now()]). - - -%%%================================================================= -%%%========= ****   PARSE TRANSFORM    **** ======================== -%%%================================================================= -parse_transform(Forms, _Options) -> -    transform(Forms, _Options). - -%% forms(Fs) -> lists:map(fun (F) -> form(F) end, Fs). - -transform(Forms, _Options)-> -    Vars0 = #vars{}, -    {ok, MungedForms, _Vars} = transform(Forms, [], Vars0), -    MungedForms. -     - -transform([Form|Forms], MungedForms, Vars) -> -    case munge(Form, Vars) of -	ignore -> -	    transform(Forms, MungedForms, Vars); -	{MungedForm, Vars2} -> -	    transform(Forms, [MungedForm|MungedForms], Vars2) -    end; -transform([], MungedForms, Vars) -> -    {ok, lists:reverse(MungedForms), Vars}. - -%% This code traverses the abstract code, stored as the abstract_code -%% chunk in the BEAM file, as described in absform(3) for Erlang/OTP R8B -%% (Vsn=abstract_v2). -%% The abstract format after preprocessing differs slightly from the abstract -%% format given eg using epp:parse_form, this has been noted in comments. -munge(Form={attribute,_,module,Module}, Vars) -> -    Vars2 = Vars#vars{module=Module}, -    {Form, Vars2}; - -munge(Form={attribute,_,no_lines,Funcs}, Vars) -> -    Vars2 = Vars#vars{no_lines=Funcs}, -    {Form, Vars2}; - -munge(Form={attribute,_,line_trace,_}, Vars) -> -    Vars2 = Vars#vars{line_trace=true}, -    {Form, Vars2}; - -munge({function,0,module_info,_Arity,_Clauses}, _Vars) -> -    ignore; % module_info will be added again when the forms are recompiled -munge(Form = {function,Line,Function,Arity,Clauses}, Vars) -> -    case lists:member({Function,Arity},Vars#vars.no_lines) of -	true -> -	    %% Line numbers in this function shall not be stored -	    {Form,Vars}; -	false -> -	    Vars2 = Vars#vars{function=Function, -			      arity=Arity, -			      lines=[]}, -	    {MungedClauses, Vars3} = munge_clauses(Clauses, Vars2, []), -	    {{function,Line,Function,Arity,MungedClauses}, Vars3} -    end; -munge(Form, Vars) -> % attributes -    {Form, Vars}. - -munge_clauses([{clause,Line,Pattern,Guards,Body}|Clauses], Vars, MClauses) -> -    {MungedGuards, _Vars} = munge_exprs(Guards, Vars#vars{is_guard=true},[]), -    {MungedBody, Vars2} = munge_body(Body, Vars, []), -    munge_clauses(Clauses, Vars2, -		  [{clause,Line,Pattern,MungedGuards,MungedBody}| -		   MClauses]); -munge_clauses([], Vars, MungedClauses) ->  -    {lists:reverse(MungedClauses), Vars}. - -munge_body([Expr|Body], Vars, MungedBody) -> -    %% Here is the place to add a call to storage function! -    Line = element(2, Expr), -    Lines = Vars#vars.lines, -    case lists:member(Line,Lines) of -	true -> % already a bump at this line! -	    {MungedExpr, Vars2} = munge_expr(Expr, Vars), -	    munge_body(Body, Vars2, [MungedExpr|MungedBody]); -	false -> -	    Bump = {call, 0, {remote,0, -			      {atom,0,?MODULE}, -			      {atom,0,?STORAGE_FUNCTION}}, -		    [{atom,0,Vars#vars.module}, -		     {atom, 0, Vars#vars.function}, -		     {integer, 0, Line}]}, -	    Lines2 = [Line|Lines], -	     -	    {MungedExpr, Vars2} = munge_expr(Expr, Vars#vars{lines=Lines2}), -	    MungedBody2 =  -		if Vars#vars.line_trace -> -			LineTrace = {call, 0, {remote,0, -					       {atom,0,?MODULE}, -					       {atom,0,trace_line}}, -				     [{atom,0,Vars#vars.module}, -				      {atom, 0, Vars#vars.function}, -				      {integer, 0, Line}]}, -			[MungedExpr,LineTrace,Bump|MungedBody]; -		   true -> -			[MungedExpr,Bump|MungedBody] -		end, -	    munge_body(Body, Vars2, MungedBody2) -    end; -munge_body([], Vars, MungedBody) -> -    {lists:reverse(MungedBody), Vars}. - -munge_expr({match,Line,ExprL,ExprR}, Vars) -> -    {MungedExprL, Vars2} = munge_expr(ExprL, Vars), -    {MungedExprR, Vars3} = munge_expr(ExprR, Vars2), -    {{match,Line,MungedExprL,MungedExprR}, Vars3}; -munge_expr({tuple,Line,Exprs}, Vars) -> -    {MungedExprs, Vars2} = munge_exprs(Exprs, Vars, []), -    {{tuple,Line,MungedExprs}, Vars2}; -munge_expr({record,Line,Expr,Exprs}, Vars) -> -    %% Only for Vsn=raw_abstract_v1 -    {MungedExprName, Vars2} = munge_expr(Expr, Vars), -    {MungedExprFields, Vars3} = munge_exprs(Exprs, Vars2, []), -    {{record,Line,MungedExprName,MungedExprFields}, Vars3}; -munge_expr({record_field,Line,ExprL,ExprR}, Vars) -> -    %% Only for Vsn=raw_abstract_v1 -    {MungedExprL, Vars2} = munge_expr(ExprL, Vars), -    {MungedExprR, Vars3} = munge_expr(ExprR, Vars2), -    {{record_field,Line,MungedExprL,MungedExprR}, Vars3}; -munge_expr({cons,Line,ExprH,ExprT}, Vars) -> -    {MungedExprH, Vars2} = munge_expr(ExprH, Vars), -    {MungedExprT, Vars3} = munge_expr(ExprT, Vars2), -    {{cons,Line,MungedExprH,MungedExprT}, Vars3}; -munge_expr({op,Line,Op,ExprL,ExprR}, Vars) -> -    {MungedExprL, Vars2} = munge_expr(ExprL, Vars), -    {MungedExprR, Vars3} = munge_expr(ExprR, Vars2), -    {{op,Line,Op,MungedExprL,MungedExprR}, Vars3}; -munge_expr({op,Line,Op,Expr}, Vars) -> -    {MungedExpr, Vars2} = munge_expr(Expr, Vars), -    {{op,Line,Op,MungedExpr}, Vars2}; -munge_expr({'catch',Line,Expr}, Vars) -> -    {MungedExpr, Vars2} = munge_expr(Expr, Vars), -    {{'catch',Line,MungedExpr}, Vars2}; -munge_expr({call,Line1,{remote,Line2,ExprM,ExprF},Exprs}, -	   Vars) when Vars#vars.is_guard==false-> -    {MungedExprM, Vars2} = munge_expr(ExprM, Vars), -    {MungedExprF, Vars3} = munge_expr(ExprF, Vars2), -    {MungedExprs, Vars4} = munge_exprs(Exprs, Vars3, []), -    {{call,Line1,{remote,Line2,MungedExprM,MungedExprF},MungedExprs}, Vars4}; -munge_expr({call,Line1,{remote,_Line2,_ExprM,ExprF},Exprs}, -	   Vars) when Vars#vars.is_guard==true -> -    %% Difference in abstract format after preprocessing: BIF calls in guards -    %% are translated to {remote,...} (which is not allowed as source form) -    %% NOT NECESSARY FOR Vsn=raw_abstract_v1 -    munge_expr({call,Line1,ExprF,Exprs}, Vars); -munge_expr({call,Line,Expr,Exprs}, Vars) -> -    {MungedExpr, Vars2} = munge_expr(Expr, Vars), -    {MungedExprs, Vars3} = munge_exprs(Exprs, Vars2, []), -    {{call,Line,MungedExpr,MungedExprs}, Vars3}; -munge_expr({lc,Line,Expr,LC}, Vars) -> -    {MungedExpr, Vars2} = munge_expr(Expr, Vars), -    {MungedLC, Vars3} = munge_lc(LC, Vars2, []), -    {{lc,Line,MungedExpr,MungedLC}, Vars3}; -munge_expr({block,Line,Body}, Vars) -> -    {MungedBody, Vars2} = munge_body(Body, Vars, []), -    {{block,Line,MungedBody}, Vars2}; -munge_expr({'if',Line,Clauses}, Vars) ->  -    {MungedClauses,Vars2} = munge_clauses(Clauses, Vars, []), -    {{'if',Line,MungedClauses}, Vars2}; -munge_expr({'case',Line,Expr,Clauses}, Vars) -> -    {MungedExpr,Vars2} = munge_expr(Expr,Vars), -    {MungedClauses,Vars3} = munge_clauses(Clauses, Vars2, []), -    {{'case',Line,MungedExpr,MungedClauses}, Vars3}; -munge_expr({'receive',Line,Clauses}, Vars) ->  -    {MungedClauses,Vars2} = munge_clauses(Clauses, Vars, []), -    {{'receive',Line,MungedClauses}, Vars2}; -munge_expr({'receive',Line,Clauses,Expr,Body}, Vars) -> -    {MungedClauses,Vars2} = munge_clauses(Clauses, Vars, []), -    {MungedExpr, Vars3} = munge_expr(Expr, Vars2), -    {MungedBody, Vars4} = munge_body(Body, Vars3, []), -    {{'receive',Line,MungedClauses,MungedExpr,MungedBody}, Vars4}; -munge_expr({'try',Line,Exprs,Clauses,CatchClauses,After}, Vars) -> -    {MungedExprs, Vars1} = munge_exprs(Exprs, Vars, []), -    {MungedClauses, Vars2} = munge_clauses(Clauses, Vars1, []), -    {MungedCatchClauses, Vars3} = munge_clauses(CatchClauses, Vars2, []), -    {MungedAfter, Vars4} = munge_body(After, Vars3, []), -    {{'try',Line,MungedExprs,MungedClauses,MungedCatchClauses,MungedAfter},  -     Vars4}; -%% Difference in abstract format after preprocessing: Funs get an extra -%% element Extra. -%% NOT NECESSARY FOR Vsn=raw_abstract_v1 -munge_expr({'fun',Line,{function,Name,Arity},_Extra}, Vars) -> -    {{'fun',Line,{function,Name,Arity}}, Vars}; -munge_expr({'fun',Line,{clauses,Clauses},_Extra}, Vars) -> -    {MungedClauses,Vars2}=munge_clauses(Clauses, Vars, []), -    {{'fun',Line,{clauses,MungedClauses}}, Vars2}; -munge_expr({'fun',Line,{clauses,Clauses}}, Vars) -> -    %% Only for Vsn=raw_abstract_v1 -    {MungedClauses,Vars2}=munge_clauses(Clauses, Vars, []), -    {{'fun',Line,{clauses,MungedClauses}}, Vars2}; -munge_expr({bc,Line,Expr,LC}, Vars) -> -    {MungedExpr, Vars2} = munge_expr(Expr, Vars), -    {MungedLC, Vars3} = munge_lc(LC, Vars2, []), -    {{bc,Line,MungedExpr,MungedLC}, Vars3}; -munge_expr(Form, Vars) -> % var|char|integer|float|string|atom|nil|bin|eof -    {Form, Vars}. - -munge_exprs([Expr|Exprs], Vars, MungedExprs) when Vars#vars.is_guard==true, -						  is_list(Expr) -> -    {MungedExpr, _Vars} = munge_exprs(Expr, Vars, []), -    munge_exprs(Exprs, Vars, [MungedExpr|MungedExprs]); -munge_exprs([Expr|Exprs], Vars, MungedExprs) -> -    {MungedExpr, Vars2} = munge_expr(Expr, Vars), -    munge_exprs(Exprs, Vars2, [MungedExpr|MungedExprs]); -munge_exprs([], Vars, MungedExprs) -> -    {lists:reverse(MungedExprs), Vars}. - -munge_lc([{generate,Line,Pattern,Expr}|LC], Vars, MungedLC) -> -    {MungedExpr, Vars2} = munge_expr(Expr, Vars), -    munge_lc(LC, Vars2, [{generate,Line,Pattern,MungedExpr}|MungedLC]); -munge_lc([{b_generate,Line,Pattern,Expr}|LC], Vars, MungedLC) -> -    {MungedExpr, Vars2} = munge_expr(Expr, Vars), -    munge_lc(LC, Vars2, [{b_generate,Line,Pattern,MungedExpr}|MungedLC]); -munge_lc([Expr|LC], Vars, MungedLC) -> -    {MungedExpr, Vars2} = munge_expr(Expr, Vars), -    munge_lc(LC, Vars2, [MungedExpr|MungedLC]); -munge_lc([], Vars, MungedLC) -> -    {lists:reverse(MungedLC), Vars}. - - - - - - - - - - diff --git a/lib/test_server/src/test_server_node.erl b/lib/test_server/src/test_server_node.erl index 1fd40d1dd9..6358efa764 100644 --- a/lib/test_server/src/test_server_node.erl +++ b/lib/test_server/src/test_server_node.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2002-2011. All Rights Reserved. +%% Copyright Ericsson AB 2002-2012. All Rights Reserved.  %%  %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -407,7 +407,7 @@ start_node_peer(SlaveName, OptList, From, TI) ->      % Support for erl_crash_dump files..      CrashFile = filename:join([TI#target_info.test_server_dir,  			       "erl_crash_dump."++cast_to_list(SlaveName)]), -    CrashArgs = lists:concat([" -env ERL_CRASH_DUMP ",CrashFile," "]), +    CrashArgs = lists:concat([" -env ERL_CRASH_DUMP \"",CrashFile,"\" "]),      FailOnError = start_node_get_option_value(fail_on_error, OptList, true),      Pa = TI#target_info.test_server_dir,      Prog0 = start_node_get_option_value(erl, OptList, default), @@ -420,7 +420,7 @@ start_node_peer(SlaveName, OptList, From, TI) ->      Cmd = lists:concat([Prog,  			" -detached ",  			TI#target_info.naming, " ", SlaveName, -			" -pa ", Pa, +			" -pa \"", Pa,"\"",  			NodeStarted,  			CrashArgs,  			" ", Args]), @@ -472,9 +472,9 @@ start_node_slave(SlaveName, OptList, From, TI) ->      CrashFile = filename:join([TI#target_info.test_server_dir,  			       "erl_crash_dump."++cast_to_list(SlaveName)]), -    CrashArgs = lists:concat([" -env ERL_CRASH_DUMP ",CrashFile," "]), +    CrashArgs = lists:concat([" -env ERL_CRASH_DUMP \"",CrashFile,"\" "]),      Pa = TI#target_info.test_server_dir, -    Args = lists:concat([" -pa ", Pa, " ", SuppliedArgs, CrashArgs]), +    Args = lists:concat([" -pa \"", Pa, "\" ", SuppliedArgs, CrashArgs]),      Prog0 = start_node_get_option_value(erl, OptList, default),      Prog = pick_erl_program(Prog0), @@ -943,12 +943,23 @@ find_rel_suse_1(Rel, RootWc) ->      end.  find_rel_suse_2(Rel, RootWc) -> -    Wc = RootWc ++ "_" ++ Rel, -    case filelib:wildcard(Wc) of -	[] -> -	    []; -	[R|_] -> -	    [filename:join([R,"bin","erl"])] +    RelDir = filename:dirname(RootWc), +    Pat = filename:basename(RootWc ++ "_" ++ Rel) ++ ".*", +    case file:list_dir(RelDir) of +	{ok,Dirs} -> +	    case lists:filter(fun(Dir) -> +				      case re:run(Dir, Pat) of +					  nomatch -> false; +					  _       -> true +				      end +			      end, Dirs) of +		[] -> +		    []; +		[R|_] -> +		    [filename:join([RelDir,R,"bin","erl"])] +	    end; +	_ -> +	    []      end.  %% suse_release() -> VersionString | none. diff --git a/lib/test_server/src/test_server_sup.erl b/lib/test_server/src/test_server_sup.erl index 77d364d5cb..9d111ff769 100644 --- a/lib/test_server/src/test_server_sup.erl +++ b/lib/test_server/src/test_server_sup.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 1998-2011. All Rights Reserved. +%% Copyright Ericsson AB 1998-2012. All Rights Reserved.  %%  %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -21,7 +21,8 @@  %%% Purpose: Test server support functions.  %%%-------------------------------------------------------------------  -module(test_server_sup). --export([timetrap/2, timetrap/3, timetrap_cancel/1, capture_get/1, messages_get/1, +-export([timetrap/2, timetrap/3, timetrap/4, +	 timetrap_cancel/1, capture_get/1, messages_get/1,  	 timecall/3, call_crash/5, app_test/2, check_new_crash_dumps/0,  	 cleanup_crash_dumps/0, crash_dump_dir/0, tar_crash_dumps/0,  	 get_username/0, get_os_family/0,  @@ -44,37 +45,49 @@  %% delays during the test (e.g. if cover is running).  timetrap(Timeout0, Pid) -> -    timetrap(Timeout0, true, Pid). +    timetrap(Timeout0, Timeout0, true, Pid).  timetrap(Timeout0, Scale, Pid) -> +    timetrap(Timeout0, Timeout0, Scale, Pid). + +timetrap(Timeout0, ReportTVal, Scale, Pid) ->      process_flag(priority, max),      Timeout = if not Scale -> Timeout0;  		 true -> test_server:timetrap_scale_factor() * Timeout0  	      end, +    TruncTO = trunc(Timeout),      receive -    after trunc(Timeout) -> -	    Line = test_server:get_loc(Pid), -	    Mon = erlang:monitor(process, Pid), -	    Trap =  -		case get(test_server_init_or_end_conf) of -		    undefined -> -			{timetrap_timeout,trunc(Timeout),Line}; -		    InitOrEnd -> -			{timetrap_timeout,trunc(Timeout),Line,InitOrEnd} -		end, -	    exit(Pid,Trap), -	    receive -		{'DOWN', Mon, process, Pid, _} -> +    after TruncTO -> +	    case is_process_alive(Pid) of +		true -> +		    TimeToReport = if Timeout0 == ReportTVal -> TruncTO; +				      true -> ReportTVal end, +		    MFLs = test_server:get_loc(Pid), +		    Mon = erlang:monitor(process, Pid), +		    Trap =  +			case get(test_server_init_or_end_conf) of +			    undefined -> +				{timetrap_timeout,TimeToReport,MFLs}; +			    InitOrEnd -> +				{timetrap_timeout,TimeToReport,MFLs,InitOrEnd} +			end, +		    exit(Pid, Trap), +		    receive +			{'DOWN', Mon, process, Pid, _} -> +			    ok +		    after 10000 -> +			    %% Pid is probably trapping exits, hit it harder... +			    catch error_logger:warning_msg( +				    "Testcase process ~p not " +				    "responding to timetrap " +				    "timeout:~n" +				    "  ~p.~n" +				    "Killing testcase...~n", +				    [Pid, Trap]), +			    exit(Pid, kill) +		    end; +		false ->  		    ok -	    after 10000 -> -		    %% Pid is probably trapping exits, hit it harder... -		    catch error_logger:warning_msg("Testcase process ~p not " -						   "responding to timetrap " -						   "timeout:~n" -						   "  ~p.~n" -						   "Killing testcase...~n", -						   [Pid, Trap]), -		    exit(Pid, kill)  	    end      end. @@ -87,8 +100,12 @@ timetrap_cancel(Handle) ->      unlink(Handle),      MonRef = erlang:monitor(process, Handle),      exit(Handle, kill), -    receive {'DOWN',MonRef,_,_,_} -> ok after 2000 -> ok end. - +    receive {'DOWN',MonRef,_,_,_} -> ok +    after +	2000 -> +	    erlang:demonitor(MonRef, [flush]), +	    ok +    end.  capture_get(Msgs) ->      receive @@ -98,7 +115,6 @@ capture_get(Msgs) ->  	    lists:reverse(Msgs)      end. -  messages_get(Msgs) ->      receive  	Msg -> @@ -107,7 +123,6 @@ messages_get(Msgs) ->  	    lists:reverse(Msgs)      end. -  timecall(M, F, A) ->      Befor = erlang:now(),      Val = apply(M, F, A), diff --git a/lib/test_server/src/ts.erl b/lib/test_server/src/ts.erl index 729a2b11fc..5fbc0ee017 100644 --- a/lib/test_server/src/ts.erl +++ b/lib/test_server/src/ts.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 1997-2011. All Rights Reserved. +%% Copyright Ericsson AB 1997-2012. All Rights Reserved.  %%  %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -27,9 +27,10 @@  -export([run/0, run/1, run/2, run/3, run/4,  	 clean/0, clean/1,  	 tests/0, tests/1, -	 install/0, install/1, install/2, index/0, +	 install/0, install/1, index/0,  	 estone/0, estone/1,  	 cross_cover_analyse/1, +	 compile_testcases/0, compile_testcases/1,  	 help/0]).  -export([i/0, l/1, r/0, r/1, r/2, r/3]). @@ -88,35 +89,25 @@  -define(     install_help,     [ -    "  ts:install()      - Install TS for local target with no Options.\n" -    "  ts:install([Options])\n", -    "                    - Install TS for local target with Options\n" -    "  ts:install({Architecture, Target_name})\n", -    "                    - Install TS for a remote target architecture.\n", -    "                      and target network name (e.g. {vxworks_cpu32, sauron}).\n", -    "  ts:install({Architecture, Target_name}, [Options])\n", -    "                    - Install TS as above, and with Options.\n", +    "  ts:install()           - Install TS with no Options.\n" +    "  ts:install([Options])  - Install TS with Options\n"      "\n",      "Installation options supported:\n",      "  {longnames, true} - Use fully qualified hostnames\n", -    "  {hosts, [HostList]}\n" -    "                    - Use theese hosts for distributed testing.\n"      "  {verbose, Level}  - Sets verbosity level for TS output (0,1,2), 0 is\n"      "                      quiet(default).\n" -    "  {slavetargets, SlaveTarges}\n" -    "                    - Available hosts for starting slave nodes for\n" -    "                      platforms which cannot have more than one erlang\n" -    "                      node per host.\n" -    "  {crossroot, TargetErlRoot}\n" -    "                    - Erlang root directory on target host\n" -    "                      Mandatory for remote targets\n" -    "  {master, {MasterHost, MasterCookie}}\n" -    "                    - Master host and cookie for targets which are\n" -    "                      started as slave nodes.\n" -    "                      erl_boot_server must be started on master before\n" -    "                      test is run.\n" -    "                      Optional, default is controller host and then\n" -    "                      erl_boot_server is started autmatically\n" +    "  {crossroot, ErlTop}\n" +    "                    - Erlang root directory on build host, ~n" +    "                      normally same value as $ERL_TOP\n" +    "  {crossenv, [{Key,Val}]}\n" +    "                    - Environmentals used by test configure on build host\n" +    "  {crossflags, FlagsString}\n" +    "                    - Flags used by test configure on build host\n" +    "  {xcomp, XCompFile}\n" +    "                    - The xcomp file to use for cross compiling the~n" +    "                      testcases. Using this option will override any~n" +    "                      cross* configurations given to ts. Note that you~n" +    "                      have to have a correct ERL_TOP as well.~n"     ]).  help() -> @@ -183,26 +174,24 @@ help(installed) ->  	 "                      cover_details. Analyses modules specified in\n"  	 "                      cross.cover.\n"  	 "                      Level can be 'overview' or 'details'.\n", +	 "  ts:compile_testcases()~n" +	 "  ts:compile_testcases(Apps)~n" +	 "                    - Compile all testcases for usage in a cross ~n" +	 "                      compile environment."  	 " \n"  	 "Installation (already done):\n"  	],      show_help([H,?install_help]).  show_help(H) -> -    io:put_chars(lists:flatten(H)). +    io:format(lists:flatten(H)).  %% Installs tests.  install() ->      ts_install:install(install_local,[]). -install({Architecture, Target_name})  -> -    ts_install:install({ts_lib:maybe_atom_to_list(Architecture),  -			ts_lib:maybe_atom_to_list(Target_name)}, []);  install(Options) when is_list(Options) ->      ts_install:install(install_local,Options). -install({Architecture, Target_name}, Options) when is_list(Options)-> -    ts_install:install({ts_lib:maybe_atom_to_list(Architecture),  -			ts_lib:maybe_atom_to_list(Target_name)}, Options).  %% Updates the local index page. @@ -301,7 +290,44 @@ run(List, Opts) when is_list(List), is_list(Opts) ->  run(Testspec, Config) when is_atom(Testspec), is_list(Config) ->      Options=check_test_get_opts(Testspec, Config),      File=atom_to_list(Testspec), -    run_test(File, [{spec,[File++".spec"]}], Options); +    WhatToDo = +	case Testspec of +	    %% Known to exist but fails generic tests below +	    emulator -> test; +	    system -> test; +	    erl_interface -> test; +	    epmd -> test; +	    _ -> +		case code:lib_dir(Testspec) of +		    {error,bad_name} -> +			%% Application does not exist +			skip; +		    Path -> +			case file:read_file_info(filename:join(Path,"ebin")) of +			    {ok,#file_info{type=directory}} -> +				%% Erlang application is built +				test; +			    _ -> +				case filelib:wildcard( +				       filename:join([Path,"priv","*.jar"])) of +				    [] -> +					%% The application is not built +					skip; +				    [_|_] -> +					%% Java application is built +					test +				end +			end +		end +	end, +    Spec = +	case WhatToDo of +	    skip -> +		create_skip_spec(Testspec, tests(Testspec)); +	    test -> +		File++".spec" +	end, +    run_test(File, [{spec,[Spec]}], Options);  %% Runs one module in a spec (interactive)  run(Testspec, Mod) when is_atom(Testspec), is_atom(Mod) ->      run_test({atom_to_list(Testspec), Mod},  @@ -332,6 +358,21 @@ run(Testspec, Mod, Case, Config) when is_atom(Testspec),      Args = [{suite,atom_to_list(Mod)}, {testcase,atom_to_list(Case)}],      run_test(atom_to_list(Testspec), Args, Options). +%% Create a spec to skip all SUITES, this is used when the application +%% to be tested is not part of the OTP release to be tested. +create_skip_spec(Testspec, SuitesToSkip) -> +    {ok,Cwd} = file:get_cwd(), +    TestspecString = atom_to_list(Testspec), +    Specname = TestspecString++"_skip.spec", +    {ok,D} = file:open(filename:join([filename:dirname(Cwd), +				      TestspecString++"_test",Specname]), +		       [write]), +    TestDir = "\"../"++TestspecString++"_test\"", +    io:format(D,"{suites, "++TestDir++", all}.~n",[]), +    io:format(D,"{skip_suites, "++TestDir++", ~w, \"Skipped as application" +	      " is not in path!\"}.",[SuitesToSkip]), +    Specname. +  %% Check testspec to be valid and get possible Options  %% from the config.  check_test_get_opts(Testspec, Config) -> @@ -705,3 +746,23 @@ cover_type(cover_details) -> details.  do_load(Mod) ->      code:purge(Mod),      code:load_file(Mod). + + +compile_testcases() -> +    compile_datadirs("../*/*_data"). + +compile_testcases(App) when is_atom(App) -> +    compile_testcases([App]); +compile_testcases([App | T]) -> +    compile_datadirs(io_lib:format("../~s_test/*_data", [App])), +    compile_testcases(T); +compile_testcases([]) -> +    ok. + +compile_datadirs(DataDirs) -> +    {ok,Variables} = file:consult("variables"), + +    lists:foreach(fun(Dir) -> +			  ts_lib:make_non_erlang(Dir, Variables) +		  end, +		  filelib:wildcard(DataDirs)). diff --git a/lib/test_server/src/ts.hrl b/lib/test_server/src/ts.hrl index 885a726c54..5e829f3575 100644 --- a/lib/test_server/src/ts.hrl +++ b/lib/test_server/src/ts.hrl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %%  -%% Copyright Ericsson AB 1997-2009. All Rights Reserved. +%% Copyright Ericsson AB 1997-2012. All Rights Reserved.  %%   %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -28,6 +28,7 @@  -define(run_summary, "suite.summary").  -define(cover_total,"total_cover.log").  -define(variables, "variables"). +-define(cross_variables, "variables-cross").  -define(LF, [10]).                              % Newline in VxWorks script  -define(CHAR_PER_LINE, 60).                     % Characters per VxWorks script building line  -define(CROSS_COOKIE, "cross").                 % cookie used when cross platform testing diff --git a/lib/test_server/src/ts_autoconf_win32.erl b/lib/test_server/src/ts_autoconf_win32.erl index 9103542fd2..4a5c5c7603 100644 --- a/lib/test_server/src/ts_autoconf_win32.erl +++ b/lib/test_server/src/ts_autoconf_win32.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %%  -%% Copyright Ericsson AB 1997-2009. All Rights Reserved. +%% Copyright Ericsson AB 1997-2012. All Rights Reserved.  %%   %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -67,6 +67,7 @@ system_type(Vars) ->  		     {5,1,_} -> "Windows XP";  		     {5,2,_} -> "Windows 2003";  		     {6,0,_} -> "Windows Vista"; +		     {6,1,_} -> "Windows 7";  		     {_,_,_} -> "Windows NCC-1701-D"  		 end;  	     {win32, windows} ->  diff --git a/lib/test_server/src/ts_erl_config.erl b/lib/test_server/src/ts_erl_config.erl index 3b41f90d55..43e56e1098 100644 --- a/lib/test_server/src/ts_erl_config.erl +++ b/lib/test_server/src/ts_erl_config.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 1997-2010. All Rights Reserved. +%% Copyright Ericsson AB 1997-2012. All Rights Reserved.  %%  %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -128,15 +128,15 @@ erts_lib(Vars,OsType) ->  		   ErtsLibInternal}  	  end,      [{erts_lib_include, -      filename:nativename(ErtsLibInclude)}, +      quote(filename:nativename(ErtsLibInclude))},       {erts_lib_include_generated, -      filename:nativename(ErtsLibIncludeGenerated)}, +      quote(filename:nativename(ErtsLibIncludeGenerated))},       {erts_lib_include_internal, -      filename:nativename(ErtsLibIncludeInternal)}, +      quote(filename:nativename(ErtsLibIncludeInternal))},       {erts_lib_include_internal_generated, -      filename:nativename(ErtsLibIncludeInternalGenerated)}, -     {erts_lib_path, filename:nativename(ErtsLibPath)}, -     {erts_lib_internal_path, filename:nativename(ErtsLibInternalPath)}, +      quote(filename:nativename(ErtsLibIncludeInternalGenerated))}, +     {erts_lib_path, quote(filename:nativename(ErtsLibPath))}, +     {erts_lib_internal_path, quote(filename:nativename(ErtsLibInternalPath))},       {erts_lib_multi_threaded, erts_lib_name(multi_threaded, OsType)},       {erts_lib_single_threaded, erts_lib_name(single_threaded, OsType)}       | Vars]. @@ -145,13 +145,13 @@ erl_include(Vars) ->      Include =   	case erl_root(Vars) of  	    {installed, Root} -> -		filename:join([Root, "usr", "include"]); +		quote(filename:join([Root, "usr", "include"]));  	    {srctree, Root, Target} -> -		filename:join([Root, "erts", "emulator", "beam"]) -		    ++ " -I" ++ filename:join([Root, "erts", "emulator"]) +		quote(filename:join([Root, "erts", "emulator", "beam"])) +		    ++ " -I" ++ quote(filename:join([Root, "erts", "emulator"]))  		    ++ system_include(Root, Vars) -		    ++ " -I" ++ filename:join([Root, "erts", "include"]) -		    ++ " -I" ++ filename:join([Root, "erts", "include", Target]) +		    ++ " -I" ++ quote(filename:join([Root, "erts", "include"])) +		    ++ " -I" ++ quote(filename:join([Root, "erts", "include", Target]))  	end,      [{erl_include, filename:nativename(Include)}|Vars]. @@ -163,7 +163,7 @@ system_include(Root, Vars) ->  	    "VxWorks" -> "sys.vxworks";  	    _ -> "sys/unix"  	end, -    " -I" ++ filename:nativename(filename:join([Root, "erts", "emulator", SysDir])). +    " -I" ++ quote(filename:nativename(filename:join([Root, "erts", "emulator", SysDir]))).  erl_interface(Vars,OsType) ->      {Incl, {LibPath, MkIncl}} = @@ -220,20 +220,16 @@ erl_interface(Vars,OsType) ->  		    _ ->   			"" % VxWorks  		end, -    CrossCompile = case OsType of -		       vxworks -> "true"; -		       _ ->       "false" -		   end, -    [{erl_interface_libpath, filename:nativename(LibPath)}, +    [{erl_interface_libpath, quote(filename:nativename(LibPath))},       {erl_interface_sock_libs, sock_libraries(OsType)},       {erl_interface_lib, Lib},       {erl_interface_eilib, Lib1},       {erl_interface_lib_drv, LibDrv},       {erl_interface_eilib_drv, Lib1Drv},       {erl_interface_threadlib, ThreadLib}, -     {erl_interface_include, filename:nativename(Incl)}, -     {erl_interface_mk_include, filename:nativename(MkIncl)}, -     {erl_interface_cross_compile, CrossCompile} | Vars]. +     {erl_interface_include, quote(filename:nativename(Incl))}, +     {erl_interface_mk_include, quote(filename:nativename(MkIncl))} +     | Vars].  ic(Vars, OsType) ->      {ClassPath, LibPath, Incl} = @@ -250,10 +246,10 @@ ic(Vars, OsType) ->  		 end,  		 filename:join(Dir, "include")}  	end, -    [{ic_classpath, filename:nativename(ClassPath)}, -     {ic_libpath, filename:nativename(LibPath)}, +    [{ic_classpath, quote(filename:nativename(ClassPath))}, +     {ic_libpath, quote(filename:nativename(LibPath))},       {ic_lib, link_library("ic", OsType)}, -     {ic_include_path, filename:nativename(Incl)}|Vars]. +     {ic_include_path, quote(filename:nativename(Incl))}|Vars].  jinterface(Vars, _OsType) ->      ClassPath = @@ -263,7 +259,7 @@ jinterface(Vars, _OsType) ->  	    Dir ->  		filename:join([Dir, "priv", "OtpErlang.jar"])  	end, -    [{jinterface_classpath, filename:nativename(ClassPath)}|Vars]. +    [{jinterface_classpath, quote(filename:nativename(ClassPath))}|Vars].  lib_dir(Vars, Lib) ->      LibLibDir = case Lib of @@ -276,8 +272,6 @@ lib_dir(Vars, Lib) ->      case {get_var(crossroot, Vars), LibLibDir} of  	{{error, _}, _} ->			%no crossroot  	    LibLibDir; -	{_, {error, _}} ->			%no lib -	    LibLibDir;  	{CrossRoot, _} ->  	    %% XXX: Ugly. So ugly I won't comment it  	    %% /Patrik @@ -299,18 +293,16 @@ lib_dir(Vars, Lib) ->      end.  erl_root(Vars) -> -    Root = code:root_dir(), -    case ts_lib:erlang_type() of +    Root = case get_var(crossroot,Vars) of +	       {error, notfound} -> code:root_dir(); +	       CrossRoot -> CrossRoot +	   end, +    case ts_lib:erlang_type(Root) of  	{srctree, _Version} ->  	    Target = get_var(target, Vars),  	    {srctree, Root, Target};  	{_, _Version} -> -	    case get_var(crossroot,Vars) of -		{error, notfound} -> -		    {installed, Root}; -		CrossRoot -> -		    {installed, CrossRoot} -	    end +	    {installed, Root}      end. @@ -362,10 +354,17 @@ ssl(Vars, _OsType) ->  	{error, bad_name} ->  	    throw({cannot_find_app, ssl});  	Dir -> -	    [{ssl_libdir, filename:nativename(Dir)}| Vars] +	    [{ssl_libdir, quote(filename:nativename(Dir))}| Vars]      end.  separators(Vars, {win32,_}) ->      [{'DS',"\\"},{'PS',";"}|Vars];  separators(Vars, _) ->      [{'DS',"/"},{'PS',":"}|Vars]. + +quote([$ |R]) -> +    "\\ "++quote(R); +quote([C|R]) -> +    [C|quote(R)]; +quote([]) -> +    []. diff --git a/lib/test_server/src/ts_install.erl b/lib/test_server/src/ts_install.erl index 9703478f20..caf00759e5 100644 --- a/lib/test_server/src/ts_install.erl +++ b/lib/test_server/src/ts_install.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 1997-2011. All Rights Reserved. +%% Copyright Ericsson AB 1997-2012. All Rights Reserved.  %%  %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -28,12 +28,25 @@ install(install_local, Options) ->      install(os:type(), Options);  install(TargetSystem, Options) -> -    io:format("Running configure for cross architecture, network target name~n" -	      "~p~n", [TargetSystem]), -    case autoconf(TargetSystem) of +    case file:consult(?variables) of +	{ok, Vars} -> +	    case proplists:get_value(cross,Vars) of +		"yes" when Options == []-> +		    target_install(Vars); +		_ -> +		    build_install(TargetSystem, Options) +	    end; +	_ -> +	    build_install(TargetSystem, Options) +    end. + + +build_install(TargetSystem, Options) -> +    XComp = parse_xcomp_file(proplists:get_value(xcomp,Options)), +    case autoconf(TargetSystem, XComp++Options) of  	{ok, Vars0} ->  	    OsType = os_type(TargetSystem), -	    Vars1 = ts_erl_config:variables(merge(Vars0,Options),OsType), +	    Vars1 = ts_erl_config:variables(Vars0++XComp++Options,OsType),  	    {Options1, Vars2} = add_vars(Vars1, Options),  	    Vars3 = lists:flatten([Options1|Vars2]),  	    write_terms(?variables, Vars3); @@ -45,32 +58,43 @@ os_type({unix,_}=OsType) -> OsType;  os_type({win32,_}=OsType) -> OsType;  os_type(_Other) -> vxworks. -merge(Vars,[]) -> -    Vars; -merge(Vars,[{crossroot,X}| Tail]) -> -    merge([{crossroot, X} | Vars], Tail); -merge(Vars,[_X | Tail]) -> -    merge(Vars,Tail). +target_install(CrossVars) -> +    io:format("Cross installation detected, skipping configure and data_dir make~n"), +    case file:rename(?variables,?cross_variables) of +	ok -> +	    ok; +	_ -> +	    io:format("Could not find variables file from cross make~n"), +	    throw(cross_installation_failed) +    end, +    CPU = proplists:get_value('CPU',CrossVars), +    OS = proplists:get_value(os,CrossVars), +    {Options,Vars} = add_vars([{cross,"yes"},{'CPU',CPU},{os,OS}],[]), +    Variables = lists:flatten([Options|Vars]), +    write_terms(?variables, Variables).  %% Autoconf for various platforms.  %% unix uses the configure script  %% win32 uses ts_autoconf_win32  %% VxWorks uses ts_autoconf_vxworks. -autoconf(TargetSystem) -> -    case autoconf1(TargetSystem) of +autoconf(TargetSystem, XComp) -> +    case autoconf1(TargetSystem, XComp) of  	ok ->  	    autoconf2(file:read_file("conf_vars"));  	Error ->  	    Error      end. -autoconf1({win32, _}) -> +autoconf1({win32, _},[{cross,"no"}]) ->      ts_autoconf_win32:configure(); -autoconf1({unix, _}) -> -    unix_autoconf(); -autoconf1(Other) -> -    ts_autoconf_vxworks:configure(Other). +autoconf1({unix, _},XCompFile) -> +    unix_autoconf(XCompFile); +autoconf1(Other,[{cross,"no"}]) -> +    ts_autoconf_vxworks:configure(Other); +autoconf1(_,_) -> +    io:format("cross compilation not supported for that this platform~n"), +    throw(cross_installation_failed).  autoconf2({ok, Bin}) ->      get_vars(binary_to_list(Bin), name, [], []); @@ -92,27 +116,40 @@ get_vars([], name, [], Result) ->  get_vars(_, _, _, _) ->      {error, fatal_bad_conf_vars}. -unix_autoconf() -> +unix_autoconf(XConf) ->      Configure = filename:absname("configure"), -    Args = case catch erlang:system_info(threads) of -	       false -> ""; -	       _ -> " --enable-shlib-thread-safety" -	   end -	++ case catch string:str(erlang:system_info(system_version), -				 "debug") > 0 of -	       false -> ""; -	       _ -> " --enable-debug-mode" -	   end, +    Flags = proplists:get_value(crossflags,XConf,[]), +    Env = proplists:get_value(crossenv,XConf,[]), +    Host = get_xcomp_flag("host", Flags), +    Build = get_xcomp_flag("build", Flags), +    Threads = [" --enable-shlib-thread-safety" || +		  erlang:system_info(threads) /= false], +    Debug = [" --enable-debug-mode" || +		string:str(erlang:system_info(system_version),"debug") > 0], +    Args = Host ++ Build ++ Threads ++ Debug,      case filelib:is_file(Configure) of  	true -> -	    Env = macosx_cflags(), -	    Port = open_port({spawn, Configure ++ Args}, -			     [stream, eof, {env,Env}]), +	    OSXEnv = macosx_cflags(), +	    io:format("Running ~sEnv: ~p~n", +		      [lists:flatten(Configure ++ Args),Env++OSXEnv]), +	    Port = open_port({spawn, lists:flatten(["\"",Configure,"\"",Args])}, +			     [stream, eof, {env,Env++OSXEnv}]),  	    ts_lib:print_data(Port);  	false ->  	    {error, no_configure_script}      end. + +get_xcomp_flag(Flag, Flags) -> +    get_xcomp_flag(Flag, Flag, Flags). +get_xcomp_flag(Flag, Tag, Flags) -> +    case proplists:get_value(Flag,Flags) of +	undefined -> ""; +	"guess" -> [" --",Tag,"=",os:cmd("$ERL_TOP/erts/autoconf/config.guess")]; +	HostVal -> [" --",Tag,"=",HostVal] +    end. + +  macosx_cflags() ->      case os:type() of  	{unix, darwin} -> @@ -125,10 +162,33 @@ macosx_cflags() ->  	    []      end. +parse_xcomp_file(undefined) -> +    [{cross,"no"}]; +parse_xcomp_file(Filepath) -> +    {ok,Bin} = file:read_file(Filepath), +    Lines = binary:split(Bin,<<"\n">>,[global,trim]), +    {Envs,Flags} = parse_xcomp_file(Lines,[],[]), +    [{cross,"yes"},{crossroot,os:getenv("ERL_TOP")}, +     {crossenv,Envs},{crossflags,Flags}]. + +parse_xcomp_file([<<A:8,_/binary>> = Line|R],Envs,Flags) +  when $A =< A, A =< $Z -> +    [Var,Value] = binary:split(Line,<<"=">>), +    parse_xcomp_file(R,[{binary_to_list(Var), +			 binary_to_list(Value)}|Envs],Flags); +parse_xcomp_file([<<"erl_xcomp_",Line/binary>>|R],Envs,Flags) -> +    [Var,Value] = binary:split(Line,<<"=">>), +    parse_xcomp_file(R,Envs,[{binary_to_list(Var), +			      binary_to_list(Value)}|Flags]); +parse_xcomp_file([_|R],Envs,Flags) -> +    parse_xcomp_file(R,Envs,Flags); +parse_xcomp_file([],Envs,Flags) -> +    {lists:reverse(Envs),lists:reverse(Flags)}. +  write_terms(Name, Terms) ->      case file:open(Name, [write]) of  	{ok, Fd} -> -	    Result = write_terms1(Fd, Terms), +	    Result = write_terms1(Fd, remove_duplicates(Terms)),  	    file:close(Fd),  	    Result;  	{error, Reason} -> @@ -141,6 +201,17 @@ write_terms1(Fd, [Term|Rest]) ->  write_terms1(_, []) ->      ok. +remove_duplicates(List) -> +    lists:reverse( +      lists:foldl(fun({Key,Val},Acc) -> +			  R = make_ref(), +			  case proplists:get_value(Key,Acc,R) of +			      R -> [{Key,Val}|Acc]; +			      _Else -> +				  Acc +			  end +		  end,[],List)). +  add_vars(Vars0, Opts0) ->      {Opts,LongNames} =  	case lists:keymember(longnames, 1, Opts0) of @@ -209,12 +280,11 @@ platform(Vars) ->      LC = lock_checking(),      MT = modified_timing(),      AsyncThreads = async_threads(), -    HeapType = heap_type_label(),      Debug = debug(),      CpuBits = word_size(),      Common = lists:concat([Hostname,"/",OsType,"/",CpuType,CpuBits,LinuxDist,  			   Schedulers,BindType,KP,IOTHR,LC,MT,AsyncThreads, -			   HeapType,Debug,ExtraLabel]), +			   Debug,ExtraLabel]),      PlatformId = lists:concat([ErlType, " ", Version, Common]),      PlatformLabel = ErlType ++ Common,      PlatformFilename = platform_as_filename(PlatformId), @@ -272,12 +342,6 @@ hostname() ->  	    "/localhost"      end. -heap_type_label() -> -    case catch erlang:system_info(heap_type) of -	hybrid -> "/Hybrid"; -	_ -> "" %private -    end. -  async_threads() ->      case catch erlang:system_info(threads) of  	true -> "/A"++integer_to_list(erlang:system_info(thread_pool_size)); diff --git a/lib/test_server/src/ts_install_cth.erl b/lib/test_server/src/ts_install_cth.erl index a41916fd0a..9b6e10e7e2 100644 --- a/lib/test_server/src/ts_install_cth.erl +++ b/lib/test_server/src/ts_install_cth.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2010-2011. All Rights Reserved. +%% Copyright Ericsson AB 2010-2012. All Rights Reserved.  %%  %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -95,17 +95,12 @@ pre_init_per_suite(_Suite,Config,State) ->      try  	{ok,Variables} =   	    file:consult(filename:join(State#state.ts_conf_dir,"variables")), - -	%% Make the stuff in all_SUITE_data if it exists -	AllDir = filename:join(DataDir,"../all_SUITE_data"), -	case filelib:is_dir(AllDir) of -	    true -> -		make_non_erlang(AllDir,Variables); -	    false -> -		ok +	case proplists:get_value(cross,Variables) of +	    "yes" -> +		ct:log("Not making data dir as tests have been cross compiled"); +	    _ -> +		ts_lib:make_non_erlang(DataDir, Variables)  	end, -	 -	make_non_erlang(DataDir, Variables),  	{add_node_name(Config, State), State}      catch Error:Reason -> @@ -219,39 +214,6 @@ terminate(_State) ->  %%% ============================================================================  %%% Local functions  %%% ============================================================================ -%% Configure and run all the Makefiles in the data dirs of the suite  -%% in question -make_non_erlang(DataDir, Variables) -> -    {ok,CurrWD} = file:get_cwd(), -    try -	file:set_cwd(DataDir), -	MakeCommand = proplists:get_value(make_command,Variables), -	 -	FirstMakefile = filename:join(DataDir,"Makefile.first"), -	case filelib:is_regular(FirstMakefile) of -	    true -> -		ct:log("Making ~p",[FirstMakefile]), -		ok = ts_make:make( -		       MakeCommand, DataDir, filename:basename(FirstMakefile)); -	    false -> -		ok -	end, -	 -	MakefileSrc = filename:join(DataDir,"Makefile.src"), -	MakefileDest = filename:join(DataDir,"Makefile"), -	case filelib:is_regular(MakefileSrc) of -	    true -> -		ok = ts_lib:subst_file(MakefileSrc,MakefileDest,Variables), -		ct:log("Making ~p",[MakefileDest]), -		ok = ts_make:make([{makefile,"Makefile"},{data_dir,DataDir}  -				   | Variables]); -	    false -> -		ok -	end -    after -	file:set_cwd(CurrWD), -	timer:sleep(100) -    end.  %% Add a nodename to config if it does not exist  add_node_name(Config, State) -> diff --git a/lib/test_server/src/ts_lib.erl b/lib/test_server/src/ts_lib.erl index 2f0a4ea8c0..ea97361bd3 100644 --- a/lib/test_server/src/ts_lib.erl +++ b/lib/test_server/src/ts_lib.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 1997-2010. All Rights Reserved. +%% Copyright Ericsson AB 1997-2012. All Rights Reserved.  %%  %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -24,10 +24,12 @@  %% Avoid warning for local function error/1 clashing with autoimported BIF.  -compile({no_auto_import,[error/1]}).  -export([error/1, var/2, erlang_type/0, +	 erlang_type/1,  	 initial_capital/1, interesting_logs/1,   	 specs/1, suites/2, last_test/1,  	 force_write_file/2, force_delete/1,  	 subst_file/3, subst/2, print_data/1, +	 make_non_erlang/2,  	 maybe_atom_to_list/1, progress/4  	]). @@ -73,8 +75,10 @@ progress(Vars, Level, Format, Args) ->  %% Returns: {Type, Version} where Type is otp|src  erlang_type() -> +    erlang_type(code:root_dir()). +erlang_type(RootDir) ->      {_, Version} = init:script_id(), -    RelDir = filename:join(code:root_dir(), "releases"), % Only in installed +    RelDir = filename:join(RootDir, "releases"), % Only in installed      case filelib:is_file(RelDir) of  	true -> {otp,Version};			% installed OTP  	false -> {srctree,Version}		% source code tree @@ -333,3 +337,45 @@ maybe_atom_to_list(To_list) when is_list(To_list) ->  maybe_atom_to_list(To_list) when is_atom(To_list)->      atom_to_list(To_list). + +%% Configure and run all the Makefiles in the data dir of the suite +%% in question +make_non_erlang(DataDir, Variables) -> +    %% Make the stuff in all_SUITE_data if it exists +    AllDir = filename:join(DataDir,"../all_SUITE_data"), +    case filelib:is_dir(AllDir) of +	true -> +	    make_non_erlang_do(AllDir,Variables); +	false -> +	    ok +    end, +    make_non_erlang_do(DataDir, Variables). + +make_non_erlang_do(DataDir, Variables) -> +    try +	MakeCommand = proplists:get_value(make_command,Variables), + +	FirstMakefile = filename:join(DataDir,"Makefile.first"), +	case filelib:is_regular(FirstMakefile) of +	    true -> +		io:format("Making ~p",[FirstMakefile]), +		ok = ts_make:make( +		       MakeCommand, DataDir, filename:basename(FirstMakefile)); +	    false -> +		ok +	end, + +	MakefileSrc = filename:join(DataDir,"Makefile.src"), +	MakefileDest = filename:join(DataDir,"Makefile"), +	case filelib:is_regular(MakefileSrc) of +	    true -> +		ok = ts_lib:subst_file(MakefileSrc,MakefileDest,Variables), +		io:format("Making ~p",[MakefileDest]), +		ok = ts_make:make([{makefile,"Makefile"},{data_dir,DataDir} +				   | Variables]); +	    false -> +		ok +	end +    after +	timer:sleep(100)  %% maybe unnecessary now when we don't do set_cwd anymore +    end. diff --git a/lib/test_server/src/ts_make.erl b/lib/test_server/src/ts_make.erl index 3df66111a3..f3266f5836 100644 --- a/lib/test_server/src/ts_make.erl +++ b/lib/test_server/src/ts_make.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %%  -%% Copyright Ericsson AB 1997-2009. All Rights Reserved. +%% Copyright Ericsson AB 1997-2012. All Rights Reserved.  %%   %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -25,12 +25,12 @@  %% Functions to be called from make test cases.  make(Config) when is_list(Config) -> -    DataDir = ?config(data_dir, Config), -    Makefile = ?config(makefile, Config), -    Make = ?config(make_command, Config), +    DataDir = proplists:get_value(data_dir, Config), +    Makefile = proplists:get_value(makefile, Config), +    Make = proplists:get_value(make_command, Config),      case make(Make, DataDir, Makefile) of  	ok -> ok; -	{error,Reason} -> ?t:fail({make_failed,Reason}) +	{error,Reason} -> exit({make_failed,Reason})      end.  unmake(Config) when is_list(Config) -> @@ -85,7 +85,7 @@ run_make_script({win32, _}, Make, Dir, Makefile) ->      {"run_make.bat",       ".\\run_make",       ["@echo off\r\n", -      "cd ", filename:nativename(Dir), "\r\n", +      "cd \"", filename:nativename(Dir), "\"\r\n",        Make, " -f ", Makefile, " \r\n",        "if errorlevel 1 echo *error*\r\n",        "if not errorlevel 1 echo *ok*\r\n"]}; @@ -93,7 +93,7 @@ run_make_script({unix, _}, Make, Dir, Makefile) ->      {"run_make",        "/bin/sh ./run_make",       ["#!/bin/sh\n", -      "cd ", Dir, "\n", +      "cd \"", Dir, "\"\n",        Make, " -f ", Makefile, " 2>&1\n",        "case $? in\n",        "  0) echo '*ok*';;\n", diff --git a/lib/test_server/src/ts_run.erl b/lib/test_server/src/ts_run.erl index 885a3c9b96..95e3c08d5b 100644 --- a/lib/test_server/src/ts_run.erl +++ b/lib/test_server/src/ts_run.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 1997-2011. All Rights Reserved. +%% Copyright Ericsson AB 1997-2012. All Rights Reserved.  %%  %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -229,7 +229,7 @@ make_command(Vars, Spec, State) ->  	   %% uncomment the line below to disable exception formatting   	   %%	   " -test_server_format_exception false",  	   " -boot start_sasl -sasl errlog_type error", -	   " -pz ",Cwd, +	   " -pz \"",Cwd,"\"",  	   " -ct_test_vars ",TestVars,  	   " -eval \"file:set_cwd(\\\"",TestDir,"\\\")\" "  	   " -eval \"ct:run_test(",  @@ -334,9 +334,9 @@ path_separator() ->      end. -make_common_test_args(Args0, Options, _Vars) -> +make_common_test_args(Args0, Options0, _Vars) ->      Trace =  -	case lists:keysearch(trace,1,Options) of +	case lists:keysearch(trace,1,Options0) of  	    {value,{trace,TI}} when is_tuple(TI); is_tuple(hd(TI)) ->  		ok = file:write_file(?tracefile,io_lib:format("~p.~n",[TI])),  		[{ct_trace,?tracefile}]; @@ -348,7 +348,7 @@ make_common_test_args(Args0, Options, _Vars) ->  		[]  	end,      Cover =  -	case lists:keysearch(cover,1,Options) of +	case lists:keysearch(cover,1,Options0) of  	    {value,{cover, App, none, _Analyse}} ->  		io:format("No cover file found for ~p~n",[App]),  		[]; @@ -358,7 +358,7 @@ make_common_test_args(Args0, Options, _Vars) ->  		[]  	end, -    Logdir = case lists:keysearch(logdir, 1, Options) of +    Logdir = case lists:keysearch(logdir, 1, Options0) of  		  {value,{logdir, _}} ->  		      [];  		  false -> @@ -373,15 +373,16 @@ make_common_test_args(Args0, Options, _Vars) ->  			{scale_timetraps, true}]  	       end, -    ConfigPath = case {os:getenv("TEST_CONFIG_PATH"), -		       lists:keysearch(config, 1, Options)} of -		     {false,{value, {config, Path}}} -> -			 Path; -		     {false,false} -> -			 "../test_server"; -		     {Path,_} -> -			 Path -		 end, +    {ConfigPath, +     Options} = case {os:getenv("TEST_CONFIG_PATH"), +		      lists:keysearch(config, 1, Options0)} of +		    {_,{value, {config, Path}}} -> +			{Path,lists:keydelete(config, 1, Options0)}; +		    {false,false} -> +			{"../test_server",Options0}; +		    {Path,_} -> +			{Path,Options0} +		end,      ConfigFiles = [{config,[filename:join(ConfigPath,File)  			    || File <- get_config_files()]}],      io_lib:format("~100000p",[Args0++Trace++Cover++Logdir++ diff --git a/lib/test_server/test/Makefile b/lib/test_server/test/Makefile index 198440bb17..afe5aff196 100644 --- a/lib/test_server/test/Makefile +++ b/lib/test_server/test/Makefile @@ -1,7 +1,7 @@  #  # %CopyrightBegin%  #  -# Copyright Ericsson AB 1997-2011. All Rights Reserved. +# Copyright Ericsson AB 1997-2012. All Rights Reserved.  #   # The contents of this file are subject to the Erlang Public License,  # Version 1.1, (the "License"); you may not use this file except in @@ -83,10 +83,10 @@ include $(ERL_TOP)/make/otp_release_targets.mk  release_spec: opt  release_tests_spec: make_emakefile -	$(INSTALL_DIR) $(RELSYSDIR) -	$(INSTALL_DATA) $(EMAKEFILE) $(ERL_FILES) $(COVERFILE) $(RELSYSDIR) -	$(INSTALL_DATA) test_server_test_lib.hrl test_server.spec test_server.cover $(RELSYSDIR) -	chmod -R u+w $(RELSYSDIR) -	@tar cf - *_SUITE_data | (cd $(RELSYSDIR); tar xf -) +	$(INSTALL_DIR) "$(RELSYSDIR)" +	$(INSTALL_DATA) $(EMAKEFILE) $(ERL_FILES) $(COVERFILE) "$(RELSYSDIR)" +	$(INSTALL_DATA) test_server_test_lib.hrl test_server.spec test_server.cover "$(RELSYSDIR)" +	chmod -R u+w "$(RELSYSDIR)" +	@tar cf - *_SUITE_data | (cd "$(RELSYSDIR)"; tar xf -)  release_docs_spec: diff --git a/lib/test_server/test/test_server_SUITE_data/Makefile.src b/lib/test_server/test/test_server_SUITE_data/Makefile.src index d5af919eec..332b855df6 100644 --- a/lib/test_server/test/test_server_SUITE_data/Makefile.src +++ b/lib/test_server/test/test_server_SUITE_data/Makefile.src @@ -1,2 +1,7 @@  all: -	erlc *.erl
\ No newline at end of file +	erlc test_server_SUITE.erl +	erlc test_server_parallel01_SUITE.erl +	erlc test_server_conf01_SUITE.erl +	erlc test_server_shuffle01_SUITE.erl +	erlc test_server_conf02_SUITE.erl +	erlc test_server_skip_SUITE.erl
\ No newline at end of file diff --git a/lib/test_server/vsn.mk b/lib/test_server/vsn.mk index 563c1b6db6..b956ebb2b3 100644 --- a/lib/test_server/vsn.mk +++ b/lib/test_server/vsn.mk @@ -1,2 +1 @@ -TEST_SERVER_VSN = 3.4.5 - +TEST_SERVER_VSN = 3.5.3 | 
