Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix compilation errors on latest 3.11.0 #1368

Closed
wants to merge 3 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion .github/workflows/coverage.yml
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,7 @@ jobs:
- "3.8"
- "3.9"
- "3.10"
- "3.11.0-alpha.5"
- "3.11.0-beta.1"
- "pypy-3.7"
exclude:
# Windows PyPy doesn't seem to work?
Expand Down
2 changes: 1 addition & 1 deletion .github/workflows/kit.yml
Original file line number Diff line number Diff line change
Expand Up @@ -239,7 +239,7 @@ jobs:
py:
# PYVERSIONS. Available versions:
# https://github.com/actions/python-versions/blob/main/versions-manifest.json
- "3.11.0-alpha.5"
- "3.11.0-beta.1"
fail-fast: false

steps:
Expand Down
5 changes: 4 additions & 1 deletion CHANGES.rst
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,10 @@ development at the same time, such as 4.5.x and 5.0.
Unreleased
----------

Nothing yet.
- Fix: Coverage.py now builds successfully on CPython 3.11 (3.11.0b1) again.
Closes `issue 1367`_. Some results for generators may have changed.

.. _issue 1367: https://github.com/nedbat/coveragepy/issues/1367


.. _changes_632:
Expand Down
2 changes: 1 addition & 1 deletion README.rst
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@ Coverage.py runs on these versions of Python:

.. PYVERSIONS

* CPython 3.7 through 3.11.0a5.
* CPython 3.7 through 3.11.0b1.
* PyPy3 7.3.8.

Documentation is on `Read the Docs`_. Code repository and issue tracker are on
Expand Down
36 changes: 24 additions & 12 deletions coverage/ctracer/tracer.c
Original file line number Diff line number Diff line change
Expand Up @@ -119,6 +119,10 @@ CTracer_dealloc(CTracer *self)
}

#if TRACE_LOG
/* Set debugging constants: a file substring and line number to start logging. */
static const char * start_file = "badasync.py";
static int start_line = 1;

static const char *
indent(int n)
{
Expand All @@ -132,9 +136,6 @@ indent(int n)
}

static BOOL logging = FALSE;
/* Set these constants to be a file substring and line number to start logging. */
static const char * start_file = "nested.py";
static int start_line = 1;

static void
CTracer_showlog(CTracer * self, int lineno, PyObject * filename, const char * msg)
Expand Down Expand Up @@ -306,6 +307,9 @@ CTracer_handle_call(CTracer *self, PyFrameObject *frame)
PyObject * plugin = NULL;
PyObject * plugin_name = NULL;
PyObject * next_tracename = NULL;
#ifdef RESUME
PyObject * pCode = NULL;
#endif

/* Borrowed references. */
PyObject * filename = NULL;
Expand Down Expand Up @@ -525,16 +529,16 @@ CTracer_handle_call(CTracer *self, PyFrameObject *frame)
*/
BOOL real_call = FALSE;

#ifdef RESUME // 3.11.0a4
#ifdef RESUME
/*
* The current opcode is guaranteed to be RESUME. The argument
* determines what kind of resume it is.
*/
PyObject * pCode = MyFrame_GetCode(frame)->co_code;
real_call = (PyBytes_AS_STRING(pCode)[MyFrame_lasti(frame) + 1] == 0);
pCode = MyCode_GetCode(MyFrame_GetCode(frame));
real_call = (PyBytes_AS_STRING(pCode)[MyFrame_GetLasti(frame) + 1] == 0);
#else
// f_lasti is -1 for a true call, and a real byte offset for a generator re-entry.
real_call = (MyFrame_lasti(frame) < 0);
real_call = (MyFrame_GetLasti(frame) < 0);
#endif

