Coverage Report

Created: 2026-05-16 06:46

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/cpython/Python/instrumentation.c
Line
Count
Source
1
#include "Python.h"
2
#include "pycore_bitutils.h"      // _Py_popcount32()
3
#include "pycore_call.h"          // _PyObject_VectorcallTstate()
4
#include "pycore_ceval.h"         // _PY_EVAL_EVENTS_BITS
5
#include "pycore_code.h"          // _PyCode_Clear_Executors()
6
#include "pycore_critical_section.h" // _Py_CRITICAL_SECTION_ASSERT_OBJECT_LOCKED()
7
#include "pycore_frame.h"         // PyFrameObject
8
#include "pycore_interpframe.h"   // _PyFrame_GetBytecode()
9
#include "pycore_long.h"          // _PyLong_GetZero()
10
#include "pycore_modsupport.h"    // _PyModule_CreateInitialized()
11
#include "pycore_namespace.h"     // _PyNamespace_New()
12
#include "pycore_opcode_metadata.h" // IS_VALID_OPCODE()
13
#include "pycore_opcode_utils.h"  // IS_CONDITIONAL_JUMP_OPCODE()
14
#include "pycore_optimizer.h"     // _PyExecutorObject
15
#include "pycore_pyatomic_ft_wrappers.h" // FT_ATOMIC_STORE_UINTPTR_RELEASE()
16
#include "pycore_pystate.h"       // _PyInterpreterState_GET()
17
#include "pycore_runtime_structs.h" // _PyCoMonitoringData
18
#include "pycore_tuple.h"         // _PyTuple_FromArraySteal()
19
20
#include "opcode_ids.h"
21
22
23
/* Uncomment this to dump debugging output when assertions fail */
24
// #define INSTRUMENT_DEBUG 1
25
26
#if defined(Py_DEBUG) && defined(Py_GIL_DISABLED)
27
28
#define ASSERT_WORLD_STOPPED_OR_LOCKED(obj)                         \
29
    if (!_PyInterpreterState_GET()->stoptheworld.world_stopped) {   \
30
        _Py_CRITICAL_SECTION_ASSERT_OBJECT_LOCKED(obj);             \
31
    }
32
#define ASSERT_WORLD_STOPPED() assert(_PyInterpreterState_GET()->stoptheworld.world_stopped);
33
34
#else
35
36
#define ASSERT_WORLD_STOPPED_OR_LOCKED(obj)
37
#define ASSERT_WORLD_STOPPED()
38
39
#endif
40
41
#ifdef Py_GIL_DISABLED
42
43
#define LOCK_CODE(code)                                             \
44
    assert(!_PyInterpreterState_GET()->stoptheworld.world_stopped); \
45
    Py_BEGIN_CRITICAL_SECTION(code)
46
47
#define UNLOCK_CODE()   Py_END_CRITICAL_SECTION()
48
49
#define MODIFY_BYTECODE(code, func, ...)                       \
50
    do {                                                       \
51
        PyCodeObject *co = (code);                             \
52
        for (Py_ssize_t i = 0; i < code->co_tlbc->size; i++) { \
53
            char *bc = co->co_tlbc->entries[i];                \
54
            if (bc == NULL) {                                  \
55
                continue;                                      \
56
            }                                                  \
57
            (func)(code, (_Py_CODEUNIT *)bc, __VA_ARGS__);           \
58
        }                                                      \
59
    } while (0)
60
61
#else
62
63
#define LOCK_CODE(code)
64
#define UNLOCK_CODE()
65
#define MODIFY_BYTECODE(code, func, ...) \
66
0
    (func)(code, _PyCode_CODE(code), __VA_ARGS__)
67
68
#endif
69
70
PyObject _PyInstrumentation_DISABLE = _PyObject_HEAD_INIT(&PyBaseObject_Type);
71
72
PyObject _PyInstrumentation_MISSING = _PyObject_HEAD_INIT(&PyBaseObject_Type);
73
74
static const int8_t EVENT_FOR_OPCODE[256] = {
75
    [RETURN_VALUE] = PY_MONITORING_EVENT_PY_RETURN,
76
    [INSTRUMENTED_RETURN_VALUE] = PY_MONITORING_EVENT_PY_RETURN,
77
    [CALL] = PY_MONITORING_EVENT_CALL,
78
    [INSTRUMENTED_CALL] = PY_MONITORING_EVENT_CALL,
79
    [CALL_KW] = PY_MONITORING_EVENT_CALL,
80
    [INSTRUMENTED_CALL_KW] = PY_MONITORING_EVENT_CALL,
81
    [CALL_FUNCTION_EX] = PY_MONITORING_EVENT_CALL,
82
    [INSTRUMENTED_CALL_FUNCTION_EX] = PY_MONITORING_EVENT_CALL,
83
    [LOAD_SUPER_ATTR] = PY_MONITORING_EVENT_CALL,
84
    [INSTRUMENTED_LOAD_SUPER_ATTR] = PY_MONITORING_EVENT_CALL,
85
    [RESUME] = -1,
86
    [YIELD_VALUE] = PY_MONITORING_EVENT_PY_YIELD,
87
    [INSTRUMENTED_YIELD_VALUE] = PY_MONITORING_EVENT_PY_YIELD,
88
    [JUMP_FORWARD] = PY_MONITORING_EVENT_JUMP,
89
    [JUMP_BACKWARD] = PY_MONITORING_EVENT_JUMP,
90
    [POP_JUMP_IF_FALSE] = PY_MONITORING_EVENT_BRANCH_RIGHT,
91
    [POP_JUMP_IF_TRUE] = PY_MONITORING_EVENT_BRANCH_RIGHT,
92
    [POP_JUMP_IF_NONE] = PY_MONITORING_EVENT_BRANCH_RIGHT,
93
    [POP_JUMP_IF_NOT_NONE] = PY_MONITORING_EVENT_BRANCH_RIGHT,
94
    [INSTRUMENTED_JUMP_FORWARD] = PY_MONITORING_EVENT_JUMP,
95
    [INSTRUMENTED_JUMP_BACKWARD] = PY_MONITORING_EVENT_JUMP,
96
    [INSTRUMENTED_POP_JUMP_IF_FALSE] = PY_MONITORING_EVENT_BRANCH_RIGHT,
97
    [INSTRUMENTED_POP_JUMP_IF_TRUE] = PY_MONITORING_EVENT_BRANCH_RIGHT,
98
    [INSTRUMENTED_POP_JUMP_IF_NONE] = PY_MONITORING_EVENT_BRANCH_RIGHT,
99
    [INSTRUMENTED_POP_JUMP_IF_NOT_NONE] = PY_MONITORING_EVENT_BRANCH_RIGHT,
100
    [FOR_ITER] = PY_MONITORING_EVENT_BRANCH_LEFT,
101
    [INSTRUMENTED_FOR_ITER] = PY_MONITORING_EVENT_BRANCH_LEFT,
102
    [POP_ITER] = PY_MONITORING_EVENT_BRANCH_RIGHT,
103
    [INSTRUMENTED_POP_ITER] = PY_MONITORING_EVENT_BRANCH_RIGHT,
104
    [END_FOR] = PY_MONITORING_EVENT_STOP_ITERATION,
105
    [INSTRUMENTED_END_FOR] = PY_MONITORING_EVENT_STOP_ITERATION,
106
    [END_SEND] = PY_MONITORING_EVENT_STOP_ITERATION,
107
    [INSTRUMENTED_END_SEND] = PY_MONITORING_EVENT_STOP_ITERATION,
108
    [NOT_TAKEN] = PY_MONITORING_EVENT_BRANCH_LEFT,
109
    [INSTRUMENTED_NOT_TAKEN] = PY_MONITORING_EVENT_BRANCH_LEFT,
110
    [END_ASYNC_FOR] = PY_MONITORING_EVENT_BRANCH_RIGHT,
111
};
112
113
static const uint8_t DE_INSTRUMENT[256] = {
114
    [INSTRUMENTED_RESUME] = RESUME,
115
    [INSTRUMENTED_RETURN_VALUE] = RETURN_VALUE,
116
    [INSTRUMENTED_CALL] = CALL,
117
    [INSTRUMENTED_CALL_KW] = CALL_KW,
118
    [INSTRUMENTED_CALL_FUNCTION_EX] = CALL_FUNCTION_EX,
119
    [INSTRUMENTED_YIELD_VALUE] = YIELD_VALUE,
120
    [INSTRUMENTED_JUMP_FORWARD] = JUMP_FORWARD,
121
    [INSTRUMENTED_JUMP_BACKWARD] = JUMP_BACKWARD,
122
    [INSTRUMENTED_POP_JUMP_IF_FALSE] = POP_JUMP_IF_FALSE,
123
    [INSTRUMENTED_POP_JUMP_IF_TRUE] = POP_JUMP_IF_TRUE,
124
    [INSTRUMENTED_POP_JUMP_IF_NONE] = POP_JUMP_IF_NONE,
125
    [INSTRUMENTED_POP_JUMP_IF_NOT_NONE] = POP_JUMP_IF_NOT_NONE,
126
    [INSTRUMENTED_FOR_ITER] = FOR_ITER,
127
    [INSTRUMENTED_POP_ITER] = POP_ITER,
128
    [INSTRUMENTED_END_FOR] = END_FOR,
129
    [INSTRUMENTED_END_SEND] = END_SEND,
130
    [INSTRUMENTED_LOAD_SUPER_ATTR] = LOAD_SUPER_ATTR,
131
    [INSTRUMENTED_NOT_TAKEN] = NOT_TAKEN,
132
    [INSTRUMENTED_END_ASYNC_FOR] = END_ASYNC_FOR,
133
};
134
135
static const uint8_t INSTRUMENTED_OPCODES[256] = {
136
    [RETURN_VALUE] = INSTRUMENTED_RETURN_VALUE,
137
    [INSTRUMENTED_RETURN_VALUE] = INSTRUMENTED_RETURN_VALUE,
138
    [CALL] = INSTRUMENTED_CALL,
139
    [INSTRUMENTED_CALL] = INSTRUMENTED_CALL,
140
    [CALL_KW] = INSTRUMENTED_CALL_KW,
141
    [INSTRUMENTED_CALL_KW] = INSTRUMENTED_CALL_KW,
142
    [CALL_FUNCTION_EX] = INSTRUMENTED_CALL_FUNCTION_EX,
143
    [INSTRUMENTED_CALL_FUNCTION_EX] = INSTRUMENTED_CALL_FUNCTION_EX,
144
    [YIELD_VALUE] = INSTRUMENTED_YIELD_VALUE,
145
    [INSTRUMENTED_YIELD_VALUE] = INSTRUMENTED_YIELD_VALUE,
146
    [RESUME] = INSTRUMENTED_RESUME,
147
    [INSTRUMENTED_RESUME] = INSTRUMENTED_RESUME,
148
    [JUMP_FORWARD] = INSTRUMENTED_JUMP_FORWARD,
149
    [INSTRUMENTED_JUMP_FORWARD] = INSTRUMENTED_JUMP_FORWARD,
150
    [JUMP_BACKWARD] = INSTRUMENTED_JUMP_BACKWARD,
151
    [INSTRUMENTED_JUMP_BACKWARD] = INSTRUMENTED_JUMP_BACKWARD,
152
    [POP_JUMP_IF_FALSE] = INSTRUMENTED_POP_JUMP_IF_FALSE,
153
    [INSTRUMENTED_POP_JUMP_IF_FALSE] = INSTRUMENTED_POP_JUMP_IF_FALSE,
154
    [POP_JUMP_IF_TRUE] = INSTRUMENTED_POP_JUMP_IF_TRUE,
155
    [INSTRUMENTED_POP_JUMP_IF_TRUE] = INSTRUMENTED_POP_JUMP_IF_TRUE,
156
    [POP_JUMP_IF_NONE] = INSTRUMENTED_POP_JUMP_IF_NONE,
157
    [INSTRUMENTED_POP_JUMP_IF_NONE] = INSTRUMENTED_POP_JUMP_IF_NONE,
158
    [POP_JUMP_IF_NOT_NONE] = INSTRUMENTED_POP_JUMP_IF_NOT_NONE,
159
    [INSTRUMENTED_POP_JUMP_IF_NOT_NONE] = INSTRUMENTED_POP_JUMP_IF_NOT_NONE,
160
    [END_FOR] = INSTRUMENTED_END_FOR,
161
    [INSTRUMENTED_END_FOR] = INSTRUMENTED_END_FOR,
162
    [END_SEND] = INSTRUMENTED_END_SEND,
163
    [INSTRUMENTED_END_SEND] = INSTRUMENTED_END_SEND,
164
    [FOR_ITER] = INSTRUMENTED_FOR_ITER,
165
    [INSTRUMENTED_FOR_ITER] = INSTRUMENTED_FOR_ITER,
166
    [POP_ITER] = INSTRUMENTED_POP_ITER,
167
    [INSTRUMENTED_POP_ITER] = INSTRUMENTED_POP_ITER,
168
    [LOAD_SUPER_ATTR] = INSTRUMENTED_LOAD_SUPER_ATTR,
169
    [INSTRUMENTED_LOAD_SUPER_ATTR] = INSTRUMENTED_LOAD_SUPER_ATTR,
170
    [NOT_TAKEN] = INSTRUMENTED_NOT_TAKEN,
171
    [INSTRUMENTED_NOT_TAKEN] = INSTRUMENTED_NOT_TAKEN,
172
    [END_ASYNC_FOR] = INSTRUMENTED_END_ASYNC_FOR,
173
    [INSTRUMENTED_END_ASYNC_FOR] = INSTRUMENTED_END_ASYNC_FOR,
174
175
    [INSTRUMENTED_LINE] = INSTRUMENTED_LINE,
176
    [INSTRUMENTED_INSTRUCTION] = INSTRUMENTED_INSTRUCTION,
177
};
178
179
static inline bool
180
opcode_has_event(int opcode)
181
0
{
182
0
    return (
183
0
        opcode != INSTRUMENTED_LINE &&
184
0
        INSTRUMENTED_OPCODES[opcode] > 0
185
0
    );
186
0
}
187
188
uint8_t
189
_PyCode_Deinstrument(uint8_t opcode)
190
0
{
191
0
    return DE_INSTRUMENT[opcode];
192
0
}
193
194
static inline bool
195
is_instrumented(int opcode)
196
0
{
197
0
    assert(opcode != 0);
198
0
    assert(opcode != RESERVED);
199
0
    return opcode < ENTER_EXECUTOR && opcode >= MIN_INSTRUMENTED_OPCODE;
200
0
}
201
202
#ifndef NDEBUG
203
static inline bool
204
monitors_equals(_Py_LocalMonitors a, _Py_LocalMonitors b)
205
{
206
    for (int i = 0; i < _PY_MONITORING_UNGROUPED_EVENTS; i++) {
207
        if (a.tools[i] != b.tools[i]) {
208
            return false;
209
        }
210
    }
211
    return true;
212
}
213
#endif
214
215
static inline _Py_LocalMonitors
216
monitors_sub(_Py_LocalMonitors a, _Py_LocalMonitors b)
217
0
{
218
0
    _Py_LocalMonitors res;
219
0
    for (int i = 0; i < _PY_MONITORING_UNGROUPED_EVENTS; i++) {
220
0
        res.tools[i] = a.tools[i] & ~b.tools[i];
221
0
    }
222
0
    return res;
223
0
}
224
225
#ifndef NDEBUG
226
static inline _Py_LocalMonitors
227
monitors_and(_Py_LocalMonitors a, _Py_LocalMonitors b)
228
{
229
    _Py_LocalMonitors res;
230
    for (int i = 0; i < _PY_MONITORING_UNGROUPED_EVENTS; i++) {
231
        res.tools[i] = a.tools[i] & b.tools[i];
232
    }
233
    return res;
234
}
235
#endif
236
237
/* The union of the *local* events in a and b.
238
 * Global events like RAISE are ignored.
239
 * Used for instrumentation, as only local
240
 * events get instrumented.
241
 */
242
static inline _Py_LocalMonitors
243
local_union(_Py_GlobalMonitors a, _Py_LocalMonitors b)
244
0
{
245
0
    _Py_LocalMonitors res;
246
0
    for (int i = 0; i < _PY_MONITORING_UNGROUPED_EVENTS; i++) {
247
0
        res.tools[i] = a.tools[i] | b.tools[i];
248
0
    }
249
0
    return res;
250
0
}
251
252
static inline bool
253
monitors_are_empty(_Py_LocalMonitors m)
254
0
{
255
0
    for (int i = 0; i < _PY_MONITORING_UNGROUPED_EVENTS; i++) {
256
0
        if (m.tools[i]) {
257
0
            return false;
258
0
        }
259
0
    }
260
0
    return true;
261
0
}
262
263
static inline bool
264
multiple_tools(_Py_LocalMonitors *m)
265
0
{
266
0
    for (int i = 0; i < _PY_MONITORING_UNGROUPED_EVENTS; i++) {
267
0
        if (_Py_popcount32(m->tools[i]) > 1) {
268
0
            return true;
269
0
        }
270
0
    }
271
0
    return false;
272
0
}
273
274
static inline _PyMonitoringEventSet
275
get_local_events(_Py_LocalMonitors *m, int tool_id)
276
0
{
277
0
    _PyMonitoringEventSet result = 0;
278
0
    for (int e = 0; e < _PY_MONITORING_UNGROUPED_EVENTS; e++) {
279
0
        if ((m->tools[e] >> tool_id) & 1) {
280
0
            result |= (1 << e);
281
0
        }
282
0
    }
283
0
    return result;
284
0
}
285
286
static inline _PyMonitoringEventSet
287
get_events(_Py_GlobalMonitors *m, int tool_id)
288
0
{
289
0
    _PyMonitoringEventSet result = 0;
290
0
    for (int e = 0; e < _PY_MONITORING_UNGROUPED_EVENTS; e++) {
291
0
        if ((m->tools[e] >> tool_id) & 1) {
292
0
            result |= (1 << e);
293
0
        }
294
0
    }
295
0
    return result;
296
0
}
297
298
/* Module code can have line 0, even though modules start at line 1,
299
 * so -1 is a legal delta. */
300
0
#define NO_LINE (-2)
301
302
/* Returns the line delta. Defined as:
303
 * if line is None:
304
 *     line_delta = NO_LINE
305
 * else:
306
 *     line_delta = line - first_line
307
 */
