|  | /* Python interface to inferiors. | 
|  |  | 
|  | Copyright (C) 2009-2025 Free Software Foundation, Inc. | 
|  |  | 
|  | This file is part of GDB. | 
|  |  | 
|  | 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/>.  */ | 
|  |  | 
|  | #include "auto-load.h" | 
|  | #include "gdbcore.h" | 
|  | #include "gdbthread.h" | 
|  | #include "inferior.h" | 
|  | #include "objfiles.h" | 
|  | #include "observable.h" | 
|  | #include "python-internal.h" | 
|  | #include "arch-utils.h" | 
|  | #include "language.h" | 
|  | #include "gdbsupport/gdb_signals.h" | 
|  | #include "py-event.h" | 
|  | #include "py-stopevent.h" | 
|  | #include "progspace-and-thread.h" | 
|  | #include "gdbsupport/unordered_map.h" | 
|  |  | 
|  | using thread_map_t | 
|  | = gdb::unordered_map<thread_info *, gdbpy_ref<thread_object>>; | 
|  |  | 
|  | struct inferior_object | 
|  | { | 
|  | PyObject_HEAD | 
|  |  | 
|  | /* The inferior we represent.  */ | 
|  | struct inferior *inferior; | 
|  |  | 
|  | /* thread_object instances under this inferior.  This owns a | 
|  | reference to each object it contains.  */ | 
|  | thread_map_t *threads; | 
|  |  | 
|  | /* Dictionary holding user-added attributes. | 
|  | This is the __dict__ attribute of the object.  */ | 
|  | PyObject *dict; | 
|  | }; | 
|  |  | 
|  | extern PyTypeObject inferior_object_type | 
|  | CPYCHECKER_TYPE_OBJECT_FOR_TYPEDEF ("inferior_object"); | 
|  |  | 
|  | /* Deleter to clean up when an inferior is removed.  */ | 
|  | struct infpy_deleter | 
|  | { | 
|  | void operator() (inferior_object *obj) | 
|  | { | 
|  | if (!gdb_python_initialized) | 
|  | return; | 
|  |  | 
|  | gdbpy_enter enter_py; | 
|  | gdbpy_ref<inferior_object> inf_obj (obj); | 
|  |  | 
|  | inf_obj->inferior = NULL; | 
|  |  | 
|  | delete inf_obj->threads; | 
|  | } | 
|  | }; | 
|  |  | 
|  | static const registry<inferior>::key<inferior_object, infpy_deleter> | 
|  | infpy_inf_data_key; | 
|  |  | 
|  | /* Require that INFERIOR be a valid inferior ID.  */ | 
|  | #define INFPY_REQUIRE_VALID(Inferior)				\ | 
|  | do {								\ | 
|  | if (!Inferior->inferior)					\ | 
|  | {								\ | 
|  | PyErr_SetString (PyExc_RuntimeError,			\ | 
|  | _("Inferior no longer exists."));	\ | 
|  | return NULL;						\ | 
|  | }								\ | 
|  | } while (0) | 
|  |  | 
|  | static void | 
|  | python_on_normal_stop (struct bpstat *bs, int print_frame) | 
|  | { | 
|  | enum gdb_signal stop_signal; | 
|  |  | 
|  | if (!gdb_python_initialized) | 
|  | return; | 
|  |  | 
|  | if (inferior_ptid == null_ptid) | 
|  | return; | 
|  |  | 
|  | stop_signal = inferior_thread ()->stop_signal (); | 
|  |  | 
|  | gdbpy_enter enter_py; | 
|  |  | 
|  | if (emit_stop_event (bs, stop_signal) < 0) | 
|  | gdbpy_print_stack (); | 
|  | } | 
|  |  | 
|  | static void | 
|  | python_on_resume (ptid_t ptid) | 
|  | { | 
|  | if (!gdb_python_initialized) | 
|  | return; | 
|  |  | 
|  | gdbpy_enter enter_py (current_inferior ()->arch ()); | 
|  |  | 
|  | if (emit_continue_event (ptid) < 0) | 
|  | gdbpy_print_stack (); | 
|  | } | 
|  |  | 
|  | /* Callback, registered as an observer, that notifies Python listeners | 
|  | when an inferior function call is about to be made. */ | 
|  |  | 
|  | static void | 
|  | python_on_inferior_call_pre (ptid_t thread, CORE_ADDR address) | 
|  | { | 
|  | gdbpy_enter enter_py (current_inferior ()->arch ()); | 
|  |  | 
|  | if (emit_inferior_call_event (INFERIOR_CALL_PRE, thread, address) < 0) | 
|  | gdbpy_print_stack (); | 
|  | } | 
|  |  | 
|  | /* Callback, registered as an observer, that notifies Python listeners | 
|  | when an inferior function call has completed. */ | 
|  |  | 
|  | static void | 
|  | python_on_inferior_call_post (ptid_t thread, CORE_ADDR address) | 
|  | { | 
|  | gdbpy_enter enter_py (current_inferior ()->arch ()); | 
|  |  | 
|  | if (emit_inferior_call_event (INFERIOR_CALL_POST, thread, address) < 0) | 
|  | gdbpy_print_stack (); | 
|  | } | 
|  |  | 
|  | /* Callback, registered as an observer, that notifies Python listeners | 
|  | when a part of memory has been modified by user action (eg via a | 
|  | 'set' command). */ | 
|  |  | 
|  | static void | 
|  | python_on_memory_change (struct inferior *inferior, CORE_ADDR addr, ssize_t len, const bfd_byte *data) | 
|  | { | 
|  | gdbpy_enter enter_py (current_inferior ()->arch ()); | 
|  |  | 
|  | if (emit_memory_changed_event (addr, len) < 0) | 
|  | gdbpy_print_stack (); | 
|  | } | 
|  |  | 
|  | /* Callback, registered as an observer, that notifies Python listeners | 
|  | when a register has been modified by user action (eg via a 'set' | 
|  | command). */ | 
|  |  | 
|  | static void | 
|  | python_on_register_change (const frame_info_ptr &frame, int regnum) | 
|  | { | 
|  | gdbpy_enter enter_py (current_inferior ()->arch ()); | 
|  |  | 
|  | if (emit_register_changed_event (frame, regnum) < 0) | 
|  | gdbpy_print_stack (); | 
|  | } | 
|  |  | 
|  | static void | 
|  | python_inferior_exit (struct inferior *inf) | 
|  | { | 
|  | const LONGEST *exit_code = NULL; | 
|  |  | 
|  | if (!gdb_python_initialized) | 
|  | return; | 
|  |  | 
|  | gdbpy_enter enter_py (current_inferior ()->arch ()); | 
|  |  | 
|  | if (inf->has_exit_code) | 
|  | exit_code = &inf->exit_code; | 
|  |  | 
|  | if (emit_exited_event (exit_code, inf) < 0) | 
|  | gdbpy_print_stack (); | 
|  | } | 
|  |  | 
|  | /* Callback used to notify Python listeners about new objfiles loaded in the | 
|  | inferior.  OBJFILE may be NULL which means that the objfile list has been | 
|  | cleared (emptied).  */ | 
|  |  | 
|  | static void | 
|  | python_new_objfile (struct objfile *objfile) | 
|  | { | 
|  | if (!gdb_python_initialized) | 
|  | return; | 
|  |  | 
|  | gdbpy_enter enter_py (objfile->arch ()); | 
|  |  | 
|  | if (emit_new_objfile_event (objfile) < 0) | 
|  | gdbpy_print_stack (); | 
|  | } | 
|  |  | 
|  | static void | 
|  | python_all_objfiles_removed (program_space *pspace) | 
|  | { | 
|  | if (!gdb_python_initialized) | 
|  | return; | 
|  |  | 
|  | gdbpy_enter enter_py (current_inferior ()->arch ()); | 
|  |  | 
|  | if (emit_clear_objfiles_event (pspace) < 0) | 
|  | gdbpy_print_stack (); | 
|  | } | 
|  |  | 
|  | /* Emit a Python event when an objfile is about to be removed.  */ | 
|  |  | 
|  | static void | 
|  | python_free_objfile (struct objfile *objfile) | 
|  | { | 
|  | if (!gdb_python_initialized) | 
|  | return; | 
|  |  | 
|  | gdbpy_enter enter_py (objfile->arch ()); | 
|  |  | 
|  | if (emit_free_objfile_event (objfile) < 0) | 
|  | gdbpy_print_stack (); | 
|  | } | 
|  |  | 
|  | /* Return a reference to the Python object of type Inferior | 
|  | representing INFERIOR.  If the object has already been created, | 
|  | return it and increment the reference count,  otherwise, create it. | 
|  | Return NULL on failure.  */ | 
|  |  | 
|  | gdbpy_ref<inferior_object> | 
|  | inferior_to_inferior_object (struct inferior *inferior) | 
|  | { | 
|  | inferior_object *inf_obj; | 
|  |  | 
|  | inf_obj = infpy_inf_data_key.get (inferior); | 
|  | if (!inf_obj) | 
|  | { | 
|  | inf_obj = PyObject_New (inferior_object, &inferior_object_type); | 
|  | if (!inf_obj) | 
|  | return NULL; | 
|  |  | 
|  | inf_obj->inferior = inferior; | 
|  | inf_obj->threads = new thread_map_t (); | 
|  | inf_obj->dict = PyDict_New (); | 
|  | if (inf_obj->dict == nullptr) | 
|  | return nullptr; | 
|  |  | 
|  | /* PyObject_New initializes the new object with a refcount of 1.  This | 
|  | counts for the reference we are keeping in the inferior data.  */ | 
|  | infpy_inf_data_key.set (inferior, inf_obj); | 
|  | } | 
|  |  | 
|  | /* We are returning a new reference.  */ | 
|  | gdb_assert (inf_obj != nullptr); | 
|  | return gdbpy_ref<inferior_object>::new_reference (inf_obj); | 
|  | } | 
|  |  | 
|  | /* Called when a new inferior is created.  Notifies any Python event | 
|  | listeners.  */ | 
|  | static void | 
|  | python_new_inferior (struct inferior *inf) | 
|  | { | 
|  | if (!gdb_python_initialized) | 
|  | return; | 
|  |  | 
|  | gdbpy_enter enter_py; | 
|  |  | 
|  | if (evregpy_no_listeners_p (gdb_py_events.new_inferior)) | 
|  | return; | 
|  |  | 
|  | gdbpy_ref<inferior_object> inf_obj = inferior_to_inferior_object (inf); | 
|  | if (inf_obj == NULL) | 
|  | { | 
|  | gdbpy_print_stack (); | 
|  | return; | 
|  | } | 
|  |  | 
|  | gdbpy_ref<> event = create_event_object (&new_inferior_event_object_type); | 
|  | if (event == NULL | 
|  | || evpy_add_attribute (event.get (), "inferior", | 
|  | (PyObject *) inf_obj.get ()) < 0 | 
|  | || evpy_emit_event (event.get (), gdb_py_events.new_inferior) < 0) | 
|  | gdbpy_print_stack (); | 
|  | } | 
|  |  | 
|  | /* Called when an inferior is removed.  Notifies any Python event | 
|  | listeners.  */ | 
|  | static void | 
|  | python_inferior_deleted (struct inferior *inf) | 
|  | { | 
|  | if (!gdb_python_initialized) | 
|  | return; | 
|  |  | 
|  | gdbpy_enter enter_py; | 
|  |  | 
|  | if (evregpy_no_listeners_p (gdb_py_events.inferior_deleted)) | 
|  | return; | 
|  |  | 
|  | gdbpy_ref<inferior_object> inf_obj = inferior_to_inferior_object (inf); | 
|  | if (inf_obj == NULL) | 
|  | { | 
|  | gdbpy_print_stack (); | 
|  | return; | 
|  | } | 
|  |  | 
|  | gdbpy_ref<> event = create_event_object (&inferior_deleted_event_object_type); | 
|  | if (event == NULL | 
|  | || evpy_add_attribute (event.get (), "inferior", | 
|  | (PyObject *) inf_obj.get ()) < 0 | 
|  | || evpy_emit_event (event.get (), gdb_py_events.inferior_deleted) < 0) | 
|  | gdbpy_print_stack (); | 
|  | } | 
|  |  | 
|  | gdbpy_ref<> | 
|  | thread_to_thread_object (thread_info *thr) | 
|  | { | 
|  | gdbpy_ref<inferior_object> inf_obj = inferior_to_inferior_object (thr->inf); | 
|  | if (inf_obj == NULL) | 
|  | return NULL; | 
|  |  | 
|  | auto thread_it = inf_obj->threads->find (thr); | 
|  | if (thread_it != inf_obj->threads->end ()) | 
|  | return gdbpy_ref<>::new_reference | 
|  | ((PyObject *) (thread_it->second.get ())); | 
|  |  | 
|  | PyErr_SetString (PyExc_SystemError, | 
|  | _("could not find gdb thread object")); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | static void | 
|  | add_thread_object (struct thread_info *tp) | 
|  | { | 
|  | inferior_object *inf_obj; | 
|  |  | 
|  | if (!gdb_python_initialized) | 
|  | return; | 
|  |  | 
|  | gdbpy_enter enter_py; | 
|  |  | 
|  | gdbpy_ref<thread_object> thread_obj = create_thread_object (tp); | 
|  | if (thread_obj == NULL) | 
|  | { | 
|  | gdbpy_print_stack (); | 
|  | return; | 
|  | } | 
|  |  | 
|  | inf_obj = (inferior_object *) thread_obj->inf_obj; | 
|  |  | 
|  | auto ins_result = inf_obj->threads->emplace | 
|  | (thread_map_t::value_type (tp, std::move (thread_obj))); | 
|  |  | 
|  | if (!ins_result.second) | 
|  | return; | 
|  |  | 
|  | if (evregpy_no_listeners_p (gdb_py_events.new_thread)) | 
|  | return; | 
|  |  | 
|  | gdbpy_ref<> event = create_thread_event_object | 
|  | (&new_thread_event_object_type, | 
|  | (PyObject *) ins_result.first->second.get ()); | 
|  |  | 
|  | if (event == NULL | 
|  | || evpy_emit_event (event.get (), gdb_py_events.new_thread) < 0) | 
|  | gdbpy_print_stack (); | 
|  | } | 
|  |  | 
|  | static void | 
|  | delete_thread_object (thread_info *tp, | 
|  | std::optional<ULONGEST> /* exit_code */, | 
|  | bool /* silent */) | 
|  | { | 
|  | if (!gdb_python_initialized) | 
|  | return; | 
|  |  | 
|  | gdbpy_enter enter_py; | 
|  |  | 
|  | gdbpy_ref<inferior_object> inf_obj = inferior_to_inferior_object (tp->inf); | 
|  | if (inf_obj == NULL) | 
|  | return; | 
|  |  | 
|  | if (emit_thread_exit_event (tp) < 0) | 
|  | gdbpy_print_stack (); | 
|  |  | 
|  | auto it = inf_obj->threads->find (tp); | 
|  | if (it != inf_obj->threads->end ()) | 
|  | { | 
|  | /* Some python code can still hold a reference to the thread_object | 
|  | instance.   Make sure to remove the link to the associated | 
|  | thread_info object as it will be freed soon.  This makes the python | 
|  | object invalid (i.e. gdb.InfThread.is_valid returns False).  */ | 
|  | it->second->thread = nullptr; | 
|  | inf_obj->threads->erase (it); | 
|  | } | 
|  | } | 
|  |  | 
|  | static PyObject * | 
|  | infpy_threads (PyObject *self, PyObject *args) | 
|  | { | 
|  | int i = 0; | 
|  | inferior_object *inf_obj = (inferior_object *) self; | 
|  | PyObject *tuple; | 
|  |  | 
|  | INFPY_REQUIRE_VALID (inf_obj); | 
|  |  | 
|  | try | 
|  | { | 
|  | update_thread_list (); | 
|  | } | 
|  | catch (const gdb_exception &except) | 
|  | { | 
|  | return gdbpy_handle_gdb_exception (nullptr, except); | 
|  | } | 
|  |  | 
|  | tuple = PyTuple_New (inf_obj->threads->size ()); | 
|  | if (!tuple) | 
|  | return NULL; | 
|  |  | 
|  | for (const thread_map_t::value_type &entry : *inf_obj->threads) | 
|  | { | 
|  | PyObject *thr = (PyObject *) entry.second.get (); | 
|  | Py_INCREF (thr); | 
|  | PyTuple_SET_ITEM (tuple, i, thr); | 
|  | i = i + 1; | 
|  | } | 
|  |  | 
|  | return tuple; | 
|  | } | 
|  |  | 
|  | static PyObject * | 
|  | infpy_get_num (PyObject *self, void *closure) | 
|  | { | 
|  | inferior_object *inf = (inferior_object *) self; | 
|  |  | 
|  | INFPY_REQUIRE_VALID (inf); | 
|  |  | 
|  | return gdb_py_object_from_longest (inf->inferior->num).release (); | 
|  | } | 
|  |  | 
|  | /* Return the gdb.TargetConnection object for this inferior, or None if a | 
|  | connection does not exist.  */ | 
|  |  | 
|  | static PyObject * | 
|  | infpy_get_connection (PyObject *self, void *closure) | 
|  | { | 
|  | inferior_object *inf = (inferior_object *) self; | 
|  |  | 
|  | INFPY_REQUIRE_VALID (inf); | 
|  |  | 
|  | process_stratum_target *target = inf->inferior->process_target (); | 
|  | return target_to_connection_object (target).release (); | 
|  | } | 
|  |  | 
|  | /* Return the connection number of the given inferior, or None if a | 
|  | connection does not exist.  */ | 
|  |  | 
|  | static PyObject * | 
|  | infpy_get_connection_num (PyObject *self, void *closure) | 
|  | { | 
|  | inferior_object *inf = (inferior_object *) self; | 
|  |  | 
|  | INFPY_REQUIRE_VALID (inf); | 
|  |  | 
|  | process_stratum_target *target = inf->inferior->process_target (); | 
|  | if (target == nullptr) | 
|  | Py_RETURN_NONE; | 
|  |  | 
|  | return gdb_py_object_from_longest (target->connection_number).release (); | 
|  | } | 
|  |  | 
|  | static PyObject * | 
|  | infpy_get_pid (PyObject *self, void *closure) | 
|  | { | 
|  | inferior_object *inf = (inferior_object *) self; | 
|  |  | 
|  | INFPY_REQUIRE_VALID (inf); | 
|  |  | 
|  | return gdb_py_object_from_longest (inf->inferior->pid).release (); | 
|  | } | 
|  |  | 
|  | static PyObject * | 
|  | infpy_get_was_attached (PyObject *self, void *closure) | 
|  | { | 
|  | inferior_object *inf = (inferior_object *) self; | 
|  |  | 
|  | INFPY_REQUIRE_VALID (inf); | 
|  | if (inf->inferior->attach_flag) | 
|  | Py_RETURN_TRUE; | 
|  | Py_RETURN_FALSE; | 
|  | } | 
|  |  | 
|  | /* Getter of gdb.Inferior.progspace.  */ | 
|  |  | 
|  | static PyObject * | 
|  | infpy_get_progspace (PyObject *self, void *closure) | 
|  | { | 
|  | inferior_object *inf = (inferior_object *) self; | 
|  |  | 
|  | INFPY_REQUIRE_VALID (inf); | 
|  |  | 
|  | program_space *pspace = inf->inferior->pspace; | 
|  | gdb_assert (pspace != nullptr); | 
|  |  | 
|  | return pspace_to_pspace_object (pspace).release (); | 
|  | } | 
|  |  | 
|  | /* Implementation of gdb.inferiors () -> (gdb.Inferior, ...). | 
|  | Returns a tuple of all inferiors.  */ | 
|  | PyObject * | 
|  | gdbpy_inferiors (PyObject *unused, PyObject *unused2) | 
|  | { | 
|  | gdbpy_ref<> list (PyList_New (0)); | 
|  | if (list == NULL) | 
|  | return NULL; | 
|  |  | 
|  | for (inferior *inf : all_inferiors ()) | 
|  | { | 
|  | gdbpy_ref<inferior_object> inferior = inferior_to_inferior_object (inf); | 
|  |  | 
|  | if (inferior == NULL) | 
|  | continue; | 
|  |  | 
|  | if (PyList_Append (list.get (), (PyObject *) inferior.get ()) != 0) | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | return PyList_AsTuple (list.get ()); | 
|  | } | 
|  |  | 
|  | /* Membuf and memory manipulation.  */ | 
|  |  | 
|  | /* Implementation of Inferior.read_memory (address, length). | 
|  | Returns a Python buffer object with LENGTH bytes of the inferior's | 
|  | memory at ADDRESS.  Both arguments are integers.  Returns NULL on error, | 
|  | with a python exception set.  */ | 
|  | static PyObject * | 
|  | infpy_read_memory (PyObject *self, PyObject *args, PyObject *kw) | 
|  | { | 
|  | inferior_object *inf = (inferior_object *) self; | 
|  | CORE_ADDR addr, length; | 
|  | gdb::unique_xmalloc_ptr<gdb_byte> buffer; | 
|  | PyObject *addr_obj, *length_obj; | 
|  | static const char *keywords[] = { "address", "length", NULL }; | 
|  |  | 
|  | INFPY_REQUIRE_VALID (inf); | 
|  |  | 
|  | if (!gdb_PyArg_ParseTupleAndKeywords (args, kw, "OO", keywords, | 
|  | &addr_obj, &length_obj)) | 
|  | return NULL; | 
|  |  | 
|  | if (get_addr_from_python (addr_obj, &addr) < 0 | 
|  | || get_addr_from_python (length_obj, &length) < 0) | 
|  | return NULL; | 
|  |  | 
|  | if (length == 0) | 
|  | { | 
|  | PyErr_SetString (PyExc_ValueError, | 
|  | _("Argument 'count' should be greater than zero")); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | void *p = malloc (length); | 
|  | if (p == nullptr) | 
|  | return PyErr_NoMemory (); | 
|  | buffer.reset ((gdb_byte *) p); | 
|  |  | 
|  | try | 
|  | { | 
|  | /* Use this scoped-restore because we want to be able to read | 
|  | memory from an unwinder.  */ | 
|  | scoped_restore_current_inferior_for_memory restore_inferior | 
|  | (inf->inferior); | 
|  |  | 
|  | read_memory (addr, buffer.get (), length); | 
|  | } | 
|  | catch (const gdb_exception &except) | 
|  | { | 
|  | return gdbpy_handle_gdb_exception (nullptr, except); | 
|  | } | 
|  |  | 
|  |  | 
|  | return gdbpy_buffer_to_membuf (std::move (buffer), addr, length); | 
|  | } | 
|  |  | 
|  | /* Implementation of Inferior.write_memory (address, buffer [, length]). | 
|  | Writes the contents of BUFFER (a Python object supporting the read | 
|  | buffer protocol) at ADDRESS in the inferior's memory.  Write LENGTH | 
|  | bytes from BUFFER, or its entire contents if the argument is not | 
|  | provided.  The function returns nothing.  Returns NULL on error, with | 
|  | a python exception set.  */ | 
|  | static PyObject * | 
|  | infpy_write_memory (PyObject *self, PyObject *args, PyObject *kw) | 
|  | { | 
|  | inferior_object *inf = (inferior_object *) self; | 
|  | Py_ssize_t buf_len; | 
|  | const gdb_byte *buffer; | 
|  | CORE_ADDR addr, length; | 
|  | PyObject *addr_obj, *length_obj = NULL; | 
|  | static const char *keywords[] = { "address", "buffer", "length", NULL }; | 
|  | Py_buffer pybuf; | 
|  |  | 
|  | INFPY_REQUIRE_VALID (inf); | 
|  |  | 
|  | if (!gdb_PyArg_ParseTupleAndKeywords (args, kw, "Os*|O", keywords, | 
|  | &addr_obj, &pybuf, &length_obj)) | 
|  | return NULL; | 
|  |  | 
|  | Py_buffer_up buffer_up (&pybuf); | 
|  | buffer = (const gdb_byte *) pybuf.buf; | 
|  | buf_len = pybuf.len; | 
|  |  | 
|  | if (get_addr_from_python (addr_obj, &addr) < 0) | 
|  | return nullptr; | 
|  |  | 
|  | if (!length_obj) | 
|  | length = buf_len; | 
|  | else if (get_addr_from_python (length_obj, &length) < 0) | 
|  | return nullptr; | 
|  |  | 
|  | try | 
|  | { | 
|  | /* It's probably not too important to avoid invalidating the | 
|  | frame cache when writing memory, but this scoped-restore is | 
|  | still used here, just to keep the code similar to other code | 
|  | in this file.  */ | 
|  | scoped_restore_current_inferior_for_memory restore_inferior | 
|  | (inf->inferior); | 
|  |  | 
|  | write_memory_with_notification (addr, buffer, length); | 
|  | } | 
|  | catch (const gdb_exception &ex) | 
|  | { | 
|  | return gdbpy_handle_gdb_exception (nullptr, ex); | 
|  | } | 
|  |  | 
|  | Py_RETURN_NONE; | 
|  | } | 
|  |  | 
|  | /* Implementation of | 
|  | Inferior.search_memory (address, length, pattern).  ADDRESS is the | 
|  | address to start the search.  LENGTH specifies the scope of the | 
|  | search from ADDRESS.  PATTERN is the pattern to search for (and | 
|  | must be a Python object supporting the buffer protocol). | 
|  | Returns a Python Long object holding the address where the pattern | 
|  | was located, or if the pattern was not found, returns None.  Returns NULL | 
|  | on error, with a python exception set.  */ | 
|  | static PyObject * | 
|  | infpy_search_memory (PyObject *self, PyObject *args, PyObject *kw) | 
|  | { | 
|  | inferior_object *inf = (inferior_object *) self; | 
|  | CORE_ADDR start_addr, length; | 
|  | static const char *keywords[] = { "address", "length", "pattern", NULL }; | 
|  | PyObject *start_addr_obj, *length_obj; | 
|  | Py_ssize_t pattern_size; | 
|  | const gdb_byte *buffer; | 
|  | CORE_ADDR found_addr; | 
|  | int found = 0; | 
|  | Py_buffer pybuf; | 
|  |  | 
|  | INFPY_REQUIRE_VALID (inf); | 
|  |  | 
|  | if (!gdb_PyArg_ParseTupleAndKeywords (args, kw, "OOs*", keywords, | 
|  | &start_addr_obj, &length_obj, | 
|  | &pybuf)) | 
|  | return NULL; | 
|  |  | 
|  | Py_buffer_up buffer_up (&pybuf); | 
|  | buffer = (const gdb_byte *) pybuf.buf; | 
|  | pattern_size = pybuf.len; | 
|  |  | 
|  | if (get_addr_from_python (start_addr_obj, &start_addr) < 0) | 
|  | return nullptr; | 
|  |  | 
|  | if (get_addr_from_python (length_obj, &length) < 0) | 
|  | return nullptr; | 
|  |  | 
|  | if (!length) | 
|  | { | 
|  | PyErr_SetString (PyExc_ValueError, | 
|  | _("Search range is empty.")); | 
|  | return nullptr; | 
|  | } | 
|  | /* Watch for overflows.  */ | 
|  | else if (length > CORE_ADDR_MAX | 
|  | || (start_addr + length - 1) < start_addr) | 
|  | { | 
|  | PyErr_SetString (PyExc_ValueError, | 
|  | _("The search range is too large.")); | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  | try | 
|  | { | 
|  | /* It's probably not too important to avoid invalidating the | 
|  | frame cache when searching memory, but this scoped-restore is | 
|  | still used here, just to keep the code similar to other code | 
|  | in this file.  */ | 
|  | scoped_restore_current_inferior_for_memory restore_inferior | 
|  | (inf->inferior); | 
|  |  | 
|  | found = target_search_memory (start_addr, length, | 
|  | buffer, pattern_size, | 
|  | &found_addr); | 
|  | } | 
|  | catch (const gdb_exception &ex) | 
|  | { | 
|  | return gdbpy_handle_gdb_exception (nullptr, ex); | 
|  | } | 
|  |  | 
|  | if (found) | 
|  | return gdb_py_object_from_ulongest (found_addr).release (); | 
|  | else | 
|  | Py_RETURN_NONE; | 
|  | } | 
|  |  | 
|  | /* Implementation of gdb.Inferior.is_valid (self) -> Boolean. | 
|  | Returns True if this inferior object still exists in GDB.  */ | 
|  |  | 
|  | static PyObject * | 
|  | infpy_is_valid (PyObject *self, PyObject *args) | 
|  | { | 
|  | inferior_object *inf = (inferior_object *) self; | 
|  |  | 
|  | if (! inf->inferior) | 
|  | Py_RETURN_FALSE; | 
|  |  | 
|  | Py_RETURN_TRUE; | 
|  | } | 
|  |  | 
|  | /* Implementation of gdb.Inferior.thread_from_handle (self, handle) | 
|  | ->  gdb.InferiorThread.  */ | 
|  |  | 
|  | static PyObject * | 
|  | infpy_thread_from_thread_handle (PyObject *self, PyObject *args, PyObject *kw) | 
|  | { | 
|  | PyObject *handle_obj; | 
|  | inferior_object *inf_obj = (inferior_object *) self; | 
|  | static const char *keywords[] = { "handle", NULL }; | 
|  |  | 
|  | INFPY_REQUIRE_VALID (inf_obj); | 
|  |  | 
|  | if (! gdb_PyArg_ParseTupleAndKeywords (args, kw, "O", keywords, &handle_obj)) | 
|  | return NULL; | 
|  |  | 
|  | const gdb_byte *bytes; | 
|  | size_t bytes_len; | 
|  | Py_buffer_up buffer_up; | 
|  | Py_buffer py_buf; | 
|  |  | 
|  | if (PyObject_CheckBuffer (handle_obj) | 
|  | && PyObject_GetBuffer (handle_obj, &py_buf, PyBUF_SIMPLE) == 0) | 
|  | { | 
|  | buffer_up.reset (&py_buf); | 
|  | bytes = (const gdb_byte *) py_buf.buf; | 
|  | bytes_len = py_buf.len; | 
|  | } | 
|  | else if (gdbpy_is_value_object (handle_obj)) | 
|  | { | 
|  | struct value *val = value_object_to_value (handle_obj); | 
|  | bytes = val->contents_all ().data (); | 
|  | bytes_len = val->type ()->length (); | 
|  | } | 
|  | else | 
|  | { | 
|  | PyErr_SetString (PyExc_TypeError, | 
|  | _("Argument 'handle' must be a thread handle object.")); | 
|  |  | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | try | 
|  | { | 
|  | struct thread_info *thread_info; | 
|  |  | 
|  | thread_info = find_thread_by_handle | 
|  | (gdb::array_view<const gdb_byte> (bytes, bytes_len), | 
|  | inf_obj->inferior); | 
|  | if (thread_info != NULL) | 
|  | return thread_to_thread_object (thread_info).release (); | 
|  | } | 
|  | catch (const gdb_exception &except) | 
|  | { | 
|  | return gdbpy_handle_gdb_exception (nullptr, except); | 
|  | } | 
|  |  | 
|  | Py_RETURN_NONE; | 
|  | } | 
|  |  | 
|  | /* Implementation of gdb.Inferior.architecture.  */ | 
|  |  | 
|  | static PyObject * | 
|  | infpy_architecture (PyObject *self, PyObject *args) | 
|  | { | 
|  | inferior_object *inf = (inferior_object *) self; | 
|  |  | 
|  | INFPY_REQUIRE_VALID (inf); | 
|  |  | 
|  | return gdbarch_to_arch_object (inf->inferior->arch ()); | 
|  | } | 
|  |  | 
|  | /* Implement repr() for gdb.Inferior.  */ | 
|  |  | 
|  | static PyObject * | 
|  | infpy_repr (PyObject *obj) | 
|  | { | 
|  | inferior_object *self = (inferior_object *) obj; | 
|  | inferior *inf = self->inferior; | 
|  |  | 
|  | if (inf == nullptr) | 
|  | return gdb_py_invalid_object_repr (obj); | 
|  |  | 
|  | return PyUnicode_FromFormat ("<gdb.Inferior num=%d, pid=%d>", | 
|  | inf->num, inf->pid); | 
|  | } | 
|  |  | 
|  | /* Implement clear_env.  */ | 
|  |  | 
|  | static PyObject * | 
|  | infpy_clear_env (PyObject *obj) | 
|  | { | 
|  | inferior_object *self = (inferior_object *) obj; | 
|  |  | 
|  | INFPY_REQUIRE_VALID (self); | 
|  |  | 
|  | self->inferior->environment.clear (); | 
|  | Py_RETURN_NONE; | 
|  | } | 
|  |  | 
|  | /* Implement set_env.  */ | 
|  |  | 
|  | static PyObject * | 
|  | infpy_set_env (PyObject *obj, PyObject *args, PyObject *kw) | 
|  | { | 
|  | inferior_object *self = (inferior_object *) obj; | 
|  | INFPY_REQUIRE_VALID (self); | 
|  |  | 
|  | const char *name, *val; | 
|  | static const char *keywords[] = { "name", "value", nullptr }; | 
|  |  | 
|  | if (!gdb_PyArg_ParseTupleAndKeywords (args, kw, "ss", keywords, | 
|  | &name, &val)) | 
|  | return nullptr; | 
|  |  | 
|  | self->inferior->environment.set (name, val); | 
|  | Py_RETURN_NONE; | 
|  | } | 
|  |  | 
|  | /* Implement unset_env.  */ | 
|  |  | 
|  | static PyObject * | 
|  | infpy_unset_env (PyObject *obj, PyObject *args, PyObject *kw) | 
|  | { | 
|  | inferior_object *self = (inferior_object *) obj; | 
|  | INFPY_REQUIRE_VALID (self); | 
|  |  | 
|  | const char *name; | 
|  | static const char *keywords[] = { "name", nullptr }; | 
|  | if (!gdb_PyArg_ParseTupleAndKeywords (args, kw, "s", keywords, &name)) | 
|  | return nullptr; | 
|  |  | 
|  | self->inferior->environment.unset (name); | 
|  | Py_RETURN_NONE; | 
|  | } | 
|  |  | 
|  | /* Getter for "arguments".  */ | 
|  |  | 
|  | static PyObject * | 
|  | infpy_get_args (PyObject *self, void *closure) | 
|  | { | 
|  | inferior_object *inf = (inferior_object *) self; | 
|  |  | 
|  | INFPY_REQUIRE_VALID (inf); | 
|  |  | 
|  | const std::string &args = inf->inferior->args (); | 
|  | if (args.empty ()) | 
|  | Py_RETURN_NONE; | 
|  |  | 
|  | return host_string_to_python_string (args.c_str ()).release (); | 
|  | } | 
|  |  | 
|  | /* Setter for "arguments".  */ | 
|  |  | 
|  | static int | 
|  | infpy_set_args (PyObject *self, PyObject *value, void *closure) | 
|  | { | 
|  | inferior_object *inf = (inferior_object *) self; | 
|  |  | 
|  | if (!inf->inferior) | 
|  | { | 
|  | PyErr_SetString (PyExc_RuntimeError, _("Inferior no longer exists.")); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | if (value == nullptr) | 
|  | { | 
|  | PyErr_SetString (PyExc_TypeError, | 
|  | _("Cannot delete 'arguments' attribute.")); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | if (gdbpy_is_string (value)) | 
|  | { | 
|  | gdb::unique_xmalloc_ptr<char> str = python_string_to_host_string (value); | 
|  | if (str == nullptr) | 
|  | return -1; | 
|  | inf->inferior->set_args (std::string (str.get ())); | 
|  | } | 
|  | else if (PySequence_Check (value)) | 
|  | { | 
|  | std::vector<gdb::unique_xmalloc_ptr<char>> args; | 
|  | Py_ssize_t len = PySequence_Size (value); | 
|  | if (len == -1) | 
|  | return -1; | 
|  | for (Py_ssize_t i = 0; i < len; ++i) | 
|  | { | 
|  | gdbpy_ref<> item (PySequence_ITEM (value, i)); | 
|  | if (item == nullptr) | 
|  | return -1; | 
|  | gdb::unique_xmalloc_ptr<char> str | 
|  | = python_string_to_host_string (item.get ()); | 
|  | if (str == nullptr) | 
|  | return -1; | 
|  | args.push_back (std::move (str)); | 
|  | } | 
|  | std::vector<char *> argvec; | 
|  | for (const auto &arg : args) | 
|  | argvec.push_back (arg.get ()); | 
|  | gdb::array_view<char * const> view (argvec.data (), argvec.size ()); | 
|  | inf->inferior->set_args (view, true); | 
|  | } | 
|  | else | 
|  | { | 
|  | PyErr_SetString (PyExc_TypeError, | 
|  | _("string or sequence required for 'arguments'")); | 
|  | return -1; | 
|  | } | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* Getter for "main_name".  */ | 
|  |  | 
|  | static PyObject * | 
|  | infpy_get_main_name (PyObject *self, void *closure) | 
|  | { | 
|  | inferior_object *inf = (inferior_object *) self; | 
|  |  | 
|  | INFPY_REQUIRE_VALID (inf); | 
|  |  | 
|  | const char *name = nullptr; | 
|  | try | 
|  | { | 
|  | /* This is unfortunate but the implementation of main_name can | 
|  | reach into memory.  It's probably not too important to avoid | 
|  | invalidating the frame cache here, but this scoped-restore is | 
|  | still used, just to keep the code similar to other code in | 
|  | this file.  */ | 
|  | scoped_restore_current_inferior_for_memory restore_inferior | 
|  | (inf->inferior); | 
|  |  | 
|  | name = main_name (); | 
|  | } | 
|  | catch (const gdb_exception &except) | 
|  | { | 
|  | /* We can just ignore this.  */ | 
|  | } | 
|  |  | 
|  | if (name == nullptr) | 
|  | Py_RETURN_NONE; | 
|  |  | 
|  | return host_string_to_python_string (name).release (); | 
|  | } | 
|  |  | 
|  | static void | 
|  | infpy_dealloc (PyObject *obj) | 
|  | { | 
|  | inferior_object *inf_obj = (inferior_object *) obj; | 
|  |  | 
|  | /* The inferior itself holds a reference to this Python object, which | 
|  | will keep the reference count of this object above zero until GDB | 
|  | deletes the inferior and py_free_inferior is called. | 
|  |  | 
|  | Once py_free_inferior has been called then the link between this | 
|  | Python object and the inferior is set to nullptr, and then the | 
|  | reference count on this Python object is decremented. | 
|  |  | 
|  | The result of all this is that the link between this Python object and | 
|  | the inferior should always have been set to nullptr before this | 
|  | function is called.  */ | 
|  | gdb_assert (inf_obj->inferior == nullptr); | 
|  |  | 
|  | Py_XDECREF (inf_obj->dict); | 
|  |  | 
|  | Py_TYPE (obj)->tp_free (obj); | 
|  | } | 
|  |  | 
|  | /* Implementation of gdb.selected_inferior() -> gdb.Inferior. | 
|  | Returns the current inferior object.  */ | 
|  |  | 
|  | PyObject * | 
|  | gdbpy_selected_inferior (PyObject *self, PyObject *args) | 
|  | { | 
|  | return ((PyObject *) | 
|  | inferior_to_inferior_object (current_inferior ()).release ()); | 
|  | } | 
|  |  | 
|  | static int CPYCHECKER_NEGATIVE_RESULT_SETS_EXCEPTION | 
|  | gdbpy_initialize_inferior (void) | 
|  | { | 
|  | if (gdbpy_type_ready (&inferior_object_type) < 0) | 
|  | return -1; | 
|  |  | 
|  | gdb::observers::new_thread.attach (add_thread_object, "py-inferior"); | 
|  | gdb::observers::thread_exit.attach (delete_thread_object, "py-inferior"); | 
|  | gdb::observers::normal_stop.attach (python_on_normal_stop, "py-inferior"); | 
|  | gdb::observers::target_resumed.attach (python_on_resume, "py-inferior"); | 
|  | gdb::observers::inferior_call_pre.attach (python_on_inferior_call_pre, | 
|  | "py-inferior"); | 
|  | gdb::observers::inferior_call_post.attach (python_on_inferior_call_post, | 
|  | "py-inferior"); | 
|  | gdb::observers::memory_changed.attach (python_on_memory_change, | 
|  | "py-inferior"); | 
|  | gdb::observers::register_changed.attach (python_on_register_change, | 
|  | "py-inferior"); | 
|  | gdb::observers::inferior_exit.attach (python_inferior_exit, "py-inferior"); | 
|  | /* Need to run after auto-load's new_objfile observer, so that | 
|  | auto-loaded pretty-printers are available.  */ | 
|  | gdb::observers::new_objfile.attach | 
|  | (python_new_objfile, "py-inferior", | 
|  | { &auto_load_new_objfile_observer_token }); | 
|  | gdb::observers::all_objfiles_removed.attach (python_all_objfiles_removed, | 
|  | "py-inferior"); | 
|  | gdb::observers::free_objfile.attach (python_free_objfile, "py-inferior"); | 
|  | gdb::observers::inferior_added.attach (python_new_inferior, "py-inferior"); | 
|  | gdb::observers::inferior_removed.attach (python_inferior_deleted, | 
|  | "py-inferior"); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | GDBPY_INITIALIZE_FILE (gdbpy_initialize_inferior); | 
|  |  | 
|  |  | 
|  |  | 
|  | static gdb_PyGetSetDef inferior_object_getset[] = | 
|  | { | 
|  | { "__dict__", gdb_py_generic_dict, nullptr, | 
|  | "The __dict__ for this inferior.", &inferior_object_type }, | 
|  | { "arguments", infpy_get_args, infpy_set_args, | 
|  | "Arguments to this program.", nullptr }, | 
|  | { "num", infpy_get_num, NULL, "ID of inferior, as assigned by GDB.", NULL }, | 
|  | { "connection", infpy_get_connection, NULL, | 
|  | "The gdb.TargetConnection for this inferior.", NULL }, | 
|  | { "connection_num", infpy_get_connection_num, NULL, | 
|  | "ID of inferior's connection, as assigned by GDB.", NULL }, | 
|  | { "pid", infpy_get_pid, NULL, "PID of inferior, as assigned by the OS.", | 
|  | NULL }, | 
|  | { "was_attached", infpy_get_was_attached, NULL, | 
|  | "True if the inferior was created using 'attach'.", NULL }, | 
|  | { "progspace", infpy_get_progspace, NULL, "Program space of this inferior" }, | 
|  | { "main_name", infpy_get_main_name, nullptr, | 
|  | "Name of 'main' function, if known.", nullptr }, | 
|  | { NULL } | 
|  | }; | 
|  |  | 
|  | static PyMethodDef inferior_object_methods[] = | 
|  | { | 
|  | { "is_valid", infpy_is_valid, METH_NOARGS, | 
|  | "is_valid () -> Boolean.\n\ | 
|  | Return true if this inferior is valid, false if not." }, | 
|  | { "threads", infpy_threads, METH_NOARGS, | 
|  | "Return all the threads of this inferior." }, | 
|  | { "read_memory", (PyCFunction) infpy_read_memory, | 
|  | METH_VARARGS | METH_KEYWORDS, | 
|  | "read_memory (address, length) -> buffer\n\ | 
|  | Return a buffer object for reading from the inferior's memory." }, | 
|  | { "write_memory", (PyCFunction) infpy_write_memory, | 
|  | METH_VARARGS | METH_KEYWORDS, | 
|  | "write_memory (address, buffer [, length])\n\ | 
|  | Write the given buffer object to the inferior's memory." }, | 
|  | { "search_memory", (PyCFunction) infpy_search_memory, | 
|  | METH_VARARGS | METH_KEYWORDS, | 
|  | "search_memory (address, length, pattern) -> long\n\ | 
|  | Return a long with the address of a match, or None." }, | 
|  | /* thread_from_thread_handle is deprecated.  */ | 
|  | { "thread_from_thread_handle", (PyCFunction) infpy_thread_from_thread_handle, | 
|  | METH_VARARGS | METH_KEYWORDS, | 
|  | "thread_from_thread_handle (handle) -> gdb.InferiorThread.\n\ | 
|  | Return thread object corresponding to thread handle.\n\ | 
|  | This method is deprecated - use thread_from_handle instead." }, | 
|  | { "thread_from_handle", (PyCFunction) infpy_thread_from_thread_handle, | 
|  | METH_VARARGS | METH_KEYWORDS, | 
|  | "thread_from_handle (handle) -> gdb.InferiorThread.\n\ | 
|  | Return thread object corresponding to thread handle." }, | 
|  | { "architecture", (PyCFunction) infpy_architecture, METH_NOARGS, | 
|  | "architecture () -> gdb.Architecture\n\ | 
|  | Return architecture of this inferior." }, | 
|  | { "clear_env", (PyCFunction) infpy_clear_env, METH_NOARGS, | 
|  | "clear_env () -> None\n\ | 
|  | Clear environment of this inferior." }, | 
|  | { "set_env", (PyCFunction) infpy_set_env, METH_VARARGS | METH_KEYWORDS, | 
|  | "set_env (name, value) -> None\n\ | 
|  | Set an environment variable of this inferior." }, | 
|  | { "unset_env", (PyCFunction) infpy_unset_env, METH_VARARGS | METH_KEYWORDS, | 
|  | "unset_env (name) -> None\n\ | 
|  | Unset an environment of this inferior." }, | 
|  | { NULL } | 
|  | }; | 
|  |  | 
|  | PyTypeObject inferior_object_type = | 
|  | { | 
|  | PyVarObject_HEAD_INIT (NULL, 0) | 
|  | "gdb.Inferior",		  /* tp_name */ | 
|  | sizeof (inferior_object),	  /* tp_basicsize */ | 
|  | 0,				  /* tp_itemsize */ | 
|  | infpy_dealloc,		  /* tp_dealloc */ | 
|  | 0,				  /* tp_print */ | 
|  | 0,				  /* tp_getattr */ | 
|  | 0,				  /* tp_setattr */ | 
|  | 0,				  /* tp_compare */ | 
|  | infpy_repr,			  /* tp_repr */ | 
|  | 0,				  /* tp_as_number */ | 
|  | 0,				  /* tp_as_sequence */ | 
|  | 0,				  /* tp_as_mapping */ | 
|  | 0,				  /* tp_hash  */ | 
|  | 0,				  /* tp_call */ | 
|  | 0,				  /* tp_str */ | 
|  | 0,				  /* tp_getattro */ | 
|  | 0,				  /* tp_setattro */ | 
|  | 0,				  /* tp_as_buffer */ | 
|  | Py_TPFLAGS_DEFAULT,		  /* tp_flags */ | 
|  | "GDB inferior object",	  /* tp_doc */ | 
|  | 0,				  /* tp_traverse */ | 
|  | 0,				  /* tp_clear */ | 
|  | 0,				  /* tp_richcompare */ | 
|  | 0,				  /* tp_weaklistoffset */ | 
|  | 0,				  /* tp_iter */ | 
|  | 0,				  /* tp_iternext */ | 
|  | inferior_object_methods,	  /* tp_methods */ | 
|  | 0,				  /* tp_members */ | 
|  | inferior_object_getset,	  /* tp_getset */ | 
|  | 0,				  /* tp_base */ | 
|  | 0,				  /* tp_dict */ | 
|  | 0,				  /* tp_descr_get */ | 
|  | 0,				  /* tp_descr_set */ | 
|  | offsetof (inferior_object, dict), /* tp_dictoffset */ | 
|  | 0,				  /* tp_init */ | 
|  | 0				  /* tp_alloc */ | 
|  | }; |