Issue 36710: Pass _PyRuntimeState as an argument rather than using the _PyRuntime global variable (original) (raw)

Created on 2019-04-24 12:58 by vstinner, last changed 2022-04-11 14:59 by admin. This issue is now closed.

Messages (40)

msg340772 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-04-24 12:58

Eric Snow moved global variables into a _PyRuntimeState structure which is made of sub-structures. There is a single instance of _PyRuntimeState: the _PyRuntime global variable.

I would like to add "_PyRuntimeState *" parameters to functions to avoid relying directly on _PyRuntime global variable. The long term goal is to have "stateless" code: don't rely on global variables, only on input parameters. In practice, we will continue to use thread local storage (TLS) to get the "current context" like the current interpreter and the current Python thread state.

msg340775 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-04-24 14:47

New changeset 8bb3230149538c25c1bacced5e64a3c071475f73 by Victor Stinner in branch 'master': bpo-36710: Add runtime parameter to _PyThreadState_Init() (GH-12935) https://github.com/python/cpython/commit/8bb3230149538c25c1bacced5e64a3c071475f73

msg340779 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-04-24 15:14

New changeset b930a2d2b1247bdba560db341ba90a9cbb538eb3 by Victor Stinner in branch 'master': bpo-36710: PyOS_AfterFork_Child() pass runtime parameter (GH-12936) https://github.com/python/cpython/commit/b930a2d2b1247bdba560db341ba90a9cbb538eb3

msg340784 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-04-24 15:24

New changeset 8e91c246e468515b877690e090c73f496552541d by Victor Stinner in branch 'master': bpo-36710: Add runtime variable to Py_FinalizeEx() (GH-12937) https://github.com/python/cpython/commit/8e91c246e468515b877690e090c73f496552541d

msg340789 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-04-24 16:23

New changeset 43125224d6da5febb34101ebfd36536d791d68cd by Victor Stinner in branch 'master': bpo-36710: Add runtime variable to Py_InitializeEx() (GH-12939) https://github.com/python/cpython/commit/43125224d6da5febb34101ebfd36536d791d68cd

msg340842 - (view)

Author: Jeroen Demeyer (jdemeyer) * (Python triager)

Date: 2019-04-25 14:49

I don't really understand the rationale for these changes. What's wrong with the global variable _PyRuntime?

What's the long-term goal for _PyRuntime? If you can't get rid of all occurrences of _PyRuntime, how does it help to get rid of some occurences?

msg340845 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-04-25 15:03

I don't really understand the rationale for these changes. What's wrong with the global variable _PyRuntime? What's the long-term goal for _PyRuntime? If you can't get rid of all occurrences of _PyRuntime, how does it help to get rid of some occurences?

The long term goal is to support multiple interpreter instances per process:

Eric Snow's PEP 554 "Multiple Interpreters in the Stdlib" https://www.python.org/dev/peps/pep-0554/

Right now, there is a single instance of _PyRuntimeState: _PyRuntime. I would be interested to have one _PyRuntimeState per interpreter.

Maybe _PyRuntimeState should be reworked in the meanwhile. Right now, it's a strange beast: it contains things which are set before Python initialization and things which are set after. It contains C types and Python objects. Maybe some parts should be moved into PyInterpreterState or even PyThreadState. I don't know at this point. It takes time to look at each individual structure field...

Anyway, more generally, IMHO it's a bad practice to rely on a global variable. Python runtime should be "stateless".

The current implementation of CPython leaks dozens of Python objects at exit. For example, I started to work on this issue while working on https://bugzilla.redhat.com/show_bug.cgi?id=1696322 : Python doesn't clear 2 warnings variables at exit. When I looked into this issue, I also noticed that _PyRuntime.gc.garbage remains alive after Py_Finalize().

That's plain wrong: all Python objects must be cleared by Py_Finalize(). Two interpreters must not share any Python object.

Well, the PEP should better explain the rationale than me :-)

--

When I wrote PR 12934, I noticed that even getting the current thread state rely on _PyRuntime:

#define _PyThreadState_GET()
((PyThreadState*)_Py_atomic_load_relaxed(&_PyRuntime.gilstate.tstate_current))

That's wrong in the case of ceval.c: it should be possible to run _PyEval_EvalFrameDefault() twice at the same time in two threads using two "isolated" interpreters.

Well, PR 12934 doesn't fix all issues. It's just one small step towards "stateless" runtime and the even more long term of having one GIL per interpeter, rather than a single GIL per process.