308
static int
309
compute_line_delta(PyCodeObject *code, int line)
310
0
{
311
0
    if (line < 0) {
312
0
        assert(line == -1);
313
0
        return NO_LINE;
314
0
    }
315
0
    int delta = line - code->co_firstlineno;
316
0
    assert(delta > NO_LINE);
317
0
    return delta;
318
0
}
319
320
static int
321
compute_line(PyCodeObject *code, int line_delta)
322
0
{
323
0
    if (line_delta == NO_LINE) {
324
0
        return -1;
325
0
    }
326
0
    assert(line_delta > NO_LINE);
327
0
    return code->co_firstlineno + line_delta;
328
0
}
329
330
int
331
_PyInstruction_GetLength(PyCodeObject *code, int offset)
332
0
{
333
0
    ASSERT_WORLD_STOPPED_OR_LOCKED(code);
334
335
0
    _Py_CODEUNIT inst = _Py_GetBaseCodeUnit(code, offset);
336
0
    return 1 + _PyOpcode_Caches[inst.op.code];
337
0
}
338
339
static inline uint8_t *
340
get_original_opcode_ptr(_PyCoLineInstrumentationData *line_data, int index)
341
0
{
342
0
    return &line_data->data[index*line_data->bytes_per_entry];
343
0
}
344
345
static inline void
346
set_original_opcode(_PyCoLineInstrumentationData *line_data, int index, uint8_t opcode)
347
0
{
348
0
    line_data->data[index*line_data->bytes_per_entry] = opcode;
349
0
}
350
351
static inline int
352
get_line_delta(_PyCoLineInstrumentationData *line_data, int index)
353
0
{
354
0
    uint8_t *ptr = &line_data->data[index*line_data->bytes_per_entry+1];
355
0
    assert(line_data->bytes_per_entry >= 2);
356
0
    uint32_t value = *ptr;
357
0
    for (int idx = 2; idx < line_data->bytes_per_entry; idx++) {
358
0
        ptr++;
359
0
        int shift = (idx-1)*8;
360
0
        value |= ((uint32_t)(*ptr)) << shift;
361
0
    }
362
0
    assert(value < INT_MAX);
363
    /* NO_LINE is stored as zero. */
364
0
    return ((int)value) + NO_LINE;
365
0
}
366
367
static inline void
368
set_line_delta(_PyCoLineInstrumentationData *line_data, int index, int line_delta)
369
0
{
370
    /* Store line_delta + 2 as we need -2 to represent no line number */
371
0
    assert(line_delta >= NO_LINE);
372
0
    uint32_t adjusted = line_delta - NO_LINE;
373
0
    uint8_t *ptr = &line_data->data[index*line_data->bytes_per_entry+1];
374
0
    assert(adjusted < (1ULL << ((line_data->bytes_per_entry-1)*8)));
375
0
    assert(line_data->bytes_per_entry >= 2);
376
0
    *ptr = adjusted & 0xff;
377
0
    for (int idx = 2; idx < line_data->bytes_per_entry; idx++) {
378
0
        ptr++;
379
0
        adjusted >>= 8;
380
0
        *ptr = adjusted & 0xff;
381
0
    }
382
0
}
383
384
#ifdef INSTRUMENT_DEBUG
385
386
static void
387
dump_instrumentation_data_tools(PyCodeObject *code, uint8_t *tools, int i, FILE*out)
388
{
389
    if (tools == NULL) {
390
        fprintf(out, "tools = NULL");
391
    }
392
    else {
393
        fprintf(out, "tools = %d", tools[i]);
394
    }
395
}
396
397
static void
398
dump_instrumentation_data_lines(PyCodeObject *code, _PyCoLineInstrumentationData *lines, int i, FILE*out)
399
{
400
    if (lines == NULL) {
401
        fprintf(out, ", lines = NULL");
402
    }
403
    else {
404
        int opcode = _PyCode_GetOriginalOpcode(lines, i);
405
        int line_delta = get_line_delta(lines, i);
406
        if (opcode == 0) {
407
            fprintf(out, ", lines = {original_opcode = No LINE (0), line_delta = %d)", line_delta);
408
        }
409
        else {
410
            fprintf(out, ", lines = {original_opcode = %s, line_delta = %d)", _PyOpcode_OpName[opcode], line_delta);
411
        }
412
    }
413
}
414
415
static void
416
dump_instrumentation_data_line_tools(PyCodeObject *code, uint8_t *line_tools, int i, FILE*out)
417
{
418
    if (line_tools == NULL) {
419
        fprintf(out, ", line_tools = NULL");
420
    }
421
    else {
422
        fprintf(out, ", line_tools = %d", line_tools[i]);
423
    }
424
}
425
426
static void
427
dump_instrumentation_data_per_instruction(PyCodeObject *code, _PyCoMonitoringData *data, int i, FILE*out)
428
{
429
    if (data->per_instruction_opcodes == NULL) {
430
        fprintf(out, ", per-inst opcode = NULL");
431
    }
432
    else {
433
        fprintf(out, ", per-inst opcode = %s", _PyOpcode_OpName[data->per_instruction_opcodes[i]]);
434
    }
435
    if (data->per_instruction_tools == NULL) {
436
        fprintf(out, ", per-inst tools = NULL");
437
    }
438
    else {
439
        fprintf(out, ", per-inst tools = %d", data->per_instruction_tools[i]);
440
    }
441
}
442
443
static void
444
dump_global_monitors(const char *prefix, _Py_GlobalMonitors monitors, FILE*out)
445
{
446
    fprintf(out, "%s monitors:\n", prefix);
447
    for (int event = 0; event < _PY_MONITORING_UNGROUPED_EVENTS; event++) {
448
        fprintf(out, "    Event %d: Tools %x\n", event, monitors.tools[event]);
449
    }
450
}
451
452
static void
453
dump_local_monitors(const char *prefix, _Py_LocalMonitors monitors, FILE*out)
454
{
455
    fprintf(out, "%s monitors:\n", prefix);
456
    for (int event = 0; event < _PY_MONITORING_UNGROUPED_EVENTS; event++) {
457
        fprintf(out, "    Event %d: Tools %x\n", event, monitors.tools[event]);
458
    }
459
}
460
461
/** NOTE:
462
 * Do not use PyCode_Addr2Line to determine the line number in instrumentation,
463
 * as `PyCode_Addr2Line` uses the monitoring data if it is available.
464
 */
465
466
467
/* No error checking -- Don't use this for anything but experimental debugging */
468
static void
469
dump_instrumentation_data(PyCodeObject *code, int star, FILE*out)
470
{
471
    _PyCoMonitoringData *data = code->_co_monitoring;
472
    fprintf(out, "\n");
473
    PyObject_Print(code->co_name, out, Py_PRINT_RAW);
474
    fprintf(out, "\n");
475
    if (data == NULL) {
476
        fprintf(out, "NULL\n");
477
        return;
478
    }
479
    dump_global_monitors("Global", _PyInterpreterState_GET()->monitors, out);
480
    dump_local_monitors("Code", data->local_monitors, out);
481
    dump_local_monitors("Active", data->active_monitors, out);
482
    int code_len = (int)Py_SIZE(code);
483
    bool starred = false;
484
    PyCodeAddressRange range;
485
    _PyCode_InitAddressRange(code, &range);
486
    for (int i = 0; i < code_len; i += _PyInstruction_GetLength(code, i)) {
487
        _Py_CODEUNIT *instr = &_PyCode_CODE(code)[i];
488
        int opcode = instr->op.code;
489
        if (i == star) {
490
            fprintf(out, "**  ");
491
            starred = true;
492
        }
493
        fprintf(out, "Offset: %d, line: %d %s: ", i, _PyCode_CheckLineNumber(i*2, &range), _PyOpcode_OpName[opcode]);
494
        dump_instrumentation_data_tools(code, data->tools, i, out);
495
        dump_instrumentation_data_lines(code, data->lines, i, out);
496
        dump_instrumentation_data_line_tools(code, data->line_tools, i, out);
497
        dump_instrumentation_data_per_instruction(code, data, i, out);
498
        fprintf(out, "\n");
499
        ;
500
    }
501
    if (!starred && star >= 0) {
502
        fprintf(out, "Error offset not at valid instruction offset: %d\n", star);
503
        fprintf(out, "    ");
504
        dump_instrumentation_data_tools(code, data->tools, star, out);
505
        dump_instrumentation_data_lines(code, data->lines, star, out);
506
        dump_instrumentation_data_line_tools(code, data->line_tools, star, out);
507
        dump_instrumentation_data_per_instruction(code, data, star, out);
508
        fprintf(out, "\n");
509
    }
510
}
511
512
#define CHECK(test) do { \
513
    if (!(test)) { \
514
        dump_instrumentation_data(code, i, stderr); \
515
    } \
516
    assert(test); \
517
} while (0)
518
519
static bool
520
valid_opcode(int opcode)
521
{
522
    if (opcode == INSTRUMENTED_LINE) {
523
        return true;
524
    }
525
    if (IS_VALID_OPCODE(opcode) &&
526
        opcode != CACHE &&
527
        opcode != RESERVED &&
528
        opcode < 254)
529
    {
530
       return true;
531
    }
532
    return false;
533
}
534
535
static void
536
sanity_check_instrumentation(PyCodeObject *code)
537
{
538
    ASSERT_WORLD_STOPPED_OR_LOCKED(code);
539
540
    _PyCoMonitoringData *data = code->_co_monitoring;
541
    if (data == NULL) {
542
        return;
543
    }
544
    _Py_GlobalMonitors global_monitors = _PyInterpreterState_GET()->monitors;
545
    _Py_LocalMonitors active_monitors;
546
    if (code->_co_monitoring) {
547
        _Py_LocalMonitors local_monitors = code->_co_monitoring->local_monitors;
548
        active_monitors = local_union(global_monitors, local_monitors);
549
    }
550
    else {
551
        _Py_LocalMonitors empty = (_Py_LocalMonitors) { 0 };
552
        active_monitors = local_union(global_monitors, empty);
553
    }
554
    assert(monitors_equals(
555
        code->_co_monitoring->active_monitors,
556
        active_monitors));
557
    int code_len = (int)Py_SIZE(code);
558
    PyCodeAddressRange range;
559
    _PyCode_InitAddressRange(co, &range);
560
    for (int i = 0; i < code_len;) {
561
        _Py_CODEUNIT *instr = &_PyCode_CODE(code)[i];
562
        int opcode = instr->op.code;
563
        int base_opcode = _Py_GetBaseCodeUnit(code, i).op.code;
564
        CHECK(valid_opcode(opcode));
565
        CHECK(valid_opcode(base_opcode));
566
        if (opcode == INSTRUMENTED_INSTRUCTION) {
567
            opcode = data->per_instruction_opcodes[i];
568
            if (!is_instrumented(opcode)) {
569
                CHECK(_PyOpcode_Deopt[opcode] == opcode);
570
            }
571
        }
572
        if (opcode == INSTRUMENTED_LINE) {
573
            CHECK(data->lines);
574
            opcode = _PyCode_GetOriginalOpcode(data->lines, i);
575
            CHECK(valid_opcode(opcode));
576
            CHECK(opcode != END_FOR);
577
            CHECK(opcode != RESUME);
578
            CHECK(opcode != RESUME_CHECK);
579
            CHECK(opcode != RESUME_CHECK_JIT);
580
            CHECK(opcode != INSTRUMENTED_RESUME);
581
            if (!is_instrumented(opcode)) {
582
                CHECK(_PyOpcode_Deopt[opcode] == opcode);
583
            }
584
            CHECK(opcode != INSTRUMENTED_LINE);
585
        }
586
        else if (data->lines) {
587
            /* If original_opcode is INSTRUMENTED_INSTRUCTION
588
             * *and* we are executing a INSTRUMENTED_LINE instruction
589
             * that has de-instrumented itself, then we will execute
590
             * an invalid INSTRUMENTED_INSTRUCTION */
591
            CHECK(_PyCode_GetOriginalOpcode(data->lines, i) != INSTRUMENTED_INSTRUCTION);
592
        }
593
        if (opcode == INSTRUMENTED_INSTRUCTION) {
594
            CHECK(data->per_instruction_opcodes[i] != 0);
595
            opcode = data->per_instruction_opcodes[i];
596
        }
597
        if (is_instrumented(opcode)) {
598
            CHECK(DE_INSTRUMENT[opcode] == base_opcode);
599
            int event = EVENT_FOR_OPCODE[DE_INSTRUMENT[opcode]];
600
            if (event < 0) {
601
                /* RESUME fixup */
602
                event = instr->op.arg ? 1: 0;
603
            }
604
            CHECK(active_monitors.tools[event] != 0);
605
        }
606
        if (data->lines && _PyCode_GetOriginalOpcode(data->lines, i)) {
607
            int line1 = compute_line(code, get_line_delta(data->lines, i));
608
            int line2 = _PyCode_CheckLineNumber(i*sizeof(_Py_CODEUNIT), &range);
609
            CHECK(line1 == line2);
610
        }
611
        CHECK(valid_opcode(opcode));
612
        if (data->tools) {
613
            uint8_t local_tools = data->tools[i];
614
            if (opcode_has_event(base_opcode)) {
615
                int event = EVENT_FOR_OPCODE[base_opcode];
616
                if (event == -1) {
617
                    /* RESUME fixup */
618
                    event = _PyCode_CODE(code)[i].op.arg;
619
                }
620
                CHECK((active_monitors.tools[event] & local_tools) == local_tools);
621
            }
622
            else {
623
                CHECK(local_tools == 0xff);
624
            }
625
        }
626
        i += _PyInstruction_GetLength(code, i);
627
        assert(i <= code_len);
628
    }
629
}
630
#else
631
632
0
#define CHECK(test) assert(test)
633
634
#endif
635
636
/* Get the underlying code unit, stripping instrumentation and ENTER_EXECUTOR */
637
_Py_CODEUNIT
638
_Py_GetBaseCodeUnit(PyCodeObject *code, int i)
639
920k
{
640
920k
    _Py_CODEUNIT *src_instr = _PyCode_CODE(code) + i;
641
920k
    _Py_CODEUNIT inst = {
642
920k
        .cache = FT_ATOMIC_LOAD_UINT16_RELAXED(*(uint16_t *)src_instr)};
643
920k
    int opcode = inst.op.code;
644
920k
    if (opcode < MIN_INSTRUMENTED_OPCODE) {
645
920k
        inst.op.code = _PyOpcode_Deopt[opcode];
646
920k
        assert(inst.op.code < MIN_SPECIALIZED_OPCODE);
647
920k
        return inst;
648
920k
    }
649
0
    if (opcode == ENTER_EXECUTOR) {
650
0
        _PyExecutorObject *exec = code->co_executors->executors[inst.op.arg];
651
0
        opcode = _PyOpcode_Deopt[exec->vm_data.opcode];
652
0
        inst.op.code = opcode;
653
0
        inst.op.arg = exec->vm_data.oparg;
654
0
        assert(inst.op.code < MIN_SPECIALIZED_OPCODE);
655
0
        return inst;
656
0
    }
657
0
    if (opcode == INSTRUMENTED_LINE) {
658
0
        opcode = _PyCode_GetOriginalOpcode(code->_co_monitoring->lines, i);
659
0
    }
660
0
    if (opcode == INSTRUMENTED_INSTRUCTION) {
661
0
        opcode = code->_co_monitoring->per_instruction_opcodes[i];
662
0
    }
663
0
    CHECK(opcode != INSTRUMENTED_INSTRUCTION);
664
0
    CHECK(opcode != INSTRUMENTED_LINE);
665
0
    int deinstrumented = DE_INSTRUMENT[opcode];
666
0
    if (deinstrumented) {
667
0
        inst.op.code = deinstrumented;
668
0
    }
669
0
    else {
670
0
        inst.op.code = _PyOpcode_Deopt[opcode];
671
0
    }
672
0
    assert(inst.op.code < MIN_SPECIALIZED_OPCODE);
673
0
    return inst;
674
0
}
675
676
static void
677
de_instrument(PyCodeObject *code, _Py_CODEUNIT *bytecode, _PyCoMonitoringData *monitoring, int i,
678
              int event)
679
0
{
680
0
    assert(event != PY_MONITORING_EVENT_INSTRUCTION);
681
0
    assert(event != PY_MONITORING_EVENT_LINE);
682
683
0
    _Py_CODEUNIT *instr = &bytecode[i];
684
0
    uint8_t *opcode_ptr = &instr->op.code;
685
0
    int opcode = *opcode_ptr;
686
0
    assert(opcode != ENTER_EXECUTOR);
687
0
    if (opcode == INSTRUMENTED_LINE) {
688
0
        opcode_ptr = get_original_opcode_ptr(monitoring->lines, i);
689
0
        opcode = *opcode_ptr;
690
0
    }
691
0
    if (opcode == INSTRUMENTED_INSTRUCTION) {
692
0
        opcode_ptr = &monitoring->per_instruction_opcodes[i];
693
0
        opcode = *opcode_ptr;
694
0
    }
695
0
    int deinstrumented = DE_INSTRUMENT[opcode];
696
0
    if (deinstrumented == 0) {
697
0
        return;
698
0
    }
699
0
    CHECK(_PyOpcode_Deopt[deinstrumented] == deinstrumented);
700
0
    FT_ATOMIC_STORE_UINT8_RELAXED(*opcode_ptr, deinstrumented);
701
0
    if (_PyOpcode_Caches[deinstrumented]) {
702
0
        FT_ATOMIC_STORE_UINT16_RELAXED(instr[1].counter.value_and_backoff,
703
0
                                       adaptive_counter_warmup().value_and_backoff);
704
0
    }
705
0
}
706
707
static void
708
de_instrument_line(PyCodeObject *code, _Py_CODEUNIT *bytecode, _PyCoMonitoringData *monitoring,
709
                   int i)
710
0
{
711
0
    _Py_CODEUNIT *instr = &bytecode[i];
712
0
    int opcode = instr->op.code;
713
0
    if (opcode != INSTRUMENTED_LINE) {
714
0
        return;
715
0
    }
716
0
    _PyCoLineInstrumentationData *lines = monitoring->lines;
717
0
    int original_opcode = _PyCode_GetOriginalOpcode(lines, i);
718
0
    if (original_opcode == INSTRUMENTED_INSTRUCTION) {
719
0
        set_original_opcode(lines, i, monitoring->per_instruction_opcodes[i]);
720
0
    }
721
0
    CHECK(original_opcode != 0);
722
0
    CHECK(original_opcode == _PyOpcode_Deopt[original_opcode]);
723
0
    FT_ATOMIC_STORE_UINT8(instr->op.code, original_opcode);
724
0
    if (_PyOpcode_Caches[original_opcode]) {
725
0
        FT_ATOMIC_STORE_UINT16_RELAXED(instr[1].counter.value_and_backoff,
726
0
                                       adaptive_counter_warmup().value_and_backoff);
727
0
    }
728
0
    assert(instr->op.code != INSTRUMENTED_LINE);
729
0
}
730
731
static void
732
de_instrument_per_instruction(PyCodeObject *code, _Py_CODEUNIT *bytecode,
733
                              _PyCoMonitoringData *monitoring, int i)