if (real_call) {
Expand All @@ -548,6 +552,9 @@ CTracer_handle_call(CTracer *self, PyFrameObject *frame)
ret = RET_OK;

error:
#ifdef RESUME
MyCode_FreeCode(pCode);
#endif
Py_XDECREF(next_tracename);
Py_XDECREF(disposition);
Py_XDECREF(plugin);
Expand Down Expand Up @@ -688,6 +695,8 @@ CTracer_handle_return(CTracer *self, PyFrameObject *frame)
{
int ret = RET_ERROR;

PyObject * pCode = NULL;

STATS( self->stats.returns++; )
/* A near-copy of this code is above in the missing-return handler. */
if (CTracer_set_pdata_stack(self) < 0) {
Expand All @@ -698,8 +707,8 @@ CTracer_handle_return(CTracer *self, PyFrameObject *frame)
if (self->pdata_stack->depth >= 0) {
if (self->tracing_arcs && self->pcur_entry->file_data) {
BOOL real_return = FALSE;
PyObject * pCode = MyFrame_GetCode(frame)->co_code;
int lasti = MyFrame_lasti(frame);
pCode = MyCode_GetCode(MyFrame_GetCode(frame));
int lasti = MyFrame_GetLasti(frame);
Py_ssize_t code_size = PyBytes_GET_SIZE(pCode);
unsigned char * code_bytes = (unsigned char *)PyBytes_AS_STRING(pCode);
#ifdef RESUME
Expand Down Expand Up @@ -759,6 +768,7 @@ CTracer_handle_return(CTracer *self, PyFrameObject *frame)

error:

MyCode_FreeCode(pCode);
return ret;
}

Expand Down Expand Up @@ -787,11 +797,13 @@ CTracer_trace(CTracer *self, PyFrameObject *frame, int what, PyObject *arg_unuse
#endif

#if WHAT_LOG
const char * w = "XXX ";
if (what <= (int)(sizeof(what_sym)/sizeof(const char *))) {
ascii = PyUnicode_AsASCIIString(MyFrame_GetCode(frame)->co_filename);
printf("%x trace: f:%x %s @ %s %d\n", (int)self, (int)frame, what_sym[what], PyBytes_AS_STRING(ascii), PyFrame_GetLineNumber(frame));
Py_DECREF(ascii);
w = what_sym[what];
}
ascii = PyUnicode_AsASCIIString(MyFrame_GetCode(frame)->co_filename);
printf("%x trace: f:%x %s @ %s %d\n", (int)self, (int)frame, what_sym[what], PyBytes_AS_STRING(ascii), PyFrame_GetLineNumber(frame));
Py_DECREF(ascii);
#endif

#if TRACE_LOG
Expand Down
18 changes: 15 additions & 3 deletions coverage/ctracer/util.h
Original file line number Diff line number Diff line change
Expand Up @@ -17,13 +17,17 @@
// to make this work, but it's all I've got until https://bugs.python.org/issue40421
// is resolved.
#include <internal/pycore_frame.h>
#define MyFrame_lasti(f) ((f)->f_frame->f_lasti * 2)
#if PY_VERSION_HEX >= 0x030B00A7
#define MyFrame_GetLasti(f) (PyFrame_GetLasti(f))
#else
#define MyFrame_GetLasti(f) ((f)->f_frame->f_lasti * 2)
#endif
#elif PY_VERSION_HEX >= 0x030A00A7
// The f_lasti field changed meaning in 3.10.0a7. It had been bytes, but
// now is instructions, so we need to adjust it to use it as a byte index.
#define MyFrame_lasti(f) ((f)->f_lasti * 2)
#define MyFrame_GetLasti(f) ((f)->f_lasti * 2)
#else
#define MyFrame_lasti(f) ((f)->f_lasti)
#define MyFrame_GetLasti(f) ((f)->f_lasti)
#endif

// Access f_code should be done through a helper starting in 3.9.
Expand All @@ -33,6 +37,14 @@
#define MyFrame_GetCode(f) ((f)->f_code)
#endif

#if PY_VERSION_HEX >= 0x030B00A7
#define MyCode_GetCode(co) (PyObject_GetAttrString((PyObject *)(co), "co_code"))
#define MyCode_FreeCode(code) Py_XDECREF(code)
#else
#define MyCode_GetCode(co) ((co)->co_code)
#define MyCode_FreeCode(code)
#endif

/* The values returned to indicate ok or error. */
#define RET_OK 0
#define RET_ERROR -1
Expand Down
2 changes: 1 addition & 1 deletion doc/index.rst
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@ supported on:

.. PYVERSIONS

* Python versions 3.7 through 3.11.0a5.
* Python versions 3.7 through 3.11.0b1.

* PyPy3 7.3.8.

Expand Down
20 changes: 20 additions & 0 deletions tests/test_arcs.py
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,12 @@
from coverage.files import abs_file


skip_cpython_92236 = pytest.mark.skipif(
env.PYVERSION >= (3, 11, 0, "beta"),
reason="Avoid a CPython bug: https://github.com/python/cpython/issues/92236",
# #92236 is fixed in https://github.com/python/cpython/pull/92722
)

class SimpleArcTest(CoverageTest):
"""Tests for coverage.py's arc measurement."""

Expand Down Expand Up @@ -605,6 +611,7 @@ def wrong_loop(x):
arcz_missing="26 3. 6.",
)

@skip_cpython_92236
def test_generator_expression(self):
# Generator expression:
self.check_coverage("""\
Expand All @@ -617,6 +624,7 @@ def test_generator_expression(self):
arcz=".1 -22 2-2 12 23 34 45 53 3.",
)

@skip_cpython_92236
def test_generator_expression_another_way(self):
# https://bugs.python.org/issue44450
# Generator expression:
Expand Down Expand Up @@ -1169,6 +1177,7 @@ def func(x):
class YieldTest(CoverageTest):
"""Arc tests for generators."""

@skip_cpython_92236
def test_yield_in_loop(self):
self.check_coverage("""\
def gen(inp):
Expand All @@ -1180,6 +1189,7 @@ def gen(inp):
arcz=".1 .2 23 2. 32 15 5.",
)

@skip_cpython_92236
def test_padded_yield_in_loop(self):
self.check_coverage("""\
def gen(inp):
Expand All @@ -1200,6 +1210,7 @@ def gen(inp):
env.PYVERSION[:5] == (3, 11, 0, 'alpha', 3),
reason="avoid 3.11 bug: bpo46225",
)
@skip_cpython_92236
def test_bug_308(self):
self.check_coverage("""\
def run():
Expand Down Expand Up @@ -1234,6 +1245,7 @@ def run():
arcz=".1 14 45 54 4. .2 2. -22 2-2",
)

@skip_cpython_92236
def test_bug_324(self):
# This code is tricky: the list() call pulls all the values from gen(),
# but each of them is a generator itself that is never iterated. As a
Expand All @@ -1252,6 +1264,7 @@ def gen(inp):
arcz_missing="-33 3-3",
)

@skip_cpython_92236
def test_coroutines(self):
self.check_coverage("""\
def double_inputs():
Expand All @@ -1271,6 +1284,7 @@ def double_inputs():
)
assert self.stdout() == "20\n12\n"

@skip_cpython_92236
def test_yield_from(self):
self.check_coverage("""\
def gen(inp):
Expand All @@ -1286,6 +1300,7 @@ def gen(inp):
arcz=".1 19 9. .2 23 34 45 56 63 37 7.",
)

@skip_cpython_92236
def test_abandoned_yield(self):
# https://github.com/nedbat/coveragepy/issues/440
self.check_coverage("""\
Expand Down Expand Up @@ -1614,6 +1629,7 @@ def test_pathologically_long_code_object(self, n):
self.check_coverage(code, arcs=[(-1, 1), (1, 2*n+4), (2*n+4, -1)])
assert self.stdout() == f"{n}\n"

@skip_cpython_92236
def test_partial_generators(self):
# https://github.com/nedbat/coveragepy/issues/475
# Line 2 is executed completely.
Expand Down Expand Up @@ -1830,6 +1846,7 @@ class AsyncTest(CoverageTest):
"""Tests of the new async and await keywords in Python 3.5"""

@xfail_eventlet_670
@skip_cpython_92236
def test_async(self):
self.check_coverage("""\
import asyncio
Expand Down Expand Up @@ -1857,6 +1874,7 @@ async def print_sum(x, y): # 8
assert self.stdout() == "Compute 1 + 2 ...\n1 + 2 = 3\n"

@xfail_eventlet_670
@skip_cpython_92236
def test_async_for(self):
self.check_coverage("""\
import asyncio
Expand Down Expand Up @@ -1932,6 +1950,7 @@ async def go():
# https://bugs.python.org/issue44621
@pytest.mark.skipif(env.PYVERSION[:2] == (3, 9), reason="avoid a 3.9 bug: 44621")
@pytest.mark.skipif(env.PYVERSION < (3, 7), reason="need asyncio.run")
@skip_cpython_92236
def test_bug_1158(self):
self.check_coverage("""\
import asyncio
Expand Down Expand Up @@ -1962,6 +1981,7 @@ async def async_test():
)
@xfail_eventlet_670
@pytest.mark.skipif(env.PYVERSION < (3, 7), reason="need asyncio.run")
@skip_cpython_92236
def test_bug_1176(self):
self.check_coverage("""\
import asyncio
Expand Down