msg340860 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-04-25 20:14

I created bpo-36724: Clear _PyRuntime at exit.

msg340871 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-04-25 23:53

New changeset 10c8e6af910e3a26e59f913a3c1e4830ca71b1af by Victor Stinner in branch 'master': bpo-36710: Add runtime variable in pystate.c (GH-12956) https://github.com/python/cpython/commit/10c8e6af910e3a26e59f913a3c1e4830ca71b1af

msg340873 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-04-26 00:32

New changeset 9db0324712f6982d89620b420f507a6aa5da152f by Victor Stinner in branch 'master': bpo-36710: Add runtime parameter in gcmodule.c (GH-12958) https://github.com/python/cpython/commit/9db0324712f6982d89620b420f507a6aa5da152f

msg340880 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-04-26 03:48

New changeset 99e69d44f499625786a2e6461a954adcd0037d69 by Victor Stinner in branch 'master': bpo-36710: Fix compiler warning on PyThreadState_Delete() (GH-12962) https://github.com/python/cpython/commit/99e69d44f499625786a2e6461a954adcd0037d69

msg340915 - (view)

Author: Jeroen Demeyer (jdemeyer) * (Python triager)

Date: 2019-04-26 12:32

The long term goal is to support multiple interpreter instances per process: Eric Snow's PEP 554 "Multiple Interpreters in the Stdlib" https://www.python.org/dev/peps/pep-0554/

Sorry, but I don't see the relation between this issue and PEP 554. It seems to me that the PEP is about making subinterpreters available from pure Python (instead of only at the C level). It doesn't say anything about the implementation of subinterpreters, which is what this issue is about.

So I'm still missing the bigger picture where this issue fits in.

The current implementation of CPython leaks dozens of Python objects at exit.

That may be an issue to be fixed, but again I don't see the relation with this issue.

msg340930 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-04-26 16:21

Jeroen Demeyer:

Sorry, but I don't see the relation between this issue and PEP 554.

The long term plan for PEP 554 is to support having one GIL per interpreter for best performances. The GIL lives in _PyRuntime.

It's not just about the GIL. Currently, the gc module stores its state into _PyRuntime. It's wrong to share a single gc state between two interpreters: each interpreter should have its own "namespace" completely isolated from the other namespaces. For example, _PyRuntime.gc.garbage is a Python list: each interpreter should have its own list.

My PR 12934 is only a first step to prepare ceval.c for that.

Said differently, if I understood correctly, each interpreter must have its own _PyRuntime instance.

Maybe tomorrow, we will keep a single _PyRuntime instance, but my work is needed to identify the relationship between the current implementation of Python and _PyRuntime.

msg340941 - (view)

Author: Jeroen Demeyer (jdemeyer) * (Python triager)

Date: 2019-04-26 20:57

So what's the relation between _PyRuntime and PyInterpreterState? If the latter is a structure per interpreter, what's the point of also making the former per interpreter? It would be better to move data from _PyRuntime to PyInterpreterState.

msg340942 - (view)

Author: Jeroen Demeyer (jdemeyer) * (Python triager)

Date: 2019-04-26 21:03

It's wrong to share a single gc state between two interpreters

And what's your solution for that? I'm not asking for a complete ready-to-implement answer, but at least a basic idea. Otherwise it's impossible for me to judge whether your PR 12934 helps with that or not.

Basically I would like to see something like PEP 579 but for this problem.

msg340945 - (view)

Author: Eric Snow (eric.snow) * (Python committer)

Date: 2019-04-26 21:46

I don't think this change is the right way to go (yet), but something related might be. First, let's be clear on the status quo for CPython. (This has gotten long, but I want to be clear.)

Status Quo

For simplicity sake, let's say nearly all the code operates relative to the 3 levels of runtime state:

Furthermore, there are 3 groups of functions in the C-API:

Most of the C-API is context-sensitive. A small portion is runtime-dependent. A handful of functions are runtime-independent (effectively otherwise stateless helper functions that only happen to be part of the C-API).

Each context-sensitive function relies on there being a "runtime context" it can use relative to the current OS thread. That context consists of the current (i.e. active) PyThreadState, the corresponding PyInterpreterState, and the global _PyRuntimeState. That context is derived from data in TSS (see caveats below). This group includes most of the C-API.