734
0
{
735
0
    _Py_CODEUNIT *instr = &bytecode[i];
736
0
    uint8_t *opcode_ptr = &instr->op.code;
737
0
    int opcode = *opcode_ptr;
738
0
    if (opcode == INSTRUMENTED_LINE) {
739
0
        opcode_ptr = get_original_opcode_ptr(monitoring->lines, i);
740
0
        opcode = *opcode_ptr;
741
0
    }
742
0
    if (opcode != INSTRUMENTED_INSTRUCTION) {
743
0
        return;
744
0
    }
745
0
    int original_opcode = monitoring->per_instruction_opcodes[i];
746
0
    CHECK(original_opcode != 0);
747
0
    CHECK(original_opcode == _PyOpcode_Deopt[original_opcode]);
748
0
    FT_ATOMIC_STORE_UINT8_RELAXED(*opcode_ptr, original_opcode);
749
0
    if (_PyOpcode_Caches[original_opcode]) {
750
0
        FT_ATOMIC_STORE_UINT16_RELAXED(instr[1].counter.value_and_backoff,
751
0
                                       adaptive_counter_warmup().value_and_backoff);
752
0
    }
753
0
    assert(*opcode_ptr != INSTRUMENTED_INSTRUCTION);
754
0
    assert(instr->op.code != INSTRUMENTED_INSTRUCTION);
755
0
}
756
757
static void
758
instrument(PyCodeObject *code, _Py_CODEUNIT *bytecode, _PyCoMonitoringData *monitoring, int i)
759
0
{
760
0
    _Py_CODEUNIT *instr = &bytecode[i];
761
0
    uint8_t *opcode_ptr = &instr->op.code;
762
0
    int opcode =*opcode_ptr;
763
0
    if (opcode == INSTRUMENTED_LINE) {
764
0
        opcode_ptr = get_original_opcode_ptr(monitoring->lines, i);
765
0
        opcode = *opcode_ptr;
766
0
    }
767
0
    if (opcode == INSTRUMENTED_INSTRUCTION) {
768
0
        opcode_ptr = &monitoring->per_instruction_opcodes[i];
769
0
        opcode = *opcode_ptr;
770
0
        CHECK(opcode != INSTRUMENTED_INSTRUCTION && opcode != INSTRUMENTED_LINE);
771
0
        CHECK(opcode == _PyOpcode_Deopt[opcode]);
772
0
    }
773
0
    CHECK(opcode != 0);
774
0
    if (!is_instrumented(opcode)) {
775
0
        int deopt = _PyOpcode_Deopt[opcode];
776
0
        int instrumented = INSTRUMENTED_OPCODES[deopt];
777
0
        assert(instrumented);
778
0
        FT_ATOMIC_STORE_UINT8_RELAXED(*opcode_ptr, instrumented);
779
0
        if (_PyOpcode_Caches[deopt]) {
780
0
            FT_ATOMIC_STORE_UINT16_RELAXED(instr[1].counter.value_and_backoff,
781
0
                                           adaptive_counter_warmup().value_and_backoff);
782
0
        }
783
0
    }
784
0
}
785
786
static void
787
instrument_line(PyCodeObject *code, _Py_CODEUNIT *bytecode, _PyCoMonitoringData *monitoring, int i)
788
0
{
789
0
    uint8_t *opcode_ptr = &bytecode[i].op.code;
790
0
    int opcode = *opcode_ptr;
791
0
    if (opcode == INSTRUMENTED_LINE) {
792
0
        return;
793
0
    }
794
0
    set_original_opcode(monitoring->lines, i, _PyOpcode_Deopt[opcode]);
795
0
    CHECK(get_line_delta(monitoring->lines, i) > NO_LINE);
796
0
    FT_ATOMIC_STORE_UINT8_RELAXED(*opcode_ptr, INSTRUMENTED_LINE);
797
0
}
798
799
static void
800
instrument_per_instruction(PyCodeObject *code, _Py_CODEUNIT *bytecode,
801
                           _PyCoMonitoringData *monitoring, int i)
802
0
{
803
0
    _Py_CODEUNIT *instr = &bytecode[i];
804
0
    uint8_t *opcode_ptr = &instr->op.code;
805
0
    int opcode = *opcode_ptr;
806
0
    if (opcode == INSTRUMENTED_LINE) {
807
0
        opcode_ptr = get_original_opcode_ptr(monitoring->lines, i);
808
0
        opcode = *opcode_ptr;
809
0
    }
810
0
    if (opcode == INSTRUMENTED_INSTRUCTION) {
811
0
        assert(monitoring->per_instruction_opcodes[i] > 0);
812
0
        return;
813
0
    }
814
0
    CHECK(opcode != 0);
815
0
    if (is_instrumented(opcode)) {
816
0
        monitoring->per_instruction_opcodes[i] = opcode;
817
0
    }
818
0
    else {
819
0
        assert(opcode != 0);
820
0
        assert(_PyOpcode_Deopt[opcode] != 0);
821
0
        assert(_PyOpcode_Deopt[opcode] != RESUME);
822
0
        monitoring->per_instruction_opcodes[i] = _PyOpcode_Deopt[opcode];
823
0
    }
824
0
    assert(monitoring->per_instruction_opcodes[i] > 0);
825
0
    FT_ATOMIC_STORE_UINT8_RELAXED(*opcode_ptr, INSTRUMENTED_INSTRUCTION);
826
0
}
827
828
static void
829
remove_tools(PyCodeObject * code, int offset, int event, int tools)
830
0
{
831
0
    ASSERT_WORLD_STOPPED_OR_LOCKED(code);
832
0
    assert(event != PY_MONITORING_EVENT_LINE);
833
0
    assert(event != PY_MONITORING_EVENT_INSTRUCTION);
834
0
    assert(PY_MONITORING_IS_INSTRUMENTED_EVENT(event));
835
0
    assert(opcode_has_event(_Py_GetBaseCodeUnit(code, offset).op.code));
836
0
    _PyCoMonitoringData *monitoring = code->_co_monitoring;
837
0
    assert(monitoring);
838
0
    bool should_de_instrument;
839
0
    if (monitoring->tools) {
840
0
        monitoring->tools[offset] &= ~tools;
841
0
        should_de_instrument = (monitoring->tools[offset] == 0);
842
0
    }
843
0
    else {
844
        /* Single tool */
845
0
        uint8_t single_tool = monitoring->active_monitors.tools[event];
846
0
        assert(_Py_popcount32(single_tool) <= 1);
847
0
        should_de_instrument = ((single_tool & tools) == single_tool);
848
0
    }
849
0
    if (should_de_instrument) {
850
0
        MODIFY_BYTECODE(code, de_instrument, monitoring, offset, event);
851
0
    }
852
0
}
853
854
#ifndef NDEBUG
855
static bool
856
tools_is_subset_for_event(PyCodeObject * code, int event, int tools)
857
{
858
    int global_tools = _PyInterpreterState_GET()->monitors.tools[event];
859
    int local_tools = code->_co_monitoring->local_monitors.tools[event];
860
    return tools == ((global_tools | local_tools) & tools);
861
}
862
#endif
863
864
static void
865
remove_line_tools(PyCodeObject * code, int offset, int tools)
866
0
{
867
0
    ASSERT_WORLD_STOPPED_OR_LOCKED(code);
868
869
0
    _PyCoMonitoringData *monitoring = code->_co_monitoring;
870
0
    assert(monitoring);
871
0
    bool should_de_instrument;
872
0
    if (monitoring->line_tools)
873
0
    {
874
0
        uint8_t *toolsptr = &monitoring->line_tools[offset];
875
0
        *toolsptr &= ~tools;
876
0
        should_de_instrument = (*toolsptr == 0);
877
0
    }
878
0
    else {
879
        /* Single tool */
880
0
        uint8_t single_tool = monitoring->active_monitors.tools[PY_MONITORING_EVENT_LINE];
881
0
        assert(_Py_popcount32(single_tool) <= 1);
882
0
        should_de_instrument = ((single_tool & tools) == single_tool);
883
0
    }
884
0
    if (should_de_instrument) {
885
0
        MODIFY_BYTECODE(code, de_instrument_line, monitoring, offset);
886
0
    }
887
0
}
888
889
static void
890
add_tools(PyCodeObject * code, int offset, int event, int tools)
891
0
{
892
0
    ASSERT_WORLD_STOPPED_OR_LOCKED(code);
893
0
    assert(event != PY_MONITORING_EVENT_LINE);
894
0
    assert(event != PY_MONITORING_EVENT_INSTRUCTION);
895
0
    assert(PY_MONITORING_IS_INSTRUMENTED_EVENT(event));
896
0
    assert(code->_co_monitoring);
897
0
    if (code->_co_monitoring &&
898
0
        code->_co_monitoring->tools
899
0
    ) {
900
0
        code->_co_monitoring->tools[offset] |= tools;
901
0
    }
902
0
    else {
903
        /* Single tool */
904
0
        assert(_Py_popcount32(tools) == 1);
905
0
        assert(tools_is_subset_for_event(code, event, tools));
906
0
    }
907
0
    MODIFY_BYTECODE(code, instrument, code->_co_monitoring, offset);
908
0
}
909
910
static void
911
add_line_tools(PyCodeObject * code, int offset, int tools)
912
0
{
913
0
    ASSERT_WORLD_STOPPED_OR_LOCKED(code);
914
915
0
    assert(tools_is_subset_for_event(code, PY_MONITORING_EVENT_LINE, tools));
916
0
    assert(code->_co_monitoring);
917
0
    if (code->_co_monitoring->line_tools) {
918
0
        code->_co_monitoring->line_tools[offset] |= tools;
919
0
    }
920
0
    else {
921
        /* Single tool */
922
0
        assert(_Py_popcount32(tools) == 1);
923
0
    }
924
0
    MODIFY_BYTECODE(code, instrument_line, code->_co_monitoring, offset);
925
0
}
926
927
928
static void
929
add_per_instruction_tools(PyCodeObject * code, int offset, int tools)
930
0
{
931
0
    ASSERT_WORLD_STOPPED_OR_LOCKED(code);
932
933
0
    assert(tools_is_subset_for_event(code, PY_MONITORING_EVENT_INSTRUCTION, tools));
934
0
    assert(code->_co_monitoring);
935
0
    if (code->_co_monitoring->per_instruction_tools) {
936
0
        code->_co_monitoring->per_instruction_tools[offset] |= tools;
937
0
    }
938
0
    else {
939
        /* Single tool */
940
0
        assert(_Py_popcount32(tools) == 1);
941
0
    }
942
0
    MODIFY_BYTECODE(code, instrument_per_instruction, code->_co_monitoring, offset);
943
0
}
944
945
946
static void
947
remove_per_instruction_tools(PyCodeObject * code, int offset, int tools)
948
0
{
949
0
    ASSERT_WORLD_STOPPED_OR_LOCKED(code);
950
951
0
    _PyCoMonitoringData *monitoring = code->_co_monitoring;
952
0
    assert(code->_co_monitoring);
953
0
    bool should_de_instrument;
954
0
    if (code->_co_monitoring->per_instruction_tools) {
955
0
        uint8_t *toolsptr = &code->_co_monitoring->per_instruction_tools[offset];
956
0
        *toolsptr &= ~tools;
957
0
        should_de_instrument = (*toolsptr == 0);
958
0
    }
959
0
    else {
960
        /* Single tool */
961
0
        uint8_t single_tool = code->_co_monitoring->active_monitors.tools[PY_MONITORING_EVENT_INSTRUCTION];
962
0
        assert(_Py_popcount32(single_tool) <= 1);
963
0
        should_de_instrument = ((single_tool & tools) == single_tool);
964
0
    }
965
0
    if (should_de_instrument) {
966
0
        MODIFY_BYTECODE(code, de_instrument_per_instruction, monitoring, offset);
967
0
    }
968
0
}
969
970
971
/* Return 1 if DISABLE returned, -1 if error, 0 otherwise */
972
static int
973
call_one_instrument(
974
    PyInterpreterState *interp, PyThreadState *tstate, PyObject **args,
975
    size_t nargsf, int8_t tool, int event)
976
0
{
977
0
    assert(0 <= tool && tool < 8);
978
0
    assert(tstate->tracing == 0);
979
0
    PyObject *instrument = interp->monitoring_callables[tool][event];
980
0
    if (instrument == NULL) {
981
0
        return 0;
982
0
    }
983
0
    int old_what = tstate->what_event;
984
0
    tstate->what_event = event;
985
0
    tstate->tracing++;
986
0
    PyObject *res = _PyObject_VectorcallTstate(tstate, instrument, args, nargsf, NULL);
987
0
    tstate->tracing--;
988
0
    tstate->what_event = old_what;
989
0
    if (res == NULL) {
990
0
        return -1;
991
0
    }
992
0
    Py_DECREF(res);
993
0
    return (res == &_PyInstrumentation_DISABLE);
994
0
}
995
996
static const int8_t MOST_SIGNIFICANT_BITS[16] = {
997
    -1, 0, 1, 1,
998
    2, 2, 2, 2,
999
    3, 3, 3, 3,
1000
    3, 3, 3, 3,
1001
};
1002
1003
/* We could use _Py_bit_length here, but that is designed for larger (32/64)
1004
 * bit ints, and can perform relatively poorly on platforms without the
1005
 * necessary intrinsics. */
1006
0
static inline int most_significant_bit(uint8_t bits) {
1007
0
    assert(bits != 0);
1008
0
    if (bits > 15) {
1009
0
        return MOST_SIGNIFICANT_BITS[bits>>4]+4;
1010
0
    }
1011
0
    return MOST_SIGNIFICANT_BITS[bits];
1012
0
}
1013
1014
static uint32_t
1015
global_version(PyInterpreterState *interp)
1016
44.4k
{
1017
44.4k
    uint32_t version = (uint32_t)_Py_atomic_load_uintptr_relaxed(
1018
44.4k
        &interp->ceval.instrumentation_version);
1019
#ifdef Py_DEBUG
1020
    PyThreadState *tstate = _PyThreadState_GET();
1021
    uint32_t thread_version =
1022
        (uint32_t)(_Py_atomic_load_uintptr_relaxed(&tstate->eval_breaker) &
1023
                   ~_PY_EVAL_EVENTS_MASK);
1024
    assert(thread_version == version);
1025
#endif
1026
44.4k
    return version;
1027
44.4k
}
1028
1029
/* Atomically set the given version in the given location, without touching
1030
   anything in _PY_EVAL_EVENTS_MASK. */
1031
static void
1032
set_version_raw(uintptr_t *ptr, uint32_t version)
1033
0
{
1034
0
    uintptr_t old = _Py_atomic_load_uintptr_relaxed(ptr);
1035
0
    uintptr_t new;
1036
0
    do {
1037
0
        new = (old & _PY_EVAL_EVENTS_MASK) | version;
1038
0
    } while (!_Py_atomic_compare_exchange_uintptr(ptr, &old, new));
1039
0
}
1040
1041
static void
1042
set_global_version(PyThreadState *tstate, uint32_t version)
1043
0
{
1044
0
    ASSERT_WORLD_STOPPED();
1045
1046
0
    assert((version & _PY_EVAL_EVENTS_MASK) == 0);
1047
0
    PyInterpreterState *interp = tstate->interp;
1048
0
    set_version_raw(&interp->ceval.instrumentation_version, version);
1049
1050
#ifdef Py_GIL_DISABLED
1051
    // Set the version on all threads in free-threaded builds.
1052
    _Py_FOR_EACH_TSTATE_BEGIN(interp, tstate) {
1053
        set_version_raw(&tstate->eval_breaker, version);
1054
    };
1055
    _Py_FOR_EACH_TSTATE_END(interp);
1056
#else
1057
    // Normal builds take the current version from instrumentation_version when
1058
    // attaching a thread, so we only have to set the current thread's version.
1059
0
    set_version_raw(&tstate->eval_breaker, version);
1060
0
#endif
1061
0
}
1062
1063
static bool
1064
is_version_up_to_date(PyCodeObject *code, PyInterpreterState *interp)
1065
44.4k
{
1066
44.4k
    ASSERT_WORLD_STOPPED_OR_LOCKED(code);
1067
44.4k
    return global_version(interp) == code->_co_instrumentation_version;
1068
44.4k
}
1069
1070
#ifndef NDEBUG
1071
static bool
1072
instrumentation_cross_checks(PyInterpreterState *interp, PyCodeObject *code)
1073
{
1074
    ASSERT_WORLD_STOPPED_OR_LOCKED(code);
1075
    _Py_LocalMonitors expected = local_union(
1076
        interp->monitors,
1077
        code->_co_monitoring->local_monitors);
1078
    return monitors_equals(code->_co_monitoring->active_monitors, expected);
1079
}
1080
1081
static int
1082
debug_check_sanity(PyInterpreterState *interp, PyCodeObject *code)
1083
{
1084
    int res;
1085
    LOCK_CODE(code);
1086
    res = is_version_up_to_date(code, interp) &&
1087
          instrumentation_cross_checks(interp, code);
1088
    UNLOCK_CODE();
1089
    return res;
1090
}
1091
1092
#endif
1093
1094
static inline uint8_t
1095
get_tools_for_instruction(PyCodeObject *code, PyInterpreterState *interp, int i, int event)
1096
0
{
1097
0
    uint8_t tools;
1098
0
    assert(event != PY_MONITORING_EVENT_LINE);
1099
0
    assert(event != PY_MONITORING_EVENT_INSTRUCTION);
1100
0
    if (event >= _PY_MONITORING_UNGROUPED_EVENTS) {
1101
0
        assert(event == PY_MONITORING_EVENT_C_RAISE ||
1102
0
                event == PY_MONITORING_EVENT_C_RETURN);
1103
0
        event = PY_MONITORING_EVENT_CALL;
1104
0
    }
1105
0
    assert(_PY_MONITORING_IS_UNGROUPED_EVENT(event));
1106
0
    CHECK(debug_check_sanity(interp, code));
1107
0
    if (PY_MONITORING_IS_INSTRUMENTED_EVENT(event)) {
1108
        /* Instrumented events use per-instruction tool bitmaps. */
1109
0
        if (code->_co_monitoring->tools) {
1110
0
            tools = code->_co_monitoring->tools[i];
1111
0
        }
1112
0
        else {
1113
0
            tools = code->_co_monitoring->active_monitors.tools[event];
1114
0
        }
1115
0
    }
1116
0
    else {
1117
        /* Other (non-instrumented) events are not tied to specific instructions;
1118
         * use the code-object-level active_monitors bitmap instead. */
1119
0
        tools = code->_co_monitoring->active_monitors.tools[event];
1120
0
    }
1121
0
    return tools;
1122
0
}
1123
1124
static const char *const event_names [] = {
1125
    [PY_MONITORING_EVENT_PY_START] = "PY_START",
1126
    [PY_MONITORING_EVENT_PY_RESUME] = "PY_RESUME",
1127
    [PY_MONITORING_EVENT_PY_RETURN] = "PY_RETURN",
1128
    [PY_MONITORING_EVENT_PY_YIELD] = "PY_YIELD",
1129
    [PY_MONITORING_EVENT_CALL] = "CALL",
1130
    [PY_MONITORING_EVENT_LINE] = "LINE",
1131
    [PY_MONITORING_EVENT_INSTRUCTION] = "INSTRUCTION",
1132
    [PY_MONITORING_EVENT_JUMP] = "JUMP",
1133
    [PY_MONITORING_EVENT_BRANCH] = "BRANCH",
1134
    [PY_MONITORING_EVENT_BRANCH_LEFT] = "BRANCH_LEFT",
1135
    [PY_MONITORING_EVENT_BRANCH_RIGHT] = "BRANCH_RIGHT",
1136
    [PY_MONITORING_EVENT_C_RETURN] = "C_RETURN",
1137
    [PY_MONITORING_EVENT_PY_THROW] = "PY_THROW",
1138
    [PY_MONITORING_EVENT_RAISE] = "RAISE",
1139
    [PY_MONITORING_EVENT_RERAISE] = "RERAISE",
1140
    [PY_MONITORING_EVENT_EXCEPTION_HANDLED] = "EXCEPTION_HANDLED",
1141
    [PY_MONITORING_EVENT_C_RAISE] = "C_RAISE",
1142
    [PY_MONITORING_EVENT_PY_UNWIND] = "PY_UNWIND",
1143
    [PY_MONITORING_EVENT_STOP_ITERATION] = "STOP_ITERATION",
1144
};
1145
1146
/* Disable an "other" (non-instrumented) event (e.g. PY_UNWIND) for a single
1147
 * tool on this code object.  Must be called with the world stopped or the
1148
 * code lock held. */
