
This patch starts from the desire to eliminate make_cleanup_ui_file_delete, but then goes beyond. It makes ui_file & friends a real C++ class hierarchy, and switches temporary ui_file-like objects to stack-based allocation. - mem_fileopen -> string_file mem_fileopen is replaced with a new string_file class that is treated as a value class created on the stack. This alone eliminates most make_cleanup_ui_file_delete calls, and, simplifies code a whole lot (diffstat shows around 1k loc dropped.) string_file's internal buffer is a std::string, thus the "string" in the name. This simplifies the implementation much, compared to mem_fileopen, which managed growing its internal buffer manually. - ui_file_as_string, ui_file_strdup, ui_file_obsavestring all gone The new string_file class has a string() method that provides direct writable access to the internal std::string buffer. This replaced ui_file_as_string, which forced a copy of the same data the stream had inside. With direct access via a writable reference, we can instead move the string out of the string_stream, avoiding deep string copying. Related, ui_file_xstrdup calls are replaced with xstrdup'ping the stream's string, and ui_file_obsavestring is replaced by obstack_copy0. With all those out of the way, getting rid of the weird ui_file_put mechanism was possible. - New ui_file::printf, ui_file::puts, etc. methods These simplify / clarify client code. I considered splitting client-code changes, like these, e.g.: - stb = mem_fileopen (); - fprintf_unfiltered (stb, "%s%s%s", - _("The valid values are:\n"), - regdesc, - _("The default is \"std\".")); + string_file stb; + stb.printf ("%s%s%s", + _("The valid values are:\n"), + regdesc, + _("The default is \"std\".")); In two steps, with the first step leaving fprintf_unfiltered (etc.) calls in place, and only afterwards do a pass to change all those to call stb.printf etc.. I didn't do that split, because (when I tried), it turned out to be pointless make-work: the first pass would have to touch the fprintf_unfiltered line anyway, to replace "stb" with "&stb". - gdb_fopen replaced with stack-based objects This avoids the need for cleanups or unique_ptr's. I.e., this: struct ui_file *file = gdb_fopen (filename, "w"); if (filename == NULL) perror_with_name (filename); cleanups = make_cleanup_ui_file_delete (file); // use file. do_cleanups (cleanups); is replaced with this: stdio_file file; if (!file.open (filename, "w")) perror_with_name (filename); // use file. - odd contorsions in null_file_write / null_file_fputs around when to call to_fputs / to_write eliminated. - Global null_stream object A few places that were allocating a ui_file in order to print to "nowhere" are adjusted to instead refer to a new 'null_stream' global stream. - TUI's tui_sfileopen eliminated. TUI's ui_file much simplified The TUI's ui_file was serving a dual purpose. It supported being used as string buffer, and supported being backed by a stdio FILE. The string buffer part is gone, replaced by using of string_file. The 'FILE *' support is now much simplified, by making the TUI's ui_file inherit from stdio_file. gdb/ChangeLog: 2017-02-02 Pedro Alves <palves@redhat.com> * ada-lang.c (type_as_string): Use string_file. * ada-valprint.c (ada_print_floating): Use string_file. * ada-varobj.c (ada_varobj_scalar_image) (ada_varobj_get_value_image): Use string_file. * aix-thread.c (aix_thread_extra_thread_info): Use string_file. * arm-tdep.c (_initialize_arm_tdep): Use string_printf. * breakpoint.c (update_inserted_breakpoint_locations) (insert_breakpoint_locations, reattach_breakpoints) (print_breakpoint_location, print_one_detail_ranged_breakpoint) (print_it_watchpoint): Use string_file. (save_breakpoints): Use stdio_file. * c-exp.y (oper): Use string_file. * cli/cli-logging.c (set_logging_redirect): Use ui_file_up and tee_file. (pop_output_files): Use delete. (handle_redirections): Use stdio_file and tee_file. * cli/cli-setshow.c (do_show_command): Use string_file. * compile/compile-c-support.c (c_compute_program): Use string_file. * compile/compile-c-symbols.c (generate_vla_size): Take a 'string_file &' instead of a 'ui_file *'. (generate_c_for_for_one_variable): Take a 'string_file &' instead of a 'ui_file *'. Use string_file. (generate_c_for_variable_locations): Take a 'string_file &' instead of a 'ui_file *'. * compile/compile-internal.h (generate_c_for_for_one_variable): Take a 'string_file &' instead of a 'ui_file *'. * compile/compile-loc2c.c (push, pushf, unary, binary) (print_label, pushf_register_address, pushf_register) (do_compile_dwarf_expr_to_c): Take a 'string_file &' instead of a 'ui_file *'. Adjust. * compile/compile.c (compile_to_object): Use string_file. * compile/compile.h (compile_dwarf_expr_to_c) (compile_dwarf_bounds_to_c): Take a 'string_file &' instead of a 'ui_file *'. * cp-support.c (inspect_type): Use string_file and obstack_copy0. (replace_typedefs_qualified_name): Use string_file and obstack_copy0. * disasm.c (gdb_pretty_print_insn): Use string_file. (gdb_disassembly): Adjust reference the null_stream global. (do_ui_file_delete): Delete. (gdb_insn_length): Use null_stream. * dummy-frame.c (maintenance_print_dummy_frames): Use stdio_file. * dwarf2loc.c (dwarf2_compile_property_to_c) (locexpr_generate_c_location, loclist_generate_c_location): Take a 'string_file &' instead of a 'ui_file *'. * dwarf2loc.h (dwarf2_compile_property_to_c): Likewise. * dwarf2read.c (do_ui_file_peek_last): Delete. (dwarf2_compute_name): Use string_file. * event-top.c (gdb_setup_readline): Use stdio_file. * gdbarch.sh (verify_gdbarch): Use string_file. * gdbtypes.c (safe_parse_type): Use null_stream. * guile/scm-breakpoint.c (gdbscm_breakpoint_commands): Use string_file. * guile/scm-disasm.c (gdbscm_print_insn_from_port): Take a 'string_file *' instead of a 'ui_file *'. (gdbscm_arch_disassemble): Use string_file. * guile/scm-frame.c (frscm_print_frame_smob): Use string_file. * guile/scm-ports.c (class ioscm_file_port): Now a class that inherits from ui_file. (ioscm_file_port_delete, ioscm_file_port_rewind) (ioscm_file_port_put): Delete. (ioscm_file_port_write): Rename to ... (ioscm_file_port::write): ... this. Remove file_port_magic checks. (ioscm_file_port_new): Delete. (ioscm_with_output_to_port_worker): Use ioscm_file_port and ui_file_up. * guile/scm-type.c (tyscm_type_name): Use string_file. * guile/scm-value.c (vlscm_print_value_smob, gdbscm_value_print): Use string_file. * infcmd.c (print_return_value_1): Use string_file. * infrun.c (print_target_wait_results): Use string_file. * language.c (add_language): Use string_file. * location.c (explicit_to_string_internal): Use string_file. * main.c (captured_main_1): Use null_file. * maint.c (maintenance_print_architecture): Use stdio_file. * mi/mi-cmd-stack.c (list_arg_or_local): Use string_file. * mi/mi-common.h (struct mi_interp) <out, err, log, targ, event_channel>: Change type to mi_console_file pointer. * mi/mi-console.c (mi_console_file_fputs, mi_console_file_flush) (mi_console_file_delete): Delete. (struct mi_console_file): Delete. (mi_console_file_magic): Delete. (mi_console_file_new): Delete. (mi_console_file::mi_console_file): New. (mi_console_file_delete): Delete. (mi_console_file_fputs): Delete. (mi_console_file::write): New. (mi_console_raw_packet): Delete. (mi_console_file::flush): New. (mi_console_file_flush): Delete. (mi_console_set_raw): Rename to ... (mi_console_file::set_raw): ... this. * mi/mi-console.h (class mi_console_file): New class. (mi_console_file_new, mi_console_set_raw): Delete. * mi/mi-interp.c (mi_interpreter_init): Use mi_console_file. (mi_set_logging): Use delete and tee_file. Adjust. * mi/mi-main.c (output_register): Use string_file. (mi_cmd_data_evaluate_expression): Use string_file. (mi_cmd_data_read_memory): Use string_file. (mi_cmd_execute, print_variable_or_computed): Use string_file. * mi/mi-out.c (mi_ui_out::main_stream): New. (mi_ui_out::rewind): Use main_stream and string_file. (mi_ui_out::put): Use main_stream and string_file. (mi_ui_out::mi_ui_out): Remove 'stream' parameter. Allocate a 'string_file' instead. (mi_out_new): Don't allocate a mem_fileopen stream here. * mi/mi-out.h (mi_ui_out::mi_ui_out): Remove 'stream' parameter. (mi_ui_out::main_stream): Declare method. * printcmd.c (eval_command): Use string_file. * psymtab.c (maintenance_print_psymbols): Use stdio_file. * python/py-arch.c (archpy_disassemble): Use string_file. * python/py-breakpoint.c (bppy_get_commands): Use string_file. * python/py-frame.c (frapy_str): Use string_file. * python/py-framefilter.c (py_print_type, py_print_single_arg): Use string_file. * python/py-type.c (typy_str): Use string_file. * python/py-unwind.c (unwind_infopy_str): Use string_file. * python/py-value.c (valpy_str): Use string_file. * record-btrace.c (btrace_insn_history): Use string_file. * regcache.c (regcache_print): Use stdio_file. * reggroups.c (maintenance_print_reggroups): Use stdio_file. * remote.c (escape_buffer): Use string_file. * rust-lang.c (rust_get_disr_info): Use string_file. * serial.c (serial_open_ops_1): Use stdio_file. (do_serial_close): Use delete. * stack.c (print_frame_arg): Use string_file. (print_frame_args): Remove local mem_fileopen stream, not used. (print_frame): Use string_file. * symmisc.c (maintenance_print_symbols): Use stdio_file. * symtab.h (struct symbol_computed_ops) <generate_c_location>: Take a 'string_file *' instead of a 'ui_file *'. * top.c (new_ui): Use stdio_file and stderr_file. (free_ui): Use delete. (execute_command_to_string): Use string_file. (quit_confirm): Use string_file. * tracepoint.c (collection_list::append_exp): Use string_file. * tui/tui-disasm.c (tui_disassemble): Use string_file. * tui/tui-file.c: Don't include "ui-file.h". (enum streamtype, struct tui_stream): Delete. (tui_file_new, tui_file_delete, tui_fileopen, tui_sfileopen) (tui_file_isatty, tui_file_rewind, tui_file_put): Delete. (tui_file::tui_file): New method. (tui_file_fputs): Delete. (tui_file_get_strbuf): Delete. (tui_file::puts): New method. (tui_file_adjust_strbuf): Delete. (tui_file_flush): Delete. (tui_file::flush): New method. * tui/tui-file.h: Tweak intro comment. Include ui-file.h. (tui_fileopen, tui_sfileopen, tui_file_get_strbuf) (tui_file_adjust_strbuf): Delete declarations. (class tui_file): New class. * tui/tui-io.c (tui_initialize_io): Use tui_file. * tui/tui-regs.c (tui_restore_gdbout): Use delete. (tui_register_format): Use string_stream. * tui/tui-stack.c (tui_make_status_line): Use string_file. (tui_get_function_from_frame): Use string_file. * typeprint.c (type_to_string): Use string_file. * ui-file.c (struct ui_file, ui_file_magic, ui_file_new): Delete. (null_stream): New global. (ui_file_delete): Delete. (ui_file::ui_file): New. (null_file_isatty): Delete. (ui_file::~ui_file): New. (null_file_rewind): Delete. (ui_file::printf): New. (null_file_put): Delete. (null_file_flush): Delete. (ui_file::putstr): New. (null_file_write): Delete. (ui_file::putstrn): New. (null_file_read): Delete. (ui_file::putc): New. (null_file_fputs): Delete. (null_file_write_async_safe): Delete. (ui_file::vprintf): New. (null_file_delete): Delete. (null_file::write): New. (null_file_fseek): Delete. (null_file::puts): New. (ui_file_data): Delete. (null_file::write_async_safe): New. (gdb_flush, ui_file_isatty): Adjust. (ui_file_put, ui_file_rewind): Delete. (ui_file_write): Adjust. (ui_file_write_for_put): Delete. (ui_file_write_async_safe, ui_file_read): Adjust. (ui_file_fseek): Delete. (fputs_unfiltered): Adjust. (set_ui_file_flush, set_ui_file_isatty, set_ui_file_rewind) (set_ui_file_put, set_ui_file_write, set_ui_file_write_async_safe) (set_ui_file_read, set_ui_file_fputs, set_ui_file_fseek) (set_ui_file_data): Delete. (string_file::~string_file, string_file::write) (struct accumulated_ui_file, do_ui_file_xstrdup, ui_file_xstrdup) (do_ui_file_as_string, ui_file_as_string): Delete. (do_ui_file_obsavestring, ui_file_obsavestring): Delete. (struct mem_file): Delete. (mem_file_new): Delete. (stdio_file::stdio_file): New. (mem_file_delete): Delete. (stdio_file::stdio_file): New. (mem_fileopen): Delete. (stdio_file::~stdio_file): New. (mem_file_rewind): Delete. (stdio_file::set_stream): New. (mem_file_put): Delete. (stdio_file::open): New. (mem_file_write): Delete. (stdio_file_magic, struct stdio_file): Delete. (stdio_file_new, stdio_file_delete, stdio_file_flush): Delete. (stdio_file::flush): New. (stdio_file_read): Rename to ... (stdio_file::read): ... this. Adjust. (stdio_file_write): Rename to ... (stdio_file::write): ... this. Adjust. (stdio_file_write_async_safe): Rename to ... (stdio_file::write_async_safe) ... this. Adjust. (stdio_file_fputs): Rename to ... (stdio_file::puts) ... this. Adjust. (stdio_file_isatty): Delete. (stdio_file_fseek): Delete. (stdio_file::isatty): New. (stderr_file_write): Rename to ... (stderr_file::write) ... this. Adjust. (stderr_file_fputs): Rename to ... (stderr_file::puts) ... this. Adjust. (stderr_fileopen, stdio_fileopen, gdb_fopen): Delete. (stderr_file::stderr_file): New. (tee_file_magic): Delete. (struct tee_file): Delete. (tee_file::tee_file): New. (tee_file_new): Delete. (tee_file::~tee_file): New. (tee_file_delete): Delete. (tee_file_flush): Rename to ... (tee_file::flush): ... this. Adjust. (tee_file_write): Rename to ... (tee_file::write): ... this. Adjust. (tee_file::write_async_safe): New. (tee_file_fputs): Rename to ... (tee_file::puts): ... this. Adjust. (tee_file_isatty): Rename to ... (tee_file::isatty): ... this. Adjust. * ui-file.h (struct obstack, struct ui_file): Don't forward-declare. (ui_file_new, ui_file_flush_ftype, set_ui_file_flush) (ui_file_write_ftype) (set_ui_file_write, ui_file_fputs_ftype, set_ui_file_fputs) (ui_file_write_async_safe_ftype, set_ui_file_write_async_safe) (ui_file_read_ftype, set_ui_file_read, ui_file_isatty_ftype) (set_ui_file_isatty, ui_file_rewind_ftype, set_ui_file_rewind) (ui_file_put_method_ftype, ui_file_put_ftype, set_ui_file_put) (ui_file_delete_ftype, set_ui_file_data, ui_file_fseek_ftype) (set_ui_file_fseek): Delete. (ui_file_data, ui_file_delete, ui_file_rewind) (struct ui_file): New. (ui_file_up): New. (class null_file): New. (null_stream): Declare. (ui_file_write_for_put, ui_file_put): Delete. (ui_file_xstrdup, ui_file_as_string, ui_file_obsavestring): Delete. (ui_file_fseek, mem_fileopen, stdio_fileopen, stderr_fileopen) (gdb_fopen, tee_file_new): Delete. (struct string_file): New. (struct stdio_file): New. (stdio_file_up): New. (struct stderr_file): New. (class tee_file): New. * ui-out.c (ui_out::field_stream): Take a 'string_file &' instead of a 'ui_file *'. Adjust. * ui-out.h (class ui_out) <field_stream>: Likewise. * utils.c (do_ui_file_delete, make_cleanup_ui_file_delete) (null_stream): Delete. (error_stream): Take a 'string_file &' instead of a 'ui_file *'. Adjust. * utils.h (struct ui_file): Delete forward declaration.. (make_cleanup_ui_file_delete, null_stream): Delete declarations. (error_stream): Take a 'string_file &' instead of a 'ui_file *'. * varobj.c (varobj_value_get_print_value): Use string_file. * xtensa-tdep.c (xtensa_verify_config): Use string_file. * gdbarch.c: Regenerate.
1667 lines
41 KiB
C
1667 lines
41 KiB
C
/* Python interface to types.
|
||
|
||
Copyright (C) 2008-2017 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 "defs.h"
|
||
#include "value.h"
|
||
#include "python-internal.h"
|
||
#include "charset.h"
|
||
#include "gdbtypes.h"
|
||
#include "cp-support.h"
|
||
#include "demangle.h"
|
||
#include "objfiles.h"
|
||
#include "language.h"
|
||
#include "vec.h"
|
||
#include "typeprint.h"
|
||
#include "py-ref.h"
|
||
|
||
typedef struct pyty_type_object
|
||
{
|
||
PyObject_HEAD
|
||
struct type *type;
|
||
|
||
/* If a Type object is associated with an objfile, it is kept on a
|
||
doubly-linked list, rooted in the objfile. This lets us copy the
|
||
underlying struct type when the objfile is deleted. */
|
||
struct pyty_type_object *prev;
|
||
struct pyty_type_object *next;
|
||
} type_object;
|
||
|
||
extern PyTypeObject type_object_type
|
||
CPYCHECKER_TYPE_OBJECT_FOR_TYPEDEF ("type_object");
|
||
|
||
/* A Field object. */
|
||
typedef struct pyty_field_object
|
||
{
|
||
PyObject_HEAD
|
||
|
||
/* Dictionary holding our attributes. */
|
||
PyObject *dict;
|
||
} field_object;
|
||
|
||
extern PyTypeObject field_object_type
|
||
CPYCHECKER_TYPE_OBJECT_FOR_TYPEDEF ("field_object");
|
||
|
||
/* A type iterator object. */
|
||
typedef struct {
|
||
PyObject_HEAD
|
||
/* The current field index. */
|
||
int field;
|
||
/* What to return. */
|
||
enum gdbpy_iter_kind kind;
|
||
/* Pointer back to the original source type object. */
|
||
struct pyty_type_object *source;
|
||
} typy_iterator_object;
|
||
|
||
extern PyTypeObject type_iterator_object_type
|
||
CPYCHECKER_TYPE_OBJECT_FOR_TYPEDEF ("typy_iterator_object");
|
||
|
||
/* This is used to initialize various gdb.TYPE_ constants. */
|
||
struct pyty_code
|
||
{
|
||
/* The code. */
|
||
enum type_code code;
|
||
/* The name. */
|
||
const char *name;
|
||
};
|
||
|
||
/* Forward declarations. */
|
||
static PyObject *typy_make_iter (PyObject *self, enum gdbpy_iter_kind kind);
|
||
|
||
#define ENTRY(X) { X, #X }
|
||
|
||
static struct pyty_code pyty_codes[] =
|
||
{
|
||
ENTRY (TYPE_CODE_BITSTRING),
|
||
ENTRY (TYPE_CODE_PTR),
|
||
ENTRY (TYPE_CODE_ARRAY),
|
||
ENTRY (TYPE_CODE_STRUCT),
|
||
ENTRY (TYPE_CODE_UNION),
|
||
ENTRY (TYPE_CODE_ENUM),
|
||
ENTRY (TYPE_CODE_FLAGS),
|
||
ENTRY (TYPE_CODE_FUNC),
|
||
ENTRY (TYPE_CODE_INT),
|
||
ENTRY (TYPE_CODE_FLT),
|
||
ENTRY (TYPE_CODE_VOID),
|
||
ENTRY (TYPE_CODE_SET),
|
||
ENTRY (TYPE_CODE_RANGE),
|
||
ENTRY (TYPE_CODE_STRING),
|
||
ENTRY (TYPE_CODE_ERROR),
|
||
ENTRY (TYPE_CODE_METHOD),
|
||
ENTRY (TYPE_CODE_METHODPTR),
|
||
ENTRY (TYPE_CODE_MEMBERPTR),
|
||
ENTRY (TYPE_CODE_REF),
|
||
ENTRY (TYPE_CODE_CHAR),
|
||
ENTRY (TYPE_CODE_BOOL),
|
||
ENTRY (TYPE_CODE_COMPLEX),
|
||
ENTRY (TYPE_CODE_TYPEDEF),
|
||
ENTRY (TYPE_CODE_NAMESPACE),
|
||
ENTRY (TYPE_CODE_DECFLOAT),
|
||
ENTRY (TYPE_CODE_INTERNAL_FUNCTION),
|
||
{ TYPE_CODE_UNDEF, NULL }
|
||
};
|
||
|
||
|
||
|
||
static void
|
||
field_dealloc (PyObject *obj)
|
||
{
|
||
field_object *f = (field_object *) obj;
|
||
|
||
Py_XDECREF (f->dict);
|
||
Py_TYPE (obj)->tp_free (obj);
|
||
}
|
||
|
||
static PyObject *
|
||
field_new (void)
|
||
{
|
||
field_object *result = PyObject_New (field_object, &field_object_type);
|
||
|
||
if (result)
|
||
{
|
||
result->dict = PyDict_New ();
|
||
if (!result->dict)
|
||
{
|
||
Py_DECREF (result);
|
||
result = NULL;
|
||
}
|
||
}
|
||
return (PyObject *) result;
|
||
}
|
||
|
||
|
||
|
||
/* Return true if OBJ is of type gdb.Field, false otherwise. */
|
||
|
||
int
|
||
gdbpy_is_field (PyObject *obj)
|
||
{
|
||
return PyObject_TypeCheck (obj, &field_object_type);
|
||
}
|
||
|
||
/* Return the code for this type. */
|
||
static PyObject *
|
||
typy_get_code (PyObject *self, void *closure)
|
||
{
|
||
struct type *type = ((type_object *) self)->type;
|
||
|
||
return PyInt_FromLong (TYPE_CODE (type));
|
||
}
|
||
|
||
/* Helper function for typy_fields which converts a single field to a
|
||
gdb.Field object. Returns NULL on error. */
|
||
|
||
static PyObject *
|
||
convert_field (struct type *type, int field)
|
||
{
|
||
gdbpy_ref result (field_new ());
|
||
|
||
if (result == NULL)
|
||
return NULL;
|
||
|
||
gdbpy_ref arg (type_to_type_object (type));
|
||
if (arg == NULL)
|
||
return NULL;
|
||
if (PyObject_SetAttrString (result.get (), "parent_type", arg.get ()) < 0)
|
||
return NULL;
|
||
|
||
if (!field_is_static (&TYPE_FIELD (type, field)))
|
||
{
|
||
const char *attrstring;
|
||
|
||
if (TYPE_CODE (type) == TYPE_CODE_ENUM)
|
||
{
|
||
arg.reset (gdb_py_long_from_longest (TYPE_FIELD_ENUMVAL (type,
|
||
field)));
|
||
attrstring = "enumval";
|
||
}
|
||
else
|
||
{
|
||
arg.reset (gdb_py_long_from_longest (TYPE_FIELD_BITPOS (type,
|
||
field)));
|
||
attrstring = "bitpos";
|
||
}
|
||
|
||
if (arg == NULL)
|
||
return NULL;
|
||
|
||
/* At least python-2.4 had the second parameter non-const. */
|
||
if (PyObject_SetAttrString (result.get (), (char *) attrstring,
|
||
arg.get ()) < 0)
|
||
return NULL;
|
||
}
|
||
|
||
arg.reset (NULL);
|
||
if (TYPE_FIELD_NAME (type, field))
|
||
{
|
||
const char *field_name = TYPE_FIELD_NAME (type, field);
|
||
|
||
if (field_name[0] != '\0')
|
||
{
|
||
arg.reset (PyString_FromString (TYPE_FIELD_NAME (type, field)));
|
||
if (arg == NULL)
|
||
return NULL;
|
||
}
|
||
}
|
||
if (arg == NULL)
|
||
{
|
||
arg.reset (Py_None);
|
||
Py_INCREF (arg.get ());
|
||
}
|
||
if (PyObject_SetAttrString (result.get (), "name", arg.get ()) < 0)
|
||
return NULL;
|
||
|
||
arg.reset (TYPE_FIELD_ARTIFICIAL (type, field) ? Py_True : Py_False);
|
||
Py_INCREF (arg.get ());
|
||
if (PyObject_SetAttrString (result.get (), "artificial", arg.get ()) < 0)
|
||
return NULL;
|
||
|
||
if (TYPE_CODE (type) == TYPE_CODE_STRUCT)
|
||
arg.reset (field < TYPE_N_BASECLASSES (type) ? Py_True : Py_False);
|
||
else
|
||
arg.reset (Py_False);
|
||
Py_INCREF (arg.get ());
|
||
if (PyObject_SetAttrString (result.get (), "is_base_class", arg.get ()) < 0)
|
||
return NULL;
|
||
|
||
arg.reset (PyLong_FromLong (TYPE_FIELD_BITSIZE (type, field)));
|
||
if (arg == NULL)
|
||
return NULL;
|
||
if (PyObject_SetAttrString (result.get (), "bitsize", arg.get ()) < 0)
|
||
return NULL;
|
||
|
||
/* A field can have a NULL type in some situations. */
|
||
if (TYPE_FIELD_TYPE (type, field) == NULL)
|
||
{
|
||
arg.reset (Py_None);
|
||
Py_INCREF (arg.get ());
|
||
}
|
||
else
|
||
arg.reset (type_to_type_object (TYPE_FIELD_TYPE (type, field)));
|
||
if (arg == NULL)
|
||
return NULL;
|
||
if (PyObject_SetAttrString (result.get (), "type", arg.get ()) < 0)
|
||
return NULL;
|
||
|
||
return result.release ();
|
||
}
|
||
|
||
/* Helper function to return the name of a field, as a gdb.Field object.
|
||
If the field doesn't have a name, None is returned. */
|
||
|
||
static PyObject *
|
||
field_name (struct type *type, int field)
|
||
{
|
||
PyObject *result;
|
||
|
||
if (TYPE_FIELD_NAME (type, field))
|
||
result = PyString_FromString (TYPE_FIELD_NAME (type, field));
|
||
else
|
||
{
|
||
result = Py_None;
|
||
Py_INCREF (result);
|
||
}
|
||
return result;
|
||
}
|
||
|
||
/* Helper function for Type standard mapping methods. Returns a
|
||
Python object for field i of the type. "kind" specifies what to
|
||
return: the name of the field, a gdb.Field object corresponding to
|
||
the field, or a tuple consisting of field name and gdb.Field
|
||
object. */
|
||
|
||
static PyObject *
|
||
make_fielditem (struct type *type, int i, enum gdbpy_iter_kind kind)
|
||
{
|
||
switch (kind)
|
||
{
|
||
case iter_items:
|
||
{
|
||
gdbpy_ref key (field_name (type, i));
|
||
if (key == NULL)
|
||
return NULL;
|
||
gdbpy_ref value (convert_field (type, i));
|
||
if (value == NULL)
|
||
return NULL;
|
||
gdbpy_ref item (PyTuple_New (2));
|
||
if (item == NULL)
|
||
return NULL;
|
||
PyTuple_SET_ITEM (item.get (), 0, key.release ());
|
||
PyTuple_SET_ITEM (item.get (), 1, value.release ());
|
||
return item.release ();
|
||
}
|
||
case iter_keys:
|
||
return field_name (type, i);
|
||
case iter_values:
|
||
return convert_field (type, i);
|
||
}
|
||
gdb_assert_not_reached ("invalid gdbpy_iter_kind");
|
||
}
|
||
|
||
/* Return a sequence of all field names, fields, or (name, field) pairs.
|
||
Each field is a gdb.Field object. */
|
||
|
||
static PyObject *
|
||
typy_fields_items (PyObject *self, enum gdbpy_iter_kind kind)
|
||
{
|
||
PyObject *py_type = self;
|
||
PyObject *result = NULL, *iter = NULL;
|
||
struct type *type = ((type_object *) py_type)->type;
|
||
struct type *checked_type = type;
|
||
|
||
TRY
|
||
{
|
||
checked_type = check_typedef (checked_type);
|
||
}
|
||
CATCH (except, RETURN_MASK_ALL)
|
||
{
|
||
GDB_PY_HANDLE_EXCEPTION (except);
|
||
}
|
||
END_CATCH
|
||
|
||
if (checked_type != type)
|
||
py_type = type_to_type_object (checked_type);
|
||
iter = typy_make_iter (py_type, kind);
|
||
if (checked_type != type)
|
||
{
|
||
/* Need to wrap this in braces because Py_DECREF isn't wrapped
|
||
in a do{}while(0). */
|
||
Py_DECREF (py_type);
|
||
}
|
||
if (iter != NULL)
|
||
{
|
||
result = PySequence_List (iter);
|
||
Py_DECREF (iter);
|
||
}
|
||
|
||
return result;
|
||
}
|
||
|
||
/* Return a sequence of all fields. Each field is a gdb.Field object. */
|
||
|
||
static PyObject *
|
||
typy_values (PyObject *self, PyObject *args)
|
||
{
|
||
return typy_fields_items (self, iter_values);
|
||
}
|
||
|
||
/* Return a sequence of all fields. Each field is a gdb.Field object.
|
||
This method is similar to typy_values, except where the supplied
|
||
gdb.Type is an array, in which case it returns a list of one entry
|
||
which is a gdb.Field object for a range (the array bounds). */
|
||
|
||
static PyObject *
|
||
typy_fields (PyObject *self, PyObject *args)
|
||
{
|
||
struct type *type = ((type_object *) self)->type;
|
||
|
||
if (TYPE_CODE (type) != TYPE_CODE_ARRAY)
|
||
return typy_fields_items (self, iter_values);
|
||
|
||
/* Array type. Handle this as a special case because the common
|
||
machinery wants struct or union or enum types. Build a list of
|
||
one entry which is the range for the array. */
|
||
gdbpy_ref r (convert_field (type, 0));
|
||
if (r == NULL)
|
||
return NULL;
|
||
|
||
return Py_BuildValue ("[O]", r.get ());
|
||
}
|
||
|
||
/* Return a sequence of all field names. Each field is a gdb.Field object. */
|
||
|
||
static PyObject *
|
||
typy_field_names (PyObject *self, PyObject *args)
|
||
{
|
||
return typy_fields_items (self, iter_keys);
|
||
}
|
||
|
||
/* Return a sequence of all (name, fields) pairs. Each field is a
|
||
gdb.Field object. */
|
||
|
||
static PyObject *
|
||
typy_items (PyObject *self, PyObject *args)
|
||
{
|
||
return typy_fields_items (self, iter_items);
|
||
}
|
||
|
||
/* Return the type's name, or None. */
|
||
|
||
static PyObject *
|
||
typy_get_name (PyObject *self, void *closure)
|
||
{
|
||
struct type *type = ((type_object *) self)->type;
|
||
|
||
if (TYPE_NAME (type) == NULL)
|
||
Py_RETURN_NONE;
|
||
return PyString_FromString (TYPE_NAME (type));
|
||
}
|
||
|
||
/* Return the type's tag, or None. */
|
||
static PyObject *
|
||
typy_get_tag (PyObject *self, void *closure)
|
||
{
|
||
struct type *type = ((type_object *) self)->type;
|
||
|
||
if (!TYPE_TAG_NAME (type))
|
||
Py_RETURN_NONE;
|
||
return PyString_FromString (TYPE_TAG_NAME (type));
|
||
}
|
||
|
||
/* Return the type, stripped of typedefs. */
|
||
static PyObject *
|
||
typy_strip_typedefs (PyObject *self, PyObject *args)
|
||
{
|
||
struct type *type = ((type_object *) self)->type;
|
||
|
||
TRY
|
||
{
|
||
type = check_typedef (type);
|
||
}
|
||
CATCH (except, RETURN_MASK_ALL)
|
||
{
|
||
GDB_PY_HANDLE_EXCEPTION (except);
|
||
}
|
||
END_CATCH
|
||
|
||
return type_to_type_object (type);
|
||
}
|
||
|
||
/* Strip typedefs and pointers/reference from a type. Then check that
|
||
it is a struct, union, or enum type. If not, raise TypeError. */
|
||
|
||
static struct type *
|
||
typy_get_composite (struct type *type)
|
||
{
|
||
|
||
for (;;)
|
||
{
|
||
TRY
|
||
{
|
||
type = check_typedef (type);
|
||
}
|
||
CATCH (except, RETURN_MASK_ALL)
|
||
{
|
||
GDB_PY_HANDLE_EXCEPTION (except);
|
||
}
|
||
END_CATCH
|
||
|
||
if (TYPE_CODE (type) != TYPE_CODE_PTR
|
||
&& TYPE_CODE (type) != TYPE_CODE_REF)
|
||
break;
|
||
type = TYPE_TARGET_TYPE (type);
|
||
}
|
||
|
||
/* If this is not a struct, union, or enum type, raise TypeError
|
||
exception. */
|
||
if (TYPE_CODE (type) != TYPE_CODE_STRUCT
|
||
&& TYPE_CODE (type) != TYPE_CODE_UNION
|
||
&& TYPE_CODE (type) != TYPE_CODE_ENUM
|
||
&& TYPE_CODE (type) != TYPE_CODE_FUNC)
|
||
{
|
||
PyErr_SetString (PyExc_TypeError,
|
||
"Type is not a structure, union, enum, or function type.");
|
||
return NULL;
|
||
}
|
||
|
||
return type;
|
||
}
|
||
|
||
/* Helper for typy_array and typy_vector. */
|
||
|
||
static PyObject *
|
||
typy_array_1 (PyObject *self, PyObject *args, int is_vector)
|
||
{
|
||
long n1, n2;
|
||
PyObject *n2_obj = NULL;
|
||
struct type *array = NULL;
|
||
struct type *type = ((type_object *) self)->type;
|
||
|
||
if (! PyArg_ParseTuple (args, "l|O", &n1, &n2_obj))
|
||
return NULL;
|
||
|
||
if (n2_obj)
|
||
{
|
||
if (!PyInt_Check (n2_obj))
|
||
{
|
||
PyErr_SetString (PyExc_RuntimeError,
|
||
_("Array bound must be an integer"));
|
||
return NULL;
|
||
}
|
||
|
||
if (! gdb_py_int_as_long (n2_obj, &n2))
|
||
return NULL;
|
||
}
|
||
else
|
||
{
|
||
n2 = n1;
|
||
n1 = 0;
|
||
}
|
||
|
||
if (n2 < n1 - 1) /* Note: An empty array has n2 == n1 - 1. */
|
||
{
|
||
PyErr_SetString (PyExc_ValueError,
|
||
_("Array length must not be negative"));
|
||
return NULL;
|
||
}
|
||
|
||
TRY
|
||
{
|
||
array = lookup_array_range_type (type, n1, n2);
|
||
if (is_vector)
|
||
make_vector_type (array);
|
||
}
|
||
CATCH (except, RETURN_MASK_ALL)
|
||
{
|
||
GDB_PY_HANDLE_EXCEPTION (except);
|
||
}
|
||
END_CATCH
|
||
|
||
return type_to_type_object (array);
|
||
}
|
||
|
||
/* Return an array type. */
|
||
|
||
static PyObject *
|
||
typy_array (PyObject *self, PyObject *args)
|
||
{
|
||
return typy_array_1 (self, args, 0);
|
||
}
|
||
|
||
/* Return a vector type. */
|
||
|
||
static PyObject *
|
||
typy_vector (PyObject *self, PyObject *args)
|
||
{
|
||
return typy_array_1 (self, args, 1);
|
||
}
|
||
|
||
/* Return a Type object which represents a pointer to SELF. */
|
||
static PyObject *
|
||
typy_pointer (PyObject *self, PyObject *args)
|
||
{
|
||
struct type *type = ((type_object *) self)->type;
|
||
|
||
TRY
|
||
{
|
||
type = lookup_pointer_type (type);
|
||
}
|
||
CATCH (except, RETURN_MASK_ALL)
|
||
{
|
||
GDB_PY_HANDLE_EXCEPTION (except);
|
||
}
|
||
END_CATCH
|
||
|
||
return type_to_type_object (type);
|
||
}
|
||
|
||
/* Return the range of a type represented by SELF. The return type is
|
||
a tuple. The first element of the tuple contains the low bound,
|
||
while the second element of the tuple contains the high bound. */
|
||
static PyObject *
|
||
typy_range (PyObject *self, PyObject *args)
|
||
{
|
||
struct type *type = ((type_object *) self)->type;
|
||
/* Initialize these to appease GCC warnings. */
|
||
LONGEST low = 0, high = 0;
|
||
|
||
if (TYPE_CODE (type) != TYPE_CODE_ARRAY
|
||
&& TYPE_CODE (type) != TYPE_CODE_STRING
|
||
&& TYPE_CODE (type) != TYPE_CODE_RANGE)
|
||
{
|
||
PyErr_SetString (PyExc_RuntimeError,
|
||
_("This type does not have a range."));
|
||
return NULL;
|
||
}
|
||
|
||
switch (TYPE_CODE (type))
|
||
{
|
||
case TYPE_CODE_ARRAY:
|
||
case TYPE_CODE_STRING:
|
||
low = TYPE_LOW_BOUND (TYPE_INDEX_TYPE (type));
|
||
high = TYPE_HIGH_BOUND (TYPE_INDEX_TYPE (type));
|
||
break;
|
||
case TYPE_CODE_RANGE:
|
||
low = TYPE_LOW_BOUND (type);
|
||
high = TYPE_HIGH_BOUND (type);
|
||
break;
|
||
}
|
||
|
||
gdbpy_ref low_bound (PyLong_FromLong (low));
|
||
if (low_bound == NULL)
|
||
return NULL;
|
||
|
||
gdbpy_ref high_bound (PyLong_FromLong (high));
|
||
if (high_bound == NULL)
|
||
return NULL;
|
||
|
||
gdbpy_ref result (PyTuple_New (2));
|
||
if (result == NULL)
|
||
return NULL;
|
||
|
||
if (PyTuple_SetItem (result.get (), 0, low_bound.release ()) != 0
|
||
|| PyTuple_SetItem (result.get (), 1, high_bound.release ()) != 0)
|
||
return NULL;
|
||
return result.release ();
|
||
}
|
||
|
||
/* Return a Type object which represents a reference to SELF. */
|
||
static PyObject *
|
||
typy_reference (PyObject *self, PyObject *args)
|
||
{
|
||
struct type *type = ((type_object *) self)->type;
|
||
|
||
TRY
|
||
{
|
||
type = lookup_reference_type (type);
|
||
}
|
||
CATCH (except, RETURN_MASK_ALL)
|
||
{
|
||
GDB_PY_HANDLE_EXCEPTION (except);
|
||
}
|
||
END_CATCH
|
||
|
||
return type_to_type_object (type);
|
||
}
|
||
|
||
/* Return a Type object which represents the target type of SELF. */
|
||
static PyObject *
|
||
typy_target (PyObject *self, PyObject *args)
|
||
{
|
||
struct type *type = ((type_object *) self)->type;
|
||
|
||
if (!TYPE_TARGET_TYPE (type))
|
||
{
|
||
PyErr_SetString (PyExc_RuntimeError,
|
||
_("Type does not have a target."));
|
||
return NULL;
|
||
}
|
||
|
||
return type_to_type_object (TYPE_TARGET_TYPE (type));
|
||
}
|
||
|
||
/* Return a const-qualified type variant. */
|
||
static PyObject *
|
||
typy_const (PyObject *self, PyObject *args)
|
||
{
|
||
struct type *type = ((type_object *) self)->type;
|
||
|
||
TRY
|
||
{
|
||
type = make_cv_type (1, 0, type, NULL);
|
||
}
|
||
CATCH (except, RETURN_MASK_ALL)
|
||
{
|
||
GDB_PY_HANDLE_EXCEPTION (except);
|
||
}
|
||
END_CATCH
|
||
|
||
return type_to_type_object (type);
|
||
}
|
||
|
||
/* Return a volatile-qualified type variant. */
|
||
static PyObject *
|
||
typy_volatile (PyObject *self, PyObject *args)
|
||
{
|
||
struct type *type = ((type_object *) self)->type;
|
||
|
||
TRY
|
||
{
|
||
type = make_cv_type (0, 1, type, NULL);
|
||
}
|
||
CATCH (except, RETURN_MASK_ALL)
|
||
{
|
||
GDB_PY_HANDLE_EXCEPTION (except);
|
||
}
|
||
END_CATCH
|
||
|
||
return type_to_type_object (type);
|
||
}
|
||
|
||
/* Return an unqualified type variant. */
|
||
static PyObject *
|
||
typy_unqualified (PyObject *self, PyObject *args)
|
||
{
|
||
struct type *type = ((type_object *) self)->type;
|
||
|
||
TRY
|
||
{
|
||
type = make_cv_type (0, 0, type, NULL);
|
||
}
|
||
CATCH (except, RETURN_MASK_ALL)
|
||
{
|
||
GDB_PY_HANDLE_EXCEPTION (except);
|
||
}
|
||
END_CATCH
|
||
|
||
return type_to_type_object (type);
|
||
}
|
||
|
||
/* Return the size of the type represented by SELF, in bytes. */
|
||
static PyObject *
|
||
typy_get_sizeof (PyObject *self, void *closure)
|
||
{
|
||
struct type *type = ((type_object *) self)->type;
|
||
|
||
TRY
|
||
{
|
||
check_typedef (type);
|
||
}
|
||
CATCH (except, RETURN_MASK_ALL)
|
||
{
|
||
}
|
||
END_CATCH
|
||
|
||
/* Ignore exceptions. */
|
||
|
||
return gdb_py_long_from_longest (TYPE_LENGTH (type));
|
||
}
|
||
|
||
static struct type *
|
||
typy_lookup_typename (const char *type_name, const struct block *block)
|
||
{
|
||
struct type *type = NULL;
|
||
|
||
TRY
|
||
{
|
||
if (startswith (type_name, "struct "))
|
||
type = lookup_struct (type_name + 7, NULL);
|
||
else if (startswith (type_name, "union "))
|
||
type = lookup_union (type_name + 6, NULL);
|
||
else if (startswith (type_name, "enum "))
|
||
type = lookup_enum (type_name + 5, NULL);
|
||
else
|
||
type = lookup_typename (python_language, python_gdbarch,
|
||
type_name, block, 0);
|
||
}
|
||
CATCH (except, RETURN_MASK_ALL)
|
||
{
|
||
GDB_PY_HANDLE_EXCEPTION (except);
|
||
}
|
||
END_CATCH
|
||
|
||
return type;
|
||
}
|
||
|
||
static struct type *
|
||
typy_lookup_type (struct demangle_component *demangled,
|
||
const struct block *block)
|
||
{
|
||
struct type *type, *rtype = NULL;
|
||
char *type_name = NULL;
|
||
enum demangle_component_type demangled_type;
|
||
|
||
/* Save the type: typy_lookup_type() may (indirectly) overwrite
|
||
memory pointed by demangled. */
|
||
demangled_type = demangled->type;
|
||
|
||
if (demangled_type == DEMANGLE_COMPONENT_POINTER
|
||
|| demangled_type == DEMANGLE_COMPONENT_REFERENCE
|
||
|| demangled_type == DEMANGLE_COMPONENT_CONST
|
||
|| demangled_type == DEMANGLE_COMPONENT_VOLATILE)
|
||
{
|
||
type = typy_lookup_type (demangled->u.s_binary.left, block);
|
||
if (! type)
|
||
return NULL;
|
||
|
||
TRY
|
||
{
|
||
/* If the demangled_type matches with one of the types
|
||
below, run the corresponding function and save the type
|
||
to return later. We cannot just return here as we are in
|
||
an exception handler. */
|
||
switch (demangled_type)
|
||
{
|
||
case DEMANGLE_COMPONENT_REFERENCE:
|
||
rtype = lookup_reference_type (type);
|
||
break;
|
||
case DEMANGLE_COMPONENT_POINTER:
|
||
rtype = lookup_pointer_type (type);
|
||
break;
|
||
case DEMANGLE_COMPONENT_CONST:
|
||
rtype = make_cv_type (1, 0, type, NULL);
|
||
break;
|
||
case DEMANGLE_COMPONENT_VOLATILE:
|
||
rtype = make_cv_type (0, 1, type, NULL);
|
||
break;
|
||
}
|
||
}
|
||
CATCH (except, RETURN_MASK_ALL)
|
||
{
|
||
GDB_PY_HANDLE_EXCEPTION (except);
|
||
}
|
||
END_CATCH
|
||
}
|
||
|
||
/* If we have a type from the switch statement above, just return
|
||
that. */
|
||
if (rtype)
|
||
return rtype;
|
||
|
||
/* We don't have a type, so lookup the type. */
|
||
type_name = cp_comp_to_string (demangled, 10);
|
||
type = typy_lookup_typename (type_name, block);
|
||
xfree (type_name);
|
||
|
||
return type;
|
||
}
|
||
|
||
/* This is a helper function for typy_template_argument that is used
|
||
when the type does not have template symbols attached. It works by
|
||
parsing the type name. This happens with compilers, like older
|
||
versions of GCC, that do not emit DW_TAG_template_*. */
|
||
|
||
static PyObject *
|
||
typy_legacy_template_argument (struct type *type, const struct block *block,
|
||
int argno)
|
||
{
|
||
int i;
|
||
struct demangle_component *demangled;
|
||
std::unique_ptr<demangle_parse_info> info;
|
||
const char *err;
|
||
struct type *argtype;
|
||
struct cleanup *cleanup;
|
||
|
||
if (TYPE_NAME (type) == NULL)
|
||
{
|
||
PyErr_SetString (PyExc_RuntimeError, _("Null type name."));
|
||
return NULL;
|
||
}
|
||
|
||
TRY
|
||
{
|
||
/* Note -- this is not thread-safe. */
|
||
info = cp_demangled_name_to_comp (TYPE_NAME (type), &err);
|
||
}
|
||
CATCH (except, RETURN_MASK_ALL)
|
||
{
|
||
GDB_PY_HANDLE_EXCEPTION (except);
|
||
}
|
||
END_CATCH
|
||
|
||
if (! info)
|
||
{
|
||
PyErr_SetString (PyExc_RuntimeError, err);
|
||
return NULL;
|
||
}
|
||
demangled = info->tree;
|
||
|
||
/* Strip off component names. */
|
||
while (demangled->type == DEMANGLE_COMPONENT_QUAL_NAME
|
||
|| demangled->type == DEMANGLE_COMPONENT_LOCAL_NAME)
|
||
demangled = demangled->u.s_binary.right;
|
||
|
||
if (demangled->type != DEMANGLE_COMPONENT_TEMPLATE)
|
||
{
|
||
PyErr_SetString (PyExc_RuntimeError, _("Type is not a template."));
|
||
return NULL;
|
||
}
|
||
|
||
/* Skip from the template to the arguments. */
|
||
demangled = demangled->u.s_binary.right;
|
||
|
||
for (i = 0; demangled && i < argno; ++i)
|
||
demangled = demangled->u.s_binary.right;
|
||
|
||
if (! demangled)
|
||
{
|
||
PyErr_Format (PyExc_RuntimeError, _("No argument %d in template."),
|
||
argno);
|
||
return NULL;
|
||
}
|
||
|
||
argtype = typy_lookup_type (demangled->u.s_binary.left, block);
|
||
if (! argtype)
|
||
return NULL;
|
||
|
||
return type_to_type_object (argtype);
|
||
}
|
||
|
||
static PyObject *
|
||
typy_template_argument (PyObject *self, PyObject *args)
|
||
{
|
||
int argno;
|
||
struct type *type = ((type_object *) self)->type;
|
||
const struct block *block = NULL;
|
||
PyObject *block_obj = NULL;
|
||
struct symbol *sym;
|
||
struct value *val = NULL;
|
||
|
||
if (! PyArg_ParseTuple (args, "i|O", &argno, &block_obj))
|
||
return NULL;
|
||
|
||
if (block_obj)
|
||
{
|
||
block = block_object_to_block (block_obj);
|
||
if (! block)
|
||
{
|
||
PyErr_SetString (PyExc_RuntimeError,
|
||
_("Second argument must be block."));
|
||
return NULL;
|
||
}
|
||
}
|
||
|
||
TRY
|
||
{
|
||
type = check_typedef (type);
|
||
if (TYPE_CODE (type) == TYPE_CODE_REF)
|
||
type = check_typedef (TYPE_TARGET_TYPE (type));
|
||
}
|
||
CATCH (except, RETURN_MASK_ALL)
|
||
{
|
||
GDB_PY_HANDLE_EXCEPTION (except);
|
||
}
|
||
END_CATCH
|
||
|
||
/* We might not have DW_TAG_template_*, so try to parse the type's
|
||
name. This is inefficient if we do not have a template type --
|
||
but that is going to wind up as an error anyhow. */
|
||
if (! TYPE_N_TEMPLATE_ARGUMENTS (type))
|
||
return typy_legacy_template_argument (type, block, argno);
|
||
|
||
if (argno >= TYPE_N_TEMPLATE_ARGUMENTS (type))
|
||
{
|
||
PyErr_Format (PyExc_RuntimeError, _("No argument %d in template."),
|
||
argno);
|
||
return NULL;
|
||
}
|
||
|
||
sym = TYPE_TEMPLATE_ARGUMENT (type, argno);
|
||
if (SYMBOL_CLASS (sym) == LOC_TYPEDEF)
|
||
return type_to_type_object (SYMBOL_TYPE (sym));
|
||
else if (SYMBOL_CLASS (sym) == LOC_OPTIMIZED_OUT)
|
||
{
|
||
PyErr_Format (PyExc_RuntimeError,
|
||
_("Template argument is optimized out"));
|
||
return NULL;
|
||
}
|
||
|
||
TRY
|
||
{
|
||
val = value_of_variable (sym, block);
|
||
}
|
||
CATCH (except, RETURN_MASK_ALL)
|
||
{
|
||
GDB_PY_HANDLE_EXCEPTION (except);
|
||
}
|
||
END_CATCH
|
||
|
||
return value_to_value_object (val);
|
||
}
|
||
|
||
static PyObject *
|
||
typy_str (PyObject *self)
|
||
{
|
||
string_file thetype;
|
||
PyObject *result;
|
||
|
||
TRY
|
||
{
|
||
LA_PRINT_TYPE (type_object_to_type (self), "", &thetype, -1, 0,
|
||
&type_print_raw_options);
|
||
}
|
||
CATCH (except, RETURN_MASK_ALL)
|
||
{
|
||
GDB_PY_HANDLE_EXCEPTION (except);
|
||
}
|
||
END_CATCH
|
||
|
||
return PyUnicode_Decode (thetype.c_str (), thetype.size (),
|
||
host_charset (), NULL);
|
||
}
|
||
|
||
/* Implement the richcompare method. */
|
||
|
||
static PyObject *
|
||
typy_richcompare (PyObject *self, PyObject *other, int op)
|
||
{
|
||
int result = Py_NE;
|
||
struct type *type1 = type_object_to_type (self);
|
||
struct type *type2 = type_object_to_type (other);
|
||
|
||
/* We can only compare ourselves to another Type object, and only
|
||
for equality or inequality. */
|
||
if (type2 == NULL || (op != Py_EQ && op != Py_NE))
|
||
{
|
||
Py_INCREF (Py_NotImplemented);
|
||
return Py_NotImplemented;
|
||
}
|
||
|
||
if (type1 == type2)
|
||
result = Py_EQ;
|
||
else
|
||
{
|
||
TRY
|
||
{
|
||
result = types_deeply_equal (type1, type2);
|
||
}
|
||
CATCH (except, RETURN_MASK_ALL)
|
||
{
|
||
/* If there is a GDB exception, a comparison is not capable
|
||
(or trusted), so exit. */
|
||
GDB_PY_HANDLE_EXCEPTION (except);
|
||
}
|
||
END_CATCH
|
||
}
|
||
|
||
if (op == (result ? Py_EQ : Py_NE))
|
||
Py_RETURN_TRUE;
|
||
Py_RETURN_FALSE;
|
||
}
|
||
|
||
|
||
|
||
static const struct objfile_data *typy_objfile_data_key;
|
||
|
||
static void
|
||
save_objfile_types (struct objfile *objfile, void *datum)
|
||
{
|
||
type_object *obj = (type_object *) datum;
|
||
htab_t copied_types;
|
||
|
||
if (!gdb_python_initialized)
|
||
return;
|
||
|
||
/* This prevents another thread from freeing the objects we're
|
||
operating on. */
|
||
gdbpy_enter enter_py (get_objfile_arch (objfile), current_language);
|
||
|
||
copied_types = create_copied_types_hash (objfile);
|
||
|
||
while (obj)
|
||
{
|
||
type_object *next = obj->next;
|
||
|
||
htab_empty (copied_types);
|
||
|
||
obj->type = copy_type_recursive (objfile, obj->type, copied_types);
|
||
|
||
obj->next = NULL;
|
||
obj->prev = NULL;
|
||
|
||
obj = next;
|
||
}
|
||
|
||
htab_delete (copied_types);
|
||
}
|
||
|
||
static void
|
||
set_type (type_object *obj, struct type *type)
|
||
{
|
||
obj->type = type;
|
||
obj->prev = NULL;
|
||
if (type && TYPE_OBJFILE (type))
|
||
{
|
||
struct objfile *objfile = TYPE_OBJFILE (type);
|
||
|
||
obj->next = ((struct pyty_type_object *)
|
||
objfile_data (objfile, typy_objfile_data_key));
|
||
if (obj->next)
|
||
obj->next->prev = obj;
|
||
set_objfile_data (objfile, typy_objfile_data_key, obj);
|
||
}
|
||
else
|
||
obj->next = NULL;
|
||
}
|
||
|
||
static void
|
||
typy_dealloc (PyObject *obj)
|
||
{
|
||
type_object *type = (type_object *) obj;
|
||
|
||
if (type->prev)
|
||
type->prev->next = type->next;
|
||
else if (type->type && TYPE_OBJFILE (type->type))
|
||
{
|
||
/* Must reset head of list. */
|
||
struct objfile *objfile = TYPE_OBJFILE (type->type);
|
||
|
||
if (objfile)
|
||
set_objfile_data (objfile, typy_objfile_data_key, type->next);
|
||
}
|
||
if (type->next)
|
||
type->next->prev = type->prev;
|
||
|
||
Py_TYPE (type)->tp_free (type);
|
||
}
|
||
|
||
/* Return number of fields ("length" of the field dictionary). */
|
||
|
||
static Py_ssize_t
|
||
typy_length (PyObject *self)
|
||
{
|
||
struct type *type = ((type_object *) self)->type;
|
||
|
||
type = typy_get_composite (type);
|
||
if (type == NULL)
|
||
return -1;
|
||
|
||
return TYPE_NFIELDS (type);
|
||
}
|
||
|
||
/* Implements boolean evaluation of gdb.Type. Handle this like other
|
||
Python objects that don't have a meaningful truth value -- all
|
||
values are true. */
|
||
|
||
static int
|
||
typy_nonzero (PyObject *self)
|
||
{
|
||
return 1;
|
||
}
|
||
|
||
/* Return optimized out value of this type. */
|
||
|
||
static PyObject *
|
||
typy_optimized_out (PyObject *self, PyObject *args)
|
||
{
|
||
struct type *type = ((type_object *) self)->type;
|
||
|
||
return value_to_value_object (allocate_optimized_out_value (type));
|
||
}
|
||
|
||
/* Return a gdb.Field object for the field named by the argument. */
|
||
|
||
static PyObject *
|
||
typy_getitem (PyObject *self, PyObject *key)
|
||
{
|
||
struct type *type = ((type_object *) self)->type;
|
||
int i;
|
||
|
||
gdb::unique_xmalloc_ptr<char> field = python_string_to_host_string (key);
|
||
if (field == NULL)
|
||
return NULL;
|
||
|
||
/* We want just fields of this type, not of base types, so instead of
|
||
using lookup_struct_elt_type, portions of that function are
|
||
copied here. */
|
||
|
||
type = typy_get_composite (type);
|
||
if (type == NULL)
|
||
return NULL;
|
||
|
||
for (i = 0; i < TYPE_NFIELDS (type); i++)
|
||
{
|
||
const char *t_field_name = TYPE_FIELD_NAME (type, i);
|
||
|
||
if (t_field_name && (strcmp_iw (t_field_name, field.get ()) == 0))
|
||
{
|
||
return convert_field (type, i);
|
||
}
|
||
}
|
||
PyErr_SetObject (PyExc_KeyError, key);
|
||
return NULL;
|
||
}
|
||
|
||
/* Implement the "get" method on the type object. This is the
|
||
same as getitem if the key is present, but returns the supplied
|
||
default value or None if the key is not found. */
|
||
|
||
static PyObject *
|
||
typy_get (PyObject *self, PyObject *args)
|
||
{
|
||
PyObject *key, *defval = Py_None, *result;
|
||
|
||
if (!PyArg_UnpackTuple (args, "get", 1, 2, &key, &defval))
|
||
return NULL;
|
||
|
||
result = typy_getitem (self, key);
|
||
if (result != NULL)
|
||
return result;
|
||
|
||
/* typy_getitem returned error status. If the exception is
|
||
KeyError, clear the exception status and return the defval
|
||
instead. Otherwise return the exception unchanged. */
|
||
if (!PyErr_ExceptionMatches (PyExc_KeyError))
|
||
return NULL;
|
||
|
||
PyErr_Clear ();
|
||
Py_INCREF (defval);
|
||
return defval;
|
||
}
|
||
|
||
/* Implement the "has_key" method on the type object. */
|
||
|
||
static PyObject *
|
||
typy_has_key (PyObject *self, PyObject *args)
|
||
{
|
||
struct type *type = ((type_object *) self)->type;
|
||
const char *field;
|
||
int i;
|
||
|
||
if (!PyArg_ParseTuple (args, "s", &field))
|
||
return NULL;
|
||
|
||
/* We want just fields of this type, not of base types, so instead of
|
||
using lookup_struct_elt_type, portions of that function are
|
||
copied here. */
|
||
|
||
type = typy_get_composite (type);
|
||
if (type == NULL)
|
||
return NULL;
|
||
|
||
for (i = 0; i < TYPE_NFIELDS (type); i++)
|
||
{
|
||
const char *t_field_name = TYPE_FIELD_NAME (type, i);
|
||
|
||
if (t_field_name && (strcmp_iw (t_field_name, field) == 0))
|
||
Py_RETURN_TRUE;
|
||
}
|
||
Py_RETURN_FALSE;
|
||
}
|
||
|
||
/* Make an iterator object to iterate over keys, values, or items. */
|
||
|
||
static PyObject *
|
||
typy_make_iter (PyObject *self, enum gdbpy_iter_kind kind)
|
||
{
|
||
typy_iterator_object *typy_iter_obj;
|
||
|
||
/* Check that "self" is a structure or union type. */
|
||
if (typy_get_composite (((type_object *) self)->type) == NULL)
|
||
return NULL;
|
||
|
||
typy_iter_obj = PyObject_New (typy_iterator_object,
|
||
&type_iterator_object_type);
|
||
if (typy_iter_obj == NULL)
|
||
return NULL;
|
||
|
||
typy_iter_obj->field = 0;
|
||
typy_iter_obj->kind = kind;
|
||
Py_INCREF (self);
|
||
typy_iter_obj->source = (type_object *) self;
|
||
|
||
return (PyObject *) typy_iter_obj;
|
||
}
|
||
|
||
/* iteritems() method. */
|
||
|
||
static PyObject *
|
||
typy_iteritems (PyObject *self, PyObject *args)
|
||
{
|
||
return typy_make_iter (self, iter_items);
|
||
}
|
||
|
||
/* iterkeys() method. */
|
||
|
||
static PyObject *
|
||
typy_iterkeys (PyObject *self, PyObject *args)
|
||
{
|
||
return typy_make_iter (self, iter_keys);
|
||
}
|
||
|
||
/* Iterating over the class, same as iterkeys except for the function
|
||
signature. */
|
||
|
||
static PyObject *
|
||
typy_iter (PyObject *self)
|
||
{
|
||
return typy_make_iter (self, iter_keys);
|
||
}
|
||
|
||
/* itervalues() method. */
|
||
|
||
static PyObject *
|
||
typy_itervalues (PyObject *self, PyObject *args)
|
||
{
|
||
return typy_make_iter (self, iter_values);
|
||
}
|
||
|
||
/* Return a reference to the type iterator. */
|
||
|
||
static PyObject *
|
||
typy_iterator_iter (PyObject *self)
|
||
{
|
||
Py_INCREF (self);
|
||
return self;
|
||
}
|
||
|
||
/* Return the next field in the iteration through the list of fields
|
||
of the type. */
|
||
|
||
static PyObject *
|
||
typy_iterator_iternext (PyObject *self)
|
||
{
|
||
typy_iterator_object *iter_obj = (typy_iterator_object *) self;
|
||
struct type *type = iter_obj->source->type;
|
||
PyObject *result;
|
||
|
||
if (iter_obj->field < TYPE_NFIELDS (type))
|
||
{
|
||
result = make_fielditem (type, iter_obj->field, iter_obj->kind);
|
||
if (result != NULL)
|
||
iter_obj->field++;
|
||
return result;
|
||
}
|
||
|
||
return NULL;
|
||
}
|
||
|
||
static void
|
||
typy_iterator_dealloc (PyObject *obj)
|
||
{
|
||
typy_iterator_object *iter_obj = (typy_iterator_object *) obj;
|
||
|
||
Py_DECREF (iter_obj->source);
|
||
}
|
||
|
||
/* Create a new Type referring to TYPE. */
|
||
PyObject *
|
||
type_to_type_object (struct type *type)
|
||
{
|
||
type_object *type_obj;
|
||
|
||
type_obj = PyObject_New (type_object, &type_object_type);
|
||
if (type_obj)
|
||
set_type (type_obj, type);
|
||
|
||
return (PyObject *) type_obj;
|
||
}
|
||
|
||
struct type *
|
||
type_object_to_type (PyObject *obj)
|
||
{
|
||
if (! PyObject_TypeCheck (obj, &type_object_type))
|
||
return NULL;
|
||
return ((type_object *) obj)->type;
|
||
}
|
||
|
||
|
||
|
||
/* Implementation of gdb.lookup_type. */
|
||
PyObject *
|
||
gdbpy_lookup_type (PyObject *self, PyObject *args, PyObject *kw)
|
||
{
|
||
static char *keywords[] = { "name", "block", NULL };
|
||
const char *type_name = NULL;
|
||
struct type *type = NULL;
|
||
PyObject *block_obj = NULL;
|
||
const struct block *block = NULL;
|
||
|
||
if (! PyArg_ParseTupleAndKeywords (args, kw, "s|O", keywords,
|
||
&type_name, &block_obj))
|
||
return NULL;
|
||
|
||
if (block_obj)
|
||
{
|
||
block = block_object_to_block (block_obj);
|
||
if (! block)
|
||
{
|
||
PyErr_SetString (PyExc_RuntimeError,
|
||
_("'block' argument must be a Block."));
|
||
return NULL;
|
||
}
|
||
}
|
||
|
||
type = typy_lookup_typename (type_name, block);
|
||
if (! type)
|
||
return NULL;
|
||
|
||
return (PyObject *) type_to_type_object (type);
|
||
}
|
||
|
||
int
|
||
gdbpy_initialize_types (void)
|
||
{
|
||
int i;
|
||
|
||
typy_objfile_data_key
|
||
= register_objfile_data_with_cleanup (save_objfile_types, NULL);
|
||
|
||
if (PyType_Ready (&type_object_type) < 0)
|
||
return -1;
|
||
if (PyType_Ready (&field_object_type) < 0)
|
||
return -1;
|
||
if (PyType_Ready (&type_iterator_object_type) < 0)
|
||
return -1;
|
||
|
||
for (i = 0; pyty_codes[i].name; ++i)
|
||
{
|
||
if (PyModule_AddIntConstant (gdb_module,
|
||
/* Cast needed for Python 2.4. */
|
||
(char *) pyty_codes[i].name,
|
||
pyty_codes[i].code) < 0)
|
||
return -1;
|
||
}
|
||
|
||
if (gdb_pymodule_addobject (gdb_module, "Type",
|
||
(PyObject *) &type_object_type) < 0)
|
||
return -1;
|
||
|
||
if (gdb_pymodule_addobject (gdb_module, "TypeIterator",
|
||
(PyObject *) &type_iterator_object_type) < 0)
|
||
return -1;
|
||
|
||
return gdb_pymodule_addobject (gdb_module, "Field",
|
||
(PyObject *) &field_object_type);
|
||
}
|
||
|
||
|
||
|
||
static PyGetSetDef type_object_getset[] =
|
||
{
|
||
{ "code", typy_get_code, NULL,
|
||
"The code for this type.", NULL },
|
||
{ "name", typy_get_name, NULL,
|
||
"The name for this type, or None.", NULL },
|
||
{ "sizeof", typy_get_sizeof, NULL,
|
||
"The size of this type, in bytes.", NULL },
|
||
{ "tag", typy_get_tag, NULL,
|
||
"The tag name for this type, or None.", NULL },
|
||
{ NULL }
|
||
};
|
||
|
||
static PyMethodDef type_object_methods[] =
|
||
{
|
||
{ "array", typy_array, METH_VARARGS,
|
||
"array ([LOW_BOUND,] HIGH_BOUND) -> Type\n\
|
||
Return a type which represents an array of objects of this type.\n\
|
||
The bounds of the array are [LOW_BOUND, HIGH_BOUND] inclusive.\n\
|
||
If LOW_BOUND is omitted, a value of zero is used." },
|
||
{ "vector", typy_vector, METH_VARARGS,
|
||
"vector ([LOW_BOUND,] HIGH_BOUND) -> Type\n\
|
||
Return a type which represents a vector of objects of this type.\n\
|
||
The bounds of the array are [LOW_BOUND, HIGH_BOUND] inclusive.\n\
|
||
If LOW_BOUND is omitted, a value of zero is used.\n\
|
||
Vectors differ from arrays in that if the current language has C-style\n\
|
||
arrays, vectors don't decay to a pointer to the first element.\n\
|
||
They are first class values." },
|
||
{ "__contains__", typy_has_key, METH_VARARGS,
|
||
"T.__contains__(k) -> True if T has a field named k, else False" },
|
||
{ "const", typy_const, METH_NOARGS,
|
||
"const () -> Type\n\
|
||
Return a const variant of this type." },
|
||
{ "optimized_out", typy_optimized_out, METH_NOARGS,
|
||
"optimized_out() -> Value\n\
|
||
Return optimized out value of this type." },
|
||
{ "fields", typy_fields, METH_NOARGS,
|
||
"fields () -> list\n\
|
||
Return a list holding all the fields of this type.\n\
|
||
Each field is a gdb.Field object." },
|
||
{ "get", typy_get, METH_VARARGS,
|
||
"T.get(k[,default]) -> returns field named k in T, if it exists;\n\
|
||
otherwise returns default, if supplied, or None if not." },
|
||
{ "has_key", typy_has_key, METH_VARARGS,
|
||
"T.has_key(k) -> True if T has a field named k, else False" },
|
||
{ "items", typy_items, METH_NOARGS,
|
||
"items () -> list\n\
|
||
Return a list of (name, field) pairs of this type.\n\
|
||
Each field is a gdb.Field object." },
|
||
{ "iteritems", typy_iteritems, METH_NOARGS,
|
||
"iteritems () -> an iterator over the (name, field)\n\
|
||
pairs of this type. Each field is a gdb.Field object." },
|
||
{ "iterkeys", typy_iterkeys, METH_NOARGS,
|
||
"iterkeys () -> an iterator over the field names of this type." },
|
||
{ "itervalues", typy_itervalues, METH_NOARGS,
|
||
"itervalues () -> an iterator over the fields of this type.\n\
|
||
Each field is a gdb.Field object." },
|
||
{ "keys", typy_field_names, METH_NOARGS,
|
||
"keys () -> list\n\
|
||
Return a list holding all the fields names of this type." },
|
||
{ "pointer", typy_pointer, METH_NOARGS,
|
||
"pointer () -> Type\n\
|
||
Return a type of pointer to this type." },
|
||
{ "range", typy_range, METH_NOARGS,
|
||
"range () -> tuple\n\
|
||
Return a tuple containing the lower and upper range for this type."},
|
||
{ "reference", typy_reference, METH_NOARGS,
|
||
"reference () -> Type\n\
|
||
Return a type of reference to this type." },
|
||
{ "strip_typedefs", typy_strip_typedefs, METH_NOARGS,
|
||
"strip_typedefs () -> Type\n\
|
||
Return a type formed by stripping this type of all typedefs."},
|
||
{ "target", typy_target, METH_NOARGS,
|
||
"target () -> Type\n\
|
||
Return the target type of this type." },
|
||
{ "template_argument", typy_template_argument, METH_VARARGS,
|
||
"template_argument (arg, [block]) -> Type\n\
|
||
Return the type of a template argument." },
|
||
{ "unqualified", typy_unqualified, METH_NOARGS,
|
||
"unqualified () -> Type\n\
|
||
Return a variant of this type without const or volatile attributes." },
|
||
{ "values", typy_values, METH_NOARGS,
|
||
"values () -> list\n\
|
||
Return a list holding all the fields of this type.\n\
|
||
Each field is a gdb.Field object." },
|
||
{ "volatile", typy_volatile, METH_NOARGS,
|
||
"volatile () -> Type\n\
|
||
Return a volatile variant of this type" },
|
||
{ NULL }
|
||
};
|
||
|
||
static PyNumberMethods type_object_as_number = {
|
||
NULL, /* nb_add */
|
||
NULL, /* nb_subtract */
|
||
NULL, /* nb_multiply */
|
||
#ifndef IS_PY3K
|
||
NULL, /* nb_divide */
|
||
#endif
|
||
NULL, /* nb_remainder */
|
||
NULL, /* nb_divmod */
|
||
NULL, /* nb_power */
|
||
NULL, /* nb_negative */
|
||
NULL, /* nb_positive */
|
||
NULL, /* nb_absolute */
|
||
typy_nonzero, /* nb_nonzero */
|
||
NULL, /* nb_invert */
|
||
NULL, /* nb_lshift */
|
||
NULL, /* nb_rshift */
|
||
NULL, /* nb_and */
|
||
NULL, /* nb_xor */
|
||
NULL, /* nb_or */
|
||
#ifdef IS_PY3K
|
||
NULL, /* nb_int */
|
||
NULL, /* reserved */
|
||
#else
|
||
NULL, /* nb_coerce */
|
||
NULL, /* nb_int */
|
||
NULL, /* nb_long */
|
||
#endif
|
||
NULL, /* nb_float */
|
||
#ifndef IS_PY3K
|
||
NULL, /* nb_oct */
|
||
NULL /* nb_hex */
|
||
#endif
|
||
};
|
||
|
||
static PyMappingMethods typy_mapping = {
|
||
typy_length,
|
||
typy_getitem,
|
||
NULL /* no "set" method */
|
||
};
|
||
|
||
PyTypeObject type_object_type =
|
||
{
|
||
PyVarObject_HEAD_INIT (NULL, 0)
|
||
"gdb.Type", /*tp_name*/
|
||
sizeof (type_object), /*tp_basicsize*/
|
||
0, /*tp_itemsize*/
|
||
typy_dealloc, /*tp_dealloc*/
|
||
0, /*tp_print*/
|
||
0, /*tp_getattr*/
|
||
0, /*tp_setattr*/
|
||
0, /*tp_compare*/
|
||
0, /*tp_repr*/
|
||
&type_object_as_number, /*tp_as_number*/
|
||
0, /*tp_as_sequence*/
|
||
&typy_mapping, /*tp_as_mapping*/
|
||
0, /*tp_hash */
|
||
0, /*tp_call*/
|
||
typy_str, /*tp_str*/
|
||
0, /*tp_getattro*/
|
||
0, /*tp_setattro*/
|
||
0, /*tp_as_buffer*/
|
||
Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_ITER, /*tp_flags*/
|
||
"GDB type object", /* tp_doc */
|
||
0, /* tp_traverse */
|
||
0, /* tp_clear */
|
||
typy_richcompare, /* tp_richcompare */
|
||
0, /* tp_weaklistoffset */
|
||
typy_iter, /* tp_iter */
|
||
0, /* tp_iternext */
|
||
type_object_methods, /* tp_methods */
|
||
0, /* tp_members */
|
||
type_object_getset, /* 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 */
|
||
0, /* tp_new */
|
||
};
|
||
|
||
static PyGetSetDef field_object_getset[] =
|
||
{
|
||
{ "__dict__", gdb_py_generic_dict, NULL,
|
||
"The __dict__ for this field.", &field_object_type },
|
||
{ NULL }
|
||
};
|
||
|
||
PyTypeObject field_object_type =
|
||
{
|
||
PyVarObject_HEAD_INIT (NULL, 0)
|
||
"gdb.Field", /*tp_name*/
|
||
sizeof (field_object), /*tp_basicsize*/
|
||
0, /*tp_itemsize*/
|
||
field_dealloc, /*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 | Py_TPFLAGS_HAVE_ITER, /*tp_flags*/
|
||
"GDB field object", /* tp_doc */
|
||
0, /* tp_traverse */
|
||
0, /* tp_clear */
|
||
0, /* tp_richcompare */
|
||
0, /* tp_weaklistoffset */
|
||
0, /* tp_iter */
|
||
0, /* tp_iternext */
|
||
0, /* tp_methods */
|
||
0, /* tp_members */
|
||
field_object_getset, /* tp_getset */
|
||
0, /* tp_base */
|
||
0, /* tp_dict */
|
||
0, /* tp_descr_get */
|
||
0, /* tp_descr_set */
|
||
offsetof (field_object, dict), /* tp_dictoffset */
|
||
0, /* tp_init */
|
||
0, /* tp_alloc */
|
||
0, /* tp_new */
|
||
};
|
||
|
||
PyTypeObject type_iterator_object_type = {
|
||
PyVarObject_HEAD_INIT (NULL, 0)
|
||
"gdb.TypeIterator", /*tp_name*/
|
||
sizeof (typy_iterator_object), /*tp_basicsize*/
|
||
0, /*tp_itemsize*/
|
||
typy_iterator_dealloc, /*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 | Py_TPFLAGS_HAVE_ITER, /*tp_flags*/
|
||
"GDB type iterator object", /*tp_doc */
|
||
0, /*tp_traverse */
|
||
0, /*tp_clear */
|
||
0, /*tp_richcompare */
|
||
0, /*tp_weaklistoffset */
|
||
typy_iterator_iter, /*tp_iter */
|
||
typy_iterator_iternext, /*tp_iternext */
|
||
0 /*tp_methods */
|
||
};
|