2021-01-01 06:58:58 +08:00
|
|
|
# Copyright (C) 2002-2021 Free Software Foundation, Inc.
|
2020-01-13 23:18:57 +08:00
|
|
|
|
|
|
|
# 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, write to the Free Software
|
|
|
|
# Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
|
|
|
|
# test debuginfod with readelf and objdump
|
|
|
|
|
|
|
|
set test "debuginfod"
|
|
|
|
|
|
|
|
if {[which debuginfod] == 0} {
|
|
|
|
unsupported "$test (not found)"
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if {[which curl] == 0} {
|
|
|
|
unsupported "$test (curl not found)"
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if { ![is_elf_format] } {
|
|
|
|
unsupported "$test (unsupported target)"
|
|
|
|
}
|
|
|
|
|
|
|
|
if { [which $OBJDUMP] == 0} {
|
|
|
|
perror "$test $OBJDUMP (does not exist)"
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if { [which $READELF] == 0} {
|
|
|
|
perror "$test $READELF (does not exist)"
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
# Compile testprog.c, move the debuginfo to a separate file and add .gnu_debuglink.
|
|
|
|
if { [target_compile $srcdir/$subdir/testprog.c tmpdir/testprog executable debug] != ""} {
|
|
|
|
fail "$test (compilation failed)"
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if { [binutils_run $OBJCOPY "--only-keep-debug tmpdir/testprog tmpdir/testprog.debug"] != "" } {
|
|
|
|
fail "$test (create separate debug info file)"
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if { [binutils_run $OBJCOPY "--strip-debug tmpdir/testprog"] != "" } {
|
|
|
|
fail "$test (strip debug info)"
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if { [binutils_run $OBJCOPY "--add-gnu-debuglink=tmpdir/testprog.debug tmpdir/testprog"] != "" } {
|
|
|
|
fail "$test (add debuglink)"
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
# Assemble an elf file with a debugaltlink
|
|
|
|
if { ![binutils_assemble $srcdir/$subdir/debuglink.s tmpdir/debuglink.o] } {
|
|
|
|
fail "$test (assemble debuglink)"
|
|
|
|
}
|
|
|
|
|
|
|
|
if { ![binutils_assemble $srcdir/$subdir/linkdebug.s tmpdir/linkdebug.debug] } {
|
|
|
|
fail "$test (assemble linkdebug)"
|
|
|
|
}
|
|
|
|
|
|
|
|
set cache [file join [pwd] "tmpdir/.debuginfod_cache"]
|
2020-02-10 23:24:57 +08:00
|
|
|
set db [file join [pwd] "tmpdir/.debuginfod.db"]
|
2020-01-13 23:18:57 +08:00
|
|
|
|
2020-02-10 23:24:57 +08:00
|
|
|
setenv DEBUGINFOD_URLS ""
|
2020-02-10 23:20:41 +08:00
|
|
|
setenv DEBUGINFOD_TIMEOUT 30
|
|
|
|
setenv DEBUGINFOD_CACHE_PATH $cache
|
2020-01-13 23:18:57 +08:00
|
|
|
|
|
|
|
# Move debug files into another directory so that readelf and objdump cannot
|
|
|
|
# find them without debuginfod.
|
|
|
|
file mkdir tmpdir/dbg
|
|
|
|
file rename -force tmpdir/testprog.debug tmpdir/dbg
|
|
|
|
file rename -force tmpdir/linkdebug.debug tmpdir/dbg
|
|
|
|
|
2020-02-10 23:24:57 +08:00
|
|
|
# Remove old cache and database if they exist.
|
2020-01-13 23:18:57 +08:00
|
|
|
file delete -force $cache
|
2020-02-10 23:24:57 +08:00
|
|
|
file delete -force $db
|
2020-01-13 23:18:57 +08:00
|
|
|
|
|
|
|
# Check whether objdump and readelf are configured with debuginfod.
|
|
|
|
# To check this we attempt to follow a broken debuglink. If configured
|
|
|
|
# with debuginfod the output will contain the debuginfod URLs that were
|
|
|
|
# queried (these queries fail since the server is not yet running).
|
|
|
|
set conf_objdump [binutils_run $OBJDUMP "-WK tmpdir/testprog"]
|
|
|
|
set conf_readelf [binutils_run $READELF "-wK tmpdir/testprog"]
|
|
|
|
|
2020-03-02 20:46:47 +08:00
|
|
|
# Find an unused port
|
|
|
|
set port 7999
|
|
|
|
set found 0
|
|
|
|
while { ! $found } {
|
|
|
|
incr port
|
|
|
|
if { $port == 65536 } {
|
|
|
|
untested "$test (no available ports)"
|
|
|
|
return
|
|
|
|
}
|
2020-01-13 23:18:57 +08:00
|
|
|
|
2020-03-02 20:46:47 +08:00
|
|
|
spawn debuginfod -vvvv -d $db -p $port -F tmpdir/dbg
|
|
|
|
expect {
|
|
|
|
"started http server on IPv4 IPv6 port=$port" {
|
|
|
|
set found 1
|
2020-01-13 23:18:57 +08:00
|
|
|
}
|
2020-03-02 20:46:47 +08:00
|
|
|
"Failed to bind to port" {
|
|
|
|
exec kill -INT -[exp_pid]
|
|
|
|
catch {close}; catch {wait -i $spawn_id}
|
|
|
|
}
|
|
|
|
timeout {
|
|
|
|
fail "$test (find port timeout)"
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
2020-01-13 23:18:57 +08:00
|
|
|
}
|
|
|
|
|
2020-02-10 23:24:57 +08:00
|
|
|
set metrics [list "ready 1" \
|
|
|
|
"thread_work_total{role=\"traverse\"} 1" \
|
|
|
|
"thread_work_pending{role=\"scan\"} 0" \
|
2020-03-02 20:46:47 +08:00
|
|
|
"thread_busy{role=\"scan\"} 0"]
|
2020-02-10 23:24:57 +08:00
|
|
|
|
|
|
|
# Check server metrics to confirm init has completed.
|
|
|
|
foreach m $metrics {
|
|
|
|
set timelim 20
|
|
|
|
while { $timelim != 0 } {
|
|
|
|
sleep 0.5
|
|
|
|
|
2020-01-13 23:18:57 +08:00
|
|
|
catch {exec curl -s http://127.0.0.1:$port/metrics} got
|
|
|
|
|
2020-02-10 23:24:57 +08:00
|
|
|
if { [regexp $m $got] } {
|
2020-01-13 23:18:57 +08:00
|
|
|
break
|
|
|
|
}
|
|
|
|
|
2020-02-10 23:24:57 +08:00
|
|
|
incr timelim -1
|
|
|
|
}
|
|
|
|
|
|
|
|
if { $timelim == 0 } {
|
|
|
|
fail "$test (server init timeout)"
|
2020-03-02 20:46:47 +08:00
|
|
|
exec kill -INT -[exp_pid]
|
|
|
|
catch {close}; catch {wait -i $spawn_id}
|
2020-01-13 23:18:57 +08:00
|
|
|
return
|
2020-02-10 23:24:57 +08:00
|
|
|
}
|
2020-01-13 23:18:57 +08:00
|
|
|
}
|
|
|
|
|
2020-02-10 23:24:57 +08:00
|
|
|
setenv DEBUGINFOD_URLS http://127.0.0.1:$port
|
|
|
|
|
2020-01-13 23:18:57 +08:00
|
|
|
# Test whether prog can fetch separate debuginfo using debuginfod
|
|
|
|
# if it's configured to do so.
|
|
|
|
proc test_fetch_debuglink { prog progargs } {
|
|
|
|
global test
|
|
|
|
global cache
|
|
|
|
|
|
|
|
set got [binutils_run $prog "$progargs tmpdir/testprog"]
|
|
|
|
|
|
|
|
if { [regexp ".*Found separate debug info file.*Contents\[^\n\]*loaded from\[^\n\]*$cache.*" $got] } {
|
|
|
|
pass "$test ($prog debuglink)"
|
|
|
|
} else {
|
|
|
|
fail "$test ($prog debuglink)"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# Test whether prog can fetch debugaltlink files using debuginfod
|
|
|
|
# if it's configured to do so.
|
|
|
|
proc test_fetch_debugaltlink { prog progargs } {
|
|
|
|
global test
|
|
|
|
global cache
|
|
|
|
|
|
|
|
set got [binutils_run $prog "$progargs tmpdir/debuglink.o"]
|
|
|
|
set buildid "00112233445566778899aabbccddeeff0123456789abcdef"
|
|
|
|
|
|
|
|
if { [regexp ".*Found separate debug info file\[^\n\]*$cache/$buildid" $got] } {
|
|
|
|
pass "$test ($prog debugaltlink)"
|
|
|
|
} else {
|
|
|
|
fail "$test ($prog debugaltlink)"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if { [regexp ".*DEBUGINFOD.*" $conf_objdump] } {
|
|
|
|
test_fetch_debuglink $OBJDUMP "-W"
|
|
|
|
test_fetch_debugaltlink $OBJDUMP "-WK"
|
|
|
|
} else {
|
|
|
|
untested "$test (objdump not configured with debuginfod)"
|
|
|
|
}
|
|
|
|
|
|
|
|
if { [regexp ".*DEBUGINFOD.*" $conf_readelf] } {
|
|
|
|
test_fetch_debuglink $READELF "-w"
|
|
|
|
test_fetch_debugaltlink $READELF "-wK"
|
|
|
|
} else {
|
|
|
|
untested "$test (readelf not configured with debuginfod)"
|
|
|
|
}
|