1149
static void
1150
remove_local_tool(PyCodeObject *code, PyInterpreterState *interp,
1151
                  int event, int tool)
1152
0
{
1153
0
    ASSERT_WORLD_STOPPED_OR_LOCKED(code);
1154
0
    assert(_PY_MONITORING_IS_UNGROUPED_EVENT(event));
1155
0
    assert(!PY_MONITORING_IS_INSTRUMENTED_EVENT(event));
1156
0
    assert(code->_co_monitoring);
1157
0
    code->_co_monitoring->local_monitors.tools[event] &= ~(1 << tool);
1158
    /* Recompute active_monitors for this event as the union of global and
1159
     * (now updated) local monitors. */
1160
0
    code->_co_monitoring->active_monitors.tools[event] =
1161
0
        interp->monitors.tools[event] |
1162
0
        code->_co_monitoring->local_monitors.tools[event];
1163
0
}
1164
1165
static int
1166
call_instrumentation_vector(
1167
    _Py_CODEUNIT *instr, PyThreadState *tstate, int event,
1168
    _PyInterpreterFrame *frame, _Py_CODEUNIT *arg2, Py_ssize_t nargs, PyObject *args[])
1169
0
{
1170
0
    if (tstate->tracing) {
1171
0
        return 0;
1172
0
    }
1173
0
    assert(!_PyErr_Occurred(tstate));
1174
0
    assert(args[0] == NULL);
1175
0
    PyCodeObject *code = _PyFrame_GetCode(frame);
1176
0
    assert(args[1] == NULL);
1177
0
    args[1] = (PyObject *)code;
1178
0
    int offset = (int)(instr - _PyFrame_GetBytecode(frame));
1179
    /* Offset visible to user should be the offset in bytes, as that is the
1180
     * convention for APIs involving code offsets. */
1181
0
    int bytes_arg2 = (int)(arg2 - _PyFrame_GetBytecode(frame)) * (int)sizeof(_Py_CODEUNIT);
1182
0
    PyObject *arg2_obj = PyLong_FromLong(bytes_arg2);
1183
0
    if (arg2_obj == NULL) {
1184
0
        return -1;
1185
0
    }
1186
0
    assert(args[2] == NULL);
1187
0
    args[2] = arg2_obj;
1188
0
    PyInterpreterState *interp = tstate->interp;
1189
0
    uint8_t tools = get_tools_for_instruction(code, interp, offset, event);
1190
0
    size_t nargsf = (size_t) nargs | PY_VECTORCALL_ARGUMENTS_OFFSET;
1191
0
    PyObject **callargs = &args[1];
1192
0
    int err = 0;
1193
0
    while (tools) {
1194
0
        int tool = most_significant_bit(tools);
1195
0
        assert(tool >= 0 && tool < 8);
1196
0
        assert(tools & (1 << tool));
1197
0
        tools ^= (1 << tool);
1198
0
        int res = call_one_instrument(interp, tstate, callargs, nargsf, tool, event);
1199
0
        if (res == 0) {
1200
            /* Nothing to do */
1201
0
        }
1202
0
        else if (res < 0) {
1203
            /* error */
1204
0
            err = -1;
1205
0
            break;
1206
0
        }
1207
0
        else {
1208
            /* DISABLE */
1209
0
            if (PY_MONITORING_IS_INSTRUMENTED_EVENT(event)) {
1210
0
                _PyEval_StopTheWorld(interp);
1211
0
                remove_tools(code, offset, event, 1 << tool);
1212
0
                _PyEval_StartTheWorld(interp);
1213
0
            }
1214
0
            else if (_PY_MONITORING_IS_UNGROUPED_EVENT(event)) {
1215
                /* Other (non-instrumented) event: disable for this code object. */
1216
0
                _PyEval_StopTheWorld(interp);
1217
0
                remove_local_tool(code, interp, event, tool);
1218
0
                _PyEval_StartTheWorld(interp);
1219
0
            }
1220
0
            else {
1221
0
                PyErr_Format(PyExc_ValueError,
1222
0
                              "Cannot disable %s events. Callback removed.",
1223
0
                             event_names[event]);
1224
                /* Clear tool to prevent infinite loop */
1225
0
                Py_CLEAR(interp->monitoring_callables[tool][event]);
1226
0
                err = -1;
1227
0
                break;
1228
0
            }
1229
0
        }
1230
0
    }
1231
0
    Py_DECREF(arg2_obj);
1232
0
    return err;
1233
0
}
1234
1235
Py_NO_INLINE int
1236
_Py_call_instrumentation(
1237
    PyThreadState *tstate, int event,
1238
    _PyInterpreterFrame *frame, _Py_CODEUNIT *instr)
1239
0
{
1240
0
    PyObject *args[3] = { NULL, NULL, NULL };
1241
0
    return call_instrumentation_vector(instr, tstate, event, frame, instr, 2, args);
1242
0
}
1243
1244
Py_NO_INLINE int
1245
_Py_call_instrumentation_arg(
1246
    PyThreadState *tstate, int event,
1247
    _PyInterpreterFrame *frame, _Py_CODEUNIT *instr, PyObject *arg)
1248
0
{
1249
0
    PyObject *args[4] = { NULL, NULL, NULL, arg };
1250
0
    return call_instrumentation_vector(instr, tstate, event, frame, instr, 3, args);
1251
0
}
1252
1253
Py_NO_INLINE int
1254
_Py_call_instrumentation_2args(
1255
    PyThreadState *tstate, int event,
1256
    _PyInterpreterFrame *frame, _Py_CODEUNIT *instr, PyObject *arg0, PyObject *arg1)
1257
0
{
1258
0
    PyObject *args[5] = { NULL, NULL, NULL, arg0, arg1 };
1259
0
    return call_instrumentation_vector(instr, tstate, event, frame, instr, 4, args);
1260
0
}
1261
1262
Py_NO_INLINE _Py_CODEUNIT *
1263
_Py_call_instrumentation_jump(
1264
    _Py_CODEUNIT *instr, PyThreadState *tstate, int event,
1265
    _PyInterpreterFrame *frame, _Py_CODEUNIT *src, _Py_CODEUNIT *dest)
1266
0
{
1267
0
    assert(event == PY_MONITORING_EVENT_JUMP ||
1268
0
           event == PY_MONITORING_EVENT_BRANCH_RIGHT ||
1269
0
           event == PY_MONITORING_EVENT_BRANCH_LEFT);
1270
0
    int to = (int)(dest - _PyFrame_GetBytecode(frame));
1271
0
    PyObject *to_obj = PyLong_FromLong(to * (int)sizeof(_Py_CODEUNIT));
1272
0
    if (to_obj == NULL) {
1273
0
        return NULL;
1274
0
    }
1275
0
    PyObject *args[4] = { NULL, NULL, NULL, to_obj };
1276
0
    _Py_CODEUNIT *instr_ptr = frame->instr_ptr;
1277
0
    int err = call_instrumentation_vector(instr, tstate, event, frame, src, 3, args);
1278
0
    Py_DECREF(to_obj);
1279
0
    if (err) {
1280
0
        return NULL;
1281
0
    }
1282
0
    if (frame->instr_ptr != instr_ptr) {
1283
        /* The callback has caused a jump (by setting the line number) */
1284
0
        return frame->instr_ptr;
1285
0
    }
1286
0
    return dest;
1287
0
}
1288
1289
static void
1290
call_instrumentation_vector_protected(
1291
    PyThreadState *tstate, int event,
1292
    _PyInterpreterFrame *frame, _Py_CODEUNIT *instr, Py_ssize_t nargs, PyObject *args[])
1293
0
{
1294
0
    assert(_PyErr_Occurred(tstate));
1295
0
    PyObject *exc = _PyErr_GetRaisedException(tstate);
1296
0
    int err = call_instrumentation_vector(instr, tstate, event, frame, instr, nargs, args);
1297
0
    if (err) {
1298
0
        Py_XDECREF(exc);
1299
0
    }
1300
0
    else {
1301
0
        _PyErr_SetRaisedException(tstate, exc);
1302
0
    }
1303
0
    assert(_PyErr_Occurred(tstate));
1304
0
}
1305
1306
Py_NO_INLINE void
1307
_Py_call_instrumentation_exc2(
1308
    PyThreadState *tstate, int event,
1309
    _PyInterpreterFrame *frame, _Py_CODEUNIT *instr, PyObject *arg0, PyObject *arg1)
