Commit 3c317e76 authored by Serhiy Storchaka's avatar Serhiy Storchaka

Issue #27286: Fixed compiling BUILD_MAP_UNPACK_WITH_CALL opcode. Calling

function with generalized unpacking (PEP 448) and conflicting keyword names
could cause undefined behavior.
parent 4f8aaf64
...@@ -223,6 +223,7 @@ _code_type = type(_write_atomic.__code__) ...@@ -223,6 +223,7 @@ _code_type = type(_write_atomic.__code__)
# Python 3.5b1 3330 (PEP 448: Additional Unpacking Generalizations) # Python 3.5b1 3330 (PEP 448: Additional Unpacking Generalizations)
# Python 3.5b2 3340 (fix dictionary display evaluation order #11205) # Python 3.5b2 3340 (fix dictionary display evaluation order #11205)
# Python 3.5b2 3350 (add GET_YIELD_FROM_ITER opcode #24400) # Python 3.5b2 3350 (add GET_YIELD_FROM_ITER opcode #24400)
# Python 3.5.2 3351 (fix BUILD_MAP_UNPACK_WITH_CALL opcode #27286)
# #
# MAGIC must change whenever the bytecode emitted by the compiler may no # MAGIC must change whenever the bytecode emitted by the compiler may no
# longer be understood by older implementations of the eval loop (usually # longer be understood by older implementations of the eval loop (usually
...@@ -231,7 +232,7 @@ _code_type = type(_write_atomic.__code__) ...@@ -231,7 +232,7 @@ _code_type = type(_write_atomic.__code__)
# Whenever MAGIC_NUMBER is changed, the ranges in the magic_values array # Whenever MAGIC_NUMBER is changed, the ranges in the magic_values array
# in PC/launcher.c must also be updated. # in PC/launcher.c must also be updated.
MAGIC_NUMBER = (3350).to_bytes(2, 'little') + b'\r\n' MAGIC_NUMBER = (3351).to_bytes(2, 'little') + b'\r\n'
_RAW_MAGIC_NUMBER = int.from_bytes(MAGIC_NUMBER, 'little') # For import.c _RAW_MAGIC_NUMBER = int.from_bytes(MAGIC_NUMBER, 'little') # For import.c
_PYCACHE = '__pycache__' _PYCACHE = '__pycache__'
......
...@@ -57,6 +57,10 @@ Here we add keyword arguments ...@@ -57,6 +57,10 @@ Here we add keyword arguments
Traceback (most recent call last): Traceback (most recent call last):
... ...
TypeError: f() got multiple values for keyword argument 'a' TypeError: f() got multiple values for keyword argument 'a'
>>> f(1, 2, a=3, **{'a': 4}, **{'a': 5})
Traceback (most recent call last):
...
TypeError: f() got multiple values for keyword argument 'a'
>>> f(1, 2, 3, *[4, 5], **{'a':6, 'b':7}) >>> f(1, 2, 3, *[4, 5], **{'a':6, 'b':7})
(1, 2, 3, 4, 5) {'a': 6, 'b': 7} (1, 2, 3, 4, 5) {'a': 6, 'b': 7}
>>> f(1, 2, 3, x=4, y=5, *(6, 7), **{'a':8, 'b': 9}) >>> f(1, 2, 3, x=4, y=5, *(6, 7), **{'a':8, 'b': 9})
......
...@@ -248,6 +248,11 @@ Overridden parameters ...@@ -248,6 +248,11 @@ Overridden parameters
... ...
TypeError: f() got multiple values for keyword argument 'x' TypeError: f() got multiple values for keyword argument 'x'
>>> f(x=5, **{'x': 3}, **{'x': 2})
Traceback (most recent call last):
...
TypeError: f() got multiple values for keyword argument 'x'
>>> f(**{1: 3}, **{1: 5}) >>> f(**{1: 3}, **{1: 5})
Traceback (most recent call last): Traceback (most recent call last):
... ...
......
...@@ -10,6 +10,10 @@ Release date: tba ...@@ -10,6 +10,10 @@ Release date: tba
Core and Builtins Core and Builtins
----------------- -----------------
- Issue #27286: Fixed compiling BUILD_MAP_UNPACK_WITH_CALL opcode. Calling
function with generalized unpacking (PEP 448) and conflicting keyword names
could cause undefined behavior.
- Issue #27066: Fixed SystemError if a custom opener (for open()) returns a - Issue #27066: Fixed SystemError if a custom opener (for open()) returns a
negative number without setting an exception. negative number without setting an exception.
......
...@@ -1081,7 +1081,7 @@ static PYC_MAGIC magic_values[] = { ...@@ -1081,7 +1081,7 @@ static PYC_MAGIC magic_values[] = {
{ 3160, 3180, L"3.2" }, { 3160, 3180, L"3.2" },
{ 3190, 3230, L"3.3" }, { 3190, 3230, L"3.3" },
{ 3250, 3310, L"3.4" }, { 3250, 3310, L"3.4" },
{ 3320, 3350, L"3.5" }, { 3320, 3351, L"3.5" },
{ 3360, 3361, L"3.6" }, { 3360, 3361, L"3.6" },
{ 0 } { 0 }
}; };
......
...@@ -3262,7 +3262,7 @@ compiler_call_helper(struct compiler *c, ...@@ -3262,7 +3262,7 @@ compiler_call_helper(struct compiler *c,
code |= 2; code |= 2;
if (nsubkwargs > 1) { if (nsubkwargs > 1) {
/* Pack it all up */ /* Pack it all up */
int function_pos = n + (code & 1) + nkw + 1; int function_pos = n + (code & 1) + 2 * nkw + 1;
ADDOP_I(c, BUILD_MAP_UNPACK_WITH_CALL, nsubkwargs | (function_pos << 8)); ADDOP_I(c, BUILD_MAP_UNPACK_WITH_CALL, nsubkwargs | (function_pos << 8));
} }
} }
......
This diff is collapsed.
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment