| # Copyright 2018-2021 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/>. |
| |
| # This test checks that the index-cache feature generates the expected files at |
| # the expected location. |
| |
| standard_testfile |
| |
| if { [build_executable "failed to prepare" $testfile $srcfile \ |
| {debug additional_flags=-Wl,--build-id}] } { |
| return |
| } |
| |
| # The index cache won't be used in certain circumstances, for which we must |
| # account in this test: |
| # |
| # - the binary already has an index section |
| # - we use the -readnow switch |
| set has_index_section [exec_has_index_section $binfile] |
| set uses_readnow [expr [string first "-readnow" $GDBFLAGS] != -1] |
| set expecting_index_cache_use [expr !$has_index_section && !$uses_readnow] |
| |
| # List the files in DIR on the host (where GDB-under-test runs). |
| # Return a list of two elements: |
| # - 0 on success, -1 on failure |
| # - the list of files on success, empty on failure |
| |
| proc ls_host { dir } { |
| lassign [remote_exec host ls "-1 $dir"] ret output |
| |
| if { $ret != 0 } { |
| fail "failed to list files on host in $dir" |
| return -1 |
| } |
| |
| # ls -1 returns a list separated by \r\n. split will return a bunch of |
| # empty entries (it treats a sequence of split characters as separate |
| # fields, plus there is a \r\n at the end of the result). Ignore empty |
| # list elements. |
| set filtered {} |
| set files [split $output \r\n] |
| |
| foreach file $files { |
| if { $file != "" } { |
| lappend filtered $file |
| } |
| } |
| |
| return [list 0 $filtered] |
| } |
| |
| # Execute "show index-cache stats" and verify the output against expected |
| # values. |
| |
| proc check_cache_stats { expected_hits expected_misses } { |
| set re [multi_line \ |
| " Cache hits .this session.: $expected_hits" \ |
| "Cache misses .this session.: $expected_misses" \ |
| ] |
| |
| gdb_test "show index-cache stats" $re "check index-cache stats" |
| } |
| |
| # Run CODE using a fresh GDB configured based on the other parameters. |
| |
| proc run_test_with_flags { cache_dir cache_enabled code } { |
| global GDBFLAGS testfile |
| |
| save_vars { GDBFLAGS } { |
| set GDBFLAGS "$GDBFLAGS -iex \"set index-cache directory $cache_dir\"" |
| set GDBFLAGS "$GDBFLAGS -iex \"set index-cache $cache_enabled\"" |
| |
| clean_restart ${testfile} |
| |
| uplevel 1 $code |
| } |
| } |
| |
| # Test administrative stuff. |
| |
| proc_with_prefix test_basic_stuff { } { |
| global testfile |
| |
| clean_restart ${testfile} |
| |
| # Check that the index cache is disabled by default. |
| gdb_test \ |
| "show index-cache" \ |
| " is currently disabled." \ |
| "index-cache is disabled by default" |
| |
| # Test that we can enable it and "show index-cache" reflects that. |
| gdb_test_no_output "set index-cache on" "enable index cache" |
| gdb_test \ |
| "show index-cache" \ |
| " is currently enabled." \ |
| "index-cache is now enabled" |
| |
| # Test the "set/show index-cache directory" commands. |
| gdb_test "set index-cache directory" "Argument required.*" "set index-cache directory without arg" |
| gdb_test_no_output "set index-cache directory /tmp" "change the index cache directory" |
| gdb_test \ |
| "show index-cache directory" \ |
| "The directory of the index cache is \"/tmp\"." \ |
| "show index cache directory" |
| } |
| |
| # Test loading a binary with the cache disabled. No file should be created. |
| |
| proc_with_prefix test_cache_disabled { cache_dir test_prefix } { |
| with_test_prefix $test_prefix { |
| lassign [ls_host $cache_dir] ret files_before |
| |
| run_test_with_flags $cache_dir off { |
| lassign [ls_host $cache_dir] ret files_after |
| |
| set nfiles_created [expr [llength $files_after] - [llength $files_before]] |
| gdb_assert "$nfiles_created == 0" "no files were created" |
| |
| check_cache_stats 0 0 |
| } |
| } |
| } |
| |
| # Test a cache miss. We expect to have at least one file in the cache if the |
| # index cache is going to be used (see expecting_index_cache_use) and a cache |
| # miss in the stats. If the cache is not going to be used, we expect to have |
| # no files and no cache hits nor misses. |
| |
| proc_with_prefix test_cache_enabled_miss { cache_dir } { |
| global testfile expecting_index_cache_use |
| |
| lassign [ls_host $cache_dir] ret files_before |
| |
| run_test_with_flags $cache_dir on { |
| |
| lassign [ls_host $cache_dir] ret files_after |
| set nfiles_created [expr [llength $files_after] - [llength $files_before]] |
| if { $expecting_index_cache_use } { |
| gdb_assert "$nfiles_created > 0" "at least one file was created" |
| } else { |
| gdb_assert "$nfiles_created == 0" "no file was created" |
| } |
| |
| set build_id [get_build_id [standard_output_file ${testfile}]] |
| if { $build_id == "" } { |
| fail "couldn't get executable build id" |
| return |
| } |
| |
| set expected_created_file [list "${build_id}.gdb-index"] |
| set found_idx [lsearch -exact $files_after $expected_created_file] |
| if { $expecting_index_cache_use } { |
| gdb_assert "$found_idx >= 0" "expected file is there" |
| } else { |
| gdb_assert "$found_idx == -1" "no index cache file generated" |
| } |
| |
| remote_exec host rm "-f $cache_dir/$expected_created_file" |
| |
| if { $expecting_index_cache_use } { |
| check_cache_stats 0 1 |
| } else { |
| check_cache_stats 0 0 |
| } |
| } |
| } |
| |
| |
| # Test a cache hit. We should have at least one file in the cache if the index |
| # cache is going to be used (see expecting_index_cache_use) and a cache hit in |
| # the stats. If the cache is not going to be used, we expect to have no files |
| # and no cache hits nor misses. |
| |
| proc_with_prefix test_cache_enabled_hit { cache_dir } { |
| global expecting_index_cache_use |
| |
| # Just to populate the cache. |
| run_test_with_flags $cache_dir on {} |
| |
| lassign [ls_host $cache_dir] ret files_before |
| |
| run_test_with_flags $cache_dir on { |
| lassign [ls_host $cache_dir] ret files_after |
| set nfiles_created [expr [llength $files_after] - [llength $files_before]] |
| gdb_assert "$nfiles_created == 0" "no files were created" |
| |
| if { $expecting_index_cache_use } { |
| check_cache_stats 1 0 |
| } else { |
| check_cache_stats 0 0 |
| } |
| } |
| } |
| |
| test_basic_stuff |
| |
| # The cache dir should be on the host (possibly remote), so we can't use the |
| # standard output directory for that (it's on the build machine). |
| lassign [remote_exec host mktemp -d] ret cache_dir |
| |
| if { $ret != 0 } { |
| fail "couldn't create temporary cache dir" |
| return |
| } |
| |
| # The ouput of mktemp contains an end of line, remove it. |
| set cache_dir [string trimright $cache_dir \r\n] |
| |
| test_cache_disabled $cache_dir "before populate" |
| test_cache_enabled_miss $cache_dir |
| test_cache_enabled_hit $cache_dir |
| |
| # Test again with the cache disabled, now that it is populated. |
| test_cache_disabled $cache_dir "after populate" |
| |
| lassign [remote_exec host sh "-c \"rm $cache_dir/*.gdb-index\""] ret |
| if { $ret != 0 && $expecting_index_cache_use } { |
| fail "couldn't remove files in temporary cache dir" |
| return |
| } |
| |
| lassign [remote_exec host rmdir "$cache_dir"] ret |
| if { $ret != 0 } { |
| fail "couldn't remove temporary cache dir" |
| return |
| } |