1310
0
{
1311
0
    assert(_PyErr_Occurred(tstate));
1312
0
    PyObject *args[5] = { NULL, NULL, NULL, arg0, arg1 };
1313
0
    call_instrumentation_vector_protected(tstate, event, frame, instr, 4, args);
1314
0
}
1315
1316
int
1317
_Py_Instrumentation_GetLine(PyCodeObject *code, _PyCoLineInstrumentationData *line_data, int index)
1318
0
{
1319
0
    assert(line_data != NULL);
1320
0
    assert(index < Py_SIZE(code));
1321
0
    int line_delta = get_line_delta(line_data, index);
1322
0
    int line = compute_line(code, line_delta);
1323
0
    return line;
1324
0
}
1325
1326
Py_NO_INLINE int
1327
_Py_call_instrumentation_line(PyThreadState *tstate, _PyInterpreterFrame* frame, _Py_CODEUNIT *instr, _Py_CODEUNIT *prev)
1328
0
{
1329
0
    PyCodeObject *code = _PyFrame_GetCode(frame);
1330
0
    assert(tstate->tracing == 0);
1331
0
    assert(debug_check_sanity(tstate->interp, code));
1332
0
    _Py_CODEUNIT *bytecode = _PyFrame_GetBytecode(frame);
1333
0
    int i = (int)(instr - bytecode);
1334
1335
0
    _PyCoMonitoringData *monitoring = code->_co_monitoring;
1336
0
    _PyCoLineInstrumentationData *line_data = monitoring->lines;
1337
0
    PyInterpreterState *interp = tstate->interp;
1338
0
    int line = _Py_Instrumentation_GetLine(code, line_data, i);
1339
0
    assert(line >= 0);
1340
0
    assert(prev != NULL);
1341
0
    int prev_index = (int)(prev - bytecode);
1342
0
    int prev_line = _Py_Instrumentation_GetLine(code, line_data, prev_index);
1343
0
    if (prev_line == line) {
1344
0
        int prev_opcode = bytecode[prev_index].op.code;
1345
        /* RESUME and INSTRUMENTED_RESUME are needed for the operation of
1346
            * instrumentation, so must never be hidden by an INSTRUMENTED_LINE.
1347
            */
1348
0
        if (prev_opcode != RESUME && prev_opcode != INSTRUMENTED_RESUME) {
1349
0
            goto done;
1350
0
        }
1351
0
    }
1352
1353
0
    uint8_t tools = code->_co_monitoring->line_tools != NULL ?
1354
0
        code->_co_monitoring->line_tools[i] :
1355
0
        (interp->monitors.tools[PY_MONITORING_EVENT_LINE] |
1356
0
         code->_co_monitoring->local_monitors.tools[PY_MONITORING_EVENT_LINE]
1357
0
        );
1358
    /* Special case sys.settrace to avoid boxing the line number,
1359
     * only to immediately unbox it. */
1360
0
    if (tools & (1 << PY_MONITORING_SYS_TRACE_ID)) {
1361
0
        if (tstate->c_tracefunc != NULL) {
1362
0
            PyFrameObject *frame_obj = _PyFrame_GetFrameObject(frame);
1363
0
            if (frame_obj == NULL) {
1364
0
                return -1;
1365
0
            }
1366
0
            if (frame_obj->f_trace_lines) {
1367
                /* Need to set tracing and what_event as if using
1368
                 * the instrumentation call. */
1369
0
                int old_what = tstate->what_event;
1370
0
                tstate->what_event = PY_MONITORING_EVENT_LINE;
1371
0
                tstate->tracing++;
1372
                /* Call c_tracefunc directly, having set the line number. */
1373
0
                Py_INCREF(frame_obj);
1374
0
                frame_obj->f_lineno = line;
1375
0
                int err = tstate->c_tracefunc(tstate->c_traceobj, frame_obj, PyTrace_LINE, Py_None);
1376
0
                frame_obj->f_lineno = 0;
1377
0
                tstate->tracing--;
1378
0
                tstate->what_event = old_what;
1379
0
                Py_DECREF(frame_obj);
1380
0
                if (err) {
1381
0
                    return -1;
1382
0
                }
1383
0
            }
1384
0
        }
1385
0
        tools &= (255 - (1 << PY_MONITORING_SYS_TRACE_ID));
1386
0
    }
1387
0
    if (tools == 0) {
1388
0
        goto done;
1389
0
    }
1390
0
    PyObject *line_obj = PyLong_FromLong(line);
1391
0
    if (line_obj == NULL) {
1392
0
        return -1;
1393
0
    }
1394
0
    PyObject *args[3] = { NULL, (PyObject *)code, line_obj };
1395
0
    do {
1396
0
        int tool = most_significant_bit(tools);
1397
0
        assert(tool >= 0 && tool < PY_MONITORING_SYS_PROFILE_ID);
1398
0
        assert(tools & (1 << tool));
1399
0
        tools &= ~(1 << tool);
1400
0
        int res = call_one_instrument(interp, tstate, &args[1],
1401
0
                                      2 | PY_VECTORCALL_ARGUMENTS_OFFSET,
1402
0
                                      tool, PY_MONITORING_EVENT_LINE);
1403
0
        if (res == 0) {
1404
            /* Nothing to do */
1405
0
        }
1406
0
        else if (res < 0) {
1407
            /* error */
1408
0
            Py_DECREF(line_obj);
1409
0
            return -1;
1410
0
        }
1411
0
        else {
1412
            /* DISABLE  */
1413
0
            PyInterpreterState *interp = tstate->interp;
1414
0
            _PyEval_StopTheWorld(interp);
1415
0
            remove_line_tools(code, i, 1 << tool);
1416
0
            _PyEval_StartTheWorld(interp);
1417
0
        }
1418
0
    } while (tools);
1419
0
    Py_DECREF(line_obj);
1420
0
    uint8_t original_opcode;
1421
0
done:
1422
0
    original_opcode = _PyCode_GetOriginalOpcode(line_data, i);
1423
0
    assert(original_opcode != 0);
1424
0
    assert(original_opcode != INSTRUMENTED_LINE);
1425
0
    assert(_PyOpcode_Deopt[original_opcode] == original_opcode);
1426
0
    return original_opcode;
1427
0
}
1428
1429
Py_NO_INLINE int
1430
_Py_call_instrumentation_instruction(PyThreadState *tstate, _PyInterpreterFrame* frame, _Py_CODEUNIT *instr)
1431
0
{
1432
0
    PyCodeObject *code = _PyFrame_GetCode(frame);
1433
0
    int offset = (int)(instr - _PyFrame_GetBytecode(frame));
1434
0
    _PyCoMonitoringData *instrumentation_data = code->_co_monitoring;
1435
0
    assert(instrumentation_data->per_instruction_opcodes);
1436
0
    int next_opcode = instrumentation_data->per_instruction_opcodes[offset];
1437
0
    if (tstate->tracing) {
1438
0
        return next_opcode;
1439
0
    }
1440
0
    assert(debug_check_sanity(tstate->interp, code));
1441
0
    PyInterpreterState *interp = tstate->interp;
1442
0
    uint8_t tools = instrumentation_data->per_instruction_tools != NULL ?
1443
0
        instrumentation_data->per_instruction_tools[offset] :
1444
0
        (interp->monitors.tools[PY_MONITORING_EVENT_INSTRUCTION] |
1445
0
         code->_co_monitoring->local_monitors.tools[PY_MONITORING_EVENT_INSTRUCTION]
1446
0
        );
1447
0
    int bytes_offset = offset * (int)sizeof(_Py_CODEUNIT);
1448
0
    PyObject *offset_obj = PyLong_FromLong(bytes_offset);
1449
0
    if (offset_obj == NULL) {
1450
0
        return -1;
1451
0
    }
1452
0
    PyObject *args[3] = { NULL, (PyObject *)code, offset_obj };
1453
0
    while (tools) {
1454
0
        int tool = most_significant_bit(tools);
1455
0
        assert(tool >= 0 && tool < 8);
1456
0
        assert(tools & (1 << tool));
1457
0
        tools &= ~(1 << tool);
1458
0
        int res = call_one_instrument(interp, tstate, &args[1],
1459
0
                                      2 | PY_VECTORCALL_ARGUMENTS_OFFSET,
1460
0
                                      tool, PY_MONITORING_EVENT_INSTRUCTION);
1461
0
        if (res == 0) {
1462
            /* Nothing to do */
1463
0
        }
1464
0
        else if (res < 0) {
1465
            /* error */
1466
0
            Py_DECREF(offset_obj);
1467
0
            return -1;
1468
0
        }
1469
0
        else {
1470
            /* DISABLE  */
1471
0
            PyInterpreterState *interp = tstate->interp;
1472
0
            _PyEval_StopTheWorld(interp);
1473
0
            remove_per_instruction_tools(code, offset, 1 << tool);
1474
0
            _PyEval_StartTheWorld(interp);
1475
0
        }
1476
0
    }
1477
0
    Py_DECREF(offset_obj);
1478
0
    assert(next_opcode != 0);
1479
0
    return next_opcode;
1480
0
}
1481
1482
static void
1483
initialize_tools(PyCodeObject *code)
1484
0
{
1485
0
    ASSERT_WORLD_STOPPED_OR_LOCKED(code);
1486
0
    uint8_t* tools = code->_co_monitoring->tools;
1487
1488
0
    assert(tools != NULL);
1489
0
    int code_len = (int)Py_SIZE(code);
1490
0
    for (int i = 0; i < code_len; i++) {
1491
0
        _Py_CODEUNIT *instr = &_PyCode_CODE(code)[i];
1492
0
        int opcode = instr->op.code;
1493
0
        assert(opcode != ENTER_EXECUTOR);
1494
0
        if (opcode == INSTRUMENTED_LINE) {
1495
0
            opcode = _PyCode_GetOriginalOpcode(code->_co_monitoring->lines, i);
1496
0
        }
1497
0
        if (opcode == INSTRUMENTED_INSTRUCTION) {
1498
0
            opcode = code->_co_monitoring->per_instruction_opcodes[i];
1499
0
        }
1500
0
        bool instrumented = is_instrumented(opcode);
1501
0
        if (instrumented) {
1502
0
            opcode = DE_INSTRUMENT[opcode];
1503
0
            assert(opcode != 0);
1504
0
        }
1505
0
        opcode = _PyOpcode_Deopt[opcode];
1506
0
        if (opcode_has_event(opcode)) {
1507
0
            if (instrumented) {
1508
0
                int8_t event;
1509
0
                if (opcode == RESUME) {
1510
0
                    event = instr->op.arg != 0;
1511
0
                }
1512
0
                else {
1513
0
                    event = EVENT_FOR_OPCODE[opcode];
1514
0
                    assert(event > 0);
1515
0
                }
1516
0
                assert(event >= 0);
1517
0
                assert(PY_MONITORING_IS_INSTRUMENTED_EVENT(event));
1518
0
                tools[i] = code->_co_monitoring->active_monitors.tools[event];
1519
0
                CHECK(tools[i] != 0);
1520
0
            }
1521
0
            else {
1522
0
                tools[i] = 0;
1523
0
            }
1524
0
        }
1525
#ifdef Py_DEBUG
1526
        /* Initialize tools for invalid locations to all ones to try to catch errors */
1527
        else {
1528
            tools[i] = 0xff;
1529
        }
1530
        for (int j = 1; j <= _PyOpcode_Caches[opcode]; j++) {
1531
            tools[i+j] = 0xff;
1532
        }
1533
#endif
1534
0
        i += _PyOpcode_Caches[opcode];
1535
0
    }
1536
0
}
1537
1538
static void
1539
initialize_lines(_PyCoLineInstrumentationData *line_data, PyCodeObject *code, int bytes_per_entry)
1540
0
{
1541
0
    ASSERT_WORLD_STOPPED_OR_LOCKED(code);
1542
0
    assert(line_data != NULL);
1543
0
    line_data->bytes_per_entry = bytes_per_entry;
1544
0
    int code_len = (int)Py_SIZE(code);
1545
0
    PyCodeAddressRange range;
1546
0
    _PyCode_InitAddressRange(code, &range);
1547
0
    int current_line = -1;
1548
0
    for (int i = 0; i < code_len; ) {
1549
0
        int opcode = _Py_GetBaseCodeUnit(code, i).op.code;
1550
0
        int line = _PyCode_CheckLineNumber(i*(int)sizeof(_Py_CODEUNIT), &range);
1551
0
        set_line_delta(line_data, i, compute_line_delta(code, line));
1552
0
        int length = _PyInstruction_GetLength(code, i);
1553
0
        if (i < code->_co_firsttraceable) {
1554
0
            set_original_opcode(line_data, i, 0);
1555
0
        }
1556
0
        else {
1557
0
            switch (opcode) {
1558
0
                case END_ASYNC_FOR:
1559
0
                case END_FOR:
1560
0
                case END_SEND:
1561
0
                case RESUME:
1562
0
                case POP_ITER:
1563
                    /* END_FOR cannot start a line, as it is skipped by FOR_ITER
1564
                    * END_SEND cannot start a line, as it is skipped by SEND
1565
                    * RESUME and POP_ITER must not be instrumented with INSTRUMENTED_LINE */
1566
0
                    set_original_opcode(line_data, i, 0);
1567
0
                    break;
1568
0
                default:
1569
                    /* Set original_opcode to the opcode iff the instruction
1570
                    * starts a line, and thus should be instrumented.
1571
                    * This saves having to perform this check every time the
1572
                    * we turn instrumentation on or off, and serves as a sanity
1573
                    * check when debugging.
1574
                    */
1575
0
                    if (line != current_line && line >= 0) {
1576
0
                        set_original_opcode(line_data, i, opcode);
1577
0
                        CHECK(get_line_delta(line_data, i) != NO_LINE);
1578
0
                    }
1579
0
                    else {
1580
0
                        set_original_opcode(line_data, i, 0);
1581
0
                    }
1582
0
                    current_line = line;
1583
0
            }
1584
0
        }
1585
0
        for (int j = 1; j < length; j++) {
1586
0
            set_original_opcode(line_data, i+j, 0);
1587
0
            set_line_delta(line_data, i+j, NO_LINE);
1588
0
        }
1589
0
        i += length;
1590
0
    }
1591
0
    for (int i = code->_co_firsttraceable; i < code_len; ) {
1592
0
        _Py_CODEUNIT inst =_Py_GetBaseCodeUnit(code, i);
1593
0
        int opcode = inst.op.code;
1594
0
        int oparg = 0;
1595
0
        while (opcode == EXTENDED_ARG) {
1596
0
            oparg = (oparg << 8) | inst.op.arg;
1597
0
            i++;
1598
0
            inst =_Py_GetBaseCodeUnit(code, i);
1599
0
            opcode = inst.op.code;
1600
0
        }
1601
0
        oparg = (oparg << 8) | inst.op.arg;
1602
0
        i += _PyInstruction_GetLength(code, i);
1603
0
        int target = -1;
1604
0
        switch (opcode) {
1605
0
            case POP_JUMP_IF_FALSE:
1606
0
            case POP_JUMP_IF_TRUE:
1607
0
            case POP_JUMP_IF_NONE:
1608
0
            case POP_JUMP_IF_NOT_NONE:
1609
0
            case JUMP_FORWARD:
1610
0
            {
1611
0
                target = i + oparg;
1612
0
                break;
1613
0
            }
1614
0
            case FOR_ITER:
1615
0
            case SEND:
1616
0
            {
1617
                /* Skip over END_FOR/END_SEND */
1618
0
                target = i + oparg + 1;
1619
0
                break;
1620
0
            }
1621
0
            case JUMP_BACKWARD:
1622
0
            case JUMP_BACKWARD_NO_INTERRUPT:
1623
0
            {
1624
0
                target = i - oparg;
1625
0
                break;
1626
0
            }
1627
0
            default:
1628
0
                continue;
1629
0
        }
1630
0
        assert(target >= 0);
1631
0
        if (get_line_delta(line_data, target) != NO_LINE) {
1632
0
            int opcode = _Py_GetBaseCodeUnit(code, target).op.code;
1633
0
            if (opcode != POP_ITER) {
1634
0
                set_original_opcode(line_data, target, opcode);
1635
0
            }
1636
0
        }
1637
0
    }
1638
    /* Scan exception table */
1639
0
    unsigned char *start = (unsigned char *)PyBytes_AS_STRING(code->co_exceptiontable);
1640
0
    unsigned char *end = start + PyBytes_GET_SIZE(code->co_exceptiontable);
1641
0
    unsigned char *scan = start;
1642
0
    while (scan < end) {
1643
0
        int start_offset, size, handler;
1644
0
        scan = parse_varint(scan, &start_offset);
1645
0
        assert(start_offset >= 0 && start_offset < code_len);
1646
0
        scan = parse_varint(scan, &size);
1647
0
        assert(size >= 0 && start_offset+size <= code_len);
1648
0
        scan = parse_varint(scan, &handler);
1649
0
        assert(handler >= 0 && handler < code_len);
1650
0
        int depth_and_lasti;
1651
0
        scan = parse_varint(scan, &depth_and_lasti);
1652
0
        int original_opcode = _Py_GetBaseCodeUnit(code, handler).op.code;
1653
        /* Skip if not the start of a line.
1654
         * END_ASYNC_FOR is a bit special as it marks the end of
1655
         * an `async for` loop, which should not generate its own
1656
         * line event. */
1657
0
        if (get_line_delta(line_data, handler) != NO_LINE && original_opcode != END_ASYNC_FOR) {
1658
0
            set_original_opcode(line_data, handler, original_opcode);
1659
0
        }
1660
0
    }
1661
0
}
1662
1663
static void
1664
initialize_line_tools(PyCodeObject *code, _Py_LocalMonitors *all_events)
1665
0
{
1666
0
    ASSERT_WORLD_STOPPED_OR_LOCKED(code);
1667
0
    uint8_t *line_tools = code->_co_monitoring->line_tools;
1668
1669
0
    assert(line_tools != NULL);
1670
0
    int code_len = (int)Py_SIZE(code);
1671
0
    for (int i = 0; i < code_len; i++) {
1672
0
        line_tools[i] = all_events->tools[PY_MONITORING_EVENT_LINE];
1673
0
    }
1674
0
}
1675
1676
static int
1677
allocate_instrumentation_data(PyCodeObject *code)
1678
0
{
1679
0
    ASSERT_WORLD_STOPPED_OR_LOCKED(code);
1680
1681
0
    if (code->_co_monitoring == NULL) {
1682
0
        _PyCoMonitoringData *monitoring = PyMem_Malloc(sizeof(_PyCoMonitoringData));
1683
0
        if (monitoring == NULL) {
1684
0
            PyErr_NoMemory();
1685
0
            return -1;
1686
0
        }
1687
0
        monitoring->local_monitors = (_Py_LocalMonitors){ 0 };
1688
0
        monitoring->active_monitors = (_Py_LocalMonitors){ 0 };
1689
0
        monitoring->tools = NULL;
1690
0
        monitoring->lines = NULL;
1691
0
        monitoring->line_tools = NULL;
1692
0
        monitoring->per_instruction_opcodes = NULL;
1693
0
        monitoring->per_instruction_tools = NULL;
1694
0
        _Py_atomic_store_ptr_release(&code->_co_monitoring, monitoring);
1695
0
    }
1696
0
    return 0;
1697
0
}
1698
1699
static int
1700
update_instrumentation_data(PyCodeObject *code, PyInterpreterState *interp)
1701
0
{
1702
0
    ASSERT_WORLD_STOPPED_OR_LOCKED(code);
1703
1704
0
    int code_len = (int)Py_SIZE(code);
1705
0
    if (allocate_instrumentation_data(code)) {
1706
0
        return -1;
1707
0
    }
1708
    // If the local monitors are out of date, clear them up
1709
0
    _Py_LocalMonitors *local_monitors = &code->_co_monitoring->local_monitors;
1710
0
    for (int i = 0; i < PY_MONITORING_TOOL_IDS; i++) {
1711
0
        if (code->_co_monitoring->tool_versions[i] != interp->monitoring_tool_versions[i]) {
1712
0
            for (int j = 0; j < _PY_MONITORING_UNGROUPED_EVENTS; j++) {
1713
0
                local_monitors->tools[j] &= ~(1 << i);
1714
0
            }
1715
0
        }
1716
0
    }
1717
1718
0
    _Py_LocalMonitors all_events = local_union(
1719
0
        interp->monitors,
1720
0
        code->_co_monitoring->local_monitors);
1721
0
    bool multitools = multiple_tools(&all_events);
1722
0
    if (code->_co_monitoring->tools == NULL && multitools) {
1723
0
        code->_co_monitoring->tools = PyMem_Malloc(code_len);
1724
0
        if (code->_co_monitoring->tools == NULL) {
1725
0
            PyErr_NoMemory();
1726
0
            return -1;
1727
0
        }
1728
0
        initialize_tools(code);
1729
0
    }
1730
0
    if (all_events.tools[PY_MONITORING_EVENT_LINE]) {
1731
0
        if (code->_co_monitoring->lines == NULL) {
1732
0
            PyCodeAddressRange range;
1733
0
            _PyCode_InitAddressRange(code, &range);
1734
0
            int max_line = code->co_firstlineno + 1;
1735
0
            _PyCode_InitAddressRange(code, &range);
1736
0
            for (int i = code->_co_firsttraceable; i < code_len; ) {
1737
0
                int line = _PyCode_CheckLineNumber(i*(int)sizeof(_Py_CODEUNIT), &range);
1738
0
                if (line > max_line) {
1739
0
                    max_line = line;
1740
0
                }
1741
0
                int length = _PyInstruction_GetLength(code, i);
1742
0
                i += length;
1743
0
            }
1744
0
            int bytes_per_entry;
1745
0
            int max_delta = max_line - code->co_firstlineno;
1746
            /* We store delta+2 in the table, so 253 is max for one byte */
1747
0
            if (max_delta < 256+NO_LINE) {
1748
0
                bytes_per_entry = 2;
1749
0
            }
1750
0
            else if (max_delta < (1 << 16)+NO_LINE) {
1751
0
                bytes_per_entry = 3;
1752
0
            }
1753
0
            else if (max_delta < (1 << 24)+NO_LINE) {
1754
0
                bytes_per_entry = 4;
1755
0
            }
1756
0
            else {
1757
0
                bytes_per_entry = 5;
1758
0
            }
1759
0
            _PyCoLineInstrumentationData *lines = PyMem_Malloc(1 + code_len * bytes_per_entry);
1760
0
            if (lines == NULL) {
1761
0
                PyErr_NoMemory();
1762
0
                return -1;
1763
0
            }
1764
0
            initialize_lines(lines, code, bytes_per_entry);
1765
0
            _Py_atomic_store_ptr_release(&code->_co_monitoring->lines, lines);
1766
0
        }
1767
0
        if (multitools && code->_co_monitoring->line_tools == NULL) {
1768
0
            code->_co_monitoring->line_tools = PyMem_Malloc(code_len);
1769
0
            if (code->_co_monitoring->line_tools == NULL) {
1770
0
                PyErr_NoMemory();
1771
0
                return -1;
1772
0
            }
1773
0
            initialize_line_tools(code, &all_events);
1774
0
        }
1775
0
    }
1776
0
    if (all_events.tools[PY_MONITORING_EVENT_INSTRUCTION]) {
1777
0
        if (code->_co_monitoring->per_instruction_opcodes == NULL) {
1778
0
            code->_co_monitoring->per_instruction_opcodes = PyMem_Malloc(code_len * sizeof(_PyCoLineInstrumentationData));
1779
0
            if (code->_co_monitoring->per_instruction_opcodes == NULL) {
1780
0
                PyErr_NoMemory();
1781
0
                return -1;
1782
0
            }
1783
            // Initialize all of the instructions so if local events change while another thread is executing
1784
            // we know what the original opcode was.
1785
0
            for (int i = 0; i < code_len; i++) {
1786
0
                int opcode = _PyCode_CODE(code)[i].op.code;
1787
0
                code->_co_monitoring->per_instruction_opcodes[i] = _PyOpcode_Deopt[opcode];
1788
0
            }
1789
0
        }
1790
0
        if (multitools && code->_co_monitoring->per_instruction_tools == NULL) {
1791
0
            code->_co_monitoring->per_instruction_tools = PyMem_Malloc(code_len);
1792
0
            if (code->_co_monitoring->per_instruction_tools == NULL) {
1793
0
                PyErr_NoMemory();
1794
0
                return -1;
1795
0
            }
1796
0
            for (int i = 0; i < code_len; i++) {
1797
0
                code->_co_monitoring->per_instruction_tools[i] = 0;
1798
0
            }
1799
0
        }
1800
0
    }
1801
0
    return 0;
1802
0
}
1803
1804
static int
1805
force_instrument_lock_held(PyCodeObject *code, PyInterpreterState *interp)
1806
0
{
1807
0
    ASSERT_WORLD_STOPPED_OR_LOCKED(code);
1808
1809
#ifdef _Py_TIER2
1810
    if (code->co_executors != NULL) {
1811
        _PyCode_Clear_Executors(code);
1812
    }
1813
    _Py_Executors_InvalidateDependency(interp, code, 1);
1814
#endif
1815
0
    int code_len = (int)Py_SIZE(code);
1816
    /* Exit early to avoid creating instrumentation
1817
     * data for potential statically allocated code
1818
     * objects.
1819
     * See https://github.com/python/cpython/issues/108390 */
1820
0
    if (code->co_flags & CO_NO_MONITORING_EVENTS) {
1821
0
        return 0;
1822
0
    }
1823
0
    if (update_instrumentation_data(code, interp)) {
1824
0
        return -1;
1825
0
    }
1826
0
    _Py_LocalMonitors active_events = local_union(
1827
0
        interp->monitors,
1828
0
        code->_co_monitoring->local_monitors);
1829
0
    _Py_LocalMonitors new_events;
1830
0
    _Py_LocalMonitors removed_events;
1831
1832
0
    bool restarted = interp->last_restart_version > code->_co_instrumentation_version;
1833
0
    if (restarted) {
1834
0
        removed_events = code->_co_monitoring->active_monitors;
1835
0
        new_events = active_events;
1836
0
    }
1837
0
    else {
1838
0
        removed_events = monitors_sub(code->_co_monitoring->active_monitors, active_events);
1839
0
        new_events = monitors_sub(active_events, code->_co_monitoring->active_monitors);
1840
0
        assert(monitors_are_empty(monitors_and(new_events, removed_events)));
1841
0
    }
1842
0
    code->_co_monitoring->active_monitors = active_events;
1843
0
    if (monitors_are_empty(new_events) && monitors_are_empty(removed_events)) {
1844
0
        goto done;
1845
0
    }
1846
    /* Insert instrumentation */
1847
0
    for (int i = code->_co_firsttraceable; i < code_len; i+= _PyInstruction_GetLength(code, i)) {
1848
0
        assert(_PyCode_CODE(code)[i].op.code != ENTER_EXECUTOR);
1849
0
        _Py_CODEUNIT instr = _Py_GetBaseCodeUnit(code, i);
1850
0
        CHECK(instr.op.code != 0);
1851
0
        int base_opcode = instr.op.code;
1852
0
        if (opcode_has_event(base_opcode)) {
1853
0
            int8_t event;
1854
0
            if (base_opcode == RESUME) {
1855
0
                event = instr.op.arg > 0;
1856
0
            }
1857
0
            else {
1858
0
                event = EVENT_FOR_OPCODE[base_opcode];
1859
0
                assert(event > 0);
1860
0
            }
1861
0
            uint8_t removed_tools = removed_events.tools[event];
1862
0
            if (removed_tools) {
1863
0
                remove_tools(code, i, event, removed_tools);
1864
0
            }
1865
0
            uint8_t new_tools = new_events.tools[event];
1866
0
            if (new_tools) {
1867
0
                add_tools(code, i, event, new_tools);
1868
0
            }
1869
0
        }
1870
0
    }
1871
1872
    // GH-103845: We need to remove both the line and instruction instrumentation before
1873
    // adding new ones, otherwise we may remove the newly added instrumentation.
1874
0
    uint8_t removed_line_tools = removed_events.tools[PY_MONITORING_EVENT_LINE];
1875
0
    uint8_t removed_per_instruction_tools = removed_events.tools[PY_MONITORING_EVENT_INSTRUCTION];
1876
1877
0
    if (removed_line_tools) {
1878
0
        _PyCoLineInstrumentationData *line_data = code->_co_monitoring->lines;
1879
0
        for (int i = code->_co_firsttraceable; i < code_len;) {
1880
0
            if (_PyCode_GetOriginalOpcode(line_data, i)) {
1881
0
                remove_line_tools(code, i, removed_line_tools);
1882
0
            }
1883
0
            i += _PyInstruction_GetLength(code, i);
1884
0
        }
1885
0
    }
1886
0
    if (removed_per_instruction_tools) {
1887
0
        for (int i = code->_co_firsttraceable; i < code_len;) {
1888
0
            int opcode = _Py_GetBaseCodeUnit(code, i).op.code;
1889
0
            if (opcode == RESUME || opcode == END_FOR) {
1890
0
                i += _PyInstruction_GetLength(code, i);
1891
0
                continue;
1892
0
            }
1893
0
            remove_per_instruction_tools(code, i, removed_per_instruction_tools);
1894
0
            i += _PyInstruction_GetLength(code, i);
1895
0
        }
1896
0
    }
1897
#ifdef INSTRUMENT_DEBUG
1898
    sanity_check_instrumentation(code);
1899
#endif
1900
1901
0
    uint8_t new_line_tools = new_events.tools[PY_MONITORING_EVENT_LINE];
1902
0
    uint8_t new_per_instruction_tools = new_events.tools[PY_MONITORING_EVENT_INSTRUCTION];
1903
1904
0
    if (new_line_tools) {
1905
0
        _PyCoLineInstrumentationData *line_data = code->_co_monitoring->lines;
1906
0
        for (int i = code->_co_firsttraceable; i < code_len;) {
1907
0
            if (_PyCode_GetOriginalOpcode(line_data, i)) {
1908
0
                add_line_tools(code, i, new_line_tools);
1909
0
            }
1910
0
            i += _PyInstruction_GetLength(code, i);
1911
0
        }
1912
0
    }
1913
0
    if (new_per_instruction_tools) {
1914
0
        for (int i = code->_co_firsttraceable; i < code_len;) {
1915
0
            int opcode = _Py_GetBaseCodeUnit(code, i).op.code;
1916
0
            if (opcode == RESUME || opcode == END_FOR) {
1917
0
                i += _PyInstruction_GetLength(code, i);
1918
0
                continue;
1919
0
            }
1920
0
            add_per_instruction_tools(code, i, new_per_instruction_tools);
1921
0
            i += _PyInstruction_GetLength(code, i);
1922
0
        }
1923
0
    }
1924
1925
0
done:
1926
0
    FT_ATOMIC_STORE_UINTPTR_RELEASE(code->_co_instrumentation_version,
1927
0
                                    global_version(interp));
1928
1929
#ifdef INSTRUMENT_DEBUG
1930
    sanity_check_instrumentation(code);
1931
#endif
1932
0
    return 0;
1933
0
}
1934
1935
static int
1936
instrument_lock_held(PyCodeObject *code, PyInterpreterState *interp)
1937
44.4k
{
1938
44.4k
    ASSERT_WORLD_STOPPED_OR_LOCKED(code);
1939
1940
44.4k
    if (is_version_up_to_date(code, interp)) {
1941
44.4k
        assert(
1942
44.4k
            interp->ceval.instrumentation_version == 0 ||
1943
44.4k
            instrumentation_cross_checks(interp, code)
1944
44.4k
        );
1945
44.4k
        return 0;
1946
44.4k
    }
1947
1948
0
    return force_instrument_lock_held(code, interp);
1949
44.4k
}
1950
1951
int
1952
_Py_Instrument(PyCodeObject *code, PyInterpreterState *interp)
1953
44.4k
{
1954
44.4k
    int res;
1955
44.4k
    LOCK_CODE(code);
1956
44.4k
    res = instrument_lock_held(code, interp);
1957
44.4k
    UNLOCK_CODE();
1958
44.4k
    return res;
1959
44.4k
}
1960
1961
#define C_RETURN_EVENTS \
1962
0
    ((1 << PY_MONITORING_EVENT_C_RETURN) | \
1963
0
     (1 << PY_MONITORING_EVENT_C_RAISE))
