# Copyright (C) 2022-2024 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 related to source code highlighting and Python. Includes a
# test for using the Pygments module as a fall back to GNU source
# highlight.
#
# This script also includes tests for handling a non-uft-8 character
# with both Pygments highlighting, and with gdb.execute (when using
# the list command).
require allow_python_tests
load_lib gdb-python.exp
standard_testfile
if { [build_executable "failed to build" ${testfile} ${srcfile}] == -1 } {
return
}
set line_number [gdb_get_line_number "List this line."]
# Helper proc. Run CMD, which should produce a source listing, and
# check if the source code is styled or not. EXPECT_STYLED indicates
# if we expect the source listing to be styled or not.
proc check_source_listing_styling { cmd expect_styled { testname "" } } {
if { $testname eq "" } {
set testname $cmd
}
set seen_style_escape false
gdb_test_multiple $cmd $testname {
-re -wrap "Python Exception.*" {
fail $gdb_test_name
return
}
-re "\033" {
set seen_style_escape true
exp_continue
}
-re "$::gdb_prompt $" {
gdb_assert { $seen_style_escape == $expect_styled } \
$gdb_test_name
}
}
}
# Check that the Python pygments module can be used for source
# highlighting when GNU source highlight is not available (or is
# disabled, as is done in this test).
proc test_pygments_styling {} {
clean_restart $::binfile
# Remote host boards disable styling via GDB's command line. Turn
# it back on now.
if {[is_remote host]} {
gdb_test "set style enabled on"
}
if { ![gdb_py_module_available "pygments"] } {
unsupported "pygments module not available"
return
}
if ![runto_main] {
return
}
gdb_test_no_output "maint set gnu-source-highlight enabled off"
gdb_test "maint flush source-cache" "Source cache flushed\\."
check_source_listing_styling "list $::line_number" true
}
# Use gdb.execute to list source code containing non-utf-8 character.
# Check that initially GDB fails to convert the source code to a
# string, then set the correct host encoding, and try again. This
# time the conversion should succeed.
proc test_gdb_execute_non_utf8_source {} {
clean_restart $::binfile
# The default host charset is utf-8, the source code contains a
# non-utf-8 character, so this will fail.
gdb_test \
"python source = gdb.execute('list $::line_number', True, True)" \
[multi_line \
"Python Exception : 'ascii' codec can't decode byte 0xc0 in position 250: ordinal not in range\\(128\\)" \
"Error occurred in Python: 'ascii' codec can't decode byte 0xc0 in position 250: ordinal not in range\\(128\\)"] \
"gdb.execute fails to convert result to string"
# Set the correct host charset, and try the conversion again.
gdb_test_no_output "set host-charset ISO-8859-1"
gdb_test_no_output \
"python source = gdb.execute('list $::line_number', True, True)" \
"gdb.execute does convert result to string"
# Check that we captured something that looks like the expected source.
gdb_test "python print(source)" ".*List this line.*"
}
# Use gdb.execute() to list source code. Alternate between asking for
# styled, and unstyled source code. In some cases we ask for the
# output to be returned via a string, and in other cases we ask for
# the output to be sent straight to stdout.
proc_with_prefix test_source_cache_style_tracking {} {
clean_restart $::binfile
# Remote host boards disable styling via GDB's command line. Turn
# it back on now.
if {[is_remote host]} {
gdb_test "set style enabled on"
}
gdb_test_no_output "set host-charset ISO-8859-1"
# Commands which return styled, and non-styled source code mixed
# together. This ensures that the source cache will need to keep
# discarding the entry with the wrong styling mode. All of these
# gdb.execute calls send their output via a string.
check_source_listing_styling \
"python print(gdb.execute('list $::line_number', to_string=True), end='')" \
false
check_source_listing_styling \
"python print(gdb.execute('list $::line_number', to_string=True, styling=True), end='')" \
true
foreach from_tty { True False } {
check_source_listing_styling \
"python print(gdb.execute('list $::line_number', $from_tty, True), end='')" \
false
check_source_listing_styling \
"python print(gdb.execute('list $::line_number', $from_tty, True, True), end='')" \
true
check_source_listing_styling \
"python print(gdb.execute('list $::line_number', $from_tty, True, False), end='')" \
false
}
# The same again, but this time the output is sent directly to
# stdout.
check_source_listing_styling \
"python gdb.execute('list $::line_number')" \
true
check_source_listing_styling \
"python gdb.execute('list $::line_number', to_string=False, styling=False)" \
false
check_source_listing_styling \
"python gdb.execute('list $::line_number', to_string=False, styling=True)" \
true
foreach from_tty { True False } {
check_source_listing_styling \
"python gdb.execute('list $::line_number', $from_tty, False, False)" \
false
check_source_listing_styling \
"python gdb.execute('list $::line_number', $from_tty, False, True)" \
true
check_source_listing_styling \
"python gdb.execute('list $::line_number', $from_tty, False)" \
true
}
}
# We need an ANSI-capable terminal to get the output, additionally we
# need to set LC_ALL so GDB knows the terminal is UTF-8 capable,
# otherwise we'll get a UnicodeEncodeError trying to encode the
# output.
with_ansi_styling_terminal {
test_pygments_styling
test_gdb_execute_non_utf8_source
test_source_cache_style_tracking
}