Each runtime-dependent function operates against one or more runtime state target, regardless of the current thread context (or even if there isn't one). The target state (e.g. PyInterpreterState) is always passed explicitly. Again, this is only a small portion of the C-API.

Caveats:

All of this is the pattern we use currently. Using TSS to identify the implicit runtime context has certain benefits and costs:

benefits:

costs:

Alternative

For every context-sensitive function we could add a new first parameter, "context", that provides the runtime context to use. That would be something like this:

struct { PyThreadState *tstate; ... } PyRuntimeContext;

The interpreter state and global runtime state would still be accessible via the same indirection we have now.

Taking this alternative would eliminate the previous costs. Having a consistent "PyRuntimeContext *context" first parameter would maintain the easy distinction from runtime-dependent functions. Asking callers to pass in the context explicitly is probably better regardless. As to backward compatibility, we could maintain a shim to bridge between the old way and the new.

About the C-global _PyRuntime

Currently the global runtime state (_PyRuntimeState) is stored in a static global C variable, _PyRuntime. I added it at the time I consolidated many of the existing C globals into a single struct. Having a C global makes it easy to do the wrong thing, so it may be good to do something else.

That would mean allocating a _PyRuntimeState on the heap early in startup and pass that around where needed. I expect that would not have any meaningful performance penalty. It would probably also simplify some of the code we currently use to manage _PyRuntime correctly.

As a bonus, this would be important if we decided that multiple-runtimes-per-process were a desirable thing. That's a neat idea, though I don't see a need currently. So on its own it's not really a justification for dropping a static _PyRuntime. :) However, I think the other reasons are enough.

Conclusions

This issue has a specific objective that I think is premature. We have an existing pattern and we should stick with that until we decide to change to a new pattern. That said, a few things should get corrected and we should investigate alternative patterns for the context-sensitive C-API.

As to getting rid of the _PyRuntime global variable in favor of putting it on the heap, I'm not opposed. However, doing so should probably be handled in a separate issue.

Here are my thoughts on actionable items:

  1. look for a better pattern for the context-sensitive C-API
  2. clearly document which of the 3 groups each C-API function belongs to
  3. add a "runtime" field to the PyInterpreterState pointing to the parent _PyRuntimeState
  4. (maybe) add a _PyRuntimeState_GET() macro, a la PyThreadState_GET()
  5. for context-sensitive C-API that uses the global runtime state, get it from the current PyInterpreterState
  6. for runtime-dependent C-API that targets the global runtime state, ensure the _PyRuntimeState is always an explicit parameter
  7. (maybe) drop _PyRuntime and create a _PyRuntimeState on the heap during startup to pass around

msg340946 - (view)

Author: Eric Snow (eric.snow) * (Python committer)

Date: 2019-04-26 21:48

FWIW, PEP 554 is part of a larger project that I've been working on (slowly) for several years now. [1] The concrete objective is to leverage subinterpreters as the mechanism by which we can achieve multi-core parallelism in Python code. Moving the GIL (and some other parts of _PyRuntimeState, as Victor indicated) down to per-interpreter state is essential to that.

However, I don't thing making _PyRuntime a per-interpreter thing is right. The runtime holds the set of interpreters, as well as any state state shared by the interpreters.

Also, to be clear, the status quo is not a problem for me, so make sure I'm not used as the justification for the change (thoughtful as that is of Victor). :)

[1] https://github.com/ericsnowcurrently/multi-core-python

msg340971 - (view)

Author: Jeroen Demeyer (jdemeyer) * (Python triager)

Date: 2019-04-27 07:21

Changing every C API function to include a state parameter looks very cumbersome. Another alternative would be to store the interpreter state in every Python object (or every class, that would be sufficient). That way, you would only need to pass context to C API functions which do not take a Python object as argument.

msg340986 - (view)

Author: Steve Dower (steve.dower) * (Python committer)

Date: 2019-04-27 14:35

Changing every API to take the context parameter would bring us into alignment with the JavaScript VMs.

I'm working on a project that embeds a few of these, as well as Python, and our thread management is much worse than their context parameter. Though I'm of course very sympathetic to the compatibility argument (but then the shims would just load the context from TSS and pass it around, so they're not too bad).

Eric's breakdown of context scopes seems spot on, and it means that we only really need the thread state to be passed around. The few places that would be satisfied by runtime state now (GIL, GC) should become interpreter state, which is most easily found from a thread state anyway.

Runtime state should eventually probably become runtime configuration (those settings we need to create interpreters) and a minimum amount of state to track live interpreters. I see no reason to pass it around anywhere other than interpreter creation, and as a transitional step toward that goal it should be accessible through the active interpreter state.

msg340996 - (view)

Author: Eric Snow (eric.snow) * (Python committer)

Date: 2019-04-27 15:59

FWIW, I don't mean to side-track this issue. If we want to have any further discussion about broader solutions then let's take this to capi-sig. In fact, I've started a thread there. I'd post the link, but I think it got stuck in moderation. :)

msg341305 - (view)

Author: Neil Schemenauer (nascheme) * (Python committer)

Date: 2019-05-02 19:07

I think there are two questions to answer. First, do we want to support multiple runtimes per process? Second, if we do, what is the best way to do that? Some people would argue that multiple runtimes are not needed or are too hard to do. Maybe they are correct, I'm not sure. We should try to get a consensus on that first question.

If we do decide to do it, then we need to answer the second question. Passing a "context" argument around seems the best solution. That is how the Java JNI does it. It sounds like that's how Javascript VMs do it too. We don't need to get creative. Look at what other VMs do and copy the best idea.

If we do decide to do it, evolving the codebase and all extension modules is going to be a massive task. I would imagine that we can have a backwards compatible API layer that uses TSS. The layer that passes context explicitly would still have to maintain the TSS. There could be a build option that turns that backwards compatibility on or off. If off, you would gain some performance advantage because TSS does not have to be kept up-to-date.

My feeling right now that even though this is a massive job, it is the correct thing to do. CPUs continue to gain cores. Improving CPython's ability to do multi-threading and multi-processing should be a priority for CPython core developers.

msg341500 - (view)

Author: Steve Dower (steve.dower) * (Python committer)

Date: 2019-05-06 13:51

I think Neil is right, though I believe we'll have a clear enough internal boundary that we should only rarely have to maintain TSS for the sake of legacy callers. The build option should just turn off the entire legacy API, which would also make it easier to remove one day.

msg342138 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-05-10 21:39

New changeset 09532feeece39d5ba68a0d47115ce1967bfbd58e by Victor Stinner in branch 'master': bpo-36710: Add 'ceval' local variable to ceval.c (GH-12934) https://github.com/python/cpython/commit/09532feeece39d5ba68a0d47115ce1967bfbd58e

msg343380 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-05-24 11:44

New changeset b4bdecd0fc9112b60a81fec171bc78bc13f2f59c by Victor Stinner in branch 'master': bpo-36710: Add tstate parameter in errors.c (GH-13540) https://github.com/python/cpython/commit/b4bdecd0fc9112b60a81fec171bc78bc13f2f59c

msg343393 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-05-24 15:02

New changeset 438a12dd9d85f463c0bb7bf1505cd87b98b98170 by Victor Stinner in branch 'master': bpo-36710: Add tstate parameter in ceval.c (GH-13547) https://github.com/python/cpython/commit/438a12dd9d85f463c0bb7bf1505cd87b98b98170

msg345537 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-06-13 20:41

New changeset 838f26402de82640698c38ea9d2be65c6cf780d6 by Victor Stinner in branch 'master': bpo-36710: Pass explicitly tstate in sysmodule.c (GH-14060) https://github.com/python/cpython/commit/838f26402de82640698c38ea9d2be65c6cf780d6

msg345538 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-06-13 20:42

I wrote my notes on this issue there: https://pythoncapi.readthedocs.io/runtime.html

msg346016 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-06-19 00:54

New changeset 0a28f8d379544eee897979da0ce99f0b449b49dd by Victor Stinner in branch 'master': bpo-36710: Add tstate parameter in import.c (GH-14218) https://github.com/python/cpython/commit/0a28f8d379544eee897979da0ce99f0b449b49dd

msg346030 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-06-19 08:36

New changeset 987a0dcfa1302df6c1ed8cf14762dc18628e3f33 by Victor Stinner in branch 'master': bpo-36710: Remove PyImport_Cleanup() function (GH-14221) https://github.com/python/cpython/commit/987a0dcfa1302df6c1ed8cf14762dc18628e3f33

msg346088 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-06-19 22:05

New changeset b45d259bdda1de2b2d369458a9ad2e4d6f750687 by Victor Stinner in branch 'master': bpo-36710: Use tstate in pylifecycle.c (GH-14249) https://github.com/python/cpython/commit/b45d259bdda1de2b2d369458a9ad2e4d6f750687

msg356496 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-11-12 22:23

I started a thread on python-dev about this issue: "Pass the Python thread state to internal C functions" https://mail.python.org/archives/list/python-dev@python.org/thread/PQBGECVGVYFTVDLBYURLCXA3T7IPEHHO/

msg357013 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-11-20 00:49

New changeset 2e96906da764402b4c8062dbf99171ca506f9e12 by Victor Stinner in branch 'master': bpo-36710: Pass tstate parameter to GC collect() (GH-17267) https://github.com/python/cpython/commit/2e96906da764402b4c8062dbf99171ca506f9e12

msg357017 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2019-11-20 01:28

New changeset 01b1cc12e7c6a3d6a3d27ba7c731687d57aae92a by Victor Stinner in branch 'master': bpo-36710: Add PyInterpreterState.runtime field (GH-17270) https://github.com/python/cpython/commit/01b1cc12e7c6a3d6a3d27ba7c731687d57aae92a

msg359915 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2020-01-13 16:08

I continued this work by passing tstate to internal C functions: bpo-38644.

I also added PyInterpreterState.runtime field, so it's now possible to retrieve the runtime from tstate:

runtime = tstate->interp->runtime;

I wrote an article on passing tstate to internal C functions:

https://vstinner.github.io/cpython-pass-tstate.html

I consider that this issue is now done. I close the issue.

msg360210 - (view)

Author: Eric Snow (eric.snow) * (Python committer)

Date: 2020-01-17 19:52

Thanks, Victor!

msg364434 - (view)

Author: Mark Shannon (Mark.Shannon) * (Python committer)

Date: 2020-03-17 15:10

Instead of passing _PyRuntimeState around everywhere, why not just let it disappear in time.

Currently _PyRuntimeState manages "global" state, mainly the GIL and some config. Once the GIL has been migrated to the sub-interpreters, the config part can be factored out and _PyRuntimeState can just disappear.

msg364438 - (view)

Author: Steve Dower (steve.dower) * (Python committer)

Date: 2020-03-17 15:20

Instead of passing _PyRuntimeState around everywhere, why not just let it disappear in time.

Agreed. It's valuable to pass the thread state, but the runtime state should only be needed to create a new thread state (and arguably not even then).

msg364440 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2020-03-17 16:04

Instead of passing _PyRuntimeState around everywhere, why not just let it disappear in time.

Passing runtime (_PyRuntimeState) is a temporary move until more and more fields are moved from _PyRuntimeState into PyInterpreterState. I just created bpo-39984 "Move some ceval fields from _PyRuntime.ceval to PyInterpreterState.ceval" yesterday ;-)