1964
1965
#define C_CALL_EVENTS \
1966
0
    (C_RETURN_EVENTS | (1 << PY_MONITORING_EVENT_CALL))
1967
1968
1969
static int
1970
instrument_all_executing_code_objects(PyInterpreterState *interp)
1971
0
{
1972
0
    ASSERT_WORLD_STOPPED();
1973
1974
0
    int err = 0;
1975
0
    _Py_FOR_EACH_TSTATE_BEGIN(interp, ts) {
1976
0
        _PyInterpreterFrame *frame = ts->current_frame;
1977
0
        while (frame) {
1978
0
            if (frame->owner < FRAME_OWNED_BY_INTERPRETER) {
1979
0
                err = instrument_lock_held(_PyFrame_GetCode(frame), interp);
1980
0
                if (err) {
1981
0
                    goto done;
1982
0
                }
1983
0
            }
1984
0
            frame = frame->previous;
1985
0
        }
1986
0
    }
1987
0
done:
1988
0
    _Py_FOR_EACH_TSTATE_END(interp);
1989
0
    return err;
1990
0
}
1991
1992
static void
1993
set_events(_Py_GlobalMonitors *m, int tool_id, _PyMonitoringEventSet events)
1994
0
{
1995
0
    assert(0 <= tool_id && tool_id < PY_MONITORING_TOOL_IDS);
1996
0
    for (int e = 0; e < _PY_MONITORING_UNGROUPED_EVENTS; e++) {
1997
0
        uint8_t *tools = &m->tools[e];
1998
0
        int active = (events >> e) & 1;
1999
0
        *tools &= ~(1 << tool_id);
2000
0
        *tools |= (active << tool_id);
2001
0
    }
2002
0
}
2003
2004
static void
2005
set_local_events(_Py_LocalMonitors *m, int tool_id, _PyMonitoringEventSet events)
2006
0
{
2007
0
    assert(0 <= tool_id && tool_id < PY_MONITORING_TOOL_IDS);
2008
0
    for (int e = 0; e < _PY_MONITORING_UNGROUPED_EVENTS; e++) {
2009
0
        uint8_t *tools = &m->tools[e];
2010
0
        int val = (events >> e) & 1;
2011
0
        *tools &= ~(1 << tool_id);
2012
0
        *tools |= (val << tool_id);
2013
0
    }
2014
0
}
2015
2016
static int
2017
check_tool(PyInterpreterState *interp, int tool_id)
2018
0
{
2019
0
    if (tool_id < PY_MONITORING_SYS_PROFILE_ID &&
2020
0
        interp->monitoring_tool_names[tool_id] == NULL)
2021
0
    {
2022
0
        PyErr_Format(PyExc_ValueError, "tool %d is not in use", tool_id);
2023
0
        return -1;
2024
0
    }
2025
0
    return 0;
2026
0
}
2027
2028
/* We share the eval-breaker with flags, so the monitoring
2029
 * version goes in the top 24 bits */
2030
0
#define MONITORING_VERSION_INCREMENT (1 << _PY_EVAL_EVENTS_BITS)
2031
2032
int
2033
_PyMonitoring_SetEvents(int tool_id, _PyMonitoringEventSet events)
2034
0
{
2035
0
    ASSERT_WORLD_STOPPED();
2036
0
    assert(0 <= tool_id && tool_id < PY_MONITORING_TOOL_IDS);
2037
0
    PyThreadState *tstate = _PyThreadState_GET();
2038
0
    PyInterpreterState *interp = tstate->interp;
2039
0
    assert(events < (1 << _PY_MONITORING_UNGROUPED_EVENTS));
2040
0
    if (check_tool(interp, tool_id)) {
2041
0
        return -1;
2042
0
    }
2043
2044
0
    uint32_t existing_events = get_events(&interp->monitors, tool_id);
2045
0
    if (existing_events == events) {
2046
0
        return 0;
2047
0
    }
2048
0
    uint32_t new_version = global_version(interp) + MONITORING_VERSION_INCREMENT;
2049
0
    if (new_version == 0) {
2050
0
        PyErr_Format(PyExc_OverflowError, "events set too many times");
2051
0
        return -1;
2052
0
    }
2053
0
    set_events(&interp->monitors, tool_id, events);
2054
0
    set_global_version(tstate, new_version);
2055
#ifdef _Py_TIER2
2056
    _Py_Executors_InvalidateAll(interp, 1);
2057
#endif
2058
0
    return instrument_all_executing_code_objects(interp);
2059
0
}
2060
2061
int
2062
_PyMonitoring_SetLocalEvents(PyCodeObject *code, int tool_id, _PyMonitoringEventSet events)
2063
0
{
2064
0
    ASSERT_WORLD_STOPPED();
2065
2066
0
    assert(0 <= tool_id && tool_id < PY_MONITORING_TOOL_IDS);
2067
0
    PyInterpreterState *interp = _PyInterpreterState_GET();
2068
0
    assert(events < (1 << _PY_MONITORING_UNGROUPED_EVENTS));
2069
0
    if (code->_co_firsttraceable >= Py_SIZE(code)) {
2070
0
        PyErr_Format(PyExc_SystemError, "cannot instrument shim code object '%U'", code->co_name);
2071
0
        return -1;
2072
0
    }
2073
0
    if (check_tool(interp, tool_id)) {
2074
0
        return -1;
2075
0
    }
2076
2077
0
    if (allocate_instrumentation_data(code)) {
2078
0
        return -1;
2079
0
    }
2080
2081
0
    code->_co_monitoring->tool_versions[tool_id] = interp->monitoring_tool_versions[tool_id];
2082
2083
0
    _Py_LocalMonitors *local = &code->_co_monitoring->local_monitors;
2084
0
    uint32_t existing_events = get_local_events(local, tool_id);
2085
0
    if (existing_events == events) {
2086
0
        return 0;
2087
0
    }
2088
0
    set_local_events(local, tool_id, events);
2089
2090
0
    return force_instrument_lock_held(code, interp);
2091
0
}
2092
2093
int
2094
_PyMonitoring_GetLocalEvents(PyCodeObject *code, int tool_id, _PyMonitoringEventSet *events)
2095
0
{
2096
0
    assert(0 <= tool_id && tool_id < PY_MONITORING_TOOL_IDS);
2097
0
    PyInterpreterState *interp = _PyInterpreterState_GET();
2098
0
    if (check_tool(interp, tool_id)) {
2099
0
        return -1;
2100
0
    }
2101
0
    if (code->_co_monitoring == NULL) {
2102
0
        *events = 0;
2103
0
        return 0;
2104
0
    }
2105
0
    _Py_LocalMonitors *local = &code->_co_monitoring->local_monitors;
2106
0
    *events = get_local_events(local, tool_id);
2107
0
    return 0;
2108
0
}
2109
2110
int _PyMonitoring_ClearToolId(int tool_id)
2111
0
{
2112
0
    assert(0 <= tool_id && tool_id < PY_MONITORING_TOOL_IDS);
2113
0
    PyInterpreterState *interp = _PyInterpreterState_GET();
2114
2115
0
    for (int i = 0; i < _PY_MONITORING_EVENTS; i++) {
2116
0
        PyObject *func = _PyMonitoring_RegisterCallback(tool_id, i, NULL);
2117
0
        if (func != NULL) {
2118
0
            Py_DECREF(func);
2119
0
        }
2120
0
    }
2121
2122
0
    _PyEval_StopTheWorld(interp);
2123
0
    if (_PyMonitoring_SetEvents(tool_id, 0) < 0) {
2124
0
        _PyEval_StartTheWorld(interp);
2125
0
        return -1;
2126
0
    }
2127
2128
0
    uint32_t version = global_version(interp) + MONITORING_VERSION_INCREMENT;
2129
0
    if (version == 0) {
2130
0
        PyErr_Format(PyExc_OverflowError, "events set too many times");
2131
0
        _PyEval_StartTheWorld(interp);
2132
0
        return -1;
2133
0
    }
2134
2135
    // monitoring_tool_versions[tool_id] is set to latest global version here to
2136
    //   1. invalidate local events on all existing code objects
2137
    //   2. be ready for the next call to set local events
2138
0
    interp->monitoring_tool_versions[tool_id] = version;
2139
2140
    // Set the new global version so all the code objects can refresh the
2141
    // instrumentation.
2142
0
    set_global_version(_PyThreadState_GET(), version);
2143
#ifdef _Py_TIER2
2144
    _Py_Executors_InvalidateAll(interp, 1);
2145
#endif
2146
0
    int res = instrument_all_executing_code_objects(interp);
2147
0
    _PyEval_StartTheWorld(interp);
2148
0
    return res;
2149
0
}
2150
2151
/*[clinic input]
2152
module monitoring
2153
[clinic start generated code]*/
2154
/*[clinic end generated code: output=da39a3ee5e6b4b0d input=37257f5987a360cf]*/
2155
/*[clinic end generated code]*/
2156
2157
#include "clinic/instrumentation.c.h"
2158
2159
static int
2160
check_valid_tool(int tool_id)
2161
0
{
2162
0
    if (tool_id < 0 || tool_id >= PY_MONITORING_SYS_PROFILE_ID) {
2163
0
        PyErr_Format(PyExc_ValueError, "invalid tool %d (must be between 0 and 5)", tool_id);
2164
0
        return -1;
2165
0
    }
2166
0
    return 0;
2167
0
}
2168
2169
/*[clinic input]
2170
monitoring.use_tool_id
2171
2172
    tool_id: int
2173
    name: object
2174
    /
2175
2176
[clinic start generated code]*/
2177
2178
static PyObject *
2179
monitoring_use_tool_id_impl(PyObject *module, int tool_id, PyObject *name)
2180
/*[clinic end generated code: output=30d76dc92b7cd653 input=ebc453761c621be1]*/
2181
0
{
2182
0
    if (check_valid_tool(tool_id))  {
2183
0
        return NULL;
2184
0
    }
2185
0
    if (!PyUnicode_Check(name)) {
2186
0
        PyErr_SetString(PyExc_ValueError, "tool name must be a str");
2187
0
        return NULL;
2188
0
    }
2189
0
    PyInterpreterState *interp = _PyInterpreterState_GET();
2190
0
    if (interp->monitoring_tool_names[tool_id] != NULL) {
2191
0
        PyErr_Format(PyExc_ValueError, "tool %d is already in use", tool_id);
2192
0
        return NULL;
2193
0
    }
2194
0
    interp->monitoring_tool_names[tool_id] = Py_NewRef(name);
2195
0
    Py_RETURN_NONE;
2196
0
}
2197
2198
/*[clinic input]
2199
monitoring.clear_tool_id
2200
2201
    tool_id: int
2202
    /
2203
2204
[clinic start generated code]*/
2205
2206
static PyObject *
2207
monitoring_clear_tool_id_impl(PyObject *module, int tool_id)
2208
/*[clinic end generated code: output=04defc23470b1be7 input=af643d6648a66163]*/
2209
0
{
2210
0
    if (check_valid_tool(tool_id))  {
2211
0
        return NULL;
2212
0
    }
2213
2214
0
    PyInterpreterState *interp = _PyInterpreterState_GET();
2215
2216
0
    if (interp->monitoring_tool_names[tool_id] != NULL) {
2217
0
        if (_PyMonitoring_ClearToolId(tool_id) < 0) {
2218
0
            return NULL;
2219
0
        }
2220
0
    }
2221
2222
0
    Py_RETURN_NONE;
2223
0
}
2224
2225
/*[clinic input]
2226
monitoring.free_tool_id
2227
2228
    tool_id: int
2229
    /
2230
2231
[clinic start generated code]*/
2232
2233
static PyObject *
2234
monitoring_free_tool_id_impl(PyObject *module, int tool_id)
2235
/*[clinic end generated code: output=86c2d2a1219a8591 input=a23fb6be3a8618e9]*/
2236
0
{
2237
0
    if (check_valid_tool(tool_id))  {
2238
0
        return NULL;
2239
0
    }
2240
0
    PyInterpreterState *interp = _PyInterpreterState_GET();
2241
2242
0
    if (interp->monitoring_tool_names[tool_id] != NULL) {
2243
0
        if (_PyMonitoring_ClearToolId(tool_id) < 0) {
2244
0
            return NULL;
2245
0
        }
2246
0
    }
2247
2248
0
    Py_CLEAR(interp->monitoring_tool_names[tool_id]);
2249
0
    Py_RETURN_NONE;
2250
0
}
2251
2252
/*[clinic input]
2253
monitoring.get_tool
2254
2255
    tool_id: int
2256
    /
2257
2258
[clinic start generated code]*/
2259
2260
static PyObject *
2261
monitoring_get_tool_impl(PyObject *module, int tool_id)
2262
/*[clinic end generated code: output=1c05a98b404a9a16 input=eeee9bebd0bcae9d]*/
2263
2264
/*[clinic end generated code]*/
2265
0
{
2266
0
    if (check_valid_tool(tool_id))  {
2267
0
        return NULL;
2268
0
    }
2269
0
    PyInterpreterState *interp = _PyInterpreterState_GET();
2270
0
    PyObject *name = interp->monitoring_tool_names[tool_id];
2271
0
    if (name == NULL) {
2272
0
        Py_RETURN_NONE;
2273
0
    }
2274
0
    return Py_NewRef(name);
2275
0
}
2276
2277
/*[clinic input]
2278
monitoring.register_callback
2279
2280
2281
    tool_id: int
2282
    event: int
2283
    func: object
2284
    /
2285
2286
[clinic start generated code]*/
2287
2288
static PyObject *
2289
monitoring_register_callback_impl(PyObject *module, int tool_id, int event,
2290
                                  PyObject *func)
2291
/*[clinic end generated code: output=e64daa363004030c input=df6d70ea4cf81007]*/
2292
0
{
2293
0
    if (check_valid_tool(tool_id))  {
2294
0
        return NULL;
2295
0
    }
2296
0
    if (_Py_popcount32(event) != 1) {
2297
0
        PyErr_SetString(PyExc_ValueError, "The callback can only be set for one event at a time");
2298
0
        return NULL;
2299
0
    }
2300
0
    int event_id = _Py_bit_length(event)-1;
2301
0
    if (event_id < 0 || event_id >= _PY_MONITORING_EVENTS) {
2302
0
        PyErr_Format(PyExc_ValueError, "invalid event %d", event);
2303
0
        return NULL;
2304
0
    }
2305
0
    if (PySys_Audit("sys.monitoring.register_callback", "O", func) < 0) {
2306
0
        return NULL;
2307
0
    }
2308
0
    if (func == Py_None) {
2309
0
        func = NULL;
2310
0
    }
2311
0
    func = _PyMonitoring_RegisterCallback(tool_id, event_id, func);
2312
0
    if (func == NULL) {
2313
0
        Py_RETURN_NONE;
2314
0
    }
2315
0
    return func;
2316
0
}
2317
2318
/*[clinic input]
2319
monitoring.get_events -> int
2320
2321
    tool_id: int
2322
    /
2323
2324
[clinic start generated code]*/
2325
2326
static int
2327
monitoring_get_events_impl(PyObject *module, int tool_id)
2328
/*[clinic end generated code: output=4450cc13f826c8c0 input=a64b238f76c4b2f7]*/
2329
0
{
2330
0
    if (check_valid_tool(tool_id))  {
2331
0
        return -1;
2332
0
    }
2333
0
    _Py_GlobalMonitors *m = &_PyInterpreterState_GET()->monitors;
2334
0
    _PyMonitoringEventSet event_set = get_events(m, tool_id);
2335
0
    return event_set;
2336
0
}
2337
2338
/*[clinic input]
2339
monitoring.set_events
2340
2341
    tool_id: int
2342
    event_set: int
2343
    /
2344
2345
[clinic start generated code]*/
2346
2347
static PyObject *
2348
monitoring_set_events_impl(PyObject *module, int tool_id, int event_set)
2349
/*[clinic end generated code: output=1916c1e49cfb5bdb input=a77ba729a242142b]*/
2350
0
{
2351
0
    if (check_valid_tool(tool_id))  {
2352
0
        return NULL;
2353
0
    }
2354
0
    if (event_set < 0 || event_set >= (1 << _PY_MONITORING_EVENTS)) {
2355
0
        PyErr_Format(PyExc_ValueError, "invalid event set 0x%x", event_set);
2356
0
        return NULL;
2357
0
    }
2358
0
    if ((event_set & C_RETURN_EVENTS) && (event_set & C_CALL_EVENTS) != C_CALL_EVENTS) {
2359
0
        PyErr_Format(PyExc_ValueError, "cannot set C_RETURN or C_RAISE events independently");
2360
0
        return NULL;
2361
0
    }
2362
0
    event_set &= ~C_RETURN_EVENTS;
2363
0
    if (event_set & (1 << PY_MONITORING_EVENT_BRANCH)) {
2364
0
        event_set &= ~(1 << PY_MONITORING_EVENT_BRANCH);
2365
0
        event_set |= (1 << PY_MONITORING_EVENT_BRANCH_RIGHT) | (1 << PY_MONITORING_EVENT_BRANCH_LEFT);
2366
0
    }
2367
0
    PyInterpreterState *interp = _PyInterpreterState_GET();
2368
0
    _PyEval_StopTheWorld(interp);
2369
0
    int err = _PyMonitoring_SetEvents(tool_id, event_set);
2370
0
    _PyEval_StartTheWorld(interp);
2371
0
    if (err) {
2372
0
        return NULL;
2373
0
    }
2374
0
    Py_RETURN_NONE;
2375
0
}
2376
2377
/*[clinic input]
2378
monitoring.get_local_events -> int
2379
2380
    tool_id: int
2381
    code: object
2382
    /
2383
2384
[clinic start generated code]*/
2385
2386
static int
2387
monitoring_get_local_events_impl(PyObject *module, int tool_id,
2388
                                 PyObject *code)
