# Copyright 2004-2020 Free Software Foundation, Inc.
#
# 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 3 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, see <http://www.gnu.org/licenses/>.
load_lib libgloss.exp
# FIXME:brobecker/2004-03-31:
# The following functions should eventually be part of dejagnu. Even after
# these functions becomes available in dejagnu, we will keep for a while
# a copy here in order to avoid increasing the dejagnu version
# requirement.
proc gdb_find_gnatmake {} {
global tool_root_dir
set root "$tool_root_dir/gcc"
set GM ""
if ![is_remote host] {
set file [lookfor_file $root gnatmake]
if { $file != "" } {
set GM "$file -I$root/ada/rts --GCC=$root/xgcc --GNATBIND=$root/gnatbind --GNATLINK=$root/gnatlink -cargs -B$root -largs --GCC=$root/xgcc -margs";
}
}
if {$GM == ""} {
set GM [transform gnatmake]
}
return $GM
}
proc gdb_find_gdc {} {
global tool_root_dir
print "Tool Root: $tool_root_dir"
if {![is_remote host]} {
set file [lookfor_file $tool_root_dir gdc]
if { $file == "" } {
set file [lookfor_file $tool_root_dir gcc/gdc]
}
if { $file != "" } {
set CC "$file -B[file dirname $file]/"
} else {
set CC [transform gdc]
}
} else {
set CC [transform gdc]
}
print "CC: $CC"
return $CC
}
proc gdb_find_gfortran {} {
global tool_root_dir
if {![is_remote host]} {
set file [lookfor_file $tool_root_dir gfortran]
if { $file == "" } {
set file [lookfor_file $tool_root_dir gcc/gfortran]
}
if { $file != "" } {
set CC "$file -B[file dirname $file]/"
} else {
set CC [transform gfortran]
}
} else {
set CC [transform gfortran]
}
return $CC
}
proc gdb_find_go {} {
global tool_root_dir
set GO ""
if {![is_remote host]} {
set file [lookfor_file $tool_root_dir gccgo]
if { $file != "" } {
set root [file dirname $file]
set GO "$file -B$root/gcc/"
}
}
if { $GO == "" } {
set GO [transform gccgo]
}
return $GO
}
proc gdb_find_go_linker {} {
return [find_go]
}
proc gdb_find_rustc {} {
global tool_root_dir
if {![is_remote host]} {
set rustc [lookfor_file $tool_root_dir rustc]
if {$rustc == ""} {
set rustc rustc
}
} else {
set rustc ""
}
if {$rustc != ""} {
append rustc " --color never"
}
return $rustc
}
proc gdb_find_ldd {} {
global LDD_FOR_TARGET
if [info exists LDD_FOR_TARGET] {
set ldd $LDD_FOR_TARGET
} else {
set ldd "ldd"
}
return $ldd
}
proc gdb_find_objcopy {} {
global OBJCOPY_FOR_TARGET
if [info exists OBJCOPY_FOR_TARGET] {
set objcopy $OBJCOPY_FOR_TARGET
} else {
set objcopy [transform objcopy]
}
return $objcopy
}
# find target objdump
proc gdb_find_objdump {} {
global OBJDUMP_FOR_TARGET
if [info exists OBJDUMP_FOR_TARGET] {
set objdump $OBJDUMP_FOR_TARGET
} else {
set objdump [transform objdump]
}
return $objdump
}
proc gdb_find_readelf {} {
global READELF_FOR_TARGET
if [info exists READELF_FOR_TARGET] {
set readelf $READELF_FOR_TARGET
} else {
set readelf [transform readelf]
}
return $readelf
}
proc gdb_find_eu-unstrip {} {
global EU_UNSTRIP_FOR_TARGET
if [info exists EU_UNSTRIP_FOR_TARGET] {
set eu_unstrip $EU_UNSTRIP_FOR_TARGET
} else {
set eu_unstrip [transform eu-unstrip]
}
return $eu_unstrip
}
# Local version of default_target_compile, to be used for languages that
# dejagnu's default_target_compile doesn't support.
proc gdb_default_target_compile_1 {source destfile type options} {
global target_triplet
global tool_root_dir
global CFLAGS_FOR_TARGET
global compiler_flags
if { $destfile == "" && $type != "preprocess" && $type != "none" } {
error "Must supply an output filename for the compile to default_target_compile"
}
set early_flags ""
set add_flags ""
set libs ""
set compiler_type "c"
set compiler ""
set linker ""
# linker_opts_order is one of "sources-then-flags", "flags-then-sources".
# The order matters for things like -Wl,--as-needed. The default is to
# preserve existing behavior.
set linker_opts_order "sources-then-flags"
set ldflags ""
set dest [target_info name]
if {[info exists CFLAGS_FOR_TARGET]} {
append add_flags " $CFLAGS_FOR_TARGET"
}
if {[info exists target_info(host,name)]} {
set host [host_info name]
} else {
set host "unix"
}
foreach i $options {
if { $i == "ada" } {
set compiler_type "ada"
if {[board_info $dest exists adaflags]} {
append add_flags " [target_info adaflags]"
}
if {[board_info $dest exists gnatmake]} {
set compiler [target_info gnatmake]
} else {
set compiler [find_gnatmake]
}
}
if { $i == "c++" } {
set compiler_type "c++"
if {[board_info $dest exists cxxflags]} {
append add_flags " [target_info cxxflags]"
}
append add_flags " [g++_include_flags]"
if {[board_info $dest exists c++compiler]} {
set compiler [target_info c++compiler]
} else {
set compiler [find_g++]
}
}
if { $i == "d" } {
set compiler_type "d"
if {[board_info $dest exists dflags]} {
append add_flags " [target_info dflags]"
}
if {[board_info $dest exists dcompiler]} {
set compiler [target_info dcompiler]
} else {
set compiler [find_gdc]
}
}
if { $i == "f77" } {
set compiler_type "f77"
if {[board_info $dest exists f77flags]} {
append add_flags " [target_info f77flags]"
}
if {[board_info $dest exists f77compiler]} {
set compiler [target_info f77compiler]
} else {
set compiler [find_g77]
}
}
if { $i == "f90" } {
set compiler_type "f90"
if {[board_info $dest exists f90flags]} {
append add_flags " [target_info f90flags]"
}
if {[board_info $dest exists f90compiler]} {
set compiler [target_info f90compiler]
} else {
set compiler [find_gfortran]
}
}
if { $i == "go" } {
set compiler_type "go"
if {[board_info $dest exists goflags]} {
append add_flags " [target_info goflags]"
}
if {[board_info $dest exists gocompiler]} {
set compiler [target_info gocompiler]
} else {
set compiler [find_go]
}
if {[board_info $dest exists golinker]} {
set linker [target_info golinker]
} else {
set linker [find_go_linker]
}
if {[board_info $dest exists golinker_opts_order]} {
set linker_opts_order [target_info golinker_opts_order]
}
}
if { $i == "rust" } {
set compiler_type "rust"
if {[board_info $dest exists rustflags]} {
append add_flags " [target_info rustflags]"
}
if {[board_info $dest exists rustflags]} {
set compiler [target_info rustflags]
} else {
set compiler [find_rustc]
}
}
if {[regexp "^dest=" $i]} {
regsub "^dest=" $i "" tmp
if {[board_info $tmp exists name]} {
set dest [board_info $tmp name]
} else {
set dest $tmp
}
}
if {[regexp "^compiler=" $i]} {
regsub "^compiler=" $i "" tmp
set compiler $tmp
}
if {[regexp "^early_flags=" $i]} {
regsub "^early_flags=" $i "" tmp
append early_flags " $tmp"
}
if {[regexp "^additional_flags=" $i]} {
regsub "^additional_flags=" $i "" tmp
append add_flags " $tmp"
}
if {[regexp "^ldflags=" $i]} {
regsub "^ldflags=" $i "" tmp
append ldflags " $tmp"
}
if {[regexp "^libs=" $i]} {
regsub "^libs=" $i "" tmp
append libs " $tmp"
}
if {[regexp "^incdir=" $i]} {
regsub "^incdir=" $i "-I" tmp
append add_flags " $tmp"
}
if {[regexp "^libdir=" $i]} {
regsub "^libdir=" $i "-L" tmp
append add_flags " $tmp"
}
if {[regexp "^ldscript=" $i]} {
regsub "^ldscript=" $i "" ldscript
}
if {[regexp "^redirect=" $i]} {
regsub "^redirect=" $i "" redirect
}
if {[regexp "^optimize=" $i]} {
regsub "^optimize=" $i "" optimize
}
if {[regexp "^timeout=" $i]} {
regsub "^timeout=" $i "" timeout
}
}
if {[board_info $host exists cflags_for_target]} {
append add_flags " [board_info $host cflags_for_target]"
}
global CC_FOR_TARGET
global CXX_FOR_TARGET
global D_FOR_TARGET
global F77_FOR_TARGET
global F90_FOR_TARGET
global GNATMAKE_FOR_TARGET
global GO_FOR_TARGET
global GO_LD_FOR_TARGET
global RUSTC_FOR_TARGET
if {[info exists GNATMAKE_FOR_TARGET]} {
if { $compiler_type == "ada" } {
set compiler $GNATMAKE_FOR_TARGET
}
}
if {[info exists CC_FOR_TARGET]} {
if { $compiler == "" } {
set compiler $CC_FOR_TARGET
}
}
if {[info exists CXX_FOR_TARGET]} {
if { $compiler_type == "c++" } {
set compiler $CXX_FOR_TARGET
}
}
if {[info exists D_FOR_TARGET]} {
if { $compiler_type == "d" } {
set compiler $D_FOR_TARGET
}
}
if {[info exists F77_FOR_TARGET]} {
if { $compiler_type == "f77" } {
set compiler $F77_FOR_TARGET
}
}
if {[info exists F90_FOR_TARGET]} {
if { $compiler_type == "f90" } {
set compiler $F90_FOR_TARGET
}
}
if { $compiler_type == "go" } {
if {[info exists GO_FOR_TARGET]} {
set compiler $GO_FOR_TARGET
}
if {[info exists GO_LD_FOR_TARGET]} {
set linker $GO_LD_FOR_TARGET
}
}
if {[info exists RUSTC_FOR_TARGET]} {
if {$compiler_type == "rust"} {
set compiler $RUSTC_FOR_TARGET
}
}
if { $type == "executable" && $linker != "" } {
set compiler $linker
}
if { $compiler == "" } {
set compiler [board_info $dest compiler]
if { $compiler == "" } {
return "default_target_compile: No compiler to compile with"
}
}
if {![is_remote host]} {
if { [which $compiler] == 0 } {
return "default_target_compile: Can't find $compiler."
}
}
if {$type == "object"} {
if {$compiler_type == "rust"} {
append add_flags "--emit obj"
} else {
append add_flags " -c"
}
}
if { $type == "preprocess" } {
append add_flags " -E"
}
if { $type == "assembly" } {
append add_flags " -S"
}
if {[board_info $dest exists cflags]} {
append add_flags " [board_info $dest cflags]"
}
if { $type == "executable" } {
if {[board_info $dest exists ldflags]} {
append add_flags " [board_info $dest ldflags]"
}
if { $compiler_type == "c++" } {
append add_flags " [g++_link_flags]"
}
if {[isnative]} {
# This is a lose.
catch "glob -nocomplain $tool_root_dir/libstdc++/libstdc++.so* $tool_root_dir/libstdc++/libstdc++.sl" tmp
if { ${tmp} != "" } {
if {[regexp ".*solaris2.*" $target_triplet]} {
# Solaris 2
append add_flags " -R$tool_root_dir/libstdc++"
} elseif {[regexp ".*(osf|irix5|linux).*" $target_triplet]} {
# OSF/1 or IRIX 5
append add_flags " -Wl,-rpath,$tool_root_dir/libstdc++"
}
}
}
}
if {![info exists ldscript]} {
set ldscript [board_info $dest ldscript]
}
foreach i $options {
if { $i == "debug" } {
if {[board_info $dest exists debug_flags]} {
append add_flags " [board_info $dest debug_flags]"
} else {
append add_flags " -g"
}
}
}
if {[info exists optimize]} {
append add_flags " $optimize"
}
if { $type == "executable" } {
append add_flags " $ldflags"
foreach x $libs {
if {[file exists $x]} {
append source " $x"
} else {
append add_flags " $x"
}
}
if {[board_info $dest exists libs]} {
append add_flags " [board_info $dest libs]"
}
# This probably isn't such a good idea, but it avoids nasty
# hackiness in the testsuites.
# The math library must be linked in before the C library. The C
# library is linked in by the linker script, so this must be before
# the linker script.
if {[board_info $dest exists mathlib]} {
append add_flags " [board_info $dest mathlib]"
} else {
append add_flags " -lm"
}
# This must be added here.
append add_flags " $ldscript"
if {[board_info $dest exists remote_link]} {
# Relink option.
append add_flags " -Wl,-r"
}
if {[board_info $dest exists output_format]} {
append add_flags " -Wl,-oformat,[board_info $dest output_format]"
}
}
if {[board_info $dest exists multilib_flags]} {
append add_flags " [board_info $dest multilib_flags]"
}
verbose "doing compile"
set sources ""
if {[is_remote host]} {
foreach x $source {
set file [remote_download host $x]
if { $file == "" } {
warning "Unable to download $x to host."
return "Unable to download $x to host."
} else {
append sources " $file"
}
}
} else {
set sources $source
}
if {[is_remote host]} {
append add_flags " -o " [file tail $destfile]
remote_file host delete [file tail $destfile]
} else {
if { $destfile != "" } {
append add_flags " -o $destfile"
}
}
# This is obscure: we put SOURCES at the end when building an
# object, because otherwise, in some situations, libtool will
# become confused about the name of the actual source file.
switch $type {
"object" {
set opts "$early_flags $add_flags $sources"
}
"executable" {
switch $linker_opts_order {
"flags-then-sources" {
set opts "$early_flags $add_flags $sources"
}
"sources-then-flags" {
set opts "$early_flags $sources $add_flags"
}
default {
error "Invalid value for board_info linker_opts_order"
}
}
}
default {
set opts "$early_flags $sources $add_flags"
}
}
if {[is_remote host]} {
if {[host_info exists use_at]} {
set fid [open "atfile" "w"]
puts $fid "$opts"
close $fid
set opts "@[remote_download host atfile]"
remote_file build delete atfile
}
}
verbose "Invoking the compiler as $compiler $opts" 2
if {[info exists redirect]} {
verbose "Redirecting output to $redirect" 2
set status [remote_exec host "$compiler $opts" "" "" $redirect]
} else {
if {[info exists timeout]} {
verbose "Setting timeout to $timeout" 2
set status [remote_exec host "$compiler $opts" "" "" "" $timeout]
} else {
set status [remote_exec host "$compiler $opts"]
}
}
set compiler_flags $opts
if {[is_remote host]} {
remote_upload host [file tail $destfile] $destfile
remote_file host delete [file tail $destfile]
}
set comp_output [prune_warnings [lindex $status 1]]
regsub "^\[\r\n\]+" $comp_output "" comp_output
if { [lindex $status 0] != 0 } {
verbose -log "compiler exited with status [lindex $status 0]"
}
if { [lindex $status 1] != "" } {
verbose -log "output is:\n[lindex $status 1]" 2
}
if { [lindex $status 0] != 0 && "${comp_output}" == "" } {
set comp_output "exit status is [lindex $status 0]"
}
return ${comp_output}
}
# If dejagnu's default_target_compile supports the language specified in
# OPTIONS, use it. Otherwise, use gdb_default_target_compile_1.
proc gdb_default_target_compile {source destfile type options} {
global use_gdb_compile
set need_local_lang 0
set need_local_early_flags 0
foreach i $options {
if { $i == "ada" || $i == "d" || $i == "go" || $i == "rust" } {
set need_local_lang [info exists use_gdb_compile($i)]
}
if { $i == "c++" } {
set need_local_lang 0
}
if { $i == "f77" || $i == "f90" } {
set need_local_lang [info exists use_gdb_compile(fortran)]
}
if { [regexp "^early_flags=" $i] } {
set need_local_early_flags 1
}
}
if { $need_local_lang || $need_local_early_flags } {
return [gdb_default_target_compile_1 $source $destfile $type $options]
}
return [dejagnu_default_target_compile $source $destfile $type $options]
}
# Array of languages for which dejagnu's default_target_compile is missing
# support.
array set use_gdb_compile [list]
# Note missing support in dejagnu's default_target_compile. This
# needs to be fixed by porting the missing support to Dejagnu.
set note_prefix "Dejagnu's default_target_compile is missing support for "
set note_suffix ", using local override"
if {[info procs find_gnatmake] == ""} {
rename gdb_find_gnatmake find_gnatmake
set use_gdb_compile(ada) 1
gdb_note [join [list $note_prefix "Ada" $note_suffix] ""]
}
if {[info procs find_gfortran] == ""} {
rename gdb_find_gfortran find_gfortran
set use_gdb_compile(fortran) 1
gdb_note [join [list $note_prefix "Fortran" $note_suffix] ""]
}
if {[info procs find_go_linker] == ""} {
rename gdb_find_go find_go
rename gdb_find_go_linker find_go_linker
set use_gdb_compile(go) 1
gdb_note [join [list $note_prefix "Go" $note_suffix] ""]
}
if {[info procs find_gdc] == ""} {
rename gdb_find_gdc find_gdc
set use_gdb_compile(d) 1
gdb_note [join [list $note_prefix "D" $note_suffix] ""]
}
if {[info procs find_rustc] == ""} {
rename gdb_find_rustc find_rustc
set use_gdb_compile(rust) 1
gdb_note [join [list $note_prefix "Rust" $note_suffix] ""]
}
# If dejagnu's default_target_compile is missing support for any language,
# override it.
if { [array size use_gdb_compile] != 0 } {
catch {rename default_target_compile dejagnu_default_target_compile}
rename gdb_default_target_compile default_target_compile
}
# Provide 'lreverse' missing in Tcl before 7.5.
if {[info procs lreverse] == ""} {
proc lreverse { arg } {
set retval {}
while { [llength $retval] < [llength $arg] } {
lappend retval [lindex $arg end-[llength $retval]]
}
return $retval
}
}
# Various ccache versions provide incorrect debug info such as ignoring
# different current directory, breaking GDB testsuite.
set env(CCACHE_DISABLE) 1
unset -nocomplain env(CCACHE_NODISABLE)