2011-07-11 Phil Muldoon <pmuldoon@redhat.com>
[deliverable/binutils-gdb.git] / gdb / python / py-inferior.c
1 /* Python interface to inferiors.
2
3 Copyright (C) 2009, 2010, 2011 Free Software Foundation, Inc.
4
5 This file is part of GDB.
6
7 This program is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 3 of the License, or
10 (at your option) any later version.
11
12 This program is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
16
17 You should have received a copy of the GNU General Public License
18 along with this program. If not, see <http://www.gnu.org/licenses/>. */
19
20 #include "defs.h"
21 #include "exceptions.h"
22 #include "gdbcore.h"
23 #include "gdbthread.h"
24 #include "inferior.h"
25 #include "observer.h"
26 #include "python-internal.h"
27 #include "arch-utils.h"
28 #include "language.h"
29 #include "gdb_signals.h"
30 #include "py-event.h"
31 #include "py-stopevent.h"
32
33 struct threadlist_entry {
34 thread_object *thread_obj;
35 struct threadlist_entry *next;
36 };
37
38 typedef struct
39 {
40 PyObject_HEAD
41
42 /* The inferior we represent. */
43 struct inferior *inferior;
44
45 /* thread_object instances under this inferior. This list owns a
46 reference to each object it contains. */
47 struct threadlist_entry *threads;
48
49 /* Number of threads in the list. */
50 int nthreads;
51 } inferior_object;
52
53 static PyTypeObject inferior_object_type;
54
55 static const struct inferior_data *infpy_inf_data_key;
56
57 typedef struct {
58 PyObject_HEAD
59 void *buffer;
60
61 /* These are kept just for mbpy_str. */
62 CORE_ADDR addr;
63 CORE_ADDR length;
64 } membuf_object;
65
66 static PyTypeObject membuf_object_type;
67
68 /* Require that INFERIOR be a valid inferior ID. */
69 #define INFPY_REQUIRE_VALID(Inferior) \
70 do { \
71 if (!Inferior->inferior) \
72 { \
73 PyErr_SetString (PyExc_RuntimeError, \
74 _("Inferior no longer exists.")); \
75 return NULL; \
76 } \
77 } while (0)
78
79 static void
80 python_on_normal_stop (struct bpstats *bs, int print_frame)
81 {
82 struct cleanup *cleanup;
83 enum target_signal stop_signal;
84
85 if (!find_thread_ptid (inferior_ptid))
86 return;
87
88 stop_signal = inferior_thread ()->suspend.stop_signal;
89
90 cleanup = ensure_python_env (get_current_arch (), current_language);
91
92 if (emit_stop_event (bs, stop_signal) < 0)
93 gdbpy_print_stack ();
94
95 do_cleanups (cleanup);
96 }
97
98 static void
99 python_on_resume (ptid_t ptid)
100 {
101 struct cleanup *cleanup;
102
103 cleanup = ensure_python_env (target_gdbarch, current_language);
104
105 if (emit_continue_event (ptid) < 0)
106 gdbpy_print_stack ();
107
108 do_cleanups (cleanup);
109 }
110
111 static void
112 python_inferior_exit (struct inferior *inf)
113 {
114 struct cleanup *cleanup;
115 const LONGEST *exit_code = NULL;
116
117 cleanup = ensure_python_env (target_gdbarch, current_language);
118
119 if (inf->has_exit_code)
120 exit_code = &inf->exit_code;
121
122 if (emit_exited_event (exit_code) < 0)
123 gdbpy_print_stack ();
124
125 do_cleanups (cleanup);
126 }
127
128 /* Return a reference to the Python object of type Inferior
129 representing INFERIOR. If the object has already been created,
130 return it and increment the reference count, otherwise, create it.
131 Return NULL on failure. */
132 PyObject *
133 inferior_to_inferior_object (struct inferior *inferior)
134 {
135 inferior_object *inf_obj;
136
137 inf_obj = inferior_data (inferior, infpy_inf_data_key);
138 if (!inf_obj)
139 {
140 struct cleanup *cleanup;
141 cleanup = ensure_python_env (python_gdbarch, python_language);
142
143 inf_obj = PyObject_New (inferior_object, &inferior_object_type);
144 if (!inf_obj)
145 {
146 do_cleanups (cleanup);
147 return NULL;
148 }
149
150 inf_obj->inferior = inferior;
151 inf_obj->threads = NULL;
152 inf_obj->nthreads = 0;
153
154 set_inferior_data (inferior, infpy_inf_data_key, inf_obj);
155
156 do_cleanups (cleanup);
157 }
158 else
159 Py_INCREF ((PyObject *)inf_obj);
160
161 return (PyObject *) inf_obj;
162 }
163
164 /* Finds the Python Inferior object for the given PID. Returns a
165 reference, or NULL if PID does not match any inferior object. */
166
167 PyObject *
168 find_inferior_object (int pid)
169 {
170 struct inflist_entry *p;
171 struct inferior *inf = find_inferior_pid (pid);
172
173 if (inf)
174 return inferior_to_inferior_object (inf);
175
176 return NULL;
177 }
178
179 thread_object *
180 find_thread_object (ptid_t ptid)
181 {
182 int pid;
183 struct threadlist_entry *thread;
184 PyObject *inf_obj;
185 thread_object *found = NULL;
186
187 pid = PIDGET (ptid);
188 if (pid == 0)
189 return NULL;
190
191 inf_obj = find_inferior_object (pid);
192
193 if (! inf_obj)
194 return NULL;
195
196 for (thread = ((inferior_object *)inf_obj)->threads; thread;
197 thread = thread->next)
198 if (ptid_equal (thread->thread_obj->thread->ptid, ptid))
199 {
200 found = thread->thread_obj;
201 break;
202 }
203
204 Py_DECREF (inf_obj);
205
206 if (found)
207 return found;
208
209 return NULL;
210 }
211
212 static void
213 add_thread_object (struct thread_info *tp)
214 {
215 struct cleanup *cleanup;
216 thread_object *thread_obj;
217 inferior_object *inf_obj;
218 struct threadlist_entry *entry;
219
220 cleanup = ensure_python_env (python_gdbarch, python_language);
221
222 thread_obj = create_thread_object (tp);
223 if (!thread_obj)
224 {
225 gdbpy_print_stack ();
226 do_cleanups (cleanup);
227 return;
228 }
229
230 inf_obj = (inferior_object *) thread_obj->inf_obj;
231
232 entry = xmalloc (sizeof (struct threadlist_entry));
233 entry->thread_obj = thread_obj;
234 entry->next = inf_obj->threads;
235
236 inf_obj->threads = entry;
237 inf_obj->nthreads++;
238
239 do_cleanups (cleanup);
240 }
241
242 static void
243 delete_thread_object (struct thread_info *tp, int ignore)
244 {
245 struct cleanup *cleanup;
246 inferior_object *inf_obj;
247 thread_object *thread_obj;
248 struct threadlist_entry **entry, *tmp;
249
250 inf_obj = (inferior_object *) find_inferior_object (PIDGET(tp->ptid));
251 if (!inf_obj)
252 return;
253
254 /* Find thread entry in its inferior's thread_list. */
255 for (entry = &inf_obj->threads; *entry != NULL; entry =
256 &(*entry)->next)
257 if ((*entry)->thread_obj->thread == tp)
258 break;
259
260 if (!*entry)
261 {
262 Py_DECREF (inf_obj);
263 return;
264 }
265
266 cleanup = ensure_python_env (python_gdbarch, python_language);
267
268 tmp = *entry;
269 tmp->thread_obj->thread = NULL;
270
271 *entry = (*entry)->next;
272 inf_obj->nthreads--;
273
274 Py_DECREF (tmp->thread_obj);
275 Py_DECREF (inf_obj);
276 xfree (tmp);
277
278 do_cleanups (cleanup);
279 }
280
281 static PyObject *
282 infpy_threads (PyObject *self, PyObject *args)
283 {
284 int i;
285 struct threadlist_entry *entry;
286 inferior_object *inf_obj = (inferior_object *) self;
287 PyObject *tuple;
288
289 INFPY_REQUIRE_VALID (inf_obj);
290
291 tuple = PyTuple_New (inf_obj->nthreads);
292 if (!tuple)
293 return NULL;
294
295 for (i = 0, entry = inf_obj->threads; i < inf_obj->nthreads;
296 i++, entry = entry->next)
297 {
298 Py_INCREF (entry->thread_obj);
299 PyTuple_SET_ITEM (tuple, i, (PyObject *) entry->thread_obj);
300 }
301
302 return tuple;
303 }
304
305 static PyObject *
306 infpy_get_num (PyObject *self, void *closure)
307 {
308 inferior_object *inf = (inferior_object *) self;
309
310 INFPY_REQUIRE_VALID (inf);
311
312 return PyLong_FromLong (inf->inferior->num);
313 }
314
315 static PyObject *
316 infpy_get_pid (PyObject *self, void *closure)
317 {
318 inferior_object *inf = (inferior_object *) self;
319
320 INFPY_REQUIRE_VALID (inf);
321
322 return PyLong_FromLong (inf->inferior->pid);
323 }
324
325 static PyObject *
326 infpy_get_was_attached (PyObject *self, void *closure)
327 {
328 inferior_object *inf = (inferior_object *) self;
329
330 INFPY_REQUIRE_VALID (inf);
331 if (inf->inferior->attach_flag)
332 Py_RETURN_TRUE;
333 Py_RETURN_FALSE;
334 }
335
336 static int
337 build_inferior_list (struct inferior *inf, void *arg)
338 {
339 PyObject *list = arg;
340 PyObject *inferior = inferior_to_inferior_object (inf);
341 int success = 0;
342
343 if (! inferior)
344 return 0;
345
346 success = PyList_Append (list, inferior);
347 Py_DECREF (inferior);
348
349 if (success)
350 return 1;
351
352 return 0;
353 }
354
355 /* Implementation of gdb.inferiors () -> (gdb.Inferior, ...).
356 Returns a tuple of all inferiors. */
357 PyObject *
358 gdbpy_inferiors (PyObject *unused, PyObject *unused2)
359 {
360 int i = 0;
361 PyObject *list, *inferior;
362 struct inferior *inf;
363
364 list = PyList_New (0);
365 if (!list)
366 return NULL;
367
368 if (iterate_over_inferiors (build_inferior_list, list))
369 {
370 Py_DECREF (list);
371 return NULL;
372 }
373
374 return PyList_AsTuple (list);
375 }
376
377 /* Membuf and memory manipulation. */
378
379 /* Implementation of gdb.read_memory (address, length).
380 Returns a Python buffer object with LENGTH bytes of the inferior's
381 memory at ADDRESS. Both arguments are integers. Returns NULL on error,
382 with a python exception set. */
383 static PyObject *
384 infpy_read_memory (PyObject *self, PyObject *args, PyObject *kw)
385 {
386 int error = 0;
387 CORE_ADDR addr, length;
388 void *buffer = NULL;
389 membuf_object *membuf_obj;
390 PyObject *addr_obj, *length_obj;
391 struct cleanup *cleanups;
392 volatile struct gdb_exception except;
393 static char *keywords[] = { "address", "length", NULL };
394
395 if (! PyArg_ParseTupleAndKeywords (args, kw, "OO", keywords,
396 &addr_obj, &length_obj))
397 return NULL;
398
399 cleanups = make_cleanup (null_cleanup, NULL);
400
401 TRY_CATCH (except, RETURN_MASK_ALL)
402 {
403 if (!get_addr_from_python (addr_obj, &addr)
404 || !get_addr_from_python (length_obj, &length))
405 {
406 error = 1;
407 break;
408 }
409
410 buffer = xmalloc (length);
411 make_cleanup (xfree, buffer);
412
413 read_memory (addr, buffer, length);
414 }
415 if (except.reason < 0)
416 {
417 do_cleanups (cleanups);
418 GDB_PY_HANDLE_EXCEPTION (except);
419 }
420
421 if (error)
422 {
423 do_cleanups (cleanups);
424 return NULL;
425 }
426
427 membuf_obj = PyObject_New (membuf_object, &membuf_object_type);
428 if (membuf_obj == NULL)
429 {
430 PyErr_SetString (PyExc_MemoryError,
431 _("Could not allocate memory buffer object."));
432 do_cleanups (cleanups);
433 return NULL;
434 }
435
436 discard_cleanups (cleanups);
437
438 membuf_obj->buffer = buffer;
439 membuf_obj->addr = addr;
440 membuf_obj->length = length;
441
442 return PyBuffer_FromReadWriteObject ((PyObject *) membuf_obj, 0,
443 Py_END_OF_BUFFER);
444 }
445
446 /* Implementation of gdb.write_memory (address, buffer [, length]).
447 Writes the contents of BUFFER (a Python object supporting the read
448 buffer protocol) at ADDRESS in the inferior's memory. Write LENGTH
449 bytes from BUFFER, or its entire contents if the argument is not
450 provided. The function returns nothing. Returns NULL on error, with
451 a python exception set. */
452 static PyObject *
453 infpy_write_memory (PyObject *self, PyObject *args, PyObject *kw)
454 {
455 Py_ssize_t buf_len;
456 int error = 0;
457 const char *buffer;
458 CORE_ADDR addr, length;
459 PyObject *addr_obj, *length_obj = NULL;
460 volatile struct gdb_exception except;
461 static char *keywords[] = { "address", "buffer", "length", NULL };
462
463
464 if (! PyArg_ParseTupleAndKeywords (args, kw, "Os#|O", keywords,
465 &addr_obj, &buffer, &buf_len,
466 &length_obj))
467 return NULL;
468
469 TRY_CATCH (except, RETURN_MASK_ALL)
470 {
471 if (!get_addr_from_python (addr_obj, &addr))
472 {
473 error = 1;
474 break;
475 }
476
477 if (!length_obj)
478 length = buf_len;
479 else if (!get_addr_from_python (length_obj, &length))
480 {
481 error = 1;
482 break;
483 }
484 write_memory (addr, buffer, length);
485 }
486 GDB_PY_HANDLE_EXCEPTION (except);
487
488 if (error)
489 return NULL;
490
491 Py_RETURN_NONE;
492 }
493
494 /* Destructor of Membuf objects. */
495 static void
496 mbpy_dealloc (PyObject *self)
497 {
498 xfree (((membuf_object *) self)->buffer);
499 self->ob_type->tp_free (self);
500 }
501
502 /* Return a description of the Membuf object. */
503 static PyObject *
504 mbpy_str (PyObject *self)
505 {
506 membuf_object *membuf_obj = (membuf_object *) self;
507
508 return PyString_FromFormat (_("Memory buffer for address %s, \
509 which is %s bytes long."),
510 paddress (python_gdbarch, membuf_obj->addr),
511 pulongest (membuf_obj->length));
512 }
513
514 static Py_ssize_t
515 get_read_buffer (PyObject *self, Py_ssize_t segment, void **ptrptr)
516 {
517 membuf_object *membuf_obj = (membuf_object *) self;
518
519 if (segment)
520 {
521 PyErr_SetString (PyExc_SystemError,
522 _("The memory buffer supports only one segment."));
523 return -1;
524 }
525
526 *ptrptr = membuf_obj->buffer;
527
528 return membuf_obj->length;
529 }
530
531 static Py_ssize_t
532 get_write_buffer (PyObject *self, Py_ssize_t segment, void **ptrptr)
533 {
534 return get_read_buffer (self, segment, ptrptr);
535 }
536
537 static Py_ssize_t
538 get_seg_count (PyObject *self, Py_ssize_t *lenp)
539 {
540 if (lenp)
541 *lenp = ((membuf_object *) self)->length;
542
543 return 1;
544 }
545
546 static Py_ssize_t
547 get_char_buffer (PyObject *self, Py_ssize_t segment, char **ptrptr)
548 {
549 void *ptr = NULL;
550 Py_ssize_t ret;
551
552 ret = get_read_buffer (self, segment, &ptr);
553 *ptrptr = (char *) ptr;
554
555 return ret;
556 }
557
558 /* Implementation of
559 gdb.search_memory (address, length, pattern). ADDRESS is the
560 address to start the search. LENGTH specifies the scope of the
561 search from ADDRESS. PATTERN is the pattern to search for (and
562 must be a Python object supporting the buffer protocol).
563 Returns a Python Long object holding the address where the pattern
564 was located, or if the pattern was not found, returns None. Returns NULL
565 on error, with a python exception set. */
566 static PyObject *
567 infpy_search_memory (PyObject *self, PyObject *args, PyObject *kw)
568 {
569 CORE_ADDR start_addr, length;
570 static char *keywords[] = { "address", "length", "pattern", NULL };
571 PyObject *pattern, *start_addr_obj, *length_obj;
572 volatile struct gdb_exception except;
573 Py_ssize_t pattern_size;
574 const void *buffer;
575 CORE_ADDR found_addr;
576 int found = 0;
577
578 if (! PyArg_ParseTupleAndKeywords (args, kw, "OOO", keywords,
579 &start_addr_obj, &length_obj,
580 &pattern))
581 return NULL;
582
583 if (get_addr_from_python (start_addr_obj, &start_addr)
584 && get_addr_from_python (length_obj, &length))
585 {
586 if (!length)
587 {
588 PyErr_SetString (PyExc_ValueError,
589 _("Search range is empty."));
590 return NULL;
591 }
592 /* Watch for overflows. */
593 else if (length > CORE_ADDR_MAX
594 || (start_addr + length - 1) < start_addr)
595 {
596 PyErr_SetString (PyExc_ValueError,
597 _("The search range is too large."));
598
599 return NULL;
600 }
601 }
602 else
603 return NULL;
604
605 if (!PyObject_CheckReadBuffer (pattern))
606 {
607 PyErr_SetString (PyExc_RuntimeError,
608 _("The pattern is not a Python buffer."));
609
610 return NULL;
611 }
612
613 if (PyObject_AsReadBuffer (pattern, &buffer, &pattern_size) == -1)
614 return NULL;
615
616 TRY_CATCH (except, RETURN_MASK_ALL)
617 {
618 found = target_search_memory (start_addr, length,
619 buffer, pattern_size,
620 &found_addr);
621 }
622 GDB_PY_HANDLE_EXCEPTION (except);
623
624 if (found)
625 return PyLong_FromLong (found_addr);
626 else
627 Py_RETURN_NONE;
628 }
629
630 /* Implementation of gdb.Inferior.is_valid (self) -> Boolean.
631 Returns True if this inferior object still exists in GDB. */
632
633 static PyObject *
634 infpy_is_valid (PyObject *self, PyObject *args)
635 {
636 inferior_object *inf = (inferior_object *) self;
637
638 if (! inf->inferior)
639 Py_RETURN_FALSE;
640
641 Py_RETURN_TRUE;
642 }
643
644 static void
645 infpy_dealloc (PyObject *obj)
646 {
647 inferior_object *inf_obj = (inferior_object *) obj;
648 struct inferior *inf = inf_obj->inferior;
649
650 if (! inf)
651 return;
652
653 set_inferior_data (inf, infpy_inf_data_key, NULL);
654 }
655
656 /* Clear the INFERIOR pointer in an Inferior object and clear the
657 thread list. */
658 static void
659 py_free_inferior (struct inferior *inf, void *datum)
660 {
661
662 struct cleanup *cleanup;
663 inferior_object *inf_obj = datum;
664 struct threadlist_entry *th_entry, *th_tmp;
665
666 cleanup = ensure_python_env (python_gdbarch, python_language);
667
668 inf_obj->inferior = NULL;
669
670 /* Deallocate threads list. */
671 for (th_entry = inf_obj->threads; th_entry != NULL;)
672 {
673 Py_DECREF (th_entry->thread_obj);
674
675 th_tmp = th_entry;
676 th_entry = th_entry->next;
677 xfree (th_tmp);
678 }
679
680 inf_obj->nthreads = 0;
681
682 Py_DECREF ((PyObject *) inf_obj);
683 do_cleanups (cleanup);
684 }
685
686 void
687 gdbpy_initialize_inferior (void)
688 {
689 if (PyType_Ready (&inferior_object_type) < 0)
690 return;
691
692 Py_INCREF (&inferior_object_type);
693 PyModule_AddObject (gdb_module, "Inferior",
694 (PyObject *) &inferior_object_type);
695
696 infpy_inf_data_key =
697 register_inferior_data_with_cleanup (py_free_inferior);
698
699 observer_attach_new_thread (add_thread_object);
700 observer_attach_thread_exit (delete_thread_object);
701 observer_attach_normal_stop (python_on_normal_stop);
702 observer_attach_target_resumed (python_on_resume);
703 observer_attach_inferior_exit (python_inferior_exit);
704
705 if (PyType_Ready (&membuf_object_type) < 0)
706 return;
707
708 Py_INCREF (&membuf_object_type);
709 PyModule_AddObject (gdb_module, "Membuf", (PyObject *)
710 &membuf_object_type);
711 }
712
713 static PyGetSetDef inferior_object_getset[] =
714 {
715 { "num", infpy_get_num, NULL, "ID of inferior, as assigned by GDB.", NULL },
716 { "pid", infpy_get_pid, NULL, "PID of inferior, as assigned by the OS.",
717 NULL },
718 { "was_attached", infpy_get_was_attached, NULL,
719 "True if the inferior was created using 'attach'.", NULL },
720 { NULL }
721 };
722
723 static PyMethodDef inferior_object_methods[] =
724 {
725 { "is_valid", infpy_is_valid, METH_NOARGS,
726 "is_valid () -> Boolean.\n\
727 Return true if this inferior is valid, false if not." },
728 { "threads", infpy_threads, METH_NOARGS,
729 "Return all the threads of this inferior." },
730 { "read_memory", (PyCFunction) infpy_read_memory,
731 METH_VARARGS | METH_KEYWORDS,
732 "read_memory (address, length) -> buffer\n\
733 Return a buffer object for reading from the inferior's memory." },
734 { "write_memory", (PyCFunction) infpy_write_memory,
735 METH_VARARGS | METH_KEYWORDS,
736 "write_memory (address, buffer [, length])\n\
737 Write the given buffer object to the inferior's memory." },
738 { "search_memory", (PyCFunction) infpy_search_memory,
739 METH_VARARGS | METH_KEYWORDS,
740 "search_memory (address, length, pattern) -> long\n\
741 Return a long with the address of a match, or None." },
742 { NULL }
743 };
744
745 static PyTypeObject inferior_object_type =
746 {
747 PyObject_HEAD_INIT (NULL)
748 0, /* ob_size */
749 "gdb.Inferior", /* tp_name */
750 sizeof (inferior_object), /* tp_basicsize */
751 0, /* tp_itemsize */
752 infpy_dealloc, /* tp_dealloc */
753 0, /* tp_print */
754 0, /* tp_getattr */
755 0, /* tp_setattr */
756 0, /* tp_compare */
757 0, /* tp_repr */
758 0, /* tp_as_number */
759 0, /* tp_as_sequence */
760 0, /* tp_as_mapping */
761 0, /* tp_hash */
762 0, /* tp_call */
763 0, /* tp_str */
764 0, /* tp_getattro */
765 0, /* tp_setattro */
766 0, /* tp_as_buffer */
767 Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_ITER, /* tp_flags */
768 "GDB inferior object", /* tp_doc */
769 0, /* tp_traverse */
770 0, /* tp_clear */
771 0, /* tp_richcompare */
772 0, /* tp_weaklistoffset */
773 0, /* tp_iter */
774 0, /* tp_iternext */
775 inferior_object_methods, /* tp_methods */
776 0, /* tp_members */
777 inferior_object_getset, /* tp_getset */
778 0, /* tp_base */
779 0, /* tp_dict */
780 0, /* tp_descr_get */
781 0, /* tp_descr_set */
782 0, /* tp_dictoffset */
783 0, /* tp_init */
784 0 /* tp_alloc */
785 };
786
787 /* Python doesn't provide a decent way to get compatibility here. */
788 #if HAVE_LIBPYTHON2_4
789 #define CHARBUFFERPROC_NAME getcharbufferproc
790 #else
791 #define CHARBUFFERPROC_NAME charbufferproc
792 #endif
793
794 static PyBufferProcs buffer_procs = {
795 get_read_buffer,
796 get_write_buffer,
797 get_seg_count,
798 /* The cast here works around a difference between Python 2.4 and
799 Python 2.5. */
800 (CHARBUFFERPROC_NAME) get_char_buffer
801 };
802
803 static PyTypeObject membuf_object_type = {
804 PyObject_HEAD_INIT (NULL)
805 0, /*ob_size*/
806 "gdb.Membuf", /*tp_name*/
807 sizeof (membuf_object), /*tp_basicsize*/
808 0, /*tp_itemsize*/
809 mbpy_dealloc, /*tp_dealloc*/
810 0, /*tp_print*/
811 0, /*tp_getattr*/
812 0, /*tp_setattr*/
813 0, /*tp_compare*/
814 0, /*tp_repr*/
815 0, /*tp_as_number*/
816 0, /*tp_as_sequence*/
817 0, /*tp_as_mapping*/
818 0, /*tp_hash */
819 0, /*tp_call*/
820 mbpy_str, /*tp_str*/
821 0, /*tp_getattro*/
822 0, /*tp_setattro*/
823 &buffer_procs, /*tp_as_buffer*/
824 Py_TPFLAGS_DEFAULT, /*tp_flags*/
825 "GDB memory buffer object", /*tp_doc*/
826 0, /* tp_traverse */
827 0, /* tp_clear */
828 0, /* tp_richcompare */
829 0, /* tp_weaklistoffset */
830 0, /* tp_iter */
831 0, /* tp_iternext */
832 0, /* tp_methods */
833 0, /* tp_members */
834 0, /* tp_getset */
835 0, /* tp_base */
836 0, /* tp_dict */
837 0, /* tp_descr_get */
838 0, /* tp_descr_set */
839 0, /* tp_dictoffset */
840 0, /* tp_init */
841 0, /* tp_alloc */
842 PyType_GenericNew /* tp_new */
843 };
This page took 0.047209 seconds and 5 git commands to generate.