2389
/*[clinic end generated code: output=d3e92c1c9c1de8f9 input=bb0f927530386a94]*/
2390
0
{
2391
0
    if (!PyCode_Check(code)) {
2392
0
        PyErr_Format(
2393
0
            PyExc_TypeError,
2394
0
            "code must be a code object"
2395
0
        );
2396
0
        return -1;
2397
0
    }
2398
0
    if (check_valid_tool(tool_id))  {
2399
0
        return -1;
2400
0
    }
2401
0
    _PyMonitoringEventSet event_set = 0;
2402
0
    _PyCoMonitoringData *data = ((PyCodeObject *)code)->_co_monitoring;
2403
0
    if (data != NULL) {
2404
0
        for (int e = 0; e < _PY_MONITORING_UNGROUPED_EVENTS; e++) {
2405
0
            if ((data->local_monitors.tools[e] >> tool_id) & 1) {
2406
0
                event_set |= (1 << e);
2407
0
            }
2408
0
        }
2409
0
    }
2410
0
    return event_set;
2411
0
}
2412
2413
/*[clinic input]
2414
monitoring.set_local_events
2415
2416
    tool_id: int
2417
    code: object
2418
    event_set: int
2419
    /
2420
2421
[clinic start generated code]*/
2422
2423
static PyObject *
2424
monitoring_set_local_events_impl(PyObject *module, int tool_id,
2425
                                 PyObject *code, int event_set)
2426
/*[clinic end generated code: output=68cc755a65dfea99 input=5655ecd78d937a29]*/
2427
0
{
2428
0
    if (!PyCode_Check(code)) {
2429
0
        PyErr_Format(
2430
0
            PyExc_TypeError,
2431
0
            "code must be a code object"
2432
0
        );
2433
0
        return NULL;
2434
0
    }
2435
0
    if (check_valid_tool(tool_id))  {
2436
0
        return NULL;
2437
0
    }
2438
0
    if ((event_set & C_RETURN_EVENTS) && (event_set & C_CALL_EVENTS) != C_CALL_EVENTS) {
2439
0
        PyErr_Format(PyExc_ValueError, "cannot set C_RETURN or C_RAISE events independently");
2440
0
        return NULL;
2441
0
    }
2442
0
    event_set &= ~C_RETURN_EVENTS;
2443
0
    if (event_set & (1 << PY_MONITORING_EVENT_BRANCH)) {
2444
0
        event_set &= ~(1 << PY_MONITORING_EVENT_BRANCH);
2445
0
        event_set |= (1 << PY_MONITORING_EVENT_BRANCH_RIGHT) | (1 << PY_MONITORING_EVENT_BRANCH_LEFT);
2446
0
    }
2447
0
    if (event_set < 0 || event_set >= (1 << _PY_MONITORING_UNGROUPED_EVENTS)) {
2448
0
        PyErr_Format(PyExc_ValueError, "invalid local event set 0x%x", event_set);
2449
0
        return NULL;
2450
0
    }
2451
2452
0
    PyInterpreterState *interp = _PyInterpreterState_GET();
2453
0
    _PyEval_StopTheWorld(interp);
2454
0
    int err = _PyMonitoring_SetLocalEvents((PyCodeObject*)code, tool_id, event_set);
2455
0
    _PyEval_StartTheWorld(interp);
2456
0
    if (err) {
2457
0
        return NULL;
2458
0
    }
2459
0
    Py_RETURN_NONE;
2460
0
}
2461
2462
/*[clinic input]
2463
monitoring.restart_events
2464
2465
[clinic start generated code]*/
2466
2467
static PyObject *
2468
monitoring_restart_events_impl(PyObject *module)
2469
/*[clinic end generated code: output=e025dd5ba33314c4 input=add8a855063c8008]*/
2470
0
{
2471
    /* We want to ensure that:
2472
     * last restart version > instrumented version for all code objects
2473
     * last restart version < current version
2474
     */
2475
0
    PyThreadState *tstate = _PyThreadState_GET();
2476
0
    PyInterpreterState *interp = tstate->interp;
2477
2478
0
    _PyEval_StopTheWorld(interp);
2479
0
    uint32_t restart_version = global_version(interp) + MONITORING_VERSION_INCREMENT;
2480
0
    uint32_t new_version = restart_version + MONITORING_VERSION_INCREMENT;
2481
0
    if (new_version <= MONITORING_VERSION_INCREMENT) {
2482
0
        _PyEval_StartTheWorld(interp);
2483
0
        PyErr_Format(PyExc_OverflowError, "events set too many times");
2484
0
        return NULL;
2485
0
    }
2486
0
    interp->last_restart_version = restart_version;
2487
0
    set_global_version(tstate, new_version);
2488
#ifdef _Py_TIER2
2489
    _Py_Executors_InvalidateAll(interp, 1);
2490
#endif
2491
0
    int res = instrument_all_executing_code_objects(interp);
2492
0
    _PyEval_StartTheWorld(interp);
2493
2494
0
    if (res) {
2495
0
        return NULL;
2496
0
    }
2497
0
    Py_RETURN_NONE;
2498
0
}
2499
2500
static int
2501
add_power2_constant(PyObject *obj, const char *name, int i)
2502
703
{
2503
703
    PyObject *val = PyLong_FromLong(1<<i);
2504
703
    if (val == NULL) {
2505
0
        return -1;
2506
0
    }
2507
703
    int err = PyObject_SetAttrString(obj, name, val);
2508
703
    Py_DECREF(val);
2509
703
    return err;
2510
703
}
2511
2512
/*[clinic input]
2513
monitoring._all_events
2514
[clinic start generated code]*/
2515
2516
static PyObject *
2517
monitoring__all_events_impl(PyObject *module)
2518
/*[clinic end generated code: output=6b7581e2dbb690f6 input=62ee9672c17b7f0e]*/
2519
0
{
2520
0
    PyInterpreterState *interp = _PyInterpreterState_GET();
2521
0
    PyObject *res = PyDict_New();
2522
0
    if (res == NULL) {
2523
0
        return NULL;
2524
0
    }
2525
0
    for (int e = 0; e < _PY_MONITORING_UNGROUPED_EVENTS; e++) {
2526
0
        uint8_t tools = interp->monitors.tools[e];
2527
0
        if (tools == 0) {
2528
0
            continue;
2529
0
        }
2530
0
        PyObject *tools_obj = PyLong_FromLong(tools);
2531
0
        assert(tools_obj != NULL);
2532
0
        int err = PyDict_SetItemString(res, event_names[e], tools_obj);
2533
0
        Py_DECREF(tools_obj);
2534
0
        if (err < 0) {
2535
0
            Py_DECREF(res);
2536
0
            return NULL;
2537
0
        }
2538
0
    }
2539
0
    return res;
2540
0
}
2541
2542
static PyMethodDef methods[] = {
2543
    MONITORING_USE_TOOL_ID_METHODDEF
2544
    MONITORING_CLEAR_TOOL_ID_METHODDEF
2545
    MONITORING_FREE_TOOL_ID_METHODDEF
2546
    MONITORING_GET_TOOL_METHODDEF
2547
    MONITORING_REGISTER_CALLBACK_METHODDEF
2548
    MONITORING_GET_EVENTS_METHODDEF
2549
    MONITORING_SET_EVENTS_METHODDEF
2550
    MONITORING_GET_LOCAL_EVENTS_METHODDEF
2551
    MONITORING_SET_LOCAL_EVENTS_METHODDEF
2552
    MONITORING_RESTART_EVENTS_METHODDEF
2553
    MONITORING__ALL_EVENTS_METHODDEF
2554
    {NULL, NULL}  // sentinel
2555
};
2556
2557
static struct PyModuleDef monitoring_module = {
2558
    PyModuleDef_HEAD_INIT,
2559
    .m_name = "sys.monitoring",
2560
    .m_size = -1, /* multiple "initialization" just copies the module dict. */
2561
    .m_methods = methods,
2562
};
2563
2564
PyObject *_Py_CreateMonitoringObject(void)
2565
37
{
2566
37
    PyObject *mod = _PyModule_CreateInitialized(&monitoring_module, PYTHON_API_VERSION);
2567
37
    if (mod == NULL) {
2568
0
        return NULL;
2569
0
    }
2570
37
    if (PyObject_SetAttrString(mod, "DISABLE", &_PyInstrumentation_DISABLE)) {
2571
0
        goto error;
2572
0
    }
2573
37
    if (PyObject_SetAttrString(mod, "MISSING", &_PyInstrumentation_MISSING)) {
2574
0
        goto error;
2575
0
    }
2576
37
    PyObject *events = _PyNamespace_New(NULL);
2577
37
    if (events == NULL) {
2578
0
        goto error;
2579
0
    }
2580
37
    int err = PyObject_SetAttrString(mod, "events", events);
2581
37
    Py_DECREF(events);
2582
37
    if (err) {
2583
0
        goto error;
2584
0
    }
2585
740
    for (int i = 0; i < _PY_MONITORING_EVENTS; i++) {
2586
703
        if (add_power2_constant(events, event_names[i], i)) {
2587
0
            goto error;
2588
0
        }
2589
703
    }
2590
37
    err = PyObject_SetAttrString(events, "NO_EVENTS", _PyLong_GetZero());
2591
37
    if (err) goto error;
2592
37
    PyObject *val = PyLong_FromLong(PY_MONITORING_DEBUGGER_ID);
2593
37
    assert(val != NULL); /* Can't return NULL because the int is small. */
2594
37
    err = PyObject_SetAttrString(mod, "DEBUGGER_ID", val);
2595
37
    Py_DECREF(val);
2596
37
    if (err) goto error;
2597
37
    val = PyLong_FromLong(PY_MONITORING_COVERAGE_ID);
2598
37
    assert(val != NULL);
2599
37
    err = PyObject_SetAttrString(mod, "COVERAGE_ID", val);
2600
37
    Py_DECREF(val);
2601
37
    if (err) goto error;
2602
37
    val = PyLong_FromLong(PY_MONITORING_PROFILER_ID);
2603
37
    assert(val != NULL);
2604
37
    err = PyObject_SetAttrString(mod, "PROFILER_ID", val);
2605
37
    Py_DECREF(val);
2606
37
    if (err) goto error;
2607
37
    val = PyLong_FromLong(PY_MONITORING_OPTIMIZER_ID);
2608
37
    assert(val != NULL);
2609
37
    err = PyObject_SetAttrString(mod, "OPTIMIZER_ID", val);
2610
37
    Py_DECREF(val);
2611
37
    if (err) goto error;
2612
37
    return mod;
2613
0
error:
2614
0
    Py_DECREF(mod);
2615
0
    return NULL;
2616
37
}
2617
2618
2619
static int
2620
capi_call_instrumentation(PyMonitoringState *state, PyObject *codelike, int32_t offset,
2621
                          PyObject **args, Py_ssize_t nargs, int event)
2622
0
{
2623
0
    PyThreadState *tstate = _PyThreadState_GET();
2624
0
    PyInterpreterState *interp = tstate->interp;
2625
2626
0
    uint8_t tools = state->active;
2627
0
    assert(args[1] == NULL);
2628
0
    args[1] = codelike;
2629
0
    if (offset < 0) {
2630
0
        PyErr_SetString(PyExc_ValueError, "offset must be non-negative");
2631
0
        return -1;
2632
0
    }
2633
0
    if (event != PY_MONITORING_EVENT_LINE) {
2634
0
        PyObject *offset_obj = PyLong_FromLong(offset);
2635
0
        if (offset_obj == NULL) {
2636
0
            return -1;
2637
0
        }
2638
0
        assert(args[2] == NULL);
2639
0
        args[2] = offset_obj;
2640
0
    }
2641
0
    size_t nargsf = (size_t) nargs | PY_VECTORCALL_ARGUMENTS_OFFSET;
2642
0
    PyObject **callargs = &args[1];
2643
0
    int err = 0;
2644
2645
0
    while (tools) {
2646
0
        int tool = most_significant_bit(tools);
2647
0
        assert(tool >= 0 && tool < 8);
2648
0
        assert(tools & (1 << tool));
2649
0
        tools ^= (1 << tool);
2650
0
        int res = call_one_instrument(interp, tstate, callargs, nargsf, tool, event);
2651
0
        if (res == 0) {
2652
            /* Nothing to do */
2653
0
        }
2654
0
        else if (res < 0) {
2655
            /* error */
2656
0
            err = -1;
2657
0
            break;
2658
0
        }
2659
0
        else {
2660
            /* DISABLE */
2661
0
            if (!PY_MONITORING_IS_INSTRUMENTED_EVENT(event)) {
2662
0
                PyErr_Format(PyExc_ValueError,
2663
0
                             "Cannot disable %s events. Callback removed.",
2664
0
                             event_names[event]);
2665
                /* Clear tool to prevent infinite loop */
2666
0
                Py_CLEAR(interp->monitoring_callables[tool][event]);
2667
0
                err = -1;
2668
0
                break;
2669
0
            }
2670
0
            else {
2671
0
                state->active &= ~(1 << tool);
2672
0
            }
2673
0
        }
2674
0
    }
2675
0
    return err;
2676
0
}
2677
2678
int
2679
PyMonitoring_EnterScope(PyMonitoringState *state_array, uint64_t *version,
2680
                         const uint8_t *event_types, Py_ssize_t length)
2681
0
{
2682
0
    PyInterpreterState *interp = _PyInterpreterState_GET();
2683
0
    if (global_version(interp) == *version) {
2684
0
        return 0;
2685
0
    }
2686
2687
0
    _Py_GlobalMonitors *m = &interp->monitors;
2688
0
    for (Py_ssize_t i = 0; i < length; i++) {
2689
0
        int event = event_types[i];
2690
0
        state_array[i].active = m->tools[event];
2691
0
    }
2692
0
    *version = global_version(interp);
2693
0
    return 0;
2694
0
}
2695
2696
int
2697
PyMonitoring_ExitScope(void)
2698
0
{
2699
0
    return 0;
2700
0
}
2701
2702
int
2703
_PyMonitoring_FirePyStartEvent(PyMonitoringState *state, PyObject *codelike, int32_t offset)
2704
0
{
2705
0
    assert(state->active);
2706
0
    PyObject *args[3] = { NULL, NULL, NULL };
2707
0
    return capi_call_instrumentation(state, codelike, offset, args, 2,
2708
0
                                     PY_MONITORING_EVENT_PY_START);
2709
0
}
2710
2711
int
2712
_PyMonitoring_FirePyResumeEvent(PyMonitoringState *state, PyObject *codelike, int32_t offset)
2713
0
{
2714
0
    assert(state->active);
2715
0
    PyObject *args[3] = { NULL, NULL, NULL };
2716
0
    return capi_call_instrumentation(state, codelike, offset, args, 2,
2717
0
                                     PY_MONITORING_EVENT_PY_RESUME);
2718
0
}
2719
2720
2721
2722
int
2723
_PyMonitoring_FirePyReturnEvent(PyMonitoringState *state, PyObject *codelike, int32_t offset,
2724
                                PyObject* retval)
2725
0
{
2726
0
    assert(state->active);
2727
0
    PyObject *args[4] = { NULL, NULL, NULL, retval };
2728
0
    return capi_call_instrumentation(state, codelike, offset, args, 3,
2729
0
                                     PY_MONITORING_EVENT_PY_RETURN);
2730
0
}
2731
2732
int
2733
_PyMonitoring_FirePyYieldEvent(PyMonitoringState *state, PyObject *codelike, int32_t offset,
2734
                               PyObject* retval)
2735
0
{
2736
0
    assert(state->active);
2737
0
    PyObject *args[4] = { NULL, NULL, NULL, retval };
2738
0
    return capi_call_instrumentation(state, codelike, offset, args, 3,
2739
0
                                     PY_MONITORING_EVENT_PY_YIELD);
2740
0
}
2741
2742
int
2743
_PyMonitoring_FireCallEvent(PyMonitoringState *state, PyObject *codelike, int32_t offset,
2744
                            PyObject* callable, PyObject *arg0)
2745
0
{
2746
0
    assert(state->active);
2747
0
    PyObject *args[5] = { NULL, NULL, NULL, callable, arg0 };
2748
0
    return capi_call_instrumentation(state, codelike, offset, args, 4,
2749
0
                                     PY_MONITORING_EVENT_CALL);
2750
0
}
2751
2752
int
2753
_PyMonitoring_FireLineEvent(PyMonitoringState *state, PyObject *codelike, int32_t offset,
2754
                            int lineno)
2755
0
{
2756
0
    assert(state->active);
2757
0
    PyObject *lno = PyLong_FromLong(lineno);
2758
0
    if (lno == NULL) {
2759
0
        return -1;
2760
0
    }
2761
0
    PyObject *args[3] = { NULL, NULL, lno };
2762
0
    int res= capi_call_instrumentation(state, codelike, offset, args, 2,
2763
0
                                       PY_MONITORING_EVENT_LINE);
2764
0
    Py_DECREF(lno);
2765
0
    return res;
2766
0
}
2767
2768
int
2769
_PyMonitoring_FireJumpEvent(PyMonitoringState *state, PyObject *codelike, int32_t offset,
2770
                            PyObject *target_offset)
2771
0
{
2772
0
    assert(state->active);
2773
0
    PyObject *args[4] = { NULL, NULL, NULL, target_offset };
2774
0
    return capi_call_instrumentation(state, codelike, offset, args, 3,
2775
0
                                     PY_MONITORING_EVENT_JUMP);
2776
0
}
2777
2778
int
2779
_PyMonitoring_FireBranchEvent(PyMonitoringState *state, PyObject *codelike, int32_t offset,
2780
                              PyObject *target_offset)
2781
0
{
2782
0
    assert(state->active);
2783
0
    PyObject *args[4] = { NULL, NULL, NULL, target_offset };
2784
0
    return capi_call_instrumentation(state, codelike, offset, args, 3,
2785
0
                                     PY_MONITORING_EVENT_BRANCH_RIGHT);
2786
0
}
2787
2788
int
2789
_PyMonitoring_FireBranchRightEvent(PyMonitoringState *state, PyObject *codelike, int32_t offset,
2790
                              PyObject *target_offset)
2791
0
{
2792
0
    assert(state->active);
2793
0
    PyObject *args[4] = { NULL, NULL, NULL, target_offset };
2794
0
    return capi_call_instrumentation(state, codelike, offset, args, 3,
2795
0
                                     PY_MONITORING_EVENT_BRANCH_RIGHT);
2796
0
}
2797
2798
int
2799
_PyMonitoring_FireBranchLeftEvent(PyMonitoringState *state, PyObject *codelike, int32_t offset,
2800
                              PyObject *target_offset)
