/* Python interface to architecture Copyright (C) 2013-2021 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 . */ #include "defs.h" #include "gdbarch.h" #include "arch-utils.h" #include "disasm.h" #include "python-internal.h" struct arch_object { PyObject_HEAD struct gdbarch *gdbarch; }; static struct gdbarch_data *arch_object_data = NULL; /* Require a valid Architecture. */ #define ARCHPY_REQUIRE_VALID(arch_obj, arch) \ do { \ arch = arch_object_to_gdbarch (arch_obj); \ if (arch == NULL) \ { \ PyErr_SetString (PyExc_RuntimeError, \ _("Architecture is invalid.")); \ return NULL; \ } \ } while (0) extern PyTypeObject arch_object_type CPYCHECKER_TYPE_OBJECT_FOR_TYPEDEF ("arch_object"); /* Associates an arch_object with GDBARCH as gdbarch_data via the gdbarch post init registration mechanism (gdbarch_data_register_post_init). */ static void * arch_object_data_init (struct gdbarch *gdbarch) { arch_object *arch_obj = PyObject_New (arch_object, &arch_object_type); if (arch_obj == NULL) return NULL; arch_obj->gdbarch = gdbarch; return (void *) arch_obj; } /* Returns the struct gdbarch value corresponding to the given Python architecture object OBJ. */ struct gdbarch * arch_object_to_gdbarch (PyObject *obj) { arch_object *py_arch = (arch_object *) obj; return py_arch->gdbarch; } /* Returns the Python architecture object corresponding to GDBARCH. Returns a new reference to the arch_object associated as data with GDBARCH. */ PyObject * gdbarch_to_arch_object (struct gdbarch *gdbarch) { PyObject *new_ref = (PyObject *) gdbarch_data (gdbarch, arch_object_data); /* new_ref could be NULL if registration of arch_object with GDBARCH failed in arch_object_data_init. */ Py_XINCREF (new_ref); return new_ref; } /* Implementation of gdb.Architecture.name (self) -> String. Returns the name of the architecture as a string value. */ static PyObject * archpy_name (PyObject *self, PyObject *args) { struct gdbarch *gdbarch = NULL; const char *name; ARCHPY_REQUIRE_VALID (self, gdbarch); name = (gdbarch_bfd_arch_info (gdbarch))->printable_name; return PyString_FromString (name); } /* Implementation of gdb.Architecture.disassemble (self, start_pc [, end_pc [,count]]) -> List. Returns a list of instructions in a memory address range. Each instruction in the list is a Python dict object. */ static PyObject * archpy_disassemble (PyObject *self, PyObject *args, PyObject *kw) { static const char *keywords[] = { "start_pc", "end_pc", "count", NULL }; CORE_ADDR start, end = 0; CORE_ADDR pc; gdb_py_ulongest start_temp; long count = 0, i; PyObject *end_obj = NULL, *count_obj = NULL; struct gdbarch *gdbarch = NULL; ARCHPY_REQUIRE_VALID (self, gdbarch); if (!gdb_PyArg_ParseTupleAndKeywords (args, kw, GDB_PY_LLU_ARG "|OO", keywords, &start_temp, &end_obj, &count_obj)) return NULL; start = start_temp; if (end_obj) { /* Make a long logic check first. In Python 3.x, internally, all integers are represented as longs. In Python 2.x, there is still a differentiation internally between a PyInt and a PyLong. Explicitly do this long check conversion first. In GDB, for Python 3.x, we #ifdef PyInt = PyLong. This check has to be done first to ensure we do not lose information in the conversion process. */ if (PyLong_Check (end_obj)) end = PyLong_AsUnsignedLongLong (end_obj); #if PY_MAJOR_VERSION == 2 else if (PyInt_Check (end_obj)) /* If the end_pc value is specified without a trailing 'L', end_obj will be an integer and not a long integer. */ end = PyInt_AsLong (end_obj); #endif else { PyErr_SetString (PyExc_TypeError, _("Argument 'end_pc' should be a (long) integer.")); return NULL; } if (end < start) { PyErr_SetString (PyExc_ValueError, _("Argument 'end_pc' should be greater than or " "equal to the argument 'start_pc'.")); return NULL; } } if (count_obj) { count = PyInt_AsLong (count_obj); if (PyErr_Occurred () || count < 0) { PyErr_SetString (PyExc_TypeError, _("Argument 'count' should be an non-negative " "integer.")); return NULL; } } gdbpy_ref<> result_list (PyList_New (0)); if (result_list == NULL) return NULL; for (pc = start, i = 0; /* All args are specified. */ (end_obj && count_obj && pc <= end && i < count) /* end_pc is specified, but no count. */ || (end_obj && count_obj == NULL && pc <= end) /* end_pc is not specified, but a count is. */ || (end_obj == NULL && count_obj && i < count) /* Both end_pc and count are not specified. */ || (end_obj == NULL && count_obj == NULL && pc == start);) { int insn_len = 0; gdbpy_ref<> insn_dict (PyDict_New ()); if (insn_dict == NULL) return NULL; if (PyList_Append (result_list.get (), insn_dict.get ())) return NULL; /* PyList_Append Sets the exception. */ string_file stb; try { insn_len = gdb_print_insn (gdbarch, pc, &stb, NULL); } catch (const gdb_exception &except) { gdbpy_convert_exception (except); return NULL; } gdbpy_ref<> pc_obj = gdb_py_object_from_ulongest (pc); if (pc_obj == nullptr) return nullptr; gdbpy_ref<> asm_obj (PyString_FromString (!stb.empty () ? stb.c_str () : "")); if (asm_obj == nullptr) return nullptr; gdbpy_ref<> len_obj = gdb_py_object_from_longest (insn_len); if (len_obj == nullptr) return nullptr; if (PyDict_SetItemString (insn_dict.get (), "addr", pc_obj.get ()) || PyDict_SetItemString (insn_dict.get (), "asm", asm_obj.get ()) || PyDict_SetItemString (insn_dict.get (), "length", len_obj.get ())) return NULL; pc += insn_len; i++; } return result_list.release (); } /* Implementation of gdb.Architecture.registers (self, reggroup) -> Iterator. Returns an iterator over register descriptors for registers in GROUP within the architecture SELF. */ static PyObject * archpy_registers (PyObject *self, PyObject *args, PyObject *kw) { static const char *keywords[] = { "reggroup", NULL }; struct gdbarch *gdbarch = NULL; const char *group_name = NULL; /* Parse method arguments. */ if (!gdb_PyArg_ParseTupleAndKeywords (args, kw, "|s", keywords, &group_name)) return NULL; /* Extract the gdbarch from the self object. */ ARCHPY_REQUIRE_VALID (self, gdbarch); return gdbpy_new_register_descriptor_iterator (gdbarch, group_name); } /* Implementation of gdb.Architecture.register_groups (self) -> Iterator. Returns an iterator that will give up all valid register groups in the architecture SELF. */ static PyObject * archpy_register_groups (PyObject *self, PyObject *args) { struct gdbarch *gdbarch = NULL; /* Extract the gdbarch from the self object. */ ARCHPY_REQUIRE_VALID (self, gdbarch); return gdbpy_new_reggroup_iterator (gdbarch); } /* Initializes the Architecture class in the gdb module. */ int gdbpy_initialize_arch (void) { arch_object_data = gdbarch_data_register_post_init (arch_object_data_init); arch_object_type.tp_new = PyType_GenericNew; if (PyType_Ready (&arch_object_type) < 0) return -1; return gdb_pymodule_addobject (gdb_module, "Architecture", (PyObject *) &arch_object_type); } static PyMethodDef arch_object_methods [] = { { "name", archpy_name, METH_NOARGS, "name () -> String.\n\ Return the name of the architecture as a string value." }, { "disassemble", (PyCFunction) archpy_disassemble, METH_VARARGS | METH_KEYWORDS, "disassemble (start_pc [, end_pc [, count]]) -> List.\n\ Return a list of at most COUNT disassembled instructions from START_PC to\n\ END_PC." }, { "registers", (PyCFunction) archpy_registers, METH_VARARGS | METH_KEYWORDS, "registers ([ group-name ]) -> Iterator.\n\ Return an iterator of register descriptors for the registers in register\n\ group GROUP-NAME." }, { "register_groups", archpy_register_groups, METH_NOARGS, "register_groups () -> Iterator.\n\ Return an iterator over all of the register groups in this architecture." }, {NULL} /* Sentinel */ }; PyTypeObject arch_object_type = { PyVarObject_HEAD_INIT (NULL, 0) "gdb.Architecture", /* tp_name */ sizeof (arch_object), /* tp_basicsize */ 0, /* tp_itemsize */ 0, /* tp_dealloc */ 0, /* tp_print */ 0, /* tp_getattr */ 0, /* tp_setattr */ 0, /* tp_compare */ 0, /* 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 architecture object", /* tp_doc */ 0, /* tp_traverse */ 0, /* tp_clear */ 0, /* tp_richcompare */ 0, /* tp_weaklistoffset */ 0, /* tp_iter */ 0, /* tp_iternext */ arch_object_methods, /* tp_methods */ 0, /* tp_members */ 0, /* tp_getset */ 0, /* tp_base */ 0, /* tp_dict */ 0, /* tp_descr_get */ 0, /* tp_descr_set */ 0, /* tp_dictoffset */ 0, /* tp_init */ 0, /* tp_alloc */ };