# This testcase is part of GDB, the GNU debugger. # Copyright 2019-2023 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 . # Test the "with" command. load_lib completion-support.exp standard_testfile .c if {[build_executable "failed to prepare" $testfile $srcfile debug]} { return -1 } clean_restart $binfile # Test "maint with". VALUES is a list of values. A nested "with" is # performed with each combination of pair of values from this list. # This exercises setting a value, and restoring it too. This is # particularly important for the "special" values like "unlimited", # which for example for var_uinteger maps to 0 at the user-visible # level, but maps to -1 internally. proc test_with {setting values} { foreach val1 $values { foreach val2 $values { gdb_test \ "maint with test-settings $setting $val1 -- maint with test-settings $setting $val2 -- p 1" \ " = 1" } } } # Test "maint with" in the error case. SETTING is the "maint set # test-setting" setting to exercise. TMP_VAL is the value to set the # setting to. EXPECTED_RE is the expected GDB output, which should be # an error of some kind. Also checks that the setting's original # value is preserved across the error. proc test_with_error {setting tmp_val expected_re} { global gdb_prompt with_test_prefix "$setting, $tmp_val" { set test "save org value" set org_val "" gdb_test_multiple "maint show test-settings $setting" $test { -re "(.*)\r\n$gdb_prompt $" { set org_val $expect_out(1,string) pass $test } } gdb_test \ "maint with test-settings $setting $tmp_val -- p 1" \ $expected_re gdb_test "maint show test-settings $setting" "^$org_val" \ "value hasn't changed across error" } } # Test "with" framework basics, using the internal "maint with # test-settings" subcommands. with_test_prefix "maint" { test_with "auto-boolean" {"on" "off" "auto"} test_with "boolean" {"" "on" "off" "0" "1" "enable" "disable"} test_with "integer" {"0" "1" "-1" "unlimited"} test_with "uinteger" {"0" "1" "unlimited"} test_with "zinteger" {"0" "1" "-1"} test_with "zuinteger" {"0" "1"} test_with "zuinteger-unlimited" {"-1" "unlimited" "0" "1"} test_with "string" {"" "foo" "\"hello world\""} test_with "string-noescape" {"" "foo" "\"hello world\""} test_with "filename" {"/foo" "bar/x/y"} test_with "optional-filename" {"" "/foo" "bar/x/y"} test_with "enum" {"xxx" "yyy"} # Check the most important error conditions. E.g., empty, # negative or "unlimited" values for settings that don't accept # those. Exhaustive error coverage of the set/with value parsing # is left to "set" testing, in gdb.base/settings.exp. test_with_error "auto-boolean" "" \ "\"on\", \"off\" or \"auto\" expected\\." test_with_error "auto-boolean" "xxx" \ "\"on\", \"off\" or \"auto\" expected\\." test_with_error "boolean" "2" "\"on\" or \"off\" expected\\." test_with_error "uinteger" "-1" "integer -1 out of range" test_with_error "uinteger" "" \ "Argument required \\(integer to set it to, or \"unlimited\"\\)\\." test_with_error "zuinteger" "-1" "integer -1 out of range" test_with_error "zuinteger" "" \ "Argument required \\(integer to set it to\\)\\." test_with_error "zuinteger-unlimited" "-2" \ "integer -2 out of range" test_with_error "zuinteger-unlimited" "" \ "Argument required \\(integer to set it to, or \"unlimited\"\\)\\." test_with_error "filename" "" \ "Argument required \\(filename to set it to\\.\\)\\." test_with_error "enum" "" \ "Requires an argument\\. Valid arguments are xxx, yyy, zzz\\." } # Basic/core tests using user-visible commands. with_test_prefix "basics" { gdb_test "print g_s" " = {a = 1, b = 2, c = 3}" gdb_test "with print pretty -- print g_s" \ [multi_line \ " = {" \ " a = 1," \ " b = 2," \ " c = 3" \ "}"] # A boolean setting. gdb_test "with non-stop on -- show non-stop" \ "Controlling the inferior in non-stop mode is on\\." gdb_test "show non-stop" \ "Controlling the inferior in non-stop mode is off\\." # Language. gdb_test "with language pascal -- show language" \ "The current source language is \"pascal\"\\." gdb_test "show language" \ "The current source language is \"auto; currently c\"\\." gdb_test "with language ada -- print g_s" \ " = \\(a => 1, b => 2, c => 3\\)" # Nested "with"s. gdb_test "with language ada -- with language c -- print g_s" \ " = {a = 1, b = 2, c = 3}" # "w" alias. gdb_test "w language pascal -- show language" \ "The current source language is \"pascal\"\\." \ "w alias works" # An early prototype of the "with" command got this wrong. gdb_test \ "w print repeats unlimited -- w print repeats 1 -- p \"1223334444\"" \ " = \"1\", '2' , '3' , '4' " } # Check a user-defined command. with_test_prefix "user-defined" { # A user defined command. set test "define usercmd" gdb_test_multiple "define usercmd" $test { -re "End with" { gdb_test \ [multi_line_input \ {print g_s} \ {end}] \ "" \ $test } } gdb_test "with language ada -- usercmd" \ " = \\(a => 1, b => 2, c => 3\\)" } # Check repeating. with_test_prefix "repeat" { clean_restart $binfile # "with" with no command reinvokes the previous command. gdb_test "with language ada" \ "No previous command to relaunch" \ "reinvoke with no previous command to relaunch" gdb_test "print g_s" " = {a = 1, b = 2, c = 3}" gdb_test "with language ada" \ " = \\(a => 1, b => 2, c => 3\\)" \ "reinvoke with language" # Same, but with "--". gdb_test "with language fortran --" \ " = \\( a = 1, b = 2, c = 3 \\)" \ "reinvoke with language and --" # Repeating repeats the original "print g_s", not the last "with" # command. set test "repeat command line" send_gdb "\n" gdb_test_multiple "" $test { -re " = {a = 1, b = 2, c = 3}\r\n$gdb_prompt $" { pass $test } } } # Basic run control. with_test_prefix "run control" { clean_restart $binfile if ![runto_main] { return } # Check "with" with a synchronous execution command. gdb_test "with disassemble-next-line on -- next" \ "return 0;.*=>.*" } # Check errors. with_test_prefix "errors" { gdb_test "with" "Missing arguments\\." # Try both an unknown root setting and an unknown prefixed # setting. The errors come from different locations in the # sources. gdb_test "with xxxx yyyy" \ "Undefined set command: \"xxxx\". Try \"help set\"\\." gdb_test "with print xxxx yyyy" \ "Undefined set print command: \"xxxx yyyy\". Try \"help set print\"\\." # Try one error case for "maint with", to make sure the right # "maintenance with" prefix is shown. gdb_test "maint with xxxx yyyy" \ "Undefined maintenance set command: \"xxxx\". Try \"help maintenance set\"\\." # Try ambiguous settings. gdb_test "with w" \ "Ambiguous set command \"w\": watchdog, width, write\\." gdb_test "with print m" \ "Ambiguous set print command \"m\": max-depth, max-symbolic-offset, memory-tag-violations\\." gdb_test "with variable xxx=1" \ "Cannot use this setting with the \"with\" command" gdb_test "with print elements -- p 1" \ "Argument required \\(integer to set it to, or \"unlimited\"\\)\\." gdb_test "with -- p 1" \ "Missing setting before '--' delimiter" # Check that the setting is restored even if the command throws. gdb_test "with print elements 1 -- unknowncommand" \ "Undefined command: \"unknowncommand\"\\. Try \"help\"\\." gdb_test "show print elements" \ "Limit on string chars or array elements to print is 200\\." } # Check completion. with_test_prefix "completion" { test_gdb_complete_unique \ "with pri" \ "with print" test_gdb_complete_unique \ "with print ele" \ "with print elements" test_gdb_complete_unique \ "with print elements u" \ "with print elements unlimited" test_gdb_complete_none \ "with print elements unlimited " test_gdb_completion_offers_commands "with print elements unlimited -- " # Check that the completer nests into the nested command line's # completer. test_gdb_complete_unique \ "with print elements unlimited -- with print ele" \ "with print elements unlimited -- with print elements" # Check completion of "maint with". "maint with" and "with"'s # completers share 99% of the code. All we need to care about # here is that the completion word point is computed correctly, so # any simple completion is sufficient. test_gdb_complete_unique \ "maint with test-set" \ "maint with test-settings" }