Once we will manage to move the GIL into PyInterpreterState, we would still have to pass PyInterpreterState or PyThreadState to function which require to access the GIL. Passing explicitly runtime is a first step to prepare to migration to PyInterpreterState or PyThreadState. My intent is to show that many functions rely on "global variables": pass these variables instead.

If you are thinking about getting the current Python thread state using a thread local storage, that's a different topic and I'm not aware of an open issue to track this idea.

Currently _PyRuntimeState manages "global" state, mainly the GIL and some config. Once the GIL has been migrated to the sub-interpreters, the config part can be factored out and _PyRuntimeState can just disappear.

I don't think that we will be able to fully remove _PyRuntimeState. It seems like the PEP 554 "Multiple Interpreters in the Stdlib" requires a registry of interpreters and it currently lives in _PyRuntimeState.

msg364444 - (view)

Author: Mark Shannon (Mark.Shannon) * (Python committer)

Date: 2020-03-17 16:39

Even if _PyRuntime ends up as just a list of interpreters and doesn't disappear completely, it won't be used anything like as much as it is now.

Many of the functions that it getting passed to will no longer need it, so why bother passing it now?

msg372170 - (view)

Author: STINNER Victor (vstinner) * (Python committer)

Date: 2020-06-23 13:55

New changeset 61b649296110dcb38e67db382a7e8427c5eb75d0 by Victor Stinner in branch 'master': bpo-36710: Pass tstate explicitly in abstract.c (GH-21075) https://github.com/python/cpython/commit/61b649296110dcb38e67db382a7e8427c5eb75d0

