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
Boxiang Sun
cython
Commits
36f9d0b8
Commit
36f9d0b8
authored
Sep 23, 2018
by
Stefan Behnel
Browse files
Options
Browse Files
Download
Plain Diff
Merge branch 'master' into release
parents
107fc454
525b8e2e
Changes
7
Show whitespace changes
Inline
Side-by-side
Showing
7 changed files
with
80 additions
and
16 deletions
+80
-16
CHANGES.rst
CHANGES.rst
+6
-2
Cython/Compiler/ExprNodes.py
Cython/Compiler/ExprNodes.py
+32
-8
Cython/Compiler/Nodes.py
Cython/Compiler/Nodes.py
+4
-2
Cython/Compiler/Parsing.py
Cython/Compiler/Parsing.py
+3
-0
docs/src/userguide/wrapping_CPlusPlus.rst
docs/src/userguide/wrapping_CPlusPlus.rst
+13
-4
tests/run/cpp_exceptions.pyx
tests/run/cpp_exceptions.pyx
+14
-0
tests/run/cpp_exceptions_helper.h
tests/run/cpp_exceptions_helper.h
+8
-0
No files found.
CHANGES.rst
View file @
36f9d0b8
...
...
@@ -21,11 +21,15 @@ Features added
* Memoryviews are supported in PEP-484/526 style type declarations.
(Github issue #2529)
* ``@cython.nogil`` is supported as a C-function decorator in Python code.
(Github issue #2557)
* Raising exceptions from nogil code will automatically acquire the GIL, instead
of requiring an explicit ``with gil`` block.
* ``@cython.nogil`` is supported as a C-function decorator in Python code.
(Github issue #2557)
* C++ functions can now be declared as potentially raising both C++ and Python
exceptions, so that Cython can handle both correctly.
(Github issue #2615)
* ``cython.inline()`` supports a direct ``language_level`` keyword argument that
was previously only available via a directive.
...
...
Cython/Compiler/ExprNodes.py
View file @
36f9d0b8
...
...
@@ -187,22 +187,44 @@ def infer_sequence_item_type(env, seq_node, index_node=None, seq_type=None):
return
item_types
.
pop
()
return
None
# Returns a block of code to translate the exception,
# plus a boolean indicating whether to check for Python exceptions.
def
get_exception_handler
(
exception_value
):
if
exception_value
is
None
:
return
"__Pyx_CppExn2PyErr();"
return
"__Pyx_CppExn2PyErr();"
,
False
elif
(
exception_value
.
type
==
PyrexTypes
.
c_char_type
and
exception_value
.
value
==
'*'
):
return
"__Pyx_CppExn2PyErr();"
,
True
elif
exception_value
.
type
.
is_pyobject
:
return
'try { throw; } catch(const std::exception& exn) { PyErr_SetString(%s, exn.what()); } catch(...) { PyErr_SetNone(%s); }'
%
(
return
(
'try { throw; } catch(const std::exception& exn) {'
'PyErr_SetString(%s, exn.what());'
'} catch(...) { PyErr_SetNone(%s); }'
%
(
exception_value
.
entry
.
cname
,
exception_value
.
entry
.
cname
)
exception_value
.
entry
.
cname
),
False
)
else
:
return
(
'%s(); if (!PyErr_Occurred())'
'PyErr_SetString(PyExc_RuntimeError, '
'"Error converting c++ exception.");'
%
(
exception_value
.
entry
.
cname
),
False
)
def
maybe_check_py_error
(
code
,
check_py_exception
,
pos
,
nogil
):
if
check_py_exception
:
if
nogil
:
code
.
putln
(
code
.
error_goto_if
(
"__Pyx_ErrOccurredWithGIL()"
,
pos
))
else
:
return
'%s(); if (!PyErr_Occurred()) PyErr_SetString(PyExc_RuntimeError , "Error converting c++ exception.");'
%
exception_value
.
entry
.
cname
code
.
putln
(
code
.
error_goto_if
(
"PyErr_Occurred()"
,
pos
))
def
translate_cpp_exception
(
code
,
pos
,
inside
,
py_result
,
exception_value
,
nogil
):
raise_py_exception
=
get_exception_handler
(
exception_value
)
raise_py_exception
,
check_py_exception
=
get_exception_handler
(
exception_value
)
code
.
putln
(
"try {"
)
code
.
putln
(
"%s"
%
inside
)
if
py_result
:
code
.
putln
(
code
.
error_goto_if_null
(
py_result
,
pos
))
maybe_check_py_error
(
code
,
check_py_exception
,
pos
,
nogil
)
code
.
putln
(
"} catch(...) {"
)
if
nogil
:
code
.
put_ensure_gil
(
declare_gilstate
=
True
)
...
...
@@ -216,12 +238,14 @@ def translate_cpp_exception(code, pos, inside, py_result, exception_value, nogil
# both have an exception declaration.
def
translate_double_cpp_exception
(
code
,
pos
,
lhs_type
,
lhs_code
,
rhs_code
,
lhs_exc_val
,
assign_exc_val
,
nogil
):
handle_lhs_exc
=
get_exception_handler
(
lhs_exc_val
)
handle_assignment_exc
=
get_exception_handler
(
assign_exc_val
)
handle_lhs_exc
,
lhc_check_py_exc
=
get_exception_handler
(
lhs_exc_val
)
handle_assignment_exc
,
assignment_check_py_exc
=
get_exception_handler
(
assign_exc_val
)
code
.
putln
(
"try {"
)
code
.
putln
(
lhs_type
.
declaration_code
(
"__pyx_local_lvalue = %s;"
%
lhs_code
))
maybe_check_py_error
(
code
,
lhc_check_py_exc
,
pos
,
nogil
)
code
.
putln
(
"try {"
)
code
.
putln
(
"__pyx_local_lvalue = %s;"
%
rhs_code
)
maybe_check_py_error
(
code
,
assignment_check_py_exc
,
pos
,
nogil
)
# Catch any exception from the overloaded assignment.
code
.
putln
(
"} catch(...) {"
)
if
nogil
:
...
...
Cython/Compiler/Nodes.py
View file @
36f9d0b8
...
...
@@ -741,9 +741,11 @@ class CFuncDeclaratorNode(CDeclaratorNode):
and
not
exc_val_type
.
is_pyobject
and
not
(
exc_val_type
.
is_cfunction
and
not
exc_val_type
.
return_type
.
is_pyobject
and
not
exc_val_type
.
args
)):
and
not
exc_val_type
.
args
)
and
not
(
exc_val_type
==
PyrexTypes
.
c_char_type
and
self
.
exception_value
.
value
==
'*'
)):
error
(
self
.
exception_value
.
pos
,
"Exception value must be a Python exception or cdef function with no arguments."
)
"Exception value must be a Python exception or cdef function with no arguments
or *
."
)
exc_val
=
self
.
exception_value
else
:
self
.
exception_value
=
self
.
exception_value
.
coerce_to
(
...
...
Cython/Compiler/Parsing.py
View file @
36f9d0b8
...
...
@@ -2937,6 +2937,9 @@ def p_exception_value_clause(s):
name
=
s
.
systring
s
.
next
()
exc_val
=
p_name
(
s
,
name
)
elif
s
.
sy
==
'*'
:
exc_val
=
ExprNodes
.
CharNode
(
s
.
position
(),
value
=
u'*'
)
s
.
next
()
else
:
if
s
.
sy
==
'?'
:
exc_check
=
1
...
...
docs/src/userguide/wrapping_CPlusPlus.rst
View file @
36f9d0b8
...
...
@@ -111,7 +111,12 @@ of the :func:`cythonize` function.
Declare a var with the wrapped C++ class
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
Now, we use cdef to declare a var of the class with the C++ ``new`` statement:
We'll create a ``.pyx`` file named ``rect.pyx`` to build our wrapper. We're
using a name other than ``Rectangle``, but if you prefer giving the same name
to the wrapper as the C++ class, see the section on
:ref:`resolving naming conflicts <resolve-conflicts>`.
Within, we use cdef to declare a var of the class with the C++ ``new`` statement:
.. literalinclude:: ../../examples/userguide/wrapping_CPlusPlus/cython_usage.pyx
...
...
@@ -164,9 +169,6 @@ instance.
.. literalinclude:: ../../examples/userguide/wrapping_CPlusPlus/rect_ptr.pyx
If you prefer giving the same name to the wrapper as the C++ class, see the
section on :ref:`resolving naming conflicts <resolve-conflicts>`.
Compilation and Importing
=========================
...
...
@@ -440,6 +442,13 @@ called, which allows one to do custom C++ to Python error "translations." If
raise_py_error does not actually raise an exception a RuntimeError will be
raised.
There is also the special form::
cdef int raise_py_or_cpp() except +*
for those functions that may raise either a Python or a C++ exception.
Static member method
--------------------
...
...
tests/run/cpp_exceptions.pyx
View file @
36f9d0b8
...
...
@@ -22,6 +22,7 @@ cdef extern from "cpp_exceptions_helper.h":
cdef
void
raise_underflow
()
except
+
cdef
raise_or_throw
(
bint
py
)
except
+
cdef
int
raise_or_throw_int
(
bint
py
)
except
+*
cdef
cppclass
Foo
:
int
bar_raw
"bar"
(
bint
fire
)
except
+
...
...
@@ -113,6 +114,19 @@ def test_func_that_can_raise_or_throw(bint py):
"""
raise_or_throw
(
py
)
def
test_func_that_can_raise_or_throw_c_return
(
bint
py
):
"""
>>> test_func_that_can_raise_or_throw_c_return(0)
Traceback (most recent call last):
...
RuntimeError: oopsie
>>> test_func_that_can_raise_or_throw_c_return(1)
Traceback (most recent call last):
...
ValueError: oopsie
"""
raise_or_throw_int
(
py
)
def
test_int_raw
(
bint
fire
):
"""
>>> test_int_raw(False)
...
...
tests/run/cpp_exceptions_helper.h
View file @
36f9d0b8
...
...
@@ -70,3 +70,11 @@ PyObject *raise_or_throw(int py) {
PyErr_SetString
(
PyExc_ValueError
,
"oopsie"
);
return
NULL
;
}
int
raise_or_throw_int
(
int
py
)
{
if
(
!
py
)
{
throw
std
::
runtime_error
(
"oopsie"
);
}
PyErr_SetString
(
PyExc_ValueError
,
"oopsie"
);
return
-
1
;
}
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