2801
0
{
2802
0
    assert(state->active);
2803
0
    PyObject *args[4] = { NULL, NULL, NULL, target_offset };
2804
0
    return capi_call_instrumentation(state, codelike, offset, args, 3,
2805
0
                                     PY_MONITORING_EVENT_BRANCH_LEFT);
2806
0
}
2807
2808
int
2809
_PyMonitoring_FireCReturnEvent(PyMonitoringState *state, PyObject *codelike, int32_t offset,
2810
                               PyObject *retval)
2811
0
{
2812
0
    assert(state->active);
2813
0
    PyObject *args[4] = { NULL, NULL, NULL, retval };
2814
0
    return capi_call_instrumentation(state, codelike, offset, args, 3,
2815
0
                                     PY_MONITORING_EVENT_C_RETURN);
2816
0
}
2817
2818
static inline int
2819
0
exception_event_setup(PyObject **exc, int event) {
2820
0
    *exc = PyErr_GetRaisedException();
2821
0
    if (*exc == NULL) {
2822
0
        PyErr_Format(PyExc_ValueError,
2823
0
                     "Firing event %d with no exception set",
2824
0
                     event);
2825
0
        return -1;
2826
0
    }
2827
0
    return 0;
2828
0
}
2829
2830
2831
static inline int
2832
0
exception_event_teardown(int err, PyObject *exc) {
2833
0
    if (err == 0) {
2834
0
        PyErr_SetRaisedException(exc);
2835
0
    }
2836
0
    else {
2837
0
        assert(PyErr_Occurred());
2838
0
        Py_XDECREF(exc);
2839
0
    }
2840
0
    return err;
2841
0
}
2842
2843
int
2844
_PyMonitoring_FirePyThrowEvent(PyMonitoringState *state, PyObject *codelike, int32_t offset)
2845
0
{
2846
0
    int event = PY_MONITORING_EVENT_PY_THROW;
2847
0
    assert(state->active);
2848
0
    PyObject *exc;
2849
0
    if (exception_event_setup(&exc, event) < 0) {
2850
0
        return -1;
2851
0
    }
2852
0
    PyObject *args[4] = { NULL, NULL, NULL, exc };
2853
0
    int err = capi_call_instrumentation(state, codelike, offset, args, 3, event);
2854
0
    return exception_event_teardown(err, exc);
2855
0
}
2856
2857
int
2858
_PyMonitoring_FireRaiseEvent(PyMonitoringState *state, PyObject *codelike, int32_t offset)
2859
0
{
2860
0
    int event = PY_MONITORING_EVENT_RAISE;
2861
0
    assert(state->active);
2862
0
    PyObject *exc;
2863
0
    if (exception_event_setup(&exc, event) < 0) {
2864
0
        return -1;
2865
0
    }
2866
0
    PyObject *args[4] = { NULL, NULL, NULL, exc };
2867
0
    int err = capi_call_instrumentation(state, codelike, offset, args, 3, event);
2868
0
    return exception_event_teardown(err, exc);
2869
0
}
2870
2871
int
2872
_PyMonitoring_FireCRaiseEvent(PyMonitoringState *state, PyObject *codelike, int32_t offset)
2873
0
{
2874
0
    int event = PY_MONITORING_EVENT_C_RAISE;
2875
0
    assert(state->active);
2876
0
    PyObject *exc;
2877
0
    if (exception_event_setup(&exc, event) < 0) {
2878
0
        return -1;
2879
0
    }
2880
0
    PyObject *args[4] = { NULL, NULL, NULL, exc };
2881
0
    int err = capi_call_instrumentation(state, codelike, offset, args, 3, event);
2882
0
    return exception_event_teardown(err, exc);
2883
0
}
2884
2885
int
2886
_PyMonitoring_FireReraiseEvent(PyMonitoringState *state, PyObject *codelike, int32_t offset)
2887
0
{
2888
0
    int event = PY_MONITORING_EVENT_RERAISE;
2889
0
    assert(state->active);
2890
0
    PyObject *exc;
2891
0
    if (exception_event_setup(&exc, event) < 0) {
2892
0
        return -1;
2893
0
    }
2894
0
    PyObject *args[4] = { NULL, NULL, NULL, exc };
2895
0
    int err = capi_call_instrumentation(state, codelike, offset, args, 3, event);
2896
0
    return exception_event_teardown(err, exc);
2897
0
}
2898
2899
int
2900
_PyMonitoring_FireExceptionHandledEvent(PyMonitoringState *state, PyObject *codelike, int32_t offset)
2901
0
{
2902
0
    int event = PY_MONITORING_EVENT_EXCEPTION_HANDLED;
2903
0
    assert(state->active);
2904
0
    PyObject *exc;
2905
0
    if (exception_event_setup(&exc, event) < 0) {
2906
0
        return -1;
2907
0
    }
2908
0
    PyObject *args[4] = { NULL, NULL, NULL, exc };
2909
0
    int err = capi_call_instrumentation(state, codelike, offset, args, 3, event);
2910
0
    return exception_event_teardown(err, exc);
2911
0
}
2912
2913
int
2914
_PyMonitoring_FirePyUnwindEvent(PyMonitoringState *state, PyObject *codelike, int32_t offset)
2915
0
{
2916
0
    int event = PY_MONITORING_EVENT_PY_UNWIND;
2917
0
    assert(state->active);
2918
0
    PyObject *exc;
2919
0
    if (exception_event_setup(&exc, event) < 0) {
2920
0
        return -1;
2921
0
    }
2922
0
    PyObject *args[4] = { NULL, NULL, NULL, exc };
2923
0
    int err = capi_call_instrumentation(state, codelike, offset, args, 3, event);
2924
0
    return exception_event_teardown(err, exc);
2925
0
}
2926
2927
int
2928
_PyMonitoring_FireStopIterationEvent(PyMonitoringState *state, PyObject *codelike, int32_t offset, PyObject *value)
2929
0
{
2930
0
    int event = PY_MONITORING_EVENT_STOP_ITERATION;
2931
0
    assert(state->active);
2932
0
    assert(!PyErr_Occurred());
2933
0
    PyErr_SetObject(PyExc_StopIteration, value);
2934
0
    PyObject *exc;
2935
0
    if (exception_event_setup(&exc, event) < 0) {
2936
0
        return -1;
2937
0
    }
2938
0
    PyObject *args[4] = { NULL, NULL, NULL, exc };
2939
0
    int err = capi_call_instrumentation(state, codelike, offset, args, 3, event);
2940
0
    Py_DECREF(exc);
2941
0
    return exception_event_teardown(err, NULL);
2942
0
}
2943
2944
2945
2946
/* Handle legacy BRANCH event */
2947
2948
typedef struct _PyLegacyBranchEventHandler {
2949
    PyObject_HEAD
2950
    vectorcallfunc vectorcall;
2951
    PyObject *handler;
2952
    bool right;
2953
    int tool_id;
2954
} _PyLegacyBranchEventHandler;
2955
2956
0
#define _PyLegacyBranchEventHandler_CAST(op)    ((_PyLegacyBranchEventHandler *)(op))
2957
2958
static void
2959
dealloc_branch_handler(PyObject *op)
2960
0
{
2961
0
    _PyLegacyBranchEventHandler *self = _PyLegacyBranchEventHandler_CAST(op);
2962
0
    Py_CLEAR(self->handler);
2963
0
    PyObject_Free(self);
2964
0
}
2965
2966
static PyTypeObject _PyLegacyBranchEventHandler_Type = {
2967
    PyVarObject_HEAD_INIT(&PyType_Type, 0)
2968
    "sys.monitoring.branch_event_handler",
2969
    sizeof(_PyLegacyBranchEventHandler),
2970
    .tp_dealloc = dealloc_branch_handler,
2971
    .tp_vectorcall_offset = offsetof(_PyLegacyBranchEventHandler, vectorcall),
2972
    .tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE |
2973
        Py_TPFLAGS_HAVE_VECTORCALL | Py_TPFLAGS_DISALLOW_INSTANTIATION,
2974
    .tp_call = PyVectorcall_Call,
2975
};
2976
2977
2978
static PyObject *
2979
branch_handler_vectorcall(
2980
    PyObject *op, PyObject *const *args,
2981
    size_t nargsf, PyObject *kwnames
2982
0
) {
2983
0
    _PyLegacyBranchEventHandler *self = _PyLegacyBranchEventHandler_CAST(op);
2984
    // Find the other instrumented instruction and remove tool
2985
    // The spec (PEP 669) allows spurious events after a DISABLE,
2986
    // so a best effort is good enough.
2987
0
    assert(PyVectorcall_NARGS(nargsf) >= 3);
2988
0
    PyCodeObject *code = (PyCodeObject *)args[0];
2989
0
    int src_offset = PyLong_AsLong(args[1]);
2990
0
    if (PyErr_Occurred()) {
2991
0
        return NULL;
2992
0
    }
2993
0
    _Py_CODEUNIT instr = _PyCode_CODE(code)[src_offset/2];
2994
0
    if (!is_instrumented(instr.op.code)) {
2995
        /* Already disabled */
2996
0
        return &_PyInstrumentation_DISABLE;
2997
0
    }
2998
0
    PyObject *res = PyObject_Vectorcall(self->handler, args, nargsf, kwnames);
2999
0
    if (res == &_PyInstrumentation_DISABLE) {
3000
        /* We need FOR_ITER and POP_JUMP_ to be the same size */
3001
0
        assert(INLINE_CACHE_ENTRIES_FOR_ITER == 1);
3002
0
        int offset;
3003
0
        int other_event;
3004
0
        if (instr.op.code == FOR_ITER) {
3005
0
            if (self->right) {
3006
0
                offset = src_offset/2;
3007
0
                other_event = PY_MONITORING_EVENT_BRANCH_LEFT;
3008
0
            }
3009
0
            else {
3010
                // We don't know where the POP_ITER is, so
3011
                // we cannot de-instrument it.
3012
0
                return res;
3013
0
            }
3014
0
        }
3015
0
        else if (IS_CONDITIONAL_JUMP_OPCODE(instr.op.code)) {
3016
0
            if (self->right) {
3017
0
                offset = src_offset/2 + 2;
3018
0
                other_event = PY_MONITORING_EVENT_BRANCH_LEFT;
3019
0
                assert(_Py_GetBaseCodeUnit(code, offset).op.code == NOT_TAKEN);
3020
0
            }
3021
0
            else {
3022
0
                offset = src_offset/2;
3023
0
                other_event = PY_MONITORING_EVENT_BRANCH_RIGHT;
3024
0
            }
3025
0
        }
3026
0
        else {
3027
            // Orphaned NOT_TAKEN -- Jump removed by the compiler
3028
0
            return res;
3029
0
        }
3030
0
        PyInterpreterState *interp = _PyInterpreterState_GET();
3031
0
        _PyEval_StopTheWorld(interp);
3032
0
        remove_tools(code, offset, other_event, 1 << self->tool_id);
3033
0
        _PyEval_StartTheWorld(interp);
3034
0
    }
3035
0
    return res;
3036
0
}
3037
3038
static PyObject *make_branch_handler(int tool_id, PyObject *handler, bool right)
3039
0
{
3040
0
    _PyLegacyBranchEventHandler *callback =
3041
0
        PyObject_NEW(_PyLegacyBranchEventHandler, &_PyLegacyBranchEventHandler_Type);
3042
0
    if (callback == NULL) {
3043
0
        return NULL;
3044
0
    }
3045
0
    callback->vectorcall = branch_handler_vectorcall;
3046
0
    callback->handler = Py_NewRef(handler);
3047
0
    callback->right = right;
3048
0
    callback->tool_id = tool_id;
3049
0
    return (PyObject *)callback;
3050
0
}
3051
3052
PyObject *
3053
_PyMonitoring_RegisterCallback(int tool_id, int event_id, PyObject *obj)
3054
0
{
3055
0
    assert(0 <= tool_id && tool_id < PY_MONITORING_TOOL_IDS);
3056
0
    assert(0 <= event_id && event_id < _PY_MONITORING_EVENTS);
3057
0
    PyObject *res;
3058
0
    if (event_id == PY_MONITORING_EVENT_BRANCH) {
3059
0
        PyObject *left, *right;
3060
0
        if (obj == NULL) {
3061
0
            left = NULL;
3062
0
            right = NULL;
3063
0
        }
3064
0
        else {
3065
0
            right = make_branch_handler(tool_id, obj, true);
3066
0
            if (right == NULL) {
3067
0
                return NULL;
3068
0
            }
3069
0
            left = make_branch_handler(tool_id, obj, false);
3070
0
            if (left == NULL) {
3071
0
                Py_DECREF(right);
3072
0
                return NULL;
3073
0
            }
3074
0
        }
3075
0
        PyInterpreterState *interp = _PyInterpreterState_GET();
3076
0
        _PyEval_StopTheWorld(interp);
3077
0
        PyObject *old_right = interp->monitoring_callables[tool_id][PY_MONITORING_EVENT_BRANCH_RIGHT];
3078
0
        interp->monitoring_callables[tool_id][PY_MONITORING_EVENT_BRANCH_RIGHT] = right;
3079
0
        res = interp->monitoring_callables[tool_id][PY_MONITORING_EVENT_BRANCH_LEFT];
3080
0
        interp->monitoring_callables[tool_id][PY_MONITORING_EVENT_BRANCH_LEFT] = left;
3081
0
        _PyEval_StartTheWorld(interp);
3082
0
        Py_XDECREF(old_right);
3083
0
    }
3084
0
    else {
3085
0
        PyInterpreterState *interp = _PyInterpreterState_GET();
3086
0
        _PyEval_StopTheWorld(interp);
3087
0
        res = interp->monitoring_callables[tool_id][event_id];
3088
0
        interp->monitoring_callables[tool_id][event_id] = Py_XNewRef(obj);
3089
0
        _PyEval_StartTheWorld(interp);
3090
0
    }
3091
0
    if (res != NULL && Py_TYPE(res) == &_PyLegacyBranchEventHandler_Type) {
3092
0
        _PyLegacyBranchEventHandler *wrapper = (_PyLegacyBranchEventHandler *)res;
3093
0
        res = Py_NewRef(wrapper->handler);
3094
0
        Py_DECREF(wrapper);
3095
0
    }
3096
0
    return res;
3097
0
}
3098
3099
/* Branch Iterator */
3100
3101
typedef struct {
3102
    PyObject_HEAD
3103
    PyCodeObject *bi_code;
3104
    int bi_offset;
3105
} branchesiterator;
3106
3107
0
#define branchesiterator_CAST(op)   ((branchesiterator *)(op))
3108
3109
static PyObject *
3110
0
int_triple(int a, int b, int c) {
3111
0
    PyObject *obja = PyLong_FromLong(a);
3112
0
    PyObject *objb = NULL;
3113
0
    PyObject *objc = NULL;
3114
0
    if (obja == NULL) {
3115
0
        goto error;
3116
0
    }
3117
0
    objb = PyLong_FromLong(b);
3118
0
    if (objb == NULL) {
3119
0
        goto error;
3120
0
    }
3121
0
    objc = PyLong_FromLong(c);
3122
0
    if (objc == NULL) {
3123
0
        goto error;
3124
0
    }
3125
0
    PyObject *array[3] = { obja, objb, objc };
3126
0
    return _PyTuple_FromArraySteal(array, 3);
3127
0
error:
3128
0
    Py_XDECREF(obja);
3129
0
    Py_XDECREF(objb);
3130
0
    Py_XDECREF(objc);
3131
0
    return NULL;
3132
0
}
3133
3134
static PyObject *
3135
branchesiter_next(PyObject *op)
3136
0
{
3137
0
    branchesiterator *bi = branchesiterator_CAST(op);
3138
0
    int offset = bi->bi_offset;
3139
0
    int oparg = 0;
3140
0
    while (offset < Py_SIZE(bi->bi_code)) {
3141
0
        _Py_CODEUNIT inst = _Py_GetBaseCodeUnit(bi->bi_code, offset);
3142
0
        int next_offset = offset + 1 + _PyOpcode_Caches[inst.op.code];
3143
0
        switch(inst.op.code) {
3144
0
            case EXTENDED_ARG:
3145
0
                oparg = (oparg << 8) | inst.op.arg;
3146
0
                break;
3147
0
            case FOR_ITER:
3148
0
                oparg = (oparg << 8) | inst.op.arg;
3149
0
                bi->bi_offset = next_offset;
3150
0
                int target = next_offset + oparg+2; // Skips END_FOR and POP_ITER
3151
0
                return int_triple(offset*2, next_offset*2, target*2);
3152
0
            case POP_JUMP_IF_FALSE:
3153
0
            case POP_JUMP_IF_TRUE:
3154
0
            case POP_JUMP_IF_NONE:
3155
0
            case POP_JUMP_IF_NOT_NONE:
3156
0
                oparg = (oparg << 8) | inst.op.arg;
3157
                /* Skip NOT_TAKEN */
3158
0
                int not_taken = next_offset + 1;
3159
0
                bi->bi_offset = not_taken;
3160
0
                return int_triple(offset*2, not_taken*2, (next_offset + oparg)*2);
3161
0
            case END_ASYNC_FOR:
3162
0
                oparg = (oparg << 8) | inst.op.arg;
3163
0
                int src_offset = next_offset - oparg;
3164
0
                bi->bi_offset = next_offset;
3165
0
                assert(_Py_GetBaseCodeUnit(bi->bi_code, src_offset).op.code == END_SEND);
3166
0
                assert(_Py_GetBaseCodeUnit(bi->bi_code, src_offset+1).op.code == NOT_TAKEN);
3167
0
                not_taken = src_offset + 2;
3168
0
                return int_triple(src_offset *2, not_taken*2, next_offset*2);
3169
0
            default:
3170
0
                oparg = 0;
3171
0
        }
3172
0
        offset = next_offset;
3173
0
    }
3174
0
    return NULL;
3175
0
}
3176
3177
static void
3178
branchesiter_dealloc(PyObject *op)
3179
0
{
3180
0
    branchesiterator *bi = branchesiterator_CAST(op);
3181
0
    Py_DECREF(bi->bi_code);
3182
0
    PyObject_Free(bi);
3183
0
}
3184
3185
static PyTypeObject _PyBranchesIterator = {
3186
    PyVarObject_HEAD_INIT(&PyType_Type, 0)
3187
    "line_iterator",                    /* tp_name */
3188
    sizeof(branchesiterator),           /* tp_basicsize */
3189
    0,                                  /* tp_itemsize */
3190
    /* methods */
3191
    .tp_dealloc = branchesiter_dealloc,
3192
    .tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE,
3193
    .tp_iter = PyObject_SelfIter,
3194
    .tp_iternext = branchesiter_next,
3195
    .tp_free = PyObject_Del,
3196
};
3197
3198
PyObject *
3199
_PyInstrumentation_BranchesIterator(PyCodeObject *code)
3200
0
{
3201
3202
0
    branchesiterator *bi = (branchesiterator *)PyType_GenericAlloc(&_PyBranchesIterator, 0);
3203
0
    if (bi == NULL) {
3204
0
        return NULL;
3205
0
    }
3206
0
    bi->bi_code = (PyCodeObject*)Py_NewRef(code);
3207
0
    bi->bi_offset = 0;
3208
0
    return (PyObject *)bi;
3209
0
}