c74f7d1c6c
This promotes BFD's struct elf_build_id to the generic struct bfd_build_id, populated when an ELF or PE BFD is read. gdb is updated to use that, and to use the build-id to find symbols for PE files also. There is currently no generic way to extract the build-id from an object file, perhaps an option to objdump to do this might make sense? On x86_64-pc-cygwin, gdb's sepdebug.exp changes: -# of unsupported tests 1 +# of expected passes 90 I don't seem to get consistent testsuite runs on i686-linux-gnu, but there don't appear to be any regressions. bfd/ChangeLog: 2015-06-10 Jon Turney <jon.turney@dronecode.org.uk> * elf-bfd.h : Remove struct elf_build_id. * bfd.c : Add struct bfd_build_id. * bfd-in2.h: Regenerate. * elf.c (elfobj_grok_gnu_build_id): Update to use bfd_build_id. * libpei.h: Add protoype and macros for bfd_XXi_slurp_codeview_record. * peXXigen.c (_bfd_XXi_slurp_codeview_record): Make public * peicode.h (pe_bfd_read_buildid): Add. (pe_bfd_object_p): Use pe_bfd_read_buildid(). gdb/ChangeLog: 2015-06-10 Jon Turney <jon.turney@dronecode.org.uk> * build-id.c: Don't include elf-bfd.h. (build_id_bfd_get): Use bfd_build_id. (build_id_verify): Ditto. * build-id.h: Ditto. (find_separate_debug_file_by_buildid): Ditto. * python/py-objfile.c: Don't include elf-bfd.h. (objfpy_get_build_id) Use bfd_build_id. (objfpy_build_id_matches, objfpy_lookup_objfile_by_build_id): Ditto. * coffread.c: Include build-id.h. (coff_symfile_read): Try find_separate_debug_file_by_buildid. gdb/doc/ChangeLog: 2015-06-10 Jon Turney <jon.turney@dronecode.org.uk> * gdb.texinfo (Separate Debug Files): Document that PE is also supported. gdb/testsuite/ChangeLog: 2015-06-10 Jon Turney <jon.turney@dronecode.org.uk> * gdb.base/sepdebug.exp: Add EXEEXT where needed. * lib/gdb.exp (get_build_id): Teach how to extract build-id from a PE file. * lib/future.exp (gdb_find_objdump): Add gdb_find_objdump. Signed-off-by: Jon Turney <jon.turney@dronecode.org.uk>
577 lines
14 KiB
Text
577 lines
14 KiB
Text
# Copyright 2004-2015 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_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_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_default_target_compile {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 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 shouldn't matter. It's done this way 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 == "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 {[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 "^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 F77_FOR_TARGET
|
|
global F90_FOR_TARGET
|
|
global GNATMAKE_FOR_TARGET
|
|
global GO_FOR_TARGET
|
|
global GO_LD_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 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 { $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"} {
|
|
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++"
|
|
} elseif {[regexp ".*hppa.*" $target_triplet]} {
|
|
# HP-UX
|
|
append add_flags " -Wl,-a,shared_archive"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
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 "$add_flags $sources"
|
|
}
|
|
"executable" {
|
|
switch $linker_opts_order {
|
|
"flags-then-sources" {
|
|
set opts "$add_flags $sources"
|
|
}
|
|
"sources-then-flags" {
|
|
set opts "$sources $add_flags"
|
|
}
|
|
default {
|
|
error "Invalid value for board_info linker_opts_order"
|
|
}
|
|
}
|
|
}
|
|
default {
|
|
set opts "$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}
|
|
}
|
|
|
|
# See if the version of dejaGNU being used to run the testsuite is
|
|
# recent enough to contain support for building Ada programs or not.
|
|
# If not, then use the functions above in place of the ones provided
|
|
# by dejaGNU. This is only temporary (brobecker/2004-03-31).
|
|
|
|
set use_gdb_compile 0
|
|
if {[info procs find_gnatmake] == ""} {
|
|
rename gdb_find_gnatmake find_gnatmake
|
|
set use_gdb_compile 1
|
|
}
|
|
|
|
if {[info procs find_gfortran] == ""} {
|
|
rename gdb_find_gfortran find_gfortran
|
|
set use_gdb_compile 1
|
|
}
|
|
|
|
if {[info procs find_go_linker] == ""} {
|
|
rename gdb_find_go find_go
|
|
rename gdb_find_go_linker find_go_linker
|
|
set use_gdb_compile 1
|
|
}
|
|
|
|
if {$use_gdb_compile} {
|
|
catch {rename 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
|
|
}
|
|
}
|