URL
https://opencores.org/ocsvn/or1k/or1k/trunk
Subversion Repositories or1k
[/] [or1k/] [trunk/] [ecos-2.0/] [tools/] [src/] [infra/] [hosttest.exp] - Rev 1254
Go to most recent revision | Compare with Previous | Blame | View Log
#===============================================================================
#
# hosttest.exp
#
# Support for host-side testing
#
#===============================================================================
######COPYRIGHTBEGIN####
#
# ----------------------------------------------------------------------------
# Copyright (C) 1998, 1999, 2000, 2001 Red Hat, Inc.
#
# This file is part of the eCos host tools.
#
# This program 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.,
# 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
#
# ----------------------------------------------------------------------------
#
######COPYRIGHTEND####
#===============================================================================
######DESCRIPTIONBEGIN####
#
# Author(s): bartv
# Contributors: bartv
# Date: 1998-11-25
# Note: Arguably this should be a loadable package
#
#####DESCRIPTIONEND####
#===============================================================================
#
# ----------------------------------------------------------------------------
# This script gets loaded by host-side DejaGnu test harnesses to provide
# various utilities for testing eCos host applications. It lives in the
# host-side infrastructure directory and gets installed in
# $(PREFIX)/share/dejagnu.
#
# The script can assume that a number of globals from the site.exp
# file have been read in. These include:
# tool - name of the tool (i.e. the package)
# srcdir - location of the source directory
# objdir - location of the build tree
# host_alias - config triplet
# host_triplet - ditto
#
# The generated Makefile has some additional information that is useful.
# CC - name of the C compiler that is used
# CXX - name of the C++ compiler
# prefix - where everything gets installed
# OBJEXT - either o or obj
# EXEEXT - either nothing or .exe
# VERSION - the version number
# CFLAGS - flags to use when compiling C code
# CXXFLAGS - flags to use when compiling C++ code
#
# Some additional variables should be present in any generated
# makefiles in the eCos tree.
# INCLUDES - header file search path
# LIBS - libraries, search paths, ...
#
# hosttest_initialize
# Perform any initialization steps that are needed. Currently this
# means reading in the Makefile from the top-level of the build tree
# and figuring out the values of CC, CXX, and anything else that is
# useful. Any errors should be reported via perror and then the
# script should exit.
#
# There is an optional argument, a list of additional variables which
# should be present in the makefile and whose values are desired.
proc hosttest_initialize { { pkg_vars {} } } {
# First check that this script is actually running inside DejaGnu
if { [info exists ::objdir] == 0 } {
puts "Variable ::objdir is not defined, is this script really running inside DejaGnu?"
exit 1
}
# The information is stored in an array hosttest_data. Make sure this
# array exists.
array set ::hosttest_data {}
# Now clear out any entries in the array
foreach entry [array names ::hosttest_data] {
unset ::hosttest_data($entry)
}
# Now read in the build tree's Makefile (and not the testsuite's Makefile)
set filename [file join $::objdir .. Makefile]
if { [file exists $filename] == 0 } {
perror "Initialization error: the build tree's Makefile $filename does not exist."
exit 1
}
set status [ catch {
set fd [open $filename r]
set contents [read $fd]
close $fd
} message]
if { $status != 0 } {
perror "Error reading $filename.\n$message"
exit 1
}
# The data is available. Search it for each of the variables of
# interest. Some variables are optional and are given default
# values.
set ::hosttest_data(CFLAGS) ""
set ::hosttest_data(CXXFLAGS) ""
set lines [split $contents "\n"]
foreach var [concat { CC CXX prefix OBJEXT EXEEXT VERSION CFLAGS CXXFLAGS INCLUDES LIBS } $pkg_vars] {
set pattern "^$var\[ \t\]*:?=\[ \t\]* (.*)\$"
set dummy ""
set match ""
foreach line $lines {
if { [regexp -- $pattern $line dummy match] == 1 } {
set ::hosttest_data($var) $match
break
}
}
if { [info exists ::hosttest_data($var)] == 0 } {
perror "Variable $var is not defined in $filename"
exit 1
}
}
# If compiling with VC++ remove any cygwin-isms from the prefix
if { [string match "cl*" $::hosttest_data(CC)] } {
set status [catch "exec cygpath -w $::hosttest_data(prefix)" message]
if { $status == 0 } {
regsub -all -- {\\} $message {/} ::hosttest_data(prefix)
} else {
perror "Converting cygwin pathname $::hosttest_data(prefix)\n$message"
exit 1
}
}
}
# ----------------------------------------------------------------------------
# hosttest_extract_version
# Assuming there has been a call to initialize, the required information
# should be available in the hosttest_data array. The initialize
# function should have aborted if the data is not available.
proc hosttest_extract_version { } {
if { [info exists ::hosttest_data(VERSION)] == 0 } {
error "No version information - host testing has not been properly initialized."
}
if { [info exists ::objdir] == 0 } {
error "Variable ::objdir is not defined, is this script really running inside DejaGnu?"
}
return $::hosttest_data(VERSION)
}
# ----------------------------------------------------------------------------
# hosttest_compile
# compile and link one or more source files. The arguments are:
# 1) the name of the test case
# 2) a list of one or more source files that need to be compiled.
# Both .c and .cxx files are supported, and the appropriate
# compiler will be used. If this list is empty then the
# code will look for a .c or a .cxx file which matches the
# name of the test executable. Source files are assumed to
# be relative to $::srcdir/$::subdir
# 3) a list (possibly empty) of directories that should be in the
# include path. The build tree's directory is automatically in
# the path, as is $(PREFIX)/include. Note that the build tree
# is actually one level above objdir, on the assumption that
# objdir is the testsuite subdirectory of the real objdir.
# 4) ditto for library search paths.
# 5) and a list of additional libraries that should be linked.
#
# Currently it is not possible to pass compiler flags since those
# might need translating between gcc and VC++. This may have to be
# resolved.
#
# Currently linking is not done via libtool. This may have to change.
#
# The various object files and the executable are placed in a directory
# testcase in the build tree, to avoid the risk of name clashes. This
# directory must not exist yet. There is a separate routine hosttest_clean
# which simply expunges the entire testcase directory.
#
# The output of a succesful compile or built is reported using
# verbose at level 2. Unsuccesful compiles or builts are reported using
# level 1.
proc hosttest_compile { name sources includes libdirs libs } {
# Make sure that the testcase directory does not yet exist, then
# create it. This guarantees a clean system and reasonable access
# permissions. Each testcase invocation should involve a call to
# the clean function.
set dirname [file join $::objdir "testcase"]
if { [file exists $dirname] != 0 } {
# An empty directory is ok.
if { [llength [glob -nocomplain -- [file join $dirname "*"]]] != 0 } {
error "hosttest_compile: $dirname already exists"
}
}
set status [catch { file mkdir $dirname } message]
if { $status != 0 } {
error "hosttest_compile: unable to create directory $dirname"
}
# The only argument that must be valid is the test name.
if { $name == "" } {
error "hosttest_compile: invalid test case name"
}
# If the list of sources is empty then look for a suitable
# file in the appropriate directory.
if { [llength $sources] == 0 } {
set filename [file join $::srcdir $::subdir "${name}.c"]
if { [file exists $filename] && [file isfile $filename] } {
lappend sources [file tail $filename]
} else {
set filename [file join $::srcdir $::subdir "${name}.cxx"]
if { [file exists $filename] && [file isfile $filename] } {
lappend sources [file tail $filename]
} else {
error "hosttest_compile: no sources listed and unable to find ${name}.c or ${name}.cxx"
}
}
}
# For each source file, generate a compile command line and try to execute
# it. The command line takes the form:
# (CC|CXX) -c (CFLAGS|CXXFLAGS) (INCDIRS) -o xxx yyy
#
# It is also useful to produce a list of the object files that need to
# linked later on, and to work out which tool should be invoked for
# linking.
set object_files {}
set has_cxx_files 0
foreach source $sources {
set commandline ""
if { [file extension $source] == ".c" } {
append commandline "$::hosttest_data(CC) -c $::hosttest_data(CFLAGS) "
} elseif { [file extension $source] == ".cxx" } {
set has_cxx_files 1
append commandline "$::hosttest_data(CXX) -c $::hosttest_data(CXXFLAGS) "
} else {
error "hosttest_compile: files of type [file extension $source] ($source) are not yet supported."
}
# Include path: start with the source tree. Then the build tree.
# Then the makefile's INCLUDES variable.
# Then any additional directories specified explicitly by the
# testcase. Finish off with the prefix. Note that header files
# in the prefix directory may be out of date, depending on whether
# or not there has been an install recently.
append commandline "-I[file join [pwd] [file dirname $::srcdir]] "
append commandline "-I[file join [pwd] [file dirname $::objdir]] "
append commandline "$::hosttest_data(INCLUDES) "
foreach dir $includes {
append commandline "-I[file join [pwd] $dir] "
}
append commandline "-I[file join [pwd] $::hosttest_data(prefix) include] "
# The output file must go into the testcase directory and have the right suffix
set objfile "[file root [file tail $source]].$::hosttest_data(OBJEXT)"
lappend object_files $objfile
if { [string match "cl*" $::hosttest_data(CC)] } {
append commandline "-Fo[file join $::objdir testcase $objfile] "
} else {
append commandline "-o [file join $::objdir testcase $objfile] "
}
# Finally provide the source file.
append commandline "[file join $::srcdir $::subdir $source]"
verbose -log -- $commandline
# Time to invoke the compiler.
set status [catch { set result [eval exec -keepnewline -- $commandline] } message]
if { $status == 0 } {
# The compile succeeded and the output is in result. Report the
# output.
verbose -log -- $result
} else {
# The compile failed and the output is in message.
verbose -log -- $message
error "hosttest_compile: failed to compile $source"
}
}
# At this stage all the source files have been compiled, a list of
# object files has been constructed, and it is known whether or
# not any of the sources were c++. Time to construct a new command
# line.
set commandline ""
if { $has_cxx_files == 0 } {
append commandline "$::hosttest_data(CC) $::hosttest_data(CFLAGS) "
} else {
append commandline "$::hosttest_data(CXX) $::hosttest_data(CXXFLAGS) "
}
set exename [file join $::objdir "testcase" "$name$::hosttest_data(EXEEXT)"]
# List all of the object files
foreach obj $object_files {
append commandline "[file join $::objdir "testcase" $obj] "
}
# Now take care of libraries and search paths. This requires different
# code for gcc and VC++.
if { [string match "cl*" $::hosttest_data(CC)] } {
append commandline "-Fe$exename "
foreach lib $libs {
append commandline "${lib}.lib "
}
append commandline "$::hosttest_data(LIBS) "
append commandline "-libpath=[file join [pwd] [file dirname $::objdir]] "
foreach dir $libdirs {
append commandline "-libpath=[file join [pwd] $dir] "
}
append commandline "-libpath=[file join [pwd] $::hosttest_data(prefix) lib] "
} else {
append commandline "-o $exename "
append commandline "-L[file join [pwd] [file dirname $::objdir]] "
foreach dir $libdirs {
append commandline "-L[file join [pwd] $dir] "
}
append commandline "-L[file join [pwd] $::hosttest_data(prefix) lib] "
foreach lib $libs {
append commandline "-l$lib "
}
append commandline "$::hosttest_data(LIBS) "
}
# We have a command line, go for it.
verbose -log -- $commandline
set status [catch { set result [eval exec -keepnewline -- $commandline] } message]
if { $status == 0 } {
# The link has succeeded, we have an executable.
verbose -log -- $result
} else {
# The link failed and the output is in message.
# Report things are per compilation failures
verbose -log -- $message
error "hosttest_compile: failed to link $exename"
}
# There should be a test executable.
}
# ----------------------------------------------------------------------------
# hosttest_clean
# Clean up a testcase directory.
proc hosttest_clean { } {
set dirname [file join $::objdir "testcase"]
if { [file exists $dirname] == 0 } {
# Something must have gone seriously wrong during the build phase,
# there is nothing there.
return
}
if { [file isdirectory $dirname] == 0 } {
error "hosttest_clean: $dirname should be a directory"
}
foreach entry [glob -nocomplain -- [file join $dirname "*"]] {
set filename [file join $dirname $entry]
if { [file isfile $filename] == 0 } {
error "hosttest_clean: $filename is not a file"
}
set status [catch { file delete -force -- $filename } message]
if { $status != 0 } {
error "hosttest_clean: unable to delete $filename, $message"
}
}
set status [catch { file delete -force -- $dirname } message]
if { $status != 0 } {
error "hosttest_clean: unable to delete directory $dirname, $message"
}
}
# ----------------------------------------------------------------------------
# Run a test executable, returning the status code and the output.
# The results are returned in variables. It is assumed that some test cases
# will fail, so raising an exception is appropriate only if something
# has gone wrong at the test harness level. The argument list
# should be the name of the test case (from which the executable file name
# can be derived) and a list of arguments.
proc hosttest_run { result_arg output_arg test args } {
upvar $result_arg result
upvar $output_arg output
# Figure out the filename corresponding to the test and make
# sure it exists.
set filename [file join $::objdir "testcase" $test]
append filename $::hosttest_data(EXEEXT)
if { ([file exists $filename] == 0) || ([file isfile $filename] == 0) } {
error "hosttest_run: testcase file $filename does not exist"
}
# There is no need to worry about interacting with the program,
# just exec it. It is a good idea to do this in the testcase directory,
# so that any core dumps get cleaned up as well.
set current_dir [pwd]
set status [ catch { cd [file join $::objdir "testcase"] } message ]
if { $status != 0 } {
error "unable to change directory to [file join $::objdir testcase]\n$message"
}
verbose -log -- $filename $args
set status [ catch { set result [eval exec -keepnewline -- $filename $args] } output]
if { $status == 0 } {
# The command has succeeded. The exit code is 0 and the output
# was returned by the exec.
set output $result
set result 0
} else {
# The command has failed. The exit code is 1 and the output is
# already in the right place.
verbose -log -- $output
set result 1
}
set status [catch { cd $current_dir } message]
if { $status != 0 } {
error "unable to change directory back to $current_dir"
}
}
# ----------------------------------------------------------------------------
# Given some test output, look through it for pass and fail messages.
# These should result in appropriate DejaGnu pass and fail calls.
# In addition, if the program exited with a non-zero status code but
# did not report any failures then a special failure is reported.
proc hosttest_handle_output { name result output } {
set passes 0
set fails 0
foreach line [split $output "\n"] {
# The output should be of one of the following forms:
# PASS:<message>
# FAIL:<message> Line: xx File: xx
# Whatever
#
# PASS and FAIL messages will be reported via DejaGnu pass and fail
# calls. Everything else gets passed to verbose, so the user gets
# to choose how much information gets reported.
set dummy ""
set match1 ""
set match2 ""
if { [regexp -- {^PASS:<(.*)>.*$} $line dummy match1] == 1 } {
pass $match1
incr passes
} elseif { [regexp -- {^FAIL:<(.*)>(.*)$} $line dummy match1 match2] == 1 } {
fail "$match1 $match2"
incr fails
} else {
verbose $line
}
}
if { ($result != 0) && ($fails == 0) } {
fail "program terminated with non-zero exit code but did not report any failures"
} elseif { ($passes == 0) && ($fails == 0) } {
unresolved "test case $name did not report any passes or failures"
}
}
# ----------------------------------------------------------------------------
# hosttest_run_test_with_filter
# This routines combines the compile, run and clean operations,
# plus it invokes a supplied callback to filter the output. The
# callback is passed three arguments: the test name, the exit code,
# and all of the program output.
proc hosttest_run_test_with_filter { name filter sources incdirs libdirs libs args } {
set result 0
set output ""
set status [ catch { hosttest_compile $name $sources $incdirs $libdirs $libs } message]
if { $status != 0 } {
fail "unable to compile test case $name, $message"
hosttest_clean
return
}
set status [ catch { hosttest_run result output $name $args } message]
if { $status != 0 } {
fail "unable to run test case $name, $message"
hosttest_clean
return
}
set status [ catch { $filter $name $result $output } message]
if { $status != 0 } {
fail "unable to parse output from test case $name"
hosttest_clean
return
}
hosttest_clean
}
# ----------------------------------------------------------------------------
# hosttest_run_simple_test
# This routine combines the compile, run, output, and clean operations.
# The arguments are the same as for compilation, plus an additional
# list for run-time parameters to the test case.
proc hosttest_run_simple_test { name sources incdirs libdirs libs args } {
set result 0
set output ""
set status [ catch { hosttest_compile $name $sources $incdirs $libdirs $libs } message]
if { $status != 0 } {
fail "unable to compile test case $name, $message"
hosttest_clean
return
}
set status [ catch { hosttest_run result output $name $args } message]
if { $status != 0 } {
fail "unable to run test case $name, $message"
hosttest_clean
return
}
set status [ catch { hosttest_handle_output $name $result $output } message]
if { $status != 0 } {
fail "unable to parse output from test case $name"
hosttest_clean
return
}
hosttest_clean
}
# ----------------------------------------------------------------------------
# Filename translation. A particular file has been created and must now
# be accessed from Tcl.
#
# Under Unix everything just works.
#
# Under Windows, well there is cygwin and there is the Windows world.
# A file may have come from either. cygtclsh80 and hence DejaGnu is not
# fully integrated with cygwin, for example it does not know about
# cygwin mount points. There are also complications because of
# volume-relative filenames.
#
# The code here tries a number of different ways of finding a file which
# matches the name. It is possible that the result is not actually what
# was intended, but hopefully this case will never arise.
proc hosttest_translate_existing_filename { name } {
if { $::tcl_platform(platform) == "unix" } {
# The file should exist. It is worth checking just in case.
if { [file exists $name] == 0 } {
return ""
} else {
return $name
}
}
if { $::tcl_platform(platform) != "windows" } {
perror "The testing framework does not know about platform $::tcl_platform(platform)"
return ""
}
# Always get rid of any backslashes, they just cause trouble
regsub -all -- {\\} $name {/} name
# If the name is already valid, great.
if { [file exists $name] } {
return $name
}
# OK, try to use cygwin's cygpath utility to convert it.
set status [catch "exec cygpath -w $name" message]
if { $status == 0 } {
set cygwin_name ""
regsub -all -- {\\} $message {/} cygwin_name
if { [file exists $cygwin_name] } {
return $cygwin_name
}
}
# Is the name volumerelative? If so work out the current volume
# from the current directory and prepend this.
if { [file pathtype $name] == "volumerelative" } {
append fullname [string range [pwd] 0 1] $name
if { [file exists $fullname] } {
return $fullname
}
}
# There are other possibilities, e.g. d:xxx indicating a file
# relative to the current directory on drive d:. For now such
# Lovecraftian abominations are ignored.
return ""
}
# ----------------------------------------------------------------------------
# Support for assertion dumps. The infrastructure allows other subsystems
# to add their own callbacks which get invoked during a panic and which
# can write additional output to the dump file. For example it would be
# possible to output full details of the current configuration. These
# routines make it easier to write test cases for such callbacks.
#
# hosttest_assert_check(result output)
# Make sure that the test case really triggered an assertion.
#
# hosttest_assert_read_dump(output)
# Identify the temporary file used for the dump, read it in, delete
# it (no point in leaving such temporaries lying around when running
# testcases) and return the contents of the file.
#
# hosttest_assert_extract_callback(dump title)
# Given a dump output as returned by read_dump, look for a section
# generated by a callback with the given title. Return the contents
# of the callback.
proc hosttest_assert_check { result output } {
if { $result == 0 } {
return 0
}
foreach line [split $output "\n"] {
if { [string match "Assertion failure*" $line] } {
return 1
}
}
return 0
}
# This routine assumes that assert_check has already been called.
proc hosttest_assert_read_dump { output } {
foreach line [split $output "\n"] {
set dummy ""
set match ""
if { [regexp -nocase -- {^writing additional output to (.*)$} $line dummy match] } {
# The filename is in match, but it may not be directly accessible.
set filename [hosttest_translate_existing_filename $match]
if { $filename == "" } {
return ""
}
set status [ catch {
set fd [open $filename r]
set data [read $fd]
close $fd
file delete $filename
} message]
if { $status != 0 } {
unresolved "Unable to process assertion dump file $filename"
unresolved "File $filename may have to be deleted manually"
return ""
}
return $data
}
}
return ""
}
# Look for the appropriate markers. Also clean up blank lines
# at the start and end.
proc hosttest_assert_extract_callback { dump title } {
set lines [split $dump "\n"]
set result ""
while { [llength $lines] > 0 } {
set line [lindex $lines 0]
set lines [lreplace $lines 0 0]
if { [regexp -nocase -- "^\# \{\{\{.*${title}.*\$" $line] } {
# Skip any blank lines at the start
while { [llength $lines] > 0 } {
set line [lindex $lines 0]
if { [regexp -- {^ *$} $line] == 0} {
break
}
set lines [lreplace $lines 0 0]
}
# Now add any lines until the close marker.
# Nested folds are not supported yet.
while { [llength $lines] > 0 } {
set line [lindex $lines 0]
set lines [lreplace $lines 0 0]
if { [regexp -nocase -- {^\# \}\}\}.*$} $line] } {
break
}
append result $line "\n"
}
return $result
}
}
return ""
}
Go to most recent revision | Compare with Previous | Blame | View Log