From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: Received: from us-smtp-delivery-124.mimecast.com (us-smtp-delivery-124.mimecast.com [170.10.129.124]) by sourceware.org (Postfix) with ESMTPS id 077B43858437 for ; Mon, 7 Mar 2022 12:33:25 +0000 (GMT) DMARC-Filter: OpenDMARC Filter v1.4.1 sourceware.org 077B43858437 Received: from mail-wr1-f71.google.com (mail-wr1-f71.google.com [209.85.221.71]) by relay.mimecast.com with ESMTP with STARTTLS (version=TLSv1.2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id us-mta-447-hFdq-WdNPf6LAeOexucEUg-1; Mon, 07 Mar 2022 07:33:23 -0500 X-MC-Unique: hFdq-WdNPf6LAeOexucEUg-1 Received: by mail-wr1-f71.google.com with SMTP id c5-20020adffb05000000b001edbbefe96dso4487693wrr.8 for ; Mon, 07 Mar 2022 04:33:23 -0800 (PST) X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20210112; h=x-gm-message-state:from:to:cc:subject:date:message-id:in-reply-to :references:mime-version:content-transfer-encoding; bh=nmGj+sPaB9Igqk3kdmD5nK6N6Cngtqd2fajDkFFv+NI=; b=oF7HVF+FWbiUZWk3SyxCE6KYA1m4dNorR+h3+4a6jSJzeW4PU2M67cQhSjS4IeK7jK bVIN8288+TU90a48mxeWNkFpl+7+ip35icQ4MExzdUQ0XJ4HpTCEYA4qM331lwY1+QgU qNnoHw/0jci3KNcyQmSIZjtI5l7Iov/cB5sqvv/JgX+dxJmQSV0tNfGkwqkEzdop3Qwc sDMMLA3HmwYn+B3VaYoLKxZRJJSqYhpGuPEhUkMZODj0rJ4dlIak9Kac28TmQtSr7qgQ zZUU4r0zDNZogjXKeTCwe4LxHnzcSZDYdMrWGM6uUn9rvw9b8ajwd//Q/NYBG44zRN2u 6hYw== X-Gm-Message-State: AOAM532jqxE472qTSSfVFxhKfECBFpO4jdCpcj46KCT13G3OTRfKGeKV c5/yp0Qm6eAbS+oeEhVG2jzdrMpmATxR/pEGHfwBuWPkjKIbB+Nn7Xo3cGbcY4VtH5uVoKMtuxX ocy9Avkw13C0Vnev0B5xzcVGHLFoso4qL5wE2Z8N3Ezs/5Va83MchRkHJA7GvsKzGlnCNatndcA == X-Received: by 2002:a05:600c:1988:b0:37c:505:d45c with SMTP id t8-20020a05600c198800b0037c0505d45cmr8822977wmq.107.1646656401518; Mon, 07 Mar 2022 04:33:21 -0800 (PST) X-Google-Smtp-Source: ABdhPJx50aSukGAEZt1N7ntPjS4nu+i88075M5bYLJk3L0kQiyFxjRNl7nOJhqOdj8mR8DFZKT995w== X-Received: by 2002:a05:600c:1988:b0:37c:505:d45c with SMTP id t8-20020a05600c198800b0037c0505d45cmr8822911wmq.107.1646656400531; Mon, 07 Mar 2022 04:33:20 -0800 (PST) Received: from localhost (host86-130-161-69.range86-130.btcentralplus.com. [86.130.161.69]) by smtp.gmail.com with ESMTPSA id m3-20020a5d6243000000b001e33760776fsm11304577wrv.10.2022.03.07.04.33.19 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Mon, 07 Mar 2022 04:33:20 -0800 (PST) From: Andrew Burgess To: gdb-patches@sourceware.org Cc: Simon Marchi , Andrew Burgess Subject: [PATCHv3] gdb/python: add gdb.format_address function Date: Mon, 7 Mar 2022 12:33:17 +0000 Message-Id: <20220307123317.3966024-1-aburgess@redhat.com> X-Mailer: git-send-email 2.25.4 In-Reply-To: <20220304105031.2706582-1-aburgess@redhat.com> References: <20220304105031.2706582-1-aburgess@redhat.com> MIME-Version: 1.0 X-Mimecast-Spam-Score: 0 X-Mimecast-Originator: redhat.com Content-Transfer-Encoding: 8bit Content-Type: text/plain; charset="US-ASCII" X-Spam-Status: No, score=-12.2 required=5.0 tests=BAYES_00, DKIM_INVALID, DKIM_SIGNED, GIT_PATCH_0, KAM_DMARC_NONE, KAM_DMARC_STATUS, KAM_SHORT, RCVD_IN_DNSWL_NONE, RCVD_IN_MSPIKE_H5, RCVD_IN_MSPIKE_WL, SPF_HELO_NONE, SPF_NONE, TXREP, T_SCC_BODY_TEXT_LINE autolearn=ham autolearn_force=no version=3.4.4 X-Spam-Checker-Version: SpamAssassin 3.4.4 (2020-01-24) on server2.sourceware.org X-BeenThere: gdb-patches@sourceware.org X-Mailman-Version: 2.1.29 Precedence: list List-Id: Gdb-patches mailing list List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , X-List-Received-Date: Mon, 07 Mar 2022 12:33:29 -0000 From: Andrew Burgess Since v2: - Almost complete rewrite, function is now gdb.format_address, and takes an address, program space, and architecture, - Updated docs to go with this, - Updated tests. Since v1: - All the doc changes have been addressed (these have already been approved), - Rebased to current master, - Added an additional test for when 'set print symbol-filename' is on. Thanks, Andrew --- Add a new function, gdb.format_address, which is a wrapper around GDB's print_address function. This method takes an address, and returns a string with the format: ADDRESS Where, ADDRESS is the original address, formatted as hexadecimal, SYMBOL is a symbol with an address lower than ADDRESS, and OFFSET is the offset from SYMBOL to ADDRESS in decimal. If there's no SYMBOL suitably close to ADDRESS then the part is not included. This is useful if a user wants to write a Python script that pretty-prints addresses, the user no longer needs to do manual symbol lookup, or worry about correctly formatting addresses. Additionally, there are some settings that effect how GDB picks SYMBOL, and whether the file name and line number should be included with the SYMBOL name, the gdb.format_address function ensures that the users Python script also benefits from these settings. The gdb.format_address by default selects SYMBOL from the current inferiors program space, and address is formatted using the architecture for the current inferior. However, a user can also explicitly pass a program space and architecture like this: gdb.format_address(ADDRESS, PROGRAM_SPACE, ARCHITECTURE) In order to format an address for a different inferior. Notes on the implementation: In py-arch.c I extended arch_object_to_gdbarch to add an assertion for the type of the PyObject being worked on. Prior to this commit all uses of arch_object_to_gdbarch were guaranteed to pass this function a gdb.Architecture object, but, with this commit, this might not be the case. So, with this commit I've made it a requirement that the PyObject be a gdb.Architecture, and this is checked with the assert. And in order that callers from other files can check if they have a gdb.Architecture object, I've added the new function gdbpy_is_architecture. In py-progspace.c I've added two new function, the first progspace_object_to_program_space, converts a PyObject of type gdb.Progspace to the associated program_space pointer, and gdbpy_is_progspace checks if a PyObject is a gdb.Progspace or not. --- gdb/NEWS | 6 + gdb/doc/python.texi | 54 ++++++ gdb/python/py-arch.c | 13 +- gdb/python/py-progspace.c | 17 ++ gdb/python/python-internal.h | 16 ++ gdb/python/python.c | 108 +++++++++++ gdb/testsuite/gdb.python/py-format-address.c | 32 ++++ .../gdb.python/py-format-address.exp | 177 ++++++++++++++++++ 8 files changed, 421 insertions(+), 2 deletions(-) create mode 100644 gdb/testsuite/gdb.python/py-format-address.c create mode 100644 gdb/testsuite/gdb.python/py-format-address.exp diff --git a/gdb/NEWS b/gdb/NEWS index dc2cac1871b..ec6a0753024 100644 --- a/gdb/NEWS +++ b/gdb/NEWS @@ -206,6 +206,12 @@ GNU/Linux/LoongArch loongarch*-*-linux* state information, or None, if there is no such additional information. + ** New function gdb.format_address(ADDRESS, PROGSPACE, ARCHITECTURE), + that formats ADDRESS as 'address ', where symbol is + looked up in PROGSPACE, and ARCHITECTURE is used to format address. + This is the same format that GDB uses when printing address, symbol, + and offset information from the disassembler. + * New features in the GDB remote stub, GDBserver ** GDBserver is now supported on OpenRISC GNU/Linux. diff --git a/gdb/doc/python.texi b/gdb/doc/python.texi index 4d9e77bf12c..721c5d3176e 100644 --- a/gdb/doc/python.texi +++ b/gdb/doc/python.texi @@ -612,6 +612,60 @@ connection objects are in no particular order in the returned list. @end defun +@defun gdb.format_address (@var{address} @r{[}, @var{progspace}, @var{architecture}@r{]}) +Return a string in the format @samp{@var{addr} +<@var{symbol}+@var{offset}>}, where @var{addr} is @var{address} +formatted in hexadecimal, @var{symbol} is the symbol whose address is +the nearest to @var{address} and below it in memory, and @var{offset} +is the offset from @var{symbol} to @var{address} in decimal. + +If no suitable @var{symbol} was found, then the +<@var{symbol}+@var{offset}> part is not included in the returned +string, instead the returned string will just contain the +@var{address} formatted as hexadecimal. How far @value{GDBN} looks +back for a suitable symbol can be controlled with @kbd{set print +max-symbolic-offset} (@pxref{Print Settings}). + +Additionally, the returned string can include file name and line +number information when @kbd{set print symbol-filename on} +(@pxref{Print Settings}), in this case the format of the returned +string is @samp{@var{addr} <@var{symbol}+@var{offset}> at +@var{filename}:@var{line-number}}. + + +The @var{progspace} is the gdb.Progspace in which @var{symbol} is +looked up, and @var{architecture} is used when formatting @var{addr}, +e.g.@: in order to determine the size of an address in bytes. + +If neither @var{progspace} or @var{architecture} are passed, then by +default @value{GDBN} will use the program space and architecture of +the currently selected inferior, thus, the following two calls are +equivalent: + +@smallexample +gdb.format_address(address) +gdb.format_address(address, + gdb.selected_inferior().progspace, + gdb.selected_inferior().architecture()) +@end smallexample + +It is not valid to only pass one of @var{progspace} or +@var{architecture}, either they must both be provided, or neither must +be provided (and the defaults will be used). + +This method uses the same mechanism for formatting address, symbol, +and offset information as core @value{GDBN} does in commands such as +@kbd{disassemble}. + +Here are some examples of the possible string formats: + +@smallexample +0x00001042 +0x00001042 +0x00001042 +@end smallexample +@end defun + @node Exception Handling @subsubsection Exception Handling @cindex python exceptions diff --git a/gdb/python/py-arch.c b/gdb/python/py-arch.c index 0f273b344e4..53906ce506e 100644 --- a/gdb/python/py-arch.c +++ b/gdb/python/py-arch.c @@ -62,16 +62,25 @@ arch_object_data_init (struct gdbarch *gdbarch) } /* Returns the struct gdbarch value corresponding to the given Python - architecture object OBJ. */ + architecture object OBJ, which must be a gdb.Architecture object. */ struct gdbarch * arch_object_to_gdbarch (PyObject *obj) { - arch_object *py_arch = (arch_object *) obj; + gdb_assert (PyObject_TypeCheck (obj, &arch_object_type)); + arch_object *py_arch = (arch_object *) obj; return py_arch->gdbarch; } +/* See python-internal.h. */ + +bool +gdbpy_is_architecture (PyObject *obj) +{ + return PyObject_TypeCheck (obj, &arch_object_type); +} + /* Returns the Python architecture object corresponding to GDBARCH. Returns a new reference to the arch_object associated as data with GDBARCH. */ diff --git a/gdb/python/py-progspace.c b/gdb/python/py-progspace.c index 1e01068c59b..f9f2a969e2b 100644 --- a/gdb/python/py-progspace.c +++ b/gdb/python/py-progspace.c @@ -504,6 +504,23 @@ pspace_to_pspace_object (struct program_space *pspace) return gdbpy_ref<>::new_reference (result); } +/* See python-internal.h. */ + +struct program_space * +progspace_object_to_program_space (PyObject *obj) +{ + gdb_assert (PyObject_TypeCheck (obj, &pspace_object_type)); + return ((pspace_object *) obj)->pspace; +} + +/* See python-internal.h. */ + +bool +gdbpy_is_progspace (PyObject *obj) +{ + return PyObject_TypeCheck (obj, &pspace_object_type); +} + void _initialize_py_progspace (); void _initialize_py_progspace () diff --git a/gdb/python/python-internal.h b/gdb/python/python-internal.h index 5e15f62f745..d2a8e4d8cf4 100644 --- a/gdb/python/python-internal.h +++ b/gdb/python/python-internal.h @@ -497,6 +497,13 @@ struct symtab_and_line *sal_object_to_symtab_and_line (PyObject *obj); struct frame_info *frame_object_to_frame_info (PyObject *frame_obj); struct gdbarch *arch_object_to_gdbarch (PyObject *obj); +/* Convert Python object OBJ to a program_space pointer. OBJ must be a + gdb.Progspace reference. Return nullptr if the gdb.Progspace is not + valid (see gdb.Progspace.is_valid), otherwise return the program_space + pointer. */ + +extern struct program_space *progspace_object_to_program_space (PyObject *obj); + void gdbpy_initialize_gdb_readline (void); int gdbpy_initialize_auto_load (void) CPYCHECKER_NEGATIVE_RESULT_SETS_EXCEPTION; @@ -824,4 +831,13 @@ typedef std::unique_ptr Py_buffer_up; extern bool gdbpy_parse_register_id (struct gdbarch *gdbarch, PyObject *pyo_reg_id, int *reg_num); +/* Return true if OBJ is a gdb.Architecture object, otherwise, return + false. */ + +extern bool gdbpy_is_architecture (PyObject *obj); + +/* Return true if OBJ is a gdb.Progspace object, otherwise, return false. */ + +extern bool gdbpy_is_progspace (PyObject *obj); + #endif /* PYTHON_PYTHON_INTERNAL_H */ diff --git a/gdb/python/python.c b/gdb/python/python.c index 79f9826365a..8abce819001 100644 --- a/gdb/python/python.c +++ b/gdb/python/python.c @@ -1294,6 +1294,107 @@ gdbpy_colorize_disasm (const std::string &content, gdbarch *gdbarch) +/* Implement gdb.format_address(ADDR,P_SPACE,ARCH). Provide access to + GDB's print_address function from Python. The returned address will + have the format '0x..... '. */ + +static PyObject * +gdbpy_format_address (PyObject *self, PyObject *args, PyObject *kw) +{ + static const char *keywords[] = + { + "address", "progspace", "architecture", nullptr + }; + PyObject *addr_obj = nullptr, *pspace_obj = nullptr, *arch_obj = nullptr; + CORE_ADDR addr; + struct gdbarch *gdbarch = nullptr; + struct program_space *pspace = nullptr; + + if (!gdb_PyArg_ParseTupleAndKeywords (args, kw, "O|OO", keywords, + &addr_obj, &pspace_obj, &arch_obj)) + return nullptr; + + if (get_addr_from_python (addr_obj, &addr) < 0) + return nullptr; + + /* If the user passed None for progspace or architecture, then we + consider this to mean "the default". Here we replace references to + None with nullptr, this means that in the following code we only have + to handle the nullptr case. These are only borrowed references, so + no decref is required here. */ + if (pspace_obj == Py_None) + pspace_obj = nullptr; + if (arch_obj == Py_None) + arch_obj = nullptr; + + if (pspace_obj == nullptr && arch_obj == nullptr) + { + /* Grab both of these from the current inferior, and its associated + default architecture. */ + pspace = current_inferior ()->pspace; + gdbarch = current_inferior ()->gdbarch; + } + else if (arch_obj == nullptr || pspace_obj == nullptr) + { + /* If the user has only given one of program space or architecture, + then don't use the default for the other. Sure we could use the + default, but it feels like there's too much scope of mistakes in + this case, so better to require the user to provide both + arguments. */ + PyErr_SetString (PyExc_ValueError, + _("The architecture and progspace arguments must both be supplied")); + return nullptr; + } + else + { + /* The user provided an address, program space, and architecture. + Just check that these objects are valid. */ + if (!gdbpy_is_progspace (pspace_obj)) + { + PyErr_SetString (PyExc_TypeError, + _("The progspace argument is not a gdb.Progspace object")); + return nullptr; + } + + pspace = progspace_object_to_program_space (pspace_obj); + if (pspace == nullptr) + { + PyErr_SetString (PyExc_ValueError, + _("The progspace argument is not valid")); + return nullptr; + } + + if (!gdbpy_is_architecture (arch_obj)) + { + PyErr_SetString (PyExc_TypeError, + _("The architecture argument is not a gdb.Architecture object")); + return nullptr; + } + + /* Architectures are never deleted once created, so gdbarch should + never come back as nullptr. */ + gdbarch = arch_object_to_gdbarch (arch_obj); + gdb_assert (gdbarch != nullptr); + } + + /* By this point we should know the program space and architecture we are + going to use. */ + gdb_assert (pspace != nullptr); + gdb_assert (gdbarch != nullptr); + + /* Unfortunately print_address relies on the current program space for + its symbol lookup. Temporarily switch now. */ + scoped_restore_current_program_space restore_progspace; + set_current_program_space (pspace); + + /* Format the address, and return it as a string. */ + string_file buf; + print_address (gdbarch, addr, &buf); + return PyString_FromString (buf.c_str ()); +} + + + /* Printing. */ /* A python function to write a single string using gdb's filtered @@ -2442,6 +2543,13 @@ Return a list of all the architecture names GDB understands." }, "connections () -> List.\n\ Return a list of gdb.TargetConnection objects." }, + { "format_address", (PyCFunction) gdbpy_format_address, + METH_VARARGS | METH_KEYWORDS, + "format_address (ADDRESS, PROG_SPACE, ARCH) -> String.\n\ +Format ADDRESS, an address within PROG_SPACE, a gdb.Progspace, using\n\ +ARCH, a gdb.Architecture to determine the address size. The format of\n\ +the returned string is 'ADDRESS ' without the quotes." }, + {NULL, NULL, 0, NULL} }; diff --git a/gdb/testsuite/gdb.python/py-format-address.c b/gdb/testsuite/gdb.python/py-format-address.c new file mode 100644 index 00000000000..6493fc4d579 --- /dev/null +++ b/gdb/testsuite/gdb.python/py-format-address.c @@ -0,0 +1,32 @@ +/* This testcase is part of GDB, the GNU debugger. + + Copyright 2022 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 . */ + +/* This test is compiled multiple times with FUNCTION_NAME defined to + different strings, this means we should (hopefully) get the same code + layout in memory, but with different strings for the function name. */ + +int +FUNCTION_NAME (void) +{ + return 0; +} + +int +main (void) +{ + return FUNCTION_NAME (); +} diff --git a/gdb/testsuite/gdb.python/py-format-address.exp b/gdb/testsuite/gdb.python/py-format-address.exp new file mode 100644 index 00000000000..5c808299d34 --- /dev/null +++ b/gdb/testsuite/gdb.python/py-format-address.exp @@ -0,0 +1,177 @@ +# Copyright 2022 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 . + +load_lib gdb-python.exp +standard_testfile + +foreach func_name { foo bar } { + if {[build_executable "build binary with ${func_name} function" \ + "$testfile-${func_name}" $srcfile \ + [list debug \ + additional_flags=-DFUNCTION_NAME=${func_name}]] == -1} { + return -1 + } +} + +set binary_foo [standard_output_file "${testfile}-foo"] +set binary_bar [standard_output_file "${testfile}-bar"] + +clean_restart $binary_foo + +# Skip all tests if Python scripting is not enabled. +if { [skip_python_tests] } { continue } + +if ![runto_main] { + return -1 +} + +# Check the gdb.format_address method when using the default values +# for the program space and architecture (these will be selected based +# on the current inferior). +set main_addr [get_hexadecimal_valueof "&main" "UNKNOWN"] +set next_addr [format 0x%x [expr $main_addr + 1]] + +foreach_with_prefix symbol_filename { on off } { + gdb_test_no_output "set print symbol-filename ${symbol_filename}" + + if { $symbol_filename == "on" } { + set filename_pattern " at \[^\r\n\]+/${srcfile}:$decimal" + } else { + set filename_pattern "" + } + + gdb_test "python print(\"Got: \" + gdb.format_address($main_addr))" \ + "Got: $main_addr " \ + "gdb.format_address, result should have no offset" + + gdb_test "python print(\"Got: \" + gdb.format_address($next_addr))" \ + "Got: $next_addr " \ + "gdb.format_address, result should have an offset" +} + +if {![is_address_zero_readable]} { + gdb_test "python print(\"Got: \" + gdb.format_address(0))" \ + "Got: 0x0" \ + "gdb.format_address for address 0" +} + +# Now check that gdb.format_address will accept the program space and +# architecture arguments correctly. +gdb_test_no_output "python inf = gdb.selected_inferior()" + +# First, pass both arguments, this should be fine. +gdb_test "python print(\"Got: \" + gdb.format_address($main_addr, inf.progspace, inf.architecture()))" \ + "Got: $main_addr
" \ + "gdb.format_address passing program space and architecture" + +# Now pass the program space and architecture as None. +# First, pass both arguments, this should be fine. +gdb_test "python print(\"Got: \" + gdb.format_address($main_addr, None, None))" \ + "Got: $main_addr
" \ + "gdb.format_address passing program space and architecture as None" + +# Now forget the architecture, this should fail. +gdb_test "python print(\"Got: \" + gdb.format_address($main_addr, inf.progspace))" \ + [multi_line \ + "ValueError: The architecture and progspace arguments must both be supplied" \ + "Error while executing Python code\\."] \ + "gdb.format_address passing program space only" + +gdb_test "python print(\"Got: \" + gdb.format_address($main_addr, inf.progspace, None))" \ + [multi_line \ + "ValueError: The architecture and progspace arguments must both be supplied" \ + "Error while executing Python code\\."] \ + "gdb.format_address passing real program space, but architecture is None" + +# Now skip the program space argument. +gdb_test "python print(\"Got: \" + gdb.format_address($main_addr, architecture=inf.architecture()))" \ + [multi_line \ + "ValueError: The architecture and progspace arguments must both be supplied" \ + "Error while executing Python code\\."] \ + "gdb.format_address passing architecture only" + +gdb_test "python print(\"Got: \" + gdb.format_address($main_addr, None, inf.architecture()))" \ + [multi_line \ + "ValueError: The architecture and progspace arguments must both be supplied" \ + "Error while executing Python code\\."] \ + "gdb.format_address passing real architecture, but progspace is None" + +# Now, before we add a second inferior, lets just check we can format +# the address of 'foo' correctly. +set foo_addr [get_hexadecimal_valueof "&foo" "UNKNOWN"] + +gdb_test "python print(\"Got: \" + gdb.format_address($foo_addr, inf.progspace, inf.architecture()))" \ + "Got: $foo_addr " \ + "gdb.format_address for foo, with just one inferior" + +# Now lets add a second inferior, using a slightly different +# executable, select that inferior, and capture a reference to the +# inferior in a Python object. +gdb_test "add-inferior -exec ${binary_bar}" ".*" \ + "add a second inferior running the bar executable" +gdb_test "inferior 2" ".*" +gdb_test_no_output "python inf2 = gdb.selected_inferior()" + +# Now we can test formatting an address from inferior 1. +gdb_test "python print(\"Got: \" + gdb.format_address($foo_addr, inf.progspace, inf.architecture()))" \ + "Got: $foo_addr " \ + "gdb.format_address for foo, while inferior 2 is selected" + +# Grab the address of 'bar'. Hopefully this will be the same address +# as 'foo', but if not, that's not the end of the world, the test just +# wont be quite as tough. +set bar_addr [get_hexadecimal_valueof "&bar" "UNKNOWN"] + +# Now format the address of bar using the default inferior and +# architecture, this should display the 'bar' symbol rather than +# 'foo'. +gdb_test "python print(\"Got: \" + gdb.format_address($bar_addr))" \ + "Got: $foo_addr " \ + "gdb.format_address for bar, while inferior 2 is selected" + +# And again, but this time, specificy the program space and +# architecture. +gdb_test "python print(\"Got: \" + gdb.format_address($bar_addr, inf2.progspace, inf2.architecture()))" \ + "Got: $foo_addr " \ + "gdb.format_address for bar, while inferior 2 is selected, pass progspace and architecture" + +# Reselect inferior 1, and then format an address from inferior 2. +gdb_test "inferior 1" ".*" +gdb_test "python print(\"Got: \" + gdb.format_address($bar_addr, inf2.progspace, inf2.architecture()))" \ + "Got: $foo_addr " \ + "gdb.format_address for bar, while inferior 1 is selected, pass progspace and architecture" + +# Try pasing incorrect object types for program space and architecture. +gdb_test "python print(\"Got: \" + gdb.format_address($bar_addr, inf2.progspace, inf2.progspace))" \ + [multi_line \ + "TypeError: The architecture argument is not a gdb.Architecture object" \ + "Error while executing Python code\\."] \ + "gdb.format_address pass wrong object type for architecture" + +gdb_test "python print(\"Got: \" + gdb.format_address($bar_addr, inf2.architecture(), inf2.architecture()))" \ + [multi_line \ + "TypeError: The progspace argument is not a gdb.Progspace object" \ + "Error while executing Python code\\."] \ + "gdb.format_address pass wrong object type for progspace" + +# Now invalidate inferior 2's program space, and try using that. +gdb_test "python pspace = inf2.progspace" +gdb_test "python arch = inf2.architecture()" +gdb_test "remove-inferior 2" +gdb_test "python print(\"Got: \" + gdb.format_address($bar_addr, pspace, arch))" \ + [multi_line \ + "ValueError: The progspace argument is not valid" \ + "Error while executing Python code\\."] \ + "gdb.format_address called with an invalid program space" -- 2.25.4