Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
C
cpython
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
cpython
Commits
a9f05d69
Commit
a9f05d69
authored
May 24, 2019
by
Victor Stinner
Committed by
GitHub
May 24, 2019
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
bpo-37032: Add CodeType.replace() method (GH-13542)
parent
561612d8
Changes
8
Hide whitespace changes
Inline
Side-by-side
Showing
8 changed files
with
401 additions
and
33 deletions
+401
-33
Doc/whatsnew/3.8.rst
Doc/whatsnew/3.8.rst
+5
-1
Lib/modulefinder.py
Lib/modulefinder.py
+1
-7
Lib/test/test_code.py
Lib/test/test_code.py
+60
-6
Lib/test/test_import/__init__.py
Lib/test/test_import/__init__.py
+1
-7
Lib/types.py
Lib/types.py
+2
-8
Misc/NEWS.d/next/Core and Builtins/2019-05-24-12-38-40.bpo-37032.T8rSH8.rst
...ore and Builtins/2019-05-24-12-38-40.bpo-37032.T8rSH8.rst
+1
-0
Objects/clinic/codeobject.c.h
Objects/clinic/codeobject.c.h
+256
-0
Objects/codeobject.c
Objects/codeobject.c
+75
-4
No files found.
Doc/whatsnew/3.8.rst
View file @
a9f05d69
...
@@ -240,6 +240,9 @@ Other Language Changes
...
@@ -240,6 +240,9 @@ Other Language Changes
and Windows use this to properly terminate scripts in interactive sessions.
and Windows use this to properly terminate scripts in interactive sessions.
(Contributed by Google via Gregory P. Smith in :issue:`1054041`.)
(Contributed by Google via Gregory P. Smith in :issue:`1054041`.)
* Added new ``replace()`` method to the code type (:class:`types.CodeType`).
(Contributed by Victor Stinner in :issue:`37032`.)
New Modules
New Modules
===========
===========
...
@@ -1051,7 +1054,8 @@ Changes in the Python API
...
@@ -1051,7 +1054,8 @@ Changes in the Python API
* :class:`types.CodeType` has a new parameter in the second position of the
* :class:`types.CodeType` has a new parameter in the second position of the
constructor (*posonlyargcount*) to support positional-only arguments defined
constructor (*posonlyargcount*) to support positional-only arguments defined
in :pep:`570`.
in :pep:`570`. A new ``replace()`` method of :class:`types.CodeType` can be
used to make the code future-proof.
Changes in the C API
Changes in the C API
...
...
Lib/modulefinder.py
View file @
a9f05d69
...
@@ -619,13 +619,7 @@ class ModuleFinder:
...
@@ -619,13 +619,7 @@ class ModuleFinder:
if
isinstance
(
consts
[
i
],
type
(
co
)):
if
isinstance
(
consts
[
i
],
type
(
co
)):
consts
[
i
]
=
self
.
replace_paths_in_code
(
consts
[
i
])
consts
[
i
]
=
self
.
replace_paths_in_code
(
consts
[
i
])
return
types
.
CodeType
(
co
.
co_argcount
,
co
.
co_posonlyargcount
,
return
co
.
replace
(
co_consts
=
tuple
(
consts
),
co_filename
=
new_filename
)
co
.
co_kwonlyargcount
,
co
.
co_nlocals
,
co
.
co_stacksize
,
co
.
co_flags
,
co
.
co_code
,
tuple
(
consts
),
co
.
co_names
,
co
.
co_varnames
,
new_filename
,
co
.
co_name
,
co
.
co_firstlineno
,
co
.
co_lnotab
,
co
.
co_freevars
,
co
.
co_cellvars
)
def
test
():
def
test
():
...
...
Lib/test/test_code.py
View file @
a9f05d69
...
@@ -174,18 +174,14 @@ class CodeTest(unittest.TestCase):
...
@@ -174,18 +174,14 @@ class CodeTest(unittest.TestCase):
@
cpython_only
@
cpython_only
def
test_closure_injection
(
self
):
def
test_closure_injection
(
self
):
# From https://bugs.python.org/issue32176
# From https://bugs.python.org/issue32176
from
types
import
FunctionType
,
CodeType
from
types
import
FunctionType
def
create_closure
(
__class__
):
def
create_closure
(
__class__
):
return
(
lambda
:
__class__
).
__closure__
return
(
lambda
:
__class__
).
__closure__
def
new_code
(
c
):
def
new_code
(
c
):
'''A new code object with a __class__ cell added to freevars'''
'''A new code object with a __class__ cell added to freevars'''
return
CodeType
(
return
c
.
replace
(
co_freevars
=
c
.
co_freevars
+
(
'__class__'
,))
c
.
co_argcount
,
c
.
co_posonlyargcount
,
c
.
co_kwonlyargcount
,
c
.
co_nlocals
,
c
.
co_stacksize
,
c
.
co_flags
,
c
.
co_code
,
c
.
co_consts
,
c
.
co_names
,
c
.
co_varnames
,
c
.
co_filename
,
c
.
co_name
,
c
.
co_firstlineno
,
c
.
co_lnotab
,
c
.
co_freevars
+
(
'__class__'
,),
c
.
co_cellvars
)
def
add_foreign_method
(
cls
,
name
,
f
):
def
add_foreign_method
(
cls
,
name
,
f
):
code
=
new_code
(
f
.
__code__
)
code
=
new_code
(
f
.
__code__
)
...
@@ -212,6 +208,64 @@ class CodeTest(unittest.TestCase):
...
@@ -212,6 +208,64 @@ class CodeTest(unittest.TestCase):
obj
=
List
([
1
,
2
,
3
])
obj
=
List
([
1
,
2
,
3
])
self
.
assertEqual
(
obj
[
0
],
"Foreign getitem: 1"
)
self
.
assertEqual
(
obj
[
0
],
"Foreign getitem: 1"
)
def
test_constructor
(
self
):
def
func
():
pass
co
=
func
.
__code__
CodeType
=
type
(
co
)
# test code constructor
return
CodeType
(
co
.
co_argcount
,
co
.
co_posonlyargcount
,
co
.
co_kwonlyargcount
,
co
.
co_nlocals
,
co
.
co_stacksize
,
co
.
co_flags
,
co
.
co_code
,
co
.
co_consts
,
co
.
co_names
,
co
.
co_varnames
,
co
.
co_filename
,
co
.
co_name
,
co
.
co_firstlineno
,
co
.
co_lnotab
,
co
.
co_freevars
,
co
.
co_cellvars
)
def
test_replace
(
self
):
def
func
():
x
=
1
return
x
code
=
func
.
__code__
# different co_name, co_varnames, co_consts
def
func2
():
y
=
2
return
y
code2
=
func
.
__code__
for
attr
,
value
in
(
(
"co_argcount"
,
0
),
(
"co_posonlyargcount"
,
0
),
(
"co_kwonlyargcount"
,
0
),
(
"co_nlocals"
,
0
),
(
"co_stacksize"
,
0
),
(
"co_flags"
,
code
.
co_flags
|
inspect
.
CO_COROUTINE
),
(
"co_firstlineno"
,
100
),
(
"co_code"
,
code2
.
co_code
),
(
"co_consts"
,
code2
.
co_consts
),
(
"co_names"
,
(
"myname"
,)),
(
"co_varnames"
,
code2
.
co_varnames
),
(
"co_freevars"
,
(
"freevar"
,)),
(
"co_cellvars"
,
(
"cellvar"
,)),
(
"co_filename"
,
"newfilename"
),
(
"co_name"
,
"newname"
),
(
"co_lnotab"
,
code2
.
co_lnotab
),
):
with
self
.
subTest
(
attr
=
attr
,
value
=
value
):
new_code
=
code
.
replace
(
**
{
attr
:
value
})
self
.
assertEqual
(
getattr
(
new_code
,
attr
),
value
)
def
isinterned
(
s
):
def
isinterned
(
s
):
return
s
is
sys
.
intern
((
'_'
+
s
+
'_'
)[
1
:
-
1
])
return
s
is
sys
.
intern
((
'_'
+
s
+
'_'
)[
1
:
-
1
])
...
...
Lib/test/test_import/__init__.py
View file @
a9f05d69
...
@@ -674,13 +674,7 @@ func_filename = func.__code__.co_filename
...
@@ -674,13 +674,7 @@ func_filename = func.__code__.co_filename
foreign_code
=
importlib
.
import_module
.
__code__
foreign_code
=
importlib
.
import_module
.
__code__
pos
=
constants
.
index
(
1
)
pos
=
constants
.
index
(
1
)
constants
[
pos
]
=
foreign_code
constants
[
pos
]
=
foreign_code
code
=
type
(
code
)(
code
.
co_argcount
,
code
.
co_posonlyargcount
,
code
=
code
.
replace
(
co_consts
=
tuple
(
constants
))
code
.
co_kwonlyargcount
,
code
.
co_nlocals
,
code
.
co_stacksize
,
code
.
co_flags
,
code
.
co_code
,
tuple
(
constants
),
code
.
co_names
,
code
.
co_varnames
,
code
.
co_filename
,
code
.
co_name
,
code
.
co_firstlineno
,
code
.
co_lnotab
,
code
.
co_freevars
,
code
.
co_cellvars
)
with
open
(
self
.
compiled_name
,
"wb"
)
as
f
:
with
open
(
self
.
compiled_name
,
"wb"
)
as
f
:
f
.
write
(
header
)
f
.
write
(
header
)
marshal
.
dump
(
code
,
f
)
marshal
.
dump
(
code
,
f
)
...
...
Lib/types.py
View file @
a9f05d69
...
@@ -262,14 +262,8 @@ def coroutine(func):
...
@@ -262,14 +262,8 @@ def coroutine(func):
if
co_flags
&
0x20
:
if
co_flags
&
0x20
:
# TODO: Implement this in C.
# TODO: Implement this in C.
co
=
func
.
__code__
co
=
func
.
__code__
func
.
__code__
=
CodeType
(
# 0x100 == CO_ITERABLE_COROUTINE
co
.
co_argcount
,
co
.
co_posonlyargcount
,
co
.
co_kwonlyargcount
,
co
.
co_nlocals
,
func
.
__code__
=
co
.
replace
(
co_flags
=
co
.
co_flags
|
0x100
)
co
.
co_stacksize
,
co
.
co_flags
|
0x100
,
# 0x100 == CO_ITERABLE_COROUTINE
co
.
co_code
,
co
.
co_consts
,
co
.
co_names
,
co
.
co_varnames
,
co
.
co_filename
,
co
.
co_name
,
co
.
co_firstlineno
,
co
.
co_lnotab
,
co
.
co_freevars
,
co
.
co_cellvars
)
return
func
return
func
# The following code is primarily to support functions that
# The following code is primarily to support functions that
...
...
Misc/NEWS.d/next/Core and Builtins/2019-05-24-12-38-40.bpo-37032.T8rSH8.rst
0 → 100644
View file @
a9f05d69
Added new ``replace()`` method to the code type (:class:`types.CodeType`).
Objects/clinic/codeobject.c.h
0 → 100644
View file @
a9f05d69
/*[clinic input]
preserve
[clinic start generated code]*/
PyDoc_STRVAR
(
code_replace__doc__
,
"replace($self, /, *, co_argcount=-1, co_posonlyargcount=-1,
\n
"
" co_kwonlyargcount=-1, co_nlocals=-1, co_stacksize=-1,
\n
"
" co_flags=-1, co_firstlineno=-1, co_code=None, co_consts=None,
\n
"
" co_names=None, co_varnames=None, co_freevars=None,
\n
"
" co_cellvars=None, co_filename=None, co_name=None,
\n
"
" co_lnotab=None)
\n
"
"--
\n
"
"
\n
"
"Return a new code object with new specified fields."
);
#define CODE_REPLACE_METHODDEF \
{"replace", (PyCFunction)(void(*)(void))code_replace, METH_FASTCALL|METH_KEYWORDS, code_replace__doc__},
static
PyObject
*
code_replace_impl
(
PyCodeObject
*
self
,
int
co_argcount
,
int
co_posonlyargcount
,
int
co_kwonlyargcount
,
int
co_nlocals
,
int
co_stacksize
,
int
co_flags
,
int
co_firstlineno
,
PyBytesObject
*
co_code
,
PyObject
*
co_consts
,
PyObject
*
co_names
,
PyObject
*
co_varnames
,
PyObject
*
co_freevars
,
PyObject
*
co_cellvars
,
PyObject
*
co_filename
,
PyObject
*
co_name
,
PyBytesObject
*
co_lnotab
);
static
PyObject
*
code_replace
(
PyCodeObject
*
self
,
PyObject
*
const
*
args
,
Py_ssize_t
nargs
,
PyObject
*
kwnames
)
{
PyObject
*
return_value
=
NULL
;
static
const
char
*
const
_keywords
[]
=
{
"co_argcount"
,
"co_posonlyargcount"
,
"co_kwonlyargcount"
,
"co_nlocals"
,
"co_stacksize"
,
"co_flags"
,
"co_firstlineno"
,
"co_code"
,
"co_consts"
,
"co_names"
,
"co_varnames"
,
"co_freevars"
,
"co_cellvars"
,
"co_filename"
,
"co_name"
,
"co_lnotab"
,
NULL
};
static
_PyArg_Parser
_parser
=
{
NULL
,
_keywords
,
"replace"
,
0
};
PyObject
*
argsbuf
[
16
];
Py_ssize_t
noptargs
=
nargs
+
(
kwnames
?
PyTuple_GET_SIZE
(
kwnames
)
:
0
)
-
0
;
int
co_argcount
=
self
->
co_argcount
;
int
co_posonlyargcount
=
self
->
co_posonlyargcount
;
int
co_kwonlyargcount
=
self
->
co_kwonlyargcount
;
int
co_nlocals
=
self
->
co_nlocals
;
int
co_stacksize
=
self
->
co_stacksize
;
int
co_flags
=
self
->
co_flags
;
int
co_firstlineno
=
self
->
co_firstlineno
;
PyBytesObject
*
co_code
=
(
PyBytesObject
*
)
self
->
co_code
;
PyObject
*
co_consts
=
self
->
co_consts
;
PyObject
*
co_names
=
self
->
co_names
;
PyObject
*
co_varnames
=
self
->
co_varnames
;
PyObject
*
co_freevars
=
self
->
co_freevars
;
PyObject
*
co_cellvars
=
self
->
co_cellvars
;
PyObject
*
co_filename
=
self
->
co_filename
;
PyObject
*
co_name
=
self
->
co_name
;
PyBytesObject
*
co_lnotab
=
(
PyBytesObject
*
)
self
->
co_lnotab
;
args
=
_PyArg_UnpackKeywords
(
args
,
nargs
,
NULL
,
kwnames
,
&
_parser
,
0
,
0
,
0
,
argsbuf
);
if
(
!
args
)
{
goto
exit
;
}
if
(
!
noptargs
)
{
goto
skip_optional_kwonly
;
}
if
(
args
[
0
])
{
if
(
PyFloat_Check
(
args
[
0
]))
{
PyErr_SetString
(
PyExc_TypeError
,
"integer argument expected, got float"
);
goto
exit
;
}
co_argcount
=
_PyLong_AsInt
(
args
[
0
]);
if
(
co_argcount
==
-
1
&&
PyErr_Occurred
())
{
goto
exit
;
}
if
(
!--
noptargs
)
{
goto
skip_optional_kwonly
;
}
}
if
(
args
[
1
])
{
if
(
PyFloat_Check
(
args
[
1
]))
{
PyErr_SetString
(
PyExc_TypeError
,
"integer argument expected, got float"
);
goto
exit
;
}
co_posonlyargcount
=
_PyLong_AsInt
(
args
[
1
]);
if
(
co_posonlyargcount
==
-
1
&&
PyErr_Occurred
())
{
goto
exit
;
}
if
(
!--
noptargs
)
{
goto
skip_optional_kwonly
;
}
}
if
(
args
[
2
])
{
if
(
PyFloat_Check
(
args
[
2
]))
{
PyErr_SetString
(
PyExc_TypeError
,
"integer argument expected, got float"
);
goto
exit
;
}
co_kwonlyargcount
=
_PyLong_AsInt
(
args
[
2
]);
if
(
co_kwonlyargcount
==
-
1
&&
PyErr_Occurred
())
{
goto
exit
;
}
if
(
!--
noptargs
)
{
goto
skip_optional_kwonly
;
}
}
if
(
args
[
3
])
{
if
(
PyFloat_Check
(
args
[
3
]))
{
PyErr_SetString
(
PyExc_TypeError
,
"integer argument expected, got float"
);
goto
exit
;
}
co_nlocals
=
_PyLong_AsInt
(
args
[
3
]);
if
(
co_nlocals
==
-
1
&&
PyErr_Occurred
())
{
goto
exit
;
}
if
(
!--
noptargs
)
{
goto
skip_optional_kwonly
;
}
}
if
(
args
[
4
])
{
if
(
PyFloat_Check
(
args
[
4
]))
{
PyErr_SetString
(
PyExc_TypeError
,
"integer argument expected, got float"
);
goto
exit
;
}
co_stacksize
=
_PyLong_AsInt
(
args
[
4
]);
if
(
co_stacksize
==
-
1
&&
PyErr_Occurred
())
{
goto
exit
;
}
if
(
!--
noptargs
)
{
goto
skip_optional_kwonly
;
}
}
if
(
args
[
5
])
{
if
(
PyFloat_Check
(
args
[
5
]))
{
PyErr_SetString
(
PyExc_TypeError
,
"integer argument expected, got float"
);
goto
exit
;
}
co_flags
=
_PyLong_AsInt
(
args
[
5
]);
if
(
co_flags
==
-
1
&&
PyErr_Occurred
())
{
goto
exit
;
}
if
(
!--
noptargs
)
{
goto
skip_optional_kwonly
;
}
}
if
(
args
[
6
])
{
if
(
PyFloat_Check
(
args
[
6
]))
{
PyErr_SetString
(
PyExc_TypeError
,
"integer argument expected, got float"
);
goto
exit
;
}
co_firstlineno
=
_PyLong_AsInt
(
args
[
6
]);
if
(
co_firstlineno
==
-
1
&&
PyErr_Occurred
())
{
goto
exit
;
}
if
(
!--
noptargs
)
{
goto
skip_optional_kwonly
;
}
}
if
(
args
[
7
])
{
if
(
!
PyBytes_Check
(
args
[
7
]))
{
_PyArg_BadArgument
(
"replace"
,
8
,
"bytes"
,
args
[
7
]);
goto
exit
;
}
co_code
=
(
PyBytesObject
*
)
args
[
7
];
if
(
!--
noptargs
)
{
goto
skip_optional_kwonly
;
}
}
if
(
args
[
8
])
{
if
(
!
PyTuple_Check
(
args
[
8
]))
{
_PyArg_BadArgument
(
"replace"
,
9
,
"tuple"
,
args
[
8
]);
goto
exit
;
}
co_consts
=
args
[
8
];
if
(
!--
noptargs
)
{
goto
skip_optional_kwonly
;
}
}
if
(
args
[
9
])
{
if
(
!
PyTuple_Check
(
args
[
9
]))
{
_PyArg_BadArgument
(
"replace"
,
10
,
"tuple"
,
args
[
9
]);
goto
exit
;
}
co_names
=
args
[
9
];
if
(
!--
noptargs
)
{
goto
skip_optional_kwonly
;
}
}
if
(
args
[
10
])
{
if
(
!
PyTuple_Check
(
args
[
10
]))
{
_PyArg_BadArgument
(
"replace"
,
11
,
"tuple"
,
args
[
10
]);
goto
exit
;
}
co_varnames
=
args
[
10
];
if
(
!--
noptargs
)
{
goto
skip_optional_kwonly
;
}
}
if
(
args
[
11
])
{
if
(
!
PyTuple_Check
(
args
[
11
]))
{
_PyArg_BadArgument
(
"replace"
,
12
,
"tuple"
,
args
[
11
]);
goto
exit
;
}
co_freevars
=
args
[
11
];
if
(
!--
noptargs
)
{
goto
skip_optional_kwonly
;
}
}
if
(
args
[
12
])
{
if
(
!
PyTuple_Check
(
args
[
12
]))
{
_PyArg_BadArgument
(
"replace"
,
13
,
"tuple"
,
args
[
12
]);
goto
exit
;
}
co_cellvars
=
args
[
12
];
if
(
!--
noptargs
)
{
goto
skip_optional_kwonly
;
}
}
if
(
args
[
13
])
{
if
(
!
PyUnicode_Check
(
args
[
13
]))
{
_PyArg_BadArgument
(
"replace"
,
14
,
"str"
,
args
[
13
]);
goto
exit
;
}
if
(
PyUnicode_READY
(
args
[
13
])
==
-
1
)
{
goto
exit
;
}
co_filename
=
args
[
13
];
if
(
!--
noptargs
)
{
goto
skip_optional_kwonly
;
}
}
if
(
args
[
14
])
{
if
(
!
PyUnicode_Check
(
args
[
14
]))
{
_PyArg_BadArgument
(
"replace"
,
15
,
"str"
,
args
[
14
]);
goto
exit
;
}
if
(
PyUnicode_READY
(
args
[
14
])
==
-
1
)
{
goto
exit
;
}
co_name
=
args
[
14
];
if
(
!--
noptargs
)
{
goto
skip_optional_kwonly
;
}
}
if
(
!
PyBytes_Check
(
args
[
15
]))
{
_PyArg_BadArgument
(
"replace"
,
16
,
"bytes"
,
args
[
15
]);
goto
exit
;
}
co_lnotab
=
(
PyBytesObject
*
)
args
[
15
];
skip_optional_kwonly:
return_value
=
code_replace_impl
(
self
,
co_argcount
,
co_posonlyargcount
,
co_kwonlyargcount
,
co_nlocals
,
co_stacksize
,
co_flags
,
co_firstlineno
,
co_code
,
co_consts
,
co_names
,
co_varnames
,
co_freevars
,
co_cellvars
,
co_filename
,
co_name
,
co_lnotab
);
exit:
return
return_value
;
}
/*[clinic end generated code: output=624ab6f2ea8f0ea4 input=a9049054013a1b77]*/
Objects/codeobject.c
View file @
a9f05d69
...
@@ -5,6 +5,7 @@
...
@@ -5,6 +5,7 @@
#include "structmember.h"
#include "structmember.h"
#include "pycore_pystate.h"
#include "pycore_pystate.h"
#include "pycore_tupleobject.h"
#include "pycore_tupleobject.h"
#include "clinic/codeobject.c.h"
/* Holder for co_extra information */
/* Holder for co_extra information */
typedef
struct
{
typedef
struct
{
...
@@ -12,6 +13,11 @@ typedef struct {
...
@@ -12,6 +13,11 @@ typedef struct {
void
*
ce_extras
[
1
];
void
*
ce_extras
[
1
];
}
_PyCodeObjectExtra
;
}
_PyCodeObjectExtra
;
/*[clinic input]
class code "PyCodeObject *" "&PyCode_Type"
[clinic start generated code]*/
/*[clinic end generated code: output=da39a3ee5e6b4b0d input=78aa5d576683bb4b]*/
/* all_name_chars(s): true iff s matches [a-zA-Z0-9_]* */
/* all_name_chars(s): true iff s matches [a-zA-Z0-9_]* */
static
int
static
int
all_name_chars
(
PyObject
*
o
)
all_name_chars
(
PyObject
*
o
)
...
@@ -109,7 +115,8 @@ PyCode_New(int argcount, int posonlyargcount, int kwonlyargcount,
...
@@ -109,7 +115,8 @@ PyCode_New(int argcount, int posonlyargcount, int kwonlyargcount,
/* Check argument types */
/* Check argument types */
if
(
argcount
<
0
||
posonlyargcount
<
0
||
kwonlyargcount
<
0
||
if
(
argcount
<
0
||
posonlyargcount
<
0
||
kwonlyargcount
<
0
||
nlocals
<
0
||
code
==
NULL
||
!
PyBytes_Check
(
code
)
||
nlocals
<
0
||
stacksize
<
0
||
flags
<
0
||
code
==
NULL
||
!
PyBytes_Check
(
code
)
||
consts
==
NULL
||
!
PyTuple_Check
(
consts
)
||
consts
==
NULL
||
!
PyTuple_Check
(
consts
)
||
names
==
NULL
||
!
PyTuple_Check
(
names
)
||
names
==
NULL
||
!
PyTuple_Check
(
names
)
||
varnames
==
NULL
||
!
PyTuple_Check
(
varnames
)
||
varnames
==
NULL
||
!
PyTuple_Check
(
varnames
)
||
...
@@ -122,9 +129,13 @@ PyCode_New(int argcount, int posonlyargcount, int kwonlyargcount,
...
@@ -122,9 +129,13 @@ PyCode_New(int argcount, int posonlyargcount, int kwonlyargcount,
return
NULL
;
return
NULL
;
}
}
/* Ensure that the filename is a ready Unicode string */
/* Ensure that strings are ready Unicode string */
if
(
PyUnicode_READY
(
filename
)
<
0
)
if
(
PyUnicode_READY
(
name
)
<
0
)
{
return
NULL
;
}
if
(
PyUnicode_READY
(
filename
)
<
0
)
{
return
NULL
;
return
NULL
;
}
intern_strings
(
names
);
intern_strings
(
names
);
intern_strings
(
varnames
);
intern_strings
(
varnames
);
...
@@ -482,7 +493,7 @@ code_dealloc(PyCodeObject *co)
...
@@ -482,7 +493,7 @@ code_dealloc(PyCodeObject *co)
}
}
static
PyObject
*
static
PyObject
*
code_sizeof
(
PyCodeObject
*
co
,
void
*
unused
)
code_sizeof
(
PyCodeObject
*
co
,
PyObject
*
Py_UNUSED
(
args
)
)
{
{
Py_ssize_t
res
=
_PyObject_SIZE
(
Py_TYPE
(
co
));
Py_ssize_t
res
=
_PyObject_SIZE
(
Py_TYPE
(
co
));
_PyCodeObjectExtra
*
co_extra
=
(
_PyCodeObjectExtra
*
)
co
->
co_extra
;
_PyCodeObjectExtra
*
co_extra
=
(
_PyCodeObjectExtra
*
)
co
->
co_extra
;
...
@@ -497,6 +508,65 @@ code_sizeof(PyCodeObject *co, void *unused)
...
@@ -497,6 +508,65 @@ code_sizeof(PyCodeObject *co, void *unused)
return
PyLong_FromSsize_t
(
res
);
return
PyLong_FromSsize_t
(
res
);
}
}
/*[clinic input]
code.replace
*
co_argcount: int(c_default="self->co_argcount") = -1
co_posonlyargcount: int(c_default="self->co_posonlyargcount") = -1
co_kwonlyargcount: int(c_default="self->co_kwonlyargcount") = -1
co_nlocals: int(c_default="self->co_nlocals") = -1
co_stacksize: int(c_default="self->co_stacksize") = -1
co_flags: int(c_default="self->co_flags") = -1
co_firstlineno: int(c_default="self->co_firstlineno") = -1
co_code: PyBytesObject(c_default="(PyBytesObject *)self->co_code") = None
co_consts: object(subclass_of="&PyTuple_Type", c_default="self->co_consts") = None
co_names: object(subclass_of="&PyTuple_Type", c_default="self->co_names") = None
co_varnames: object(subclass_of="&PyTuple_Type", c_default="self->co_varnames") = None
co_freevars: object(subclass_of="&PyTuple_Type", c_default="self->co_freevars") = None
co_cellvars: object(subclass_of="&PyTuple_Type", c_default="self->co_cellvars") = None
co_filename: unicode(c_default="self->co_filename") = None
co_name: unicode(c_default="self->co_name") = None
co_lnotab: PyBytesObject(c_default="(PyBytesObject *)self->co_lnotab") = None
Return a new code object with new specified fields.
[clinic start generated code]*/
static
PyObject
*
code_replace_impl
(
PyCodeObject
*
self
,
int
co_argcount
,
int
co_posonlyargcount
,
int
co_kwonlyargcount
,
int
co_nlocals
,
int
co_stacksize
,
int
co_flags
,
int
co_firstlineno
,
PyBytesObject
*
co_code
,
PyObject
*
co_consts
,
PyObject
*
co_names
,
PyObject
*
co_varnames
,
PyObject
*
co_freevars
,
PyObject
*
co_cellvars
,
PyObject
*
co_filename
,
PyObject
*
co_name
,
PyBytesObject
*
co_lnotab
)
/*[clinic end generated code: output=25c8e303913bcace input=77189e46579ec426]*/
{
#define CHECK_INT_ARG(ARG) \
if (ARG < 0) { \
PyErr_SetString(PyExc_ValueError, \
#ARG " must be a positive integer"); \
return NULL; \
}
CHECK_INT_ARG
(
co_argcount
);
CHECK_INT_ARG
(
co_posonlyargcount
);
CHECK_INT_ARG
(
co_kwonlyargcount
);
CHECK_INT_ARG
(
co_nlocals
);
CHECK_INT_ARG
(
co_stacksize
);
CHECK_INT_ARG
(
co_flags
);
CHECK_INT_ARG
(
co_firstlineno
);
#undef CHECK_INT_ARG
return
(
PyObject
*
)
PyCode_New
(
co_argcount
,
co_posonlyargcount
,
co_kwonlyargcount
,
co_nlocals
,
co_stacksize
,
co_flags
,
(
PyObject
*
)
co_code
,
co_consts
,
co_names
,
co_varnames
,
co_freevars
,
co_cellvars
,
co_filename
,
co_name
,
co_firstlineno
,
(
PyObject
*
)
co_lnotab
);
}
static
PyObject
*
static
PyObject
*
code_repr
(
PyCodeObject
*
co
)
code_repr
(
PyCodeObject
*
co
)
{
{
...
@@ -751,6 +821,7 @@ code_hash(PyCodeObject *co)
...
@@ -751,6 +821,7 @@ code_hash(PyCodeObject *co)
static
struct
PyMethodDef
code_methods
[]
=
{
static
struct
PyMethodDef
code_methods
[]
=
{
{
"__sizeof__"
,
(
PyCFunction
)
code_sizeof
,
METH_NOARGS
},
{
"__sizeof__"
,
(
PyCFunction
)
code_sizeof
,
METH_NOARGS
},
CODE_REPLACE_METHODDEF
{
NULL
,
NULL
}
/* sentinel */
{
NULL
,
NULL
}
/* sentinel */
};
};
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment