Home | History | Annotate | Line # | Download | only in gdb.guile
      1 # Copyright (C) 2008-2025 Free Software Foundation, Inc.
      2 
      3 # This program is free software; you can redistribute it and/or modify
      4 # it under the terms of the GNU General Public License as published by
      5 # the Free Software Foundation; either version 3 of the License, or
      6 # (at your option) any later version.
      7 #
      8 # This program is distributed in the hope that it will be useful,
      9 # but WITHOUT ANY WARRANTY; without even the implied warranty of
     10 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
     11 # GNU General Public License for more details.
     12 #
     13 # You should have received a copy of the GNU General Public License
     14 # along with this program.  If not, see <http://www.gnu.org/licenses/>.
     15 
     16 # This file is part of the GDB testsuite.
     17 # It tests the mechanism exposing values to Guile.
     18 
     19 load_lib gdb-guile.exp
     20 
     21 require allow_guile_tests
     22 
     23 standard_testfile
     24 
     25 set has_argv0 [gdb_has_argv0]
     26 
     27 # Build inferior to language specification.
     28 # LANG is one of "c" or "c++".
     29 proc build_inferior {exefile lang} {
     30     global srcdir subdir srcfile testfile hex
     31 
     32     # Use different names for .o files based on the language.
     33     # For Fission, the debug info goes in foo.dwo and we don't want,
     34     # for example, a C++ compile to clobber the dwo of a C compile.
     35     # ref: http://gcc.gnu.org/wiki/DebugFission
     36     switch ${lang} {
     37 	"c" { set filename ${testfile}.o }
     38 	"c++" { set filename ${testfile}-cxx.o }
     39     }
     40     set objfile [standard_output_file $filename]
     41 
     42     if { [gdb_compile "${srcdir}/${subdir}/${srcfile}" "${objfile}" object "debug $lang"] != ""
     43 	 || [gdb_compile "${objfile}" "${exefile}" executable "debug $lang"] != "" } {
     44 	untested "failed to compile in $lang mode"
     45 	return -1
     46     }
     47     return 0
     48 }
     49 
     50 proc test_value_in_inferior {} {
     51     global gdb_prompt
     52     global testfile
     53 
     54     gdb_breakpoint [gdb_get_line_number "break to inspect struct and union"]
     55 
     56     gdb_continue_to_breakpoint "break to inspect struct and union"
     57 
     58     # Just get inferior variable s in the value history, available to guile.
     59     gdb_test "print s" "= {a = 3, b = 5}" ""
     60 
     61     gdb_scm_test_silent_cmd "gu (define s (history-ref 0))" "set s"
     62 
     63     gdb_test "gu (print (value-field s \"a\"))" \
     64 	"= 3" "access element inside struct using string name"
     65 
     66     # Append value in the value history.
     67     gdb_scm_test_silent_cmd "gu (define i (history-append! (make-value 42)))" \
     68 	"append 42"
     69 
     70     gdb_test "gu i" "\[0-9\]+"
     71     gdb_test "gu (history-ref i)" "#<gdb:value 42>"
     72     gdb_test "p \$" "= 42"
     73 
     74     # Verify the recorded history value survives a gc.
     75     gdb_test_no_output "guile (gc)"
     76     gdb_test "p \$\$" "= 42"
     77 
     78     # Make sure 'history-append!' rejects non-value objects.
     79     gdb_test "gu (history-append! 123)" \
     80 	"ERROR:.* Wrong type argument.*" "history-append! type error"
     81 
     82     # Test dereferencing the argv pointer.
     83 
     84     # Just get inferior variable argv the value history, available to guile.
     85     gdb_test "print argv" "= \\(char \\*\\*\\) 0x.*" ""
     86 
     87     gdb_scm_test_silent_cmd "gu (define argv (history-ref 0))" \
     88 	"set argv"
     89     gdb_scm_test_silent_cmd "gu (define arg0 (value-dereference argv))" \
     90 	"set arg0"
     91 
     92     # Check that the dereferenced value is sane.
     93     global has_argv0
     94     set test "verify dereferenced value"
     95     if { $has_argv0 } {
     96 	gdb_test_no_output "set print elements unlimited" ""
     97 	gdb_test_no_output "set print repeats unlimited" ""
     98 	gdb_test "gu (print arg0)" "0x.*$testfile\"" $test
     99     } else {
    100 	unsupported $test
    101     }
    102 
    103     # Smoke-test value-optimized-out?.
    104     gdb_test "gu (print (value-optimized-out? arg0))" \
    105 	"= #f" "Test value-optimized-out?"
    106 
    107     # Test address attribute.
    108     gdb_test "gu (print (value-address arg0))" \
    109 	"= 0x\[\[:xdigit:\]\]+" "Test address attribute"
    110     # Test address attribute is #f in a non-addressable value.
    111     gdb_test "gu (print (value-address (make-value 42)))" \
    112 	"= #f" "Test address attribute in non-addressable value"
    113 
    114     # Test displaying a variable that is temporarily at a bad address.
    115     # But if we can examine what's at memory address 0, then we'll also be
    116     # able to display it without error.  Don't run the test in that case.
    117     set can_read_0 [is_address_zero_readable]
    118 
    119     # Test memory error.
    120     set test "parse_and_eval with memory error"
    121     if {$can_read_0} {
    122 	untested $test
    123     } else {
    124 	gdb_test "gu (print (parse-and-eval \"*(int*)0\"))" \
    125 	    "ERROR: Cannot access memory at address 0x0.*" $test
    126     }
    127 
    128     # Test Guile lazy value handling
    129     set test "memory error and lazy values"
    130     if {$can_read_0} {
    131 	untested $test
    132     } else {
    133 	gdb_test_no_output "gu (define inval (parse-and-eval \"*(int*)0\"))"
    134 	gdb_test "gu (print (value-lazy? inval))" \
    135 	    "#t"
    136 	gdb_test "gu (define inval2 (value-add inval 1))" \
    137 	    "ERROR: Cannot access memory at address 0x0.*" \
    138 	    "$test, using value in value-add"
    139 	gdb_test "gu (value-fetch-lazy! inval))" \
    140 	    "ERROR: Cannot access memory at address 0x0.*" \
    141 	    "$test, using value in value-fetch-lazy!"
    142     }
    143     gdb_test_no_output "gu (define argc-lazy (parse-and-eval \"argc\"))"
    144     gdb_test_no_output "gu (define argc-notlazy (parse-and-eval \"argc\"))"
    145     gdb_test_no_output "gu (value-fetch-lazy! argc-notlazy)"
    146     gdb_test "gu (print (value-lazy? argc-lazy))" "= #t" \
    147 	"argc-lazy is initially lazy"
    148     gdb_test "gu (print (value-lazy? argc-notlazy))" "= #f"
    149     gdb_test "print argc" "= 1" "sanity check argc"
    150     gdb_test "gu (print (value-lazy? argc-lazy))" "= #t" \
    151 	"argc-lazy is still lazy after argc is printed"
    152     gdb_test_no_output "set argc=2"
    153     gdb_test "gu (print argc-notlazy)" "= 1"
    154     gdb_test "gu (print argc-lazy)" "= 2"
    155     gdb_test "gu (print (value-lazy? argc-lazy))" "= #f" \
    156 	"argc-lazy is no longer lazy"
    157 
    158     # Test string fetches, both partial and whole.
    159     gdb_test "print st" "\"divide et impera\""
    160     gdb_scm_test_silent_cmd "gu (define st (history-ref 0))" \
    161 	"inf: get st value from history"
    162     gdb_test "gu (print (value->string st))" \
    163 	"= divide et impera"  "Test string with no length"
    164     gdb_test "gu (print (value->string st #:length -1))" \
    165 	"= divide et impera" "Test string (length = -1) is all of the string"
    166     gdb_test "gu (print (value->string st #:length 6))" \
    167 	"= divide"
    168     gdb_test "gu (print (string-append \"---\" (value->string st #:length 0) \"---\"))" \
    169 	"= ------" "Test string (length = 0) is empty"
    170     gdb_test "gu (print (string-length (value->string st #:length 0)))" \
    171 	"= 0" "Test length is 0"
    172 
    173     # Fetch a string that has embedded nulls.
    174     gdb_test "print nullst" "\"divide\\\\000et\\\\000impera\".*"
    175     gdb_scm_test_silent_cmd "gu (define nullst (history-ref 0))" \
    176 	"inf: get nullst value from history"
    177     gdb_test "gu (print (value->string nullst))" \
    178 	"divide" "Test string to first null"
    179     gdb_scm_test_silent_cmd "gu (set! nullst (value->string nullst #:length 9))" \
    180 	"get string beyond null"
    181     gdb_test "gu (print nullst)" \
    182 	"= divide\\\\000et"
    183 
    184     gdb_scm_test_silent_cmd "gu (define argv-ref (value-reference-value argv))" \
    185 	"test value-reference-value"
    186     gdb_test "gu (equal? argv (value-referenced-value argv-ref))" "#t"
    187     gdb_test "gu (eqv? (type-code (value-type argv-ref)) TYPE_CODE_REF)" "#t"
    188 
    189     gdb_scm_test_silent_cmd "gu (define argv-rref (value-rvalue-reference-value argv))" \
    190 	"test value-rvalue-reference-value"
    191     gdb_test "gu (equal? argv (value-referenced-value argv-rref))" "#t"
    192     gdb_test "gu (eqv? (type-code (value-type argv-rref)) TYPE_CODE_RVALUE_REF)" "#t"
    193 
    194     gdb_test "gu (equal? (value-type (value-const-value argv)) (type-const (value-type argv)))" \
    195 	"#t"
    196 }
    197 
    198 proc test_strings {} {
    199     gdb_test "gu (make-value \"test\")" "#<gdb:value \"test\">" "make string"
    200 
    201     # Test string conversion errors.
    202     set save_charset [get_target_charset]
    203     gdb_test_no_output "set target-charset UTF-8"
    204 
    205     gdb_test_no_output "gu (set-port-conversion-strategy! #f 'error)"
    206     gdb_test "gu (print (value->string (make-value (string #\\x1234)) #:encoding \"ASCII\"))" \
    207 	"ERROR.*decoding-error.*" \
    208 	"value->string with default #:errors = 'error"
    209 
    210     # There is no 'escape strategy for C->SCM string conversions, but it's
    211     # still a legitimate value for %default-port-conversion-strategy.
    212     # GDB handles this by, umm, substituting 'substitute.
    213     # Use this case to also handle "#:errors #f" which explicitly says
    214     # "use %default-port-conversion-strategy".
    215     gdb_test_no_output "gu (set-port-conversion-strategy! #f 'escape)"
    216     gdb_test "gu (print (value->string (make-value (string #\\x1234)) #:encoding \"ASCII\" #:errors #f))" \
    217 	"= \[?\]{3}" "value->string with default #:errors = 'escape"
    218 
    219     # This is last in the default conversion tests so that
    220     # %default-port-conversion-strategy ends up with the default value.
    221     gdb_test_no_output "gu (set-port-conversion-strategy! #f 'substitute)"
    222     gdb_test "gu (print (value->string (make-value (string #\\x1234)) #:encoding \"ASCII\"))" \
    223 	"= \[?\]{3}" "value->string with default #:errors = 'substitute"
    224 
    225     gdb_test "gu (print (value->string (make-value (string #\\x1234)) #:encoding \"ASCII\" #:errors 'error))" \
    226 	"ERROR.*decoding-error.*" "value->string #:errors 'error"
    227     gdb_test "gu (print (value->string (make-value (string #\\x1234)) #:encoding \"ASCII\" #:errors 'substitute))" \
    228 	"= \[?\]{3}" "value->string #:errors 'substitute"
    229     gdb_test "gu (print (value->string (make-value \"abc\") #:errors \"foo\"))" \
    230 	"ERROR.*invalid error kind.*" "bad value for #:errors"
    231 
    232     gdb_test_no_output "set target-charset $save_charset" \
    233 	"restore target-charset"
    234 }
    235 
    236 proc test_inferior_function_call {} {
    237     global gdb_prompt hex decimal
    238 
    239     # Correct inferior call without arguments.
    240     gdb_test "p/x fp1" "= $hex.*"
    241     gdb_scm_test_silent_cmd "gu (define fp1 (history-ref 0))" \
    242 	"get fp1 value from history"
    243     gdb_scm_test_silent_cmd "gu (set! fp1 (value-dereference fp1))" \
    244 	"dereference fp1"
    245     gdb_test "gu (print (value-call fp1 '()))" \
    246 	"= void"
    247 
    248     # Correct inferior call with arguments.
    249     gdb_test "p/x fp2" "= $hex.*" \
    250 	"place fp2 into value history, the first time"
    251     gdb_scm_test_silent_cmd "gu (define fp2 (history-ref 0))" \
    252 	"get fp2 value from history"
    253     gdb_scm_test_silent_cmd "gu (set! fp2 (value-dereference fp2))" \
    254 	"dereference fp2"
    255     gdb_test "gu (print (value-call fp2 (list 10 20)))" \
    256 	"= 30"
    257 
    258     # Incorrect to call an int value.
    259     gdb_test "p i" "= $decimal.*"
    260     gdb_scm_test_silent_cmd "gu (define i (history-ref 0))" \
    261 	"inf call: get i value from history"
    262     gdb_test "gu (print (value-call i '()))" \
    263 	"ERROR: .*: Wrong type argument in position 1 \\(expecting function \\(value of TYPE_CODE_FUNC\\)\\): .*"
    264 
    265     # Incorrect number of arguments.
    266     gdb_test "p/x fp2" "= $hex.*" \
    267 	"place fp2 into value history, the second time"
    268     gdb_scm_test_silent_cmd "gu (define fp3 (history-ref 0))" \
    269 	"get fp3 value from history"
    270     gdb_scm_test_silent_cmd "gu (set! fp3 (value-dereference fp3))" \
    271 	"dereference fp3"
    272     gdb_test "gu (print (value-call fp3 (list 10)))" \
    273 	"ERROR: Too few arguments in function call.*"
    274 }
    275 
    276 proc test_value_after_death {} {
    277     # Construct a type while the inferior is still running.
    278     gdb_scm_test_silent_cmd "gu (define ptrtype (lookup-type \"PTR\"))" \
    279 	"create PTR type"
    280 
    281     # Kill the inferior and remove the symbols.
    282     gdb_test "kill" "" "kill the inferior" \
    283 	"Kill the program being debugged. .y or n. $" \
    284 	"y"
    285     gdb_test "file" "" "discard the symbols" \
    286 	"Discard symbol table from.*y or n. $" \
    287 	"y"
    288 
    289     # First do a garbage collect to delete anything unused.  PR 16612.
    290     gdb_scm_test_silent_cmd "gu (gc)" "garbage collect"
    291 
    292     # Now create a value using that type.  Relies on arg0, created by
    293     # test_value_in_inferior.
    294     gdb_scm_test_silent_cmd "gu (define castval (value-cast arg0 (type-pointer ptrtype)))" \
    295 	"cast arg0 to PTR"
    296 
    297     # Make sure the type is deleted.
    298     gdb_scm_test_silent_cmd "gu (set! ptrtype #f)" \
    299 	"delete PTR type"
    300 
    301     # Now see if the value's type is still valid.
    302     gdb_test "gu (print (value-type castval))" \
    303 	"= PTR ." "print value's type"
    304 }
    305 
    306 # Regression test for invalid subscript operations.  The bug was that
    307 # the type of the value was not being checked before allowing a
    308 # subscript operation to proceed.
    309 
    310 proc test_subscript_regression {exefile lang} {
    311     # Start with a fresh gdb.
    312     clean_restart
    313     gdb_load ${exefile}
    314 
    315     if ![gdb_guile_runto_main ] {
    316 	return
    317     }
    318 
    319     if {$lang == "c++"} {
    320 	gdb_breakpoint [gdb_get_line_number "break to inspect pointer by reference"]
    321 	gdb_continue_to_breakpoint "break to inspect pointer by reference"
    322 
    323 	gdb_scm_test_silent_cmd "print rptr_int" \
    324 	    "Obtain address"
    325 	gdb_scm_test_silent_cmd "gu (define rptr (history-ref 0))" \
    326 	    "set rptr"
    327 	gdb_test "gu (print (value-subscript rptr 0))" \
    328 	    "= 2" "Check pointer passed as reference"
    329 
    330 	# Just the most basic test of dynamic_cast -- it is checked in
    331 	# the C++ tests.
    332 	gdb_test "gu (print (value->bool (value-dynamic-cast (parse-and-eval \"base\") (type-pointer (lookup-type \"Derived\")))))" \
    333 	    "= #t"
    334 
    335 	# Likewise.
    336 	gdb_test "gu (print (value-dynamic-type (parse-and-eval \"base\")))" \
    337 	    "= Derived \[*\]"
    338 	gdb_test "gu (print (value-dynamic-type (parse-and-eval \"base_ref\")))" \
    339 	    "= Derived \[&\]"
    340 	# A static type case.
    341 	gdb_test "gu (print (value-dynamic-type (parse-and-eval \"5\")))" \
    342 	    "= int"
    343     }
    344 
    345     gdb_breakpoint [gdb_get_line_number "break to inspect struct and union"]
    346     gdb_continue_to_breakpoint "break to inspect struct and union in $lang"
    347 
    348     gdb_scm_test_silent_cmd "gu (define intv (make-value 1))" \
    349 	"Create int value for subscript test"
    350     gdb_scm_test_silent_cmd "gu (define stringv (make-value \"foo\"))" \
    351 	"Create string value for subscript test"
    352 
    353     # Try to access an int with a subscript.  This should fail.
    354     gdb_test "gu (print intv)" \
    355 	"= 1" "Baseline print of an int Guile value"
    356     gdb_test "gu (print (value-subscript intv 0))" \
    357 	"ERROR: Cannot subscript requested type.*" \
    358 	"Attempt to access an integer with a subscript"
    359 
    360     # Try to access a string with a subscript.  This should pass.
    361     gdb_test "gu (print stringv)" \
    362 	"= \"foo\"" "Baseline print of a string Guile value"
    363     gdb_test "gu (print (value-subscript stringv 0))" \
    364 	"= 102 'f'" "Attempt to access a string with a subscript"
    365 
    366     # Try to access an int array via a pointer with a subscript.
    367     # This should pass.
    368     gdb_scm_test_silent_cmd "print p" "Build pointer to array"
    369     gdb_scm_test_silent_cmd "gu (define pointer (history-ref 0))" "set pointer"
    370     gdb_test "gu (print (value-subscript pointer 0))" \
    371 	"= 1" "Access array via pointer with int subscript"
    372     gdb_test "gu (print (value-subscript pointer intv))" \
    373 	"= 2" "Access array via pointer with value subscript"
    374 
    375     # Try to access a single dimension array with a subscript to the
    376     # result.  This should fail.
    377     gdb_test "gu (print (value-subscript (value-subscript pointer intv) 0))" \
    378 	"ERROR: Cannot subscript requested type.*" \
    379 	"Attempt to access an integer with a subscript 2"
    380 
    381     # Lastly, test subscript access to an array with multiple
    382     # dimensions.  This should pass.
    383     gdb_scm_test_silent_cmd "print {\"fu \",\"foo\",\"bar\"}" "Build array"
    384     gdb_scm_test_silent_cmd "gu (define marray (history-ref 0))" ""
    385     gdb_test "gu (print (value-subscript (value-subscript marray 1) 2))" \
    386 	"o." "Test multiple subscript"
    387 }
    388 
    389 # A few tests of gdb:parse-and-eval.
    390 
    391 proc test_parse_and_eval {} {
    392     gdb_test "gu (print (parse-and-eval \"23\"))" \
    393 	"= 23" "parse-and-eval constant test"
    394     gdb_test "gu (print (parse-and-eval \"5 + 7\"))" \
    395 	"= 12" "parse-and-eval simple expression test"
    396     gdb_test "gu (raw-print (parse-and-eval \"5 + 7\"))" \
    397 	"#<gdb:value 12>" "parse-and-eval type test"
    398 }
    399 
    400 # Test that values are hashable.
    401 # N.B.: While smobs are hashable, the hash is really non-existent,
    402 # they all get hashed to the same value.  Guile may provide a hash function
    403 # for smobs in a future release.  In the meantime one should use a custom
    404 # hash table that uses gdb:hash-gsmob.
    405 
    406 proc test_value_hash {} {
    407     gdb_test_multiline "Simple Guile value dictionary" \
    408 	"guile" "" \
    409 	"(define one (make-value 1))" "" \
    410 	"(define two (make-value 2))" "" \
    411 	"(define three (make-value 3))" "" \
    412         "(define vdict (make-hash-table 5))" "" \
    413 	"(hash-set! vdict one \"one str\")" "" \
    414 	"(hash-set! vdict two \"two str\")" "" \
    415 	"(hash-set! vdict three \"three str\")" "" \
    416 	"end"
    417     gdb_test "gu (print (hash-ref vdict one))" \
    418 	"one str" "Test dictionary hash 1"
    419     gdb_test "gu (print (hash-ref vdict two))" \
    420 	"two str" "Test dictionary hash 2"
    421     gdb_test "gu (print (hash-ref vdict three))" \
    422 	"three str" "Test dictionary hash 3"
    423 }
    424 
    425 # Build C version of executable.  C++ is built later.
    426 if { [build_inferior "${binfile}" "c"] < 0 } {
    427     return
    428 }
    429 
    430 # Start with a fresh gdb.
    431 clean_restart ${::testfile}
    432 
    433 gdb_install_guile_utils
    434 gdb_install_guile_module
    435 
    436 test_parse_and_eval
    437 test_value_hash
    438 
    439 # The following tests require execution.
    440 
    441 if ![gdb_guile_runto_main] {
    442     return
    443 }
    444 
    445 test_value_in_inferior
    446 test_inferior_function_call
    447 test_strings
    448 test_value_after_death
    449 
    450 # Test either C or C++ values.
    451 
    452 test_subscript_regression "${binfile}" "c"
    453 
    454 if {[allow_cplus_tests]} {
    455     if { [build_inferior "${binfile}-cxx" "c++"] < 0 } {
    456 	return
    457     }
    458     with_test_prefix "c++" {
    459 	test_subscript_regression "${binfile}-cxx" "c++"
    460     }
    461 }
    462