homepage

This issue tracker has been migrated to GitHub , and is currently read-only.
For more information, see the GitHub FAQs in the Python's Developer Guide.

classification
Title: Remove private _PyLong_Zero and _PyLong_One variables
Type: Stage: resolved
Components: Interpreter Core Versions: Python 3.10
process
Status: closed Resolution: fixed
Dependencies: Superseder:
Assigned To: Nosy List: corona10, mark.dickinson, miss-islington, pablogsal, rhettinger, serhiy.storchaka, vstinner
Priority: normal Keywords: patch

Created on 2020年10月26日 21:59 by vstinner, last changed 2022年04月11日 14:59 by admin. This issue is now closed.

Pull Requests
URL Status Linked Edit
PR 22993 merged vstinner, 2020年10月26日 22:14
PR 22995 merged vstinner, 2020年10月26日 23:33
PR 22998 merged vstinner, 2020年10月27日 01:41
PR 23003 merged vstinner, 2020年10月27日 16:16
PR 23008 merged vstinner, 2020年10月27日 20:31
PR 26391 merged vstinner, 2021年05月26日 22:31
PR 26393 merged miss-islington, 2021年05月26日 22:51
PR 30656 merged rhettinger, 2022年01月18日 07:03
Messages (16)
msg379691 - (view) Author: STINNER Victor (vstinner) * (Python committer) Date: 2020年10月26日 21:59
In bpo-38858, I made the small integer singletons per interpreter: commit 630c8df5cf126594f8c1c4579c1888ca80a29d59. _PyLong_Zero and _PyLong_One variables are still shared by all interpreters, whereas subinterpreters must not share Python objects: see bpo-40533.
I propose to add new _PyLong_GetZero() and _PyLong_GetOne() functions to replace _PyLong_Zero and _PyLong_One variables. These functions will retrieve the singletons from tstate->interp->small_ints.
msg379700 - (view) Author: STINNER Victor (vstinner) * (Python committer) Date: 2020年10月26日 23:00
New changeset 8e3b9f92835654943bb59d9658bb52e1b0f40a22 by Victor Stinner in branch 'master':
bpo-42161: Add _PyLong_GetZero() and _PyLong_GetOne() (GH-22993)
https://github.com/python/cpython/commit/8e3b9f92835654943bb59d9658bb52e1b0f40a22
msg379709 - (view) Author: STINNER Victor (vstinner) * (Python committer) Date: 2020年10月27日 01:24
New changeset c9bc290dd6e3994a4ead2a224178bcba86f0c0e4 by Victor Stinner in branch 'master':
bpo-42161: Use _PyLong_GetZero() and _PyLong_GetOne() (GH-22995)
https://github.com/python/cpython/commit/c9bc290dd6e3994a4ead2a224178bcba86f0c0e4
msg379768 - (view) Author: STINNER Victor (vstinner) * (Python committer) Date: 2020年10月27日 16:13
New changeset 37834136d0afe51d274bfc79d8705514cbe73727 by Victor Stinner in branch 'master':
bpo-42161: Modules/ uses _PyLong_GetZero() and _PyLong_GetOne() (GH-22998)
https://github.com/python/cpython/commit/37834136d0afe51d274bfc79d8705514cbe73727
msg379798 - (view) Author: Raymond Hettinger (rhettinger) * (Python committer) Date: 2020年10月27日 20:15
Why did you put _PyLong_GetOne() inside the loop for the fast path and outside the loop for the slow path?
==========================================================================
diff --git a/Modules/_collectionsmodule.c b/Modules/_collectionsmodule.c
index 8990071f51..0e6c64d1a6 100644
--- a/Modules/_collectionsmodule.c
+++ b/Modules/_collectionsmodule.c
@@ -2278,6 +2278,8 @@ _collections__count_elements_impl(PyObject *module, PyObject *mapping,
 PyObject *dict_get;
 PyObject *mapping_setitem;
 PyObject *dict_setitem;
+ PyObject *zero = _PyLong_GetZero(); // borrowed reference
+ PyObject *one = _PyLong_GetOne(); // borrowed reference
 it = PyObject_GetIter(iterable);
 if (it == NULL)
@@ -2324,10 +2326,10 @@ _collections__count_elements_impl(PyObject *module, PyObject *mapping,
 if (oldval == NULL) {
 if (PyErr_Occurred())
 goto done;
- if (_PyDict_SetItem_KnownHash(mapping, key, _PyLong_GetOne(), hash) < 0)
+ if (_PyDict_SetItem_KnownHash(mapping, key, one, hash) < 0)
 goto done;
 } else {
- newval = PyNumber_Add(oldval, _PyLong_GetOne());
+ newval = PyNumber_Add(oldval, one);
 if (newval == NULL)
 goto done;
 if (_PyDict_SetItem_KnownHash(mapping, key, newval, hash) < 0)
@@ -2341,8 +2343,6 @@ _collections__count_elements_impl(PyObject *module, PyObject *mapping,
 if (bound_get == NULL)
 goto done;
- PyObject *zero = _PyLong_GetZero(); // borrowed reference
- PyObject *one = _PyLong_GetOne(); // borrowed reference
 while (1) {
 key = PyIter_Next(it);
 if (key == NULL)
msg379799 - (view) Author: STINNER Victor (vstinner) * (Python committer) Date: 2020年10月27日 20:32
> Why did you put _PyLong_GetOne() inside the loop for the fast path and outside the loop for the slow path?
Oh, I didn't notice that the first part was also a loop. I wrote PR 23008 to move the call out of the loop.
I tried to avoid calling the function if it's possible that the variable it not used. But here, it's always used, so it's relevant to move the loop invariant out of the loop ;-)
msg379800 - (view) Author: STINNER Victor (vstinner) * (Python committer) Date: 2020年10月27日 20:34
New changeset c310185c081110741fae914c06c7aaf673ad3d0d by Victor Stinner in branch 'master':
bpo-42161: Remove private _PyLong_Zero and _PyLong_One (GH-23003)
https://github.com/python/cpython/commit/c310185c081110741fae914c06c7aaf673ad3d0d
msg379804 - (view) Author: STINNER Victor (vstinner) * (Python committer) Date: 2020年10月27日 21:24
New changeset 35b95aaf21534e4a8e3370dfd6f7482265316c9e by Victor Stinner in branch 'master':
bpo-42161: Micro-optimize _collections._count_elements() (GH-23008)
https://github.com/python/cpython/commit/35b95aaf21534e4a8e3370dfd6f7482265316c9e
msg379805 - (view) Author: STINNER Victor (vstinner) * (Python committer) Date: 2020年10月27日 21:25
Thanks Raymond, I fixed the code.
I close the issue, I removed _PyLong_Zero and _PyLong_One variables.
msg388988 - (view) Author: Raymond Hettinger (rhettinger) * (Python committer) Date: 2021年03月18日 04:42
> Thanks Raymond, I fixed the code.
Can you fix all the other cases where this is used in inner-loop; otherwise, it is undoing everyone else's optimizations and fast paths.
Also, it seems the that primary motivation for this is support subinterpreters. That PEP hasn't been approved, so we should not be making code worse until we know there is going to some offsetting benefit.
For example, the inner-loop code in math_lcm() used to have "res == _PyLong_Zero" which was fast a register-to-register comparison (1 cycle at worst and typically 0 cycles with branch prediction). Also there used to be zero memory accesses. The new code has five sequentially dependent operations and four memory accesses (not what we want in an inner-loop):
 movq __PyRuntime@GOTPCREL(%rip), %rax
 movq 616(%rax), %rax
 movq 16(%rax), %rax
 cmpq %r12, 3560(%rax)
 jne L326
Ideally, the whole PR should be reverted until the subinterpreter PEP is approved. If not, then at least the changes should be made more carefully, hoisting the new call out of the hot loop:
+ zero = _PyLong_GetZero();
 for (i = 1; i < nargs; i++) {
 x = PyNumber_Index(args[i]);
 if (x == NULL) {
 Py_DECREF(res);
 return NULL;
 }
- if (res == _PyLong_GetZero()) {
+ if (res == zero) {
 /* Fast path: just check arguments.
 It is okay to use identity comparison here. */
 Py_DECREF(x);
 continue;
 }
 Py_SETREF(res, long_lcm(res, x));
 Py_DECREF(x);
 if (res == NULL) {
 return NULL;
 }
 }
 return res;
msg389002 - (view) Author: STINNER Victor (vstinner) * (Python committer) Date: 2021年03月18日 09:00
Since it seems like you have already a ready patch to optimize the code, so go ahead and merge it.
msg389049 - (view) Author: Raymond Hettinger (rhettinger) * (Python committer) Date: 2021年03月18日 23:49
I don't have a PR in-hand. I just ran across this when trying to explain 3.9 vs 3.10 timings and M-1 vs Intel timings. Ideally, all of these PRs should be reverted. Short of that, each change needs to be reviewed to see if it created extra work inside a loop. There are 35 calls to PyLong_GetOne and 3 for PyLong_GetZero. Each of those should be checked.
msg394496 - (view) Author: STINNER Victor (vstinner) * (Python committer) Date: 2021年05月26日 22:51
New changeset 3e7ee02327db13e4337374597cdc4458ecb9e3ad by Victor Stinner in branch 'main':
bpo-42161: mathmodule.c: move _PyLong_GetOne() loop invariant (GH-26391)
https://github.com/python/cpython/commit/3e7ee02327db13e4337374597cdc4458ecb9e3ad
msg394498 - (view) Author: STINNER Victor (vstinner) * (Python committer) Date: 2021年05月26日 22:54
Raymond: "Can you fix all the other cases where this is used in inner-loop; otherwise, it is undoing everyone else's optimizations and fast paths."
Done, thanks for the report.
msg394499 - (view) Author: STINNER Victor (vstinner) * (Python committer) Date: 2021年05月26日 23:13
New changeset 4115996342278de7c2a1b59ac348322e7a4e9072 by Miss Islington (bot) in branch '3.10':
bpo-42161: mathmodule.c: move _PyLong_GetOne() loop invariant (GH-26391) (GH-26393)
https://github.com/python/cpython/commit/4115996342278de7c2a1b59ac348322e7a4e9072
msg410847 - (view) Author: Raymond Hettinger (rhettinger) * (Python committer) Date: 2022年01月18日 08:02
New changeset 243c31667cc15a9a338330ad9b2a29b1cd1c76ec by Raymond Hettinger in branch 'main':
bpo-42161: Hoist the _PyLong_GetOne() call out of the inner loop. (GH-30656)
https://github.com/python/cpython/commit/243c31667cc15a9a338330ad9b2a29b1cd1c76ec
History
Date User Action Args
2022年04月11日 14:59:37adminsetgithub: 86327
2022年01月18日 08:02:43rhettingersetmessages: + msg410847
2022年01月18日 07:03:34rhettingersetpull_requests: + pull_request28857
2021年05月26日 23:13:24vstinnersetmessages: + msg394499
2021年05月26日 22:54:42vstinnersetstatus: open -> closed

messages: + msg394498
stage: patch review -> resolved
2021年05月26日 22:51:16miss-islingtonsetnosy: + miss-islington
pull_requests: + pull_request24986
2021年05月26日 22:51:14vstinnersetmessages: + msg394496
2021年05月26日 22:31:15vstinnersetstage: resolved -> patch review
pull_requests: + pull_request24984
2021年03月18日 23:49:45rhettingersetmessages: + msg389049
2021年03月18日 11:40:07mark.dickinsonsetnosy: + mark.dickinson
2021年03月18日 09:00:36vstinnersetmessages: + msg389002
2021年03月18日 04:42:41rhettingersetstatus: closed -> open
nosy: + serhiy.storchaka, pablogsal
messages: + msg388988

2020年10月27日 21:25:31vstinnersetstatus: open -> closed
resolution: fixed
messages: + msg379805

stage: patch review -> resolved
2020年10月27日 21:24:41vstinnersetmessages: + msg379804
2020年10月27日 20:34:46vstinnersetmessages: + msg379800
2020年10月27日 20:32:54vstinnersetmessages: + msg379799
2020年10月27日 20:31:28vstinnersetpull_requests: + pull_request21924
2020年10月27日 20:15:19rhettingersetnosy: + rhettinger
messages: + msg379798
2020年10月27日 16:16:49vstinnersetpull_requests: + pull_request21918
2020年10月27日 16:13:12vstinnersetmessages: + msg379768
2020年10月27日 01:41:28vstinnersetpull_requests: + pull_request21913
2020年10月27日 01:24:58vstinnersetmessages: + msg379709
2020年10月26日 23:39:34corona10setnosy: + corona10
2020年10月26日 23:33:25vstinnersetpull_requests: + pull_request21909
2020年10月26日 23:00:10vstinnersetmessages: + msg379700
2020年10月26日 22:14:23vstinnersetkeywords: + patch
stage: patch review
pull_requests: + pull_request21907
2020年10月26日 21:59:37vstinnercreate

AltStyle によって変換されたページ (->オリジナル) /