mirror of
https://github.com/python/cpython.git
synced 2026-04-20 02:40:59 +00:00
Treat the debug offset tables read from a target process as untrusted input and validate them before the unwinder uses any reported sizes or offsets. Add a shared validator in debug_offsets_validation.h and run it once when _Py_DebugOffsets is loaded and once when AsyncioDebug is loaded. The checks cover section sizes used for fixed local buffers and every offset that is later dereferenced against a local buffer or local object view. This keeps the bounds checks out of the sampling hot path while rejecting malformed tables up front.
483 lines
19 KiB
C
483 lines
19 KiB
C
#ifndef Py_REMOTE_DEBUG_OFFSETS_VALIDATION_H
|
|
#define Py_REMOTE_DEBUG_OFFSETS_VALIDATION_H
|
|
|
|
/*
|
|
* The remote debugging tables are read from the target process and must be
|
|
* treated as untrusted input. This header centralizes the one-time validation
|
|
* that runs immediately after those tables are read, before the unwinder uses
|
|
* any reported sizes or offsets to copy remote structs into fixed local
|
|
* buffers or to interpret those local copies.
|
|
*
|
|
* The key rule is simple: every offset that is later dereferenced against a
|
|
* local buffer or local object view must appear in one of the field lists
|
|
* below. Validation then checks two bounds for each field:
|
|
*
|
|
* 1. The field must fit within the section size reported by the target.
|
|
* 2. The same field must also fit within the local buffer or local layout the
|
|
* debugger will actually use.
|
|
*
|
|
* Sections that are copied into fixed local buffers also have their reported
|
|
* size checked against the corresponding local buffer size up front.
|
|
*
|
|
* This is intentionally front-loaded. Once validation succeeds, the hot path
|
|
* can keep using the raw offsets without adding per-sample bounds checks.
|
|
*
|
|
* Maintenance rule: if either exported table grows, the static_asserts below
|
|
* should yell at you. When that happens, update the matching field lists in
|
|
* this file in the same change. And if you add a new field that the unwinder
|
|
* is going to poke at later, put it in the right list here too, so nobody has
|
|
* to rediscover this the annoying way.
|
|
*/
|
|
#define FIELD_SIZE(type, member) sizeof(((type *)0)->member)
|
|
|
|
enum {
|
|
PY_REMOTE_DEBUG_OFFSETS_TOTAL_SIZE = 840,
|
|
PY_REMOTE_ASYNC_DEBUG_OFFSETS_TOTAL_SIZE = 104,
|
|
};
|
|
|
|
/*
|
|
* These asserts are the coordination tripwire for table growth. If either
|
|
* exported table changes size, update the validation lists below in the same
|
|
* change.
|
|
*/
|
|
static_assert(
|
|
sizeof(_Py_DebugOffsets) == PY_REMOTE_DEBUG_OFFSETS_TOTAL_SIZE,
|
|
"Update _remote_debugging validation for _Py_DebugOffsets");
|
|
static_assert(
|
|
sizeof(struct _Py_AsyncioModuleDebugOffsets) ==
|
|
PY_REMOTE_ASYNC_DEBUG_OFFSETS_TOTAL_SIZE,
|
|
"Update _remote_debugging validation for _Py_AsyncioModuleDebugOffsets");
|
|
|
|
/*
|
|
* This logic lives in a private header because it is shared by module.c and
|
|
* asyncio.c. Keep the helpers static inline so they stay local to those users
|
|
* without adding another compilation unit or exported symbols.
|
|
*/
|
|
static inline int
|
|
validate_section_size(const char *section_name, uint64_t size)
|
|
{
|
|
if (size == 0) {
|
|
PyErr_Format(
|
|
PyExc_RuntimeError,
|
|
"Invalid debug offsets: %s.size must be greater than zero",
|
|
section_name);
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static inline int
|
|
validate_read_size(const char *section_name, uint64_t size, size_t buffer_size)
|
|
{
|
|
if (validate_section_size(section_name, size) < 0) {
|
|
return -1;
|
|
}
|
|
if (size > buffer_size) {
|
|
PyErr_Format(
|
|
PyExc_RuntimeError,
|
|
"Invalid debug offsets: %s.size=%llu exceeds local buffer size %zu",
|
|
section_name,
|
|
(unsigned long long)size,
|
|
buffer_size);
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static inline int
|
|
validate_span(
|
|
const char *field_name,
|
|
uint64_t offset,
|
|
size_t width,
|
|
uint64_t limit,
|
|
const char *limit_name)
|
|
{
|
|
uint64_t span = (uint64_t)width;
|
|
if (span > limit || offset > limit - span) {
|
|
PyErr_Format(
|
|
PyExc_RuntimeError,
|
|
"Invalid debug offsets: %s=%llu with width %zu exceeds %s %llu",
|
|
field_name,
|
|
(unsigned long long)offset,
|
|
width,
|
|
limit_name,
|
|
(unsigned long long)limit);
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static inline int
|
|
validate_alignment(
|
|
const char *field_name,
|
|
uint64_t offset,
|
|
size_t alignment)
|
|
{
|
|
if (alignment > 1 && offset % alignment != 0) {
|
|
PyErr_Format(
|
|
PyExc_RuntimeError,
|
|
"Invalid debug offsets: %s=%llu is not aligned to %zu bytes",
|
|
field_name,
|
|
(unsigned long long)offset,
|
|
alignment);
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static inline int
|
|
validate_field(
|
|
const char *field_name,
|
|
uint64_t reported_size,
|
|
uint64_t offset,
|
|
size_t width,
|
|
size_t alignment,
|
|
size_t buffer_size)
|
|
{
|
|
if (validate_alignment(field_name, offset, alignment) < 0) {
|
|
return -1;
|
|
}
|
|
if (validate_span(field_name, offset, width, reported_size, "reported size") < 0) {
|
|
return -1;
|
|
}
|
|
return validate_span(field_name, offset, width, buffer_size, "local buffer size");
|
|
}
|
|
|
|
static inline int
|
|
validate_nested_field(
|
|
const char *field_name,
|
|
uint64_t reported_size,
|
|
uint64_t base_offset,
|
|
uint64_t nested_offset,
|
|
size_t width,
|
|
size_t alignment,
|
|
size_t buffer_size)
|
|
{
|
|
if (base_offset > UINT64_MAX - nested_offset) {
|
|
PyErr_Format(
|
|
PyExc_RuntimeError,
|
|
"Invalid debug offsets: %s overflows the offset calculation",
|
|
field_name);
|
|
return -1;
|
|
}
|
|
return validate_field(
|
|
field_name,
|
|
reported_size,
|
|
base_offset + nested_offset,
|
|
width,
|
|
alignment,
|
|
buffer_size);
|
|
}
|
|
|
|
static inline int
|
|
validate_fixed_field(
|
|
const char *field_name,
|
|
uint64_t offset,
|
|
size_t width,
|
|
size_t alignment,
|
|
size_t buffer_size)
|
|
{
|
|
if (validate_alignment(field_name, offset, alignment) < 0) {
|
|
return -1;
|
|
}
|
|
return validate_span(field_name, offset, width, buffer_size, "local buffer size");
|
|
}
|
|
|
|
#define PY_REMOTE_DEBUG_VALIDATE_SECTION(section) \
|
|
do { \
|
|
if (validate_section_size(#section, debug_offsets->section.size) < 0) { \
|
|
return -1; \
|
|
} \
|
|
} while (0)
|
|
|
|
#define PY_REMOTE_DEBUG_VALIDATE_READ_SECTION(section, buffer_size) \
|
|
do { \
|
|
if (validate_read_size(#section, debug_offsets->section.size, buffer_size) < 0) { \
|
|
return -1; \
|
|
} \
|
|
} while (0)
|
|
|
|
#define PY_REMOTE_DEBUG_VALIDATE_FIELD(section, field, field_size, field_alignment, buffer_size) \
|
|
do { \
|
|
if (validate_field( \
|
|
#section "." #field, \
|
|
debug_offsets->section.size, \
|
|
debug_offsets->section.field, \
|
|
field_size, \
|
|
field_alignment, \
|
|
buffer_size) < 0) { \
|
|
return -1; \
|
|
} \
|
|
} while (0)
|
|
|
|
#define PY_REMOTE_DEBUG_VALIDATE_NESTED_FIELD(section, base, nested_section, field, field_size, field_alignment, buffer_size) \
|
|
do { \
|
|
if (validate_nested_field( \
|
|
#section "." #base "." #field, \
|
|
debug_offsets->section.size, \
|
|
debug_offsets->section.base, \
|
|
debug_offsets->nested_section.field, \
|
|
field_size, \
|
|
field_alignment, \
|
|
buffer_size) < 0) { \
|
|
return -1; \
|
|
} \
|
|
} while (0)
|
|
|
|
#define PY_REMOTE_DEBUG_VALIDATE_FIXED_FIELD(section, field, field_size, field_alignment, buffer_size) \
|
|
do { \
|
|
if (validate_fixed_field( \
|
|
#section "." #field, \
|
|
debug_offsets->section.field, \
|
|
field_size, \
|
|
field_alignment, \
|
|
buffer_size) < 0) { \
|
|
return -1; \
|
|
} \
|
|
} while (0)
|
|
|
|
/*
|
|
* Each list below must include every offset that is later dereferenced against
|
|
* a local buffer or local object view. The validator checks that each field
|
|
* stays within both the remote table's reported section size and the local
|
|
* buffer size we use when reading that section. If a new dereferenced field is
|
|
* added to the offset tables, add it to the matching list here.
|
|
*
|
|
* Sections not listed here are present in the offset tables but not used by
|
|
* the unwinder, so no validation is needed for them.
|
|
*/
|
|
#define PY_REMOTE_DEBUG_RUNTIME_STATE_FIELDS(APPLY, buffer_size) \
|
|
APPLY(runtime_state, interpreters_head, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size)
|
|
|
|
#define PY_REMOTE_DEBUG_THREAD_STATE_FIELDS(APPLY, buffer_size) \
|
|
APPLY(thread_state, native_thread_id, sizeof(unsigned long), _Alignof(long), buffer_size); \
|
|
APPLY(thread_state, interp, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(thread_state, datastack_chunk, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(thread_state, status, FIELD_SIZE(PyThreadState, _status), _Alignof(unsigned int), buffer_size); \
|
|
APPLY(thread_state, holds_gil, sizeof(int), _Alignof(int), buffer_size); \
|
|
APPLY(thread_state, gil_requested, sizeof(int), _Alignof(int), buffer_size); \
|
|
APPLY(thread_state, current_exception, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(thread_state, thread_id, sizeof(unsigned long), _Alignof(long), buffer_size); \
|
|
APPLY(thread_state, next, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(thread_state, current_frame, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(thread_state, base_frame, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(thread_state, last_profiled_frame, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size)
|
|
|
|
#define PY_REMOTE_DEBUG_INTERPRETER_STATE_FIELDS(APPLY, buffer_size) \
|
|
APPLY(interpreter_state, id, sizeof(int64_t), _Alignof(int64_t), buffer_size); \
|
|
APPLY(interpreter_state, next, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(interpreter_state, threads_head, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(interpreter_state, threads_main, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(interpreter_state, gil_runtime_state_locked, sizeof(int), _Alignof(int), buffer_size); \
|
|
APPLY(interpreter_state, gil_runtime_state_holder, sizeof(PyThreadState *), _Alignof(PyThreadState *), buffer_size); \
|
|
APPLY(interpreter_state, code_object_generation, sizeof(uint64_t), _Alignof(uint64_t), buffer_size); \
|
|
APPLY(interpreter_state, tlbc_generation, sizeof(uint32_t), _Alignof(uint32_t), buffer_size)
|
|
|
|
#define PY_REMOTE_DEBUG_INTERPRETER_FRAME_FIELDS(APPLY, buffer_size) \
|
|
APPLY(interpreter_frame, previous, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(interpreter_frame, executable, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(interpreter_frame, instr_ptr, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(interpreter_frame, owner, sizeof(char), _Alignof(char), buffer_size); \
|
|
APPLY(interpreter_frame, stackpointer, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(interpreter_frame, tlbc_index, sizeof(int32_t), _Alignof(int32_t), buffer_size)
|
|
|
|
#define PY_REMOTE_DEBUG_CODE_OBJECT_FIELDS(APPLY, buffer_size) \
|
|
APPLY(code_object, qualname, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(code_object, filename, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(code_object, linetable, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(code_object, firstlineno, sizeof(int), _Alignof(int), buffer_size); \
|
|
APPLY(code_object, co_code_adaptive, sizeof(char), _Alignof(char), buffer_size); \
|
|
APPLY(code_object, co_tlbc, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size)
|
|
|
|
#define PY_REMOTE_DEBUG_SET_OBJECT_FIELDS(APPLY, buffer_size) \
|
|
APPLY(set_object, used, sizeof(Py_ssize_t), _Alignof(Py_ssize_t), buffer_size); \
|
|
APPLY(set_object, mask, sizeof(Py_ssize_t), _Alignof(Py_ssize_t), buffer_size); \
|
|
APPLY(set_object, table, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size)
|
|
|
|
#define PY_REMOTE_DEBUG_LONG_OBJECT_FIELDS(APPLY, buffer_size) \
|
|
APPLY(long_object, lv_tag, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(long_object, ob_digit, sizeof(digit), _Alignof(digit), buffer_size)
|
|
|
|
#define PY_REMOTE_DEBUG_BYTES_OBJECT_FIELDS(APPLY, buffer_size) \
|
|
APPLY(bytes_object, ob_size, sizeof(Py_ssize_t), _Alignof(Py_ssize_t), buffer_size); \
|
|
APPLY(bytes_object, ob_sval, sizeof(char), _Alignof(char), buffer_size)
|
|
|
|
#define PY_REMOTE_DEBUG_UNICODE_OBJECT_FIELDS(APPLY, buffer_size) \
|
|
APPLY(unicode_object, length, sizeof(Py_ssize_t), _Alignof(Py_ssize_t), buffer_size); \
|
|
APPLY(unicode_object, asciiobject_size, sizeof(char), _Alignof(char), buffer_size)
|
|
|
|
#define PY_REMOTE_DEBUG_GEN_OBJECT_FIELDS(APPLY, buffer_size) \
|
|
APPLY(gen_object, gi_frame_state, sizeof(int8_t), _Alignof(int8_t), buffer_size); \
|
|
APPLY(gen_object, gi_iframe, FIELD_SIZE(PyGenObject, gi_iframe), _Alignof(_PyInterpreterFrame), buffer_size)
|
|
|
|
#define PY_REMOTE_DEBUG_TASK_OBJECT_FIELDS(APPLY, buffer_size) \
|
|
APPLY(asyncio_task_object, task_name, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(asyncio_task_object, task_awaited_by, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(asyncio_task_object, task_is_task, sizeof(char), _Alignof(char), buffer_size); \
|
|
APPLY(asyncio_task_object, task_awaited_by_is_set, sizeof(char), _Alignof(char), buffer_size); \
|
|
APPLY(asyncio_task_object, task_coro, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(asyncio_task_object, task_node, SIZEOF_LLIST_NODE, _Alignof(struct llist_node), buffer_size)
|
|
|
|
#define PY_REMOTE_DEBUG_ASYNC_INTERPRETER_STATE_FIELDS(APPLY, buffer_size) \
|
|
APPLY(asyncio_interpreter_state, asyncio_tasks_head, SIZEOF_LLIST_NODE, _Alignof(struct llist_node), buffer_size)
|
|
|
|
#define PY_REMOTE_DEBUG_ASYNC_THREAD_STATE_FIELDS(APPLY, buffer_size) \
|
|
APPLY(asyncio_thread_state, asyncio_running_loop, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(asyncio_thread_state, asyncio_running_task, sizeof(uintptr_t), _Alignof(uintptr_t), buffer_size); \
|
|
APPLY(asyncio_thread_state, asyncio_tasks_head, SIZEOF_LLIST_NODE, _Alignof(struct llist_node), buffer_size)
|
|
|
|
/* Called once after reading _Py_DebugOffsets, before any hot-path use. */
|
|
static inline int
|
|
_PyRemoteDebug_ValidateDebugOffsetsLayout(struct _Py_DebugOffsets *debug_offsets)
|
|
{
|
|
/* Validate every field the unwinder dereferences against a local buffer
|
|
* or local object view. Fields used only for remote address arithmetic
|
|
* (e.g. runtime_state.interpreters_head) are also checked as a sanity
|
|
* bound on the offset value. */
|
|
PY_REMOTE_DEBUG_VALIDATE_SECTION(runtime_state);
|
|
PY_REMOTE_DEBUG_RUNTIME_STATE_FIELDS(
|
|
PY_REMOTE_DEBUG_VALIDATE_FIELD,
|
|
sizeof(_PyRuntimeState));
|
|
|
|
PY_REMOTE_DEBUG_VALIDATE_SECTION(interpreter_state);
|
|
PY_REMOTE_DEBUG_INTERPRETER_STATE_FIELDS(
|
|
PY_REMOTE_DEBUG_VALIDATE_FIELD,
|
|
INTERP_STATE_BUFFER_SIZE);
|
|
|
|
PY_REMOTE_DEBUG_VALIDATE_READ_SECTION(thread_state, SIZEOF_THREAD_STATE);
|
|
PY_REMOTE_DEBUG_THREAD_STATE_FIELDS(
|
|
PY_REMOTE_DEBUG_VALIDATE_FIELD,
|
|
SIZEOF_THREAD_STATE);
|
|
PY_REMOTE_DEBUG_VALIDATE_FIXED_FIELD(
|
|
err_stackitem,
|
|
exc_value,
|
|
sizeof(uintptr_t),
|
|
_Alignof(uintptr_t),
|
|
sizeof(_PyErr_StackItem));
|
|
PY_REMOTE_DEBUG_VALIDATE_NESTED_FIELD(
|
|
thread_state,
|
|
exc_state,
|
|
err_stackitem,
|
|
exc_value,
|
|
sizeof(uintptr_t),
|
|
_Alignof(uintptr_t),
|
|
SIZEOF_THREAD_STATE);
|
|
|
|
PY_REMOTE_DEBUG_VALIDATE_READ_SECTION(gc, SIZEOF_GC_RUNTIME_STATE);
|
|
PY_REMOTE_DEBUG_VALIDATE_FIELD(
|
|
gc,
|
|
frame,
|
|
sizeof(uintptr_t),
|
|
_Alignof(uintptr_t),
|
|
SIZEOF_GC_RUNTIME_STATE);
|
|
PY_REMOTE_DEBUG_VALIDATE_NESTED_FIELD(
|
|
interpreter_state,
|
|
gc,
|
|
gc,
|
|
frame,
|
|
sizeof(uintptr_t),
|
|
_Alignof(uintptr_t),
|
|
INTERP_STATE_BUFFER_SIZE);
|
|
|
|
PY_REMOTE_DEBUG_VALIDATE_SECTION(interpreter_frame);
|
|
PY_REMOTE_DEBUG_INTERPRETER_FRAME_FIELDS(
|
|
PY_REMOTE_DEBUG_VALIDATE_FIELD,
|
|
SIZEOF_INTERP_FRAME);
|
|
|
|
PY_REMOTE_DEBUG_VALIDATE_SECTION(code_object);
|
|
PY_REMOTE_DEBUG_CODE_OBJECT_FIELDS(
|
|
PY_REMOTE_DEBUG_VALIDATE_FIELD,
|
|
SIZEOF_CODE_OBJ);
|
|
|
|
PY_REMOTE_DEBUG_VALIDATE_SECTION(pyobject);
|
|
PY_REMOTE_DEBUG_VALIDATE_FIELD(
|
|
pyobject,
|
|
ob_type,
|
|
sizeof(uintptr_t),
|
|
_Alignof(uintptr_t),
|
|
SIZEOF_PYOBJECT);
|
|
|
|
PY_REMOTE_DEBUG_VALIDATE_SECTION(type_object);
|
|
PY_REMOTE_DEBUG_VALIDATE_FIELD(
|
|
type_object,
|
|
tp_flags,
|
|
sizeof(unsigned long),
|
|
_Alignof(unsigned long),
|
|
SIZEOF_TYPE_OBJ);
|
|
|
|
PY_REMOTE_DEBUG_VALIDATE_SECTION(set_object);
|
|
PY_REMOTE_DEBUG_SET_OBJECT_FIELDS(
|
|
PY_REMOTE_DEBUG_VALIDATE_FIELD,
|
|
SIZEOF_SET_OBJ);
|
|
|
|
PY_REMOTE_DEBUG_VALIDATE_READ_SECTION(long_object, SIZEOF_LONG_OBJ);
|
|
PY_REMOTE_DEBUG_LONG_OBJECT_FIELDS(
|
|
PY_REMOTE_DEBUG_VALIDATE_FIELD,
|
|
SIZEOF_LONG_OBJ);
|
|
|
|
PY_REMOTE_DEBUG_VALIDATE_SECTION(bytes_object);
|
|
PY_REMOTE_DEBUG_BYTES_OBJECT_FIELDS(
|
|
PY_REMOTE_DEBUG_VALIDATE_FIELD,
|
|
SIZEOF_BYTES_OBJ);
|
|
|
|
PY_REMOTE_DEBUG_VALIDATE_SECTION(unicode_object);
|
|
PY_REMOTE_DEBUG_UNICODE_OBJECT_FIELDS(
|
|
PY_REMOTE_DEBUG_VALIDATE_FIELD,
|
|
SIZEOF_UNICODE_OBJ);
|
|
|
|
PY_REMOTE_DEBUG_VALIDATE_SECTION(gen_object);
|
|
PY_REMOTE_DEBUG_GEN_OBJECT_FIELDS(
|
|
PY_REMOTE_DEBUG_VALIDATE_FIELD,
|
|
SIZEOF_GEN_OBJ);
|
|
|
|
PY_REMOTE_DEBUG_VALIDATE_FIXED_FIELD(
|
|
llist_node,
|
|
next,
|
|
sizeof(uintptr_t),
|
|
_Alignof(uintptr_t),
|
|
SIZEOF_LLIST_NODE);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* Called once when AsyncioDebug is loaded, before any task inspection uses it. */
|
|
static inline int
|
|
_PyRemoteDebug_ValidateAsyncDebugOffsets(
|
|
struct _Py_AsyncioModuleDebugOffsets *debug_offsets)
|
|
{
|
|
PY_REMOTE_DEBUG_VALIDATE_READ_SECTION(asyncio_task_object, SIZEOF_TASK_OBJ);
|
|
PY_REMOTE_DEBUG_TASK_OBJECT_FIELDS(
|
|
PY_REMOTE_DEBUG_VALIDATE_FIELD,
|
|
SIZEOF_TASK_OBJ);
|
|
PY_REMOTE_DEBUG_VALIDATE_SECTION(asyncio_interpreter_state);
|
|
PY_REMOTE_DEBUG_ASYNC_INTERPRETER_STATE_FIELDS(
|
|
PY_REMOTE_DEBUG_VALIDATE_FIELD,
|
|
sizeof(PyInterpreterState));
|
|
PY_REMOTE_DEBUG_VALIDATE_SECTION(asyncio_thread_state);
|
|
PY_REMOTE_DEBUG_ASYNC_THREAD_STATE_FIELDS(
|
|
PY_REMOTE_DEBUG_VALIDATE_FIELD,
|
|
sizeof(_PyThreadStateImpl));
|
|
return 0;
|
|
}
|
|
|
|
#undef PY_REMOTE_DEBUG_VALIDATE_SECTION
|
|
#undef PY_REMOTE_DEBUG_VALIDATE_READ_SECTION
|
|
#undef PY_REMOTE_DEBUG_VALIDATE_FIELD
|
|
#undef PY_REMOTE_DEBUG_VALIDATE_NESTED_FIELD
|
|
#undef PY_REMOTE_DEBUG_VALIDATE_FIXED_FIELD
|
|
#undef PY_REMOTE_DEBUG_RUNTIME_STATE_FIELDS
|
|
#undef PY_REMOTE_DEBUG_THREAD_STATE_FIELDS
|
|
#undef PY_REMOTE_DEBUG_INTERPRETER_STATE_FIELDS
|
|
#undef PY_REMOTE_DEBUG_INTERPRETER_FRAME_FIELDS
|
|
#undef PY_REMOTE_DEBUG_CODE_OBJECT_FIELDS
|
|
#undef PY_REMOTE_DEBUG_SET_OBJECT_FIELDS
|
|
#undef PY_REMOTE_DEBUG_LONG_OBJECT_FIELDS
|
|
#undef PY_REMOTE_DEBUG_BYTES_OBJECT_FIELDS
|
|
#undef PY_REMOTE_DEBUG_UNICODE_OBJECT_FIELDS
|
|
#undef PY_REMOTE_DEBUG_GEN_OBJECT_FIELDS
|
|
#undef PY_REMOTE_DEBUG_TASK_OBJECT_FIELDS
|
|
#undef PY_REMOTE_DEBUG_ASYNC_INTERPRETER_STATE_FIELDS
|
|
#undef PY_REMOTE_DEBUG_ASYNC_THREAD_STATE_FIELDS
|
|
#undef FIELD_SIZE
|
|
|
|
#endif
|