History

Date

User

Action

Args

2022-04-11 14:59:14

admin

set

github: 80891

2020-06-23 13:55:15

vstinner

set

messages: +

2020-06-23 13:20:12

vstinner

set

pull_requests: + <pull%5Frequest20243>

2020-03-17 16:39:34

Mark.Shannon

set

messages: +

2020-03-17 16:04:10

vstinner

set

messages: +

2020-03-17 15:20:23

steve.dower

set

messages: +

2020-03-17 15:10:36

Mark.Shannon

set

nosy: + Mark.Shannon
messages: +

2020-01-17 19:52:11

eric.snow

set

messages: +

2020-01-13 16:08:27

vstinner

set

status: open -> closed
resolution: fixed
messages: +

stage: patch review -> resolved

2019-11-20 01:28:11

vstinner

set

messages: +

2019-11-20 00:51:21

vstinner

set

pull_requests: + <pull%5Frequest16763>

2019-11-20 00:49:36

vstinner

set

messages: +

2019-11-20 00:19:32

vstinner

set

pull_requests: + <pull%5Frequest16759>

2019-11-12 22:23:23

vstinner

set

messages: +

2019-08-22 03:37:53

phsilva

set

nosy: + phsilva

2019-06-19 22:05:27

vstinner

set

messages: +

