Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
C
cython
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
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Gwenaël Samain
cython
Commits
583504ae
Commit
583504ae
authored
Nov 28, 2018
by
Boxiang Sun
Committed by
gsamain
Jul 08, 2019
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Add object and type of nogil extension
parent
2aaf13f7
Changes
1
Show whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
214 additions
and
0 deletions
+214
-0
Cython/Compiler/PyrexTypes.py
Cython/Compiler/PyrexTypes.py
+214
-0
No files found.
Cython/Compiler/PyrexTypes.py
View file @
583504ae
...
@@ -1189,6 +1189,218 @@ class PyObjectType(PyrexType):
...
@@ -1189,6 +1189,218 @@ class PyObjectType(PyrexType):
return
cname
return
cname
class
CythonObjectType
(
PyrexType
):
#
# Base class for all nogil extension object types (reference-counted).
#
name
=
"cythonobject"
is_pyobject
=
0
default_value
=
"0"
declaration_value
=
"0"
buffer_defaults
=
None
is_extern
=
False
is_subclassed
=
False
is_gc_simple
=
False
def
__str__
(
self
):
return
"Cython object"
def
__repr__
(
self
):
return
"<CythonObjectType>"
def
can_coerce_to_pyobject
(
self
,
env
):
return
False
def
can_coerce_from_pyobject
(
self
,
env
):
return
False
def
default_coerced_ctype
(
self
):
"""The default C type that this Python type coerces to, or None."""
return
None
def
assignable_from
(
self
,
src_type
):
# except for pointers, conversion will be attempted
# return not src_type.is_ptr or src_type.is_string or src_type.is_pyunicode_ptr
return
False
def
declaration_code
(
self
,
entity_code
,
for_display
=
0
,
dll_linkage
=
None
,
pyrex
=
0
):
raise
NotImplementedError
(
"Calling declartion_code on Cython object"
)
if
pyrex
or
for_display
:
base_code
=
"object"
else
:
base_code
=
public_decl
(
"PyObject"
,
dll_linkage
)
entity_code
=
"*%s"
%
entity_code
return
self
.
base_declaration_code
(
base_code
,
entity_code
)
def
as_pyobject
(
self
,
cname
):
raise
NotImplementedError
(
"Calling as_pyobject on Cython object"
)
if
(
not
self
.
is_complete
())
or
self
.
is_extension_type
:
return
"(PyObject *)"
+
cname
else
:
return
cname
def
py_type_name
(
self
):
raise
NotImplementedError
(
"Calling py_type_name on Cython object"
)
return
"cythonobject"
def
__lt__
(
self
,
other
):
"""
Make sure we sort highest, as instance checking on py_type_name
('object') is always true
"""
return
False
def
global_init_code
(
self
,
entry
,
code
):
raise
NotImplementedError
(
"Calling global_init_code on Cython object"
)
code
.
put_init_var_to_py_none
(
entry
,
nanny
=
False
)
def
check_for_null_code
(
self
,
cname
):
raise
NotImplementedError
(
"Calling check_for_null_code on Cython object"
)
return
cname
class
CythonExtensionType
(
CythonObjectType
):
#
# A Cython extension type with nogil flag
# TODO: This type may need big amend
#
# name string
# scope CClassScope Attribute namespace
# visibility string
# typedef_flag boolean
# base_type PyExtensionType or None
# nogil boolean
# module_name string or None Qualified name of defining module
# objstruct_cname string Name of PyObject struct
# objtypedef_cname string Name of PyObject struct typedef
# typeobj_cname string or None C code fragment referring to type object
# typeptr_cname string or None Name of pointer to external type object
# vtabslot_cname string Name of C method table member
# vtabstruct_cname string Name of C method table struct
# vtabptr_cname string Name of pointer to C method table
# vtable_cname string Name of C method table definition
# early_init boolean Whether to initialize early (as opposed to during module execution).
# defered_declarations [thunk] Used to declare class hierarchies in order
# check_size 'warn', 'error', 'ignore' What to do if tp_basicsize does not match
is_extension_type
=
1
is_struct
=
1
is_struct_or_union
=
1
nogil
=
1
is_pyobject
=
0
has_attributes
=
1
early_init
=
1
objtypedef_cname
=
None
def
__init__
(
self
,
name
,
typedef_flag
,
base_type
,
is_external
=
0
,
check_size
=
None
):
self
.
name
=
name
self
.
scope
=
None
self
.
typedef_flag
=
typedef_flag
if
base_type
is
not
None
:
base_type
.
is_subclassed
=
True
self
.
base_type
=
base_type
self
.
nogil
=
False
self
.
module_name
=
None
self
.
objstruct_cname
=
None
self
.
typeobj_cname
=
None
self
.
typeptr_cname
=
None
self
.
vtabslot_cname
=
None
self
.
vtabstruct_cname
=
None
self
.
vtabptr_cname
=
None
self
.
vtable_cname
=
None
self
.
is_external
=
is_external
self
.
check_size
=
check_size
or
'warn'
self
.
defered_declarations
=
[]
def
set_scope
(
self
,
scope
):
self
.
scope
=
scope
if
scope
:
scope
.
parent_type
=
self
def
needs_nonecheck
(
self
):
return
True
def
subtype_of_resolved_type
(
self
,
other_type
):
if
other_type
.
is_extension_type
or
other_type
.
is_builtin_type
:
return
self
is
other_type
or
(
self
.
base_type
and
self
.
base_type
.
subtype_of
(
other_type
))
else
:
return
other_type
is
py_object_type
def
typeobj_is_available
(
self
):
# Do we have a pointer to the type object?
return
self
.
typeptr_cname
def
typeobj_is_imported
(
self
):
# If we don't know the C name of the type object but we do
# know which module it's defined in, it will be imported.
return
self
.
typeobj_cname
is
None
and
self
.
module_name
is
not
None
def
assignable_from
(
self
,
src_type
):
if
self
==
src_type
:
return
True
if
isinstance
(
src_type
,
PyExtensionType
):
if
src_type
.
base_type
is
not
None
:
return
self
.
assignable_from
(
src_type
.
base_type
)
if
isinstance
(
src_type
,
BuiltinObjectType
):
# FIXME: This is an ugly special case that we currently
# keep supporting. It allows users to specify builtin
# types as external extension types, while keeping them
# compatible with the real builtin types. We already
# generate a warning for it. Big TODO: remove!
return
(
self
.
module_name
==
'__builtin__'
and
self
.
name
==
src_type
.
name
)
return
False
def
declaration_code
(
self
,
entity_code
,
for_display
=
0
,
dll_linkage
=
None
,
pyrex
=
0
,
deref
=
0
):
if
pyrex
or
for_display
:
base_code
=
self
.
name
else
:
if
self
.
typedef_flag
:
objstruct
=
self
.
objstruct_cname
else
:
objstruct
=
"struct %s"
%
self
.
objstruct_cname
base_code
=
public_decl
(
objstruct
,
dll_linkage
)
if
deref
:
assert
not
entity_code
else
:
entity_code
=
"*%s"
%
entity_code
return
self
.
base_declaration_code
(
base_code
,
entity_code
)
def
literal_code
(
self
,
value
):
return
'(struct %s *)malloc(sizeof(struct %s))'
%
(
self
.
objstruct_cname
,
self
.
objstruct_cname
)
def
type_test_code
(
self
,
py_arg
,
notnone
=
False
):
none_check
=
"((%s) == Py_None)"
%
py_arg
type_check
=
"likely(__Pyx_TypeTest(%s, %s))"
%
(
py_arg
,
self
.
typeptr_cname
)
if
notnone
:
return
type_check
else
:
return
"likely(%s || %s)"
%
(
none_check
,
type_check
)
def
attributes_known
(
self
):
return
self
.
scope
is
not
None
def
__str__
(
self
):
return
self
.
name
def
__repr__
(
self
):
return
"<CythonExtensionType %s%s>"
%
(
self
.
scope
.
class_name
,
(
""
,
" typedef"
)[
self
.
typedef_flag
])
def
py_type_name
(
self
):
if
not
self
.
module_name
:
return
self
.
name
return
"__import__(%r, None, None, ['']).%s"
%
(
self
.
module_name
,
self
.
name
)
builtin_types_that_cannot_create_refcycles
=
set
([
builtin_types_that_cannot_create_refcycles
=
set
([
'object'
,
'bool'
,
'int'
,
'long'
,
'float'
,
'complex'
,
'object'
,
'bool'
,
'int'
,
'long'
,
'float'
,
'complex'
,
'bytearray'
,
'bytes'
,
'unicode'
,
'str'
,
'basestring'
'bytearray'
,
'bytes'
,
'unicode'
,
'str'
,
'basestring'
...
@@ -1347,6 +1559,7 @@ class PyExtensionType(PyObjectType):
...
@@ -1347,6 +1559,7 @@ class PyExtensionType(PyObjectType):
# visibility string
# visibility string
# typedef_flag boolean
# typedef_flag boolean
# base_type PyExtensionType or None
# base_type PyExtensionType or None
# nogil boolean
# module_name string or None Qualified name of defining module
# module_name string or None Qualified name of defining module
# objstruct_cname string Name of PyObject struct
# objstruct_cname string Name of PyObject struct
# objtypedef_cname string Name of PyObject struct typedef
# objtypedef_cname string Name of PyObject struct typedef
...
@@ -1373,6 +1586,7 @@ class PyExtensionType(PyObjectType):
...
@@ -1373,6 +1586,7 @@ class PyExtensionType(PyObjectType):
if
base_type
is
not
None
:
if
base_type
is
not
None
:
base_type
.
is_subclassed
=
True
base_type
.
is_subclassed
=
True
self
.
base_type
=
base_type
self
.
base_type
=
base_type
self
.
nogil
=
False
self
.
module_name
=
None
self
.
module_name
=
None
self
.
objstruct_cname
=
None
self
.
objstruct_cname
=
None
self
.
typeobj_cname
=
None
self
.
typeobj_cname
=
None
...
...
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