2019-06-19 21:57:31

vstinner

set

pull_requests: + <pull%5Frequest14082>

2019-06-19 21:16:04

vstinner

set

pull_requests: + <pull%5Frequest14081>

2019-06-19 08:36:25

vstinner

set

messages: +

2019-06-19 01:01:03

vstinner

set

pull_requests: + <pull%5Frequest14059>

2019-06-19 00:54:42

vstinner

set

messages: +

2019-06-19 00:39:09

vstinner

set

pull_requests: + <pull%5Frequest14056>

2019-06-13 20:42:49

vstinner

set

messages: +

2019-06-13 20:41:47

vstinner

set

messages: +

2019-06-13 16:15:02

vstinner

set

pull_requests: + <pull%5Frequest13922>

2019-05-24 15:02:03

vstinner

set

messages: +

2019-05-24 13:16:28

vstinner

set

pull_requests: + <pull%5Frequest13459>

2019-05-24 11:44:26

vstinner

set

messages: +

2019-05-24 08:57:37

vstinner

set

pull_requests: + <pull%5Frequest13454>

2019-05-10 21:39:12

vstinner

set

messages: +

2019-05-06 13:51:58

steve.dower

set

messages: +

2019-05-02 19:07:42

nascheme

set

nosy: + nascheme
messages: +

2019-04-27 15:59:38

eric.snow

set

messages: +

2019-04-27 14:35:33

steve.dower

set

messages: +

2019-04-27 07:21:53

jdemeyer

set

messages: +

2019-04-26 21:48:28

eric.snow

set

messages: +

2019-04-26 21:46:45

eric.snow

set

nosy: + ncoghlan, steve.dower
messages: +

2019-04-26 21:03:51

jdemeyer

set

messages: +

2019-04-26 20:57:31

jdemeyer

set

messages: +

2019-04-26 16:21:23

vstinner

set

messages: +

2019-04-26 12:32:55

jdemeyer

set

messages: +

2019-04-26 03:48:54

vstinner

set

messages: +

2019-04-26 03:25:10

vstinner

set

pull_requests: + <pull%5Frequest12888>

2019-04-26 00:32:05

vstinner

set

messages: +

2019-04-25 23:53:21

vstinner

set

messages: +

2019-04-25 23:52:09

vstinner

set

pull_requests: + <pull%5Frequest12884>

2019-04-25 23:38:27

vstinner

set

pull_requests: + <pull%5Frequest12882>

2019-04-25 20:14:17

vstinner

set

messages: +

2019-04-25 15:03:25

vstinner

set

messages: +

2019-04-25 14:49:40

jdemeyer

set

nosy: + jdemeyer
messages: +

2019-04-24 17:20:23

eric.snow

set

nosy: + eric.snow

2019-04-24 16:23:57

vstinner

set

messages: +

2019-04-24 15:28:18

vstinner

set

pull_requests: + <pull%5Frequest12863>

2019-04-24 15:24:06

vstinner

set

messages: +

2019-04-24 15:14:42

vstinner

set

messages: +

2019-04-24 14:58:47

vstinner

set

pull_requests: + <pull%5Frequest12861>

2019-04-24 14:50:13

vstinner

set

pull_requests: + <pull%5Frequest12860>

2019-04-24 14:47:50

vstinner

set

messages: +

2019-04-24 14:32:39

vstinner

set

pull_requests: + <pull%5Frequest12859>

2019-04-24 14:02:02

vstinner

set

keywords: + patch
stage: patch review
pull_requests: + <pull%5Frequest12858>

2019-04-24 12:58:00

vstinner

create