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
Labels
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Commits
Open sidebar
nexedi
cython
Commits
dfaa72eb
Commit
dfaa72eb
authored
Aug 22, 2008
by
Stefan Behnel
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
fixed error handling for required keyword arguments
cosolidate exception messages cleanup of utility code
parent
b6b0f9a0
Changes
1
Show whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
71 additions
and
84 deletions
+71
-84
Cython/Compiler/Nodes.py
Cython/Compiler/Nodes.py
+71
-84
No files found.
Cython/Compiler/Nodes.py
View file @
dfaa72eb
...
@@ -1336,18 +1336,10 @@ class DefNode(FuncDefNode):
...
@@ -1336,18 +1336,10 @@ class DefNode(FuncDefNode):
self
.
declare_pyfunction
(
env
)
self
.
declare_pyfunction
(
env
)
self
.
analyse_signature
(
env
)
self
.
analyse_signature
(
env
)
self
.
return_type
=
self
.
entry
.
signature
.
return_type
()
self
.
return_type
=
self
.
entry
.
signature
.
return_type
()
env
.
use_utility_code
(
raise_argtuple_too_long_utility_code
)
env
.
use_utility_code
(
raise_argtuple_invalid_utility_code
)
env
.
use_utility_code
(
keyword_check_utility_code
)
env
.
use_utility_code
(
raise_keyword_required_utility_code
)
env
.
use_utility_code
(
check_double_keywords_utility_code
)
if
self
.
signature_has_generic_args
():
if
self
.
star_arg
:
env
.
use_utility_code
(
get_stararg_utility_code
)
if
not
self
.
signature_has_nongeneric_args
():
env
.
use_utility_code
(
keyword_string_check_utility_code
)
elif
self
.
starstar_arg
:
env
.
use_utility_code
(
split_keywords_utility_code
)
if
self
.
num_required_kw_args
:
if
self
.
num_required_kw_args
:
env
.
use_utility_code
(
check_keywords_utility_code
)
env
.
use_utility_code
(
check_
required_
keywords_utility_code
)
def
analyse_signature
(
self
,
env
):
def
analyse_signature
(
self
,
env
):
any_type_tests_needed
=
0
any_type_tests_needed
=
0
...
@@ -1560,17 +1552,14 @@ class DefNode(FuncDefNode):
...
@@ -1560,17 +1552,14 @@ class DefNode(FuncDefNode):
Naming
.
kwdlist_cname
)
Naming
.
kwdlist_cname
)
for
arg
in
self
.
args
:
for
arg
in
self
.
args
:
if
arg
.
is_generic
:
if
arg
.
is_generic
:
code
.
put
(
code
.
put
(
'"%s",'
%
arg
.
name
.
utf8encode
())
'"%s",'
%
arg
.
name
.
utf8encode
())
if
arg
.
kw_only
and
not
arg
.
default
:
if
arg
.
kw_only
and
not
arg
.
default
:
has_reqd_kwds
=
1
has_reqd_kwds
=
1
flag
=
"1"
flag
=
"1"
else
:
else
:
flag
=
"0"
flag
=
"0"
reqd_kw_flags
.
append
(
flag
)
reqd_kw_flags
.
append
(
flag
)
code
.
putln
(
code
.
putln
(
"0};"
)
"0};"
)
if
has_reqd_kwds
:
if
has_reqd_kwds
:
flags_name
=
Naming
.
reqd_kwds_cname
flags_name
=
Naming
.
reqd_kwds_cname
self
.
reqd_kw_flags_cname
=
flags_name
self
.
reqd_kw_flags_cname
=
flags_name
...
@@ -1696,6 +1685,8 @@ class DefNode(FuncDefNode):
...
@@ -1696,6 +1685,8 @@ class DefNode(FuncDefNode):
if
not
self
.
star_arg
:
if
not
self
.
star_arg
:
self
.
generate_positional_args_check
(
0
,
code
)
self
.
generate_positional_args_check
(
0
,
code
)
code
.
globalstate
.
use_utility_code
(
keyword_string_check_utility_code
)
code
.
putln
(
code
.
putln
(
"if (unlikely(%s) && unlikely(!__Pyx_CheckKeywordStrings(%s,
\
"
%s
\
"
, %d))) return %s;"
%
(
"if (unlikely(%s) && unlikely(!__Pyx_CheckKeywordStrings(%s,
\
"
%s
\
"
, %d))) return %s;"
%
(
Naming
.
kwds_cname
,
Naming
.
kwds_cname
,
self
.
name
,
Naming
.
kwds_cname
,
Naming
.
kwds_cname
,
self
.
name
,
...
@@ -1721,6 +1712,7 @@ class DefNode(FuncDefNode):
...
@@ -1721,6 +1712,7 @@ class DefNode(FuncDefNode):
def
generate_stararg_getting_code
(
self
,
max_positional_args
,
code
):
def
generate_stararg_getting_code
(
self
,
max_positional_args
,
code
):
if
self
.
star_arg
:
if
self
.
star_arg
:
code
.
globalstate
.
use_utility_code
(
split_stararg_utility_code
)
star_arg_cname
=
self
.
star_arg
.
entry
.
cname
star_arg_cname
=
self
.
star_arg
.
entry
.
cname
code
.
putln
(
"if (likely(PyTuple_GET_SIZE(%s) <= %d)) {"
%
(
code
.
putln
(
"if (likely(PyTuple_GET_SIZE(%s) <= %d)) {"
%
(
Naming
.
args_cname
,
max_positional_args
))
Naming
.
args_cname
,
max_positional_args
))
...
@@ -1738,6 +1730,7 @@ class DefNode(FuncDefNode):
...
@@ -1738,6 +1730,7 @@ class DefNode(FuncDefNode):
self
.
star_arg
.
entry
.
xdecref_cleanup
=
0
self
.
star_arg
.
entry
.
xdecref_cleanup
=
0
if
self
.
starstar_arg
:
if
self
.
starstar_arg
:
code
.
globalstate
.
use_utility_code
(
split_keywords_utility_code
)
handle_error
=
1
handle_error
=
1
code
.
put
(
code
.
put
(
'if (unlikely(__Pyx_SplitKeywords(&%s, %s, &%s, %s, PyTuple_GET_SIZE(%s), "%s") < 0)) '
%
(
'if (unlikely(__Pyx_SplitKeywords(&%s, %s, &%s, %s, PyTuple_GET_SIZE(%s), "%s") < 0)) '
%
(
...
@@ -1772,6 +1765,9 @@ class DefNode(FuncDefNode):
...
@@ -1772,6 +1765,9 @@ class DefNode(FuncDefNode):
if
self
.
star_arg
or
self
.
starstar_arg
:
if
self
.
star_arg
or
self
.
starstar_arg
:
self
.
generate_stararg_getting_code
(
max_positional_args
,
code
)
self
.
generate_stararg_getting_code
(
max_positional_args
,
code
)
code
.
globalstate
.
use_utility_code
(
raise_double_keywords_utility_code
)
code
.
globalstate
.
use_utility_code
(
keyword_check_utility_code
)
# --- optimised code when we receive keyword arguments
# --- optimised code when we receive keyword arguments
code
.
putln
(
"if (unlikely(%s) && (PyDict_Size(%s) > 0)) {"
%
(
code
.
putln
(
"if (unlikely(%s) && (PyDict_Size(%s) > 0)) {"
%
(
Naming
.
kwds_cname
,
Naming
.
kwds_cname
))
Naming
.
kwds_cname
,
Naming
.
kwds_cname
))
...
@@ -1787,7 +1783,7 @@ class DefNode(FuncDefNode):
...
@@ -1787,7 +1783,7 @@ class DefNode(FuncDefNode):
Naming
.
args_cname
)
Naming
.
args_cname
)
code
.
putln
(
"if (unlikely(PyDict_GetItemString(%s, %s[arg]))) {"
%
(
code
.
putln
(
"if (unlikely(PyDict_GetItemString(%s, %s[arg]))) {"
%
(
Naming
.
kwds_cname
,
Naming
.
kwdlist_cname
))
Naming
.
kwds_cname
,
Naming
.
kwdlist_cname
))
code
.
put
(
'__Pyx_RaiseDoubleKeywordsError("%s", %s[arg]); '
%
(
code
.
put
(
'__Pyx_RaiseDoubleKeywordsError("%s", %s[arg]); '
%
(
self
.
name
.
utf8encode
(),
Naming
.
kwdlist_cname
))
self
.
name
.
utf8encode
(),
Naming
.
kwdlist_cname
))
code
.
putln
(
code
.
error_goto
(
self
.
pos
))
code
.
putln
(
code
.
error_goto
(
self
.
pos
))
code
.
putln
(
'}'
)
code
.
putln
(
'}'
)
...
@@ -1799,6 +1795,12 @@ class DefNode(FuncDefNode):
...
@@ -1799,6 +1795,12 @@ class DefNode(FuncDefNode):
code
.
putln
(
'values[arg] = PyDict_GetItemString(%s, %s[arg]);'
%
(
code
.
putln
(
'values[arg] = PyDict_GetItemString(%s, %s[arg]);'
%
(
Naming
.
kwds_cname
,
Naming
.
kwdlist_cname
))
Naming
.
kwds_cname
,
Naming
.
kwdlist_cname
))
code
.
putln
(
'if (values[arg]) kw_args--;'
);
code
.
putln
(
'if (values[arg]) kw_args--;'
);
if
self
.
num_required_kw_args
:
code
.
putln
(
'else if (%s[arg]) {'
%
Naming
.
reqd_kwds_cname
)
code
.
put
(
'__Pyx_RaiseKeywordRequired("%s", %s[arg]); '
%
(
self
.
name
.
utf8encode
(),
Naming
.
kwdlist_cname
))
code
.
putln
(
code
.
error_goto
(
self
.
pos
))
code
.
putln
(
'}'
)
code
.
putln
(
'}'
)
code
.
putln
(
'}'
)
# raise an error if not all keywords were read
# raise an error if not all keywords were read
...
@@ -1810,28 +1812,12 @@ class DefNode(FuncDefNode):
...
@@ -1810,28 +1812,12 @@ class DefNode(FuncDefNode):
# convert arg values to their final type and assign them
# convert arg values to their final type and assign them
default_seen
=
False
default_seen
=
False
i
=
0
for
i
,
arg
in
enumerate
(
tuple
(
positional_args
)
+
tuple
(
kw_only_args
)):
for
arg
in
self
.
args
:
if
arg
.
is_self_arg
:
continue
if
arg
.
default
:
if
arg
.
default
:
default_seen
=
True
if
default_seen
:
code
.
putln
(
"if (values[%d]) {"
%
i
)
code
.
putln
(
"if (values[%d]) {"
%
i
)
self
.
generate_arg_assignment
(
arg
,
"values[%d]"
%
i
,
code
)
self
.
generate_arg_assignment
(
arg
,
"values[%d]"
%
i
,
code
)
if
default_seen
:
if
arg
.
default
:
if
not
arg
.
default
:
code
.
putln
(
'} else {'
)
if
arg
.
kw_only
:
code
.
put
(
'PyErr_Format(PyExc_TypeError, "%s() needs keyword-only argument %s"); '
%
(
self
.
name
.
utf8encode
(),
arg
.
name
.
utf8encode
()))
else
:
code
.
put
(
'__Pyx_RaiseArgtupleInvalid("%s", %d, %d, %d); '
%
(
self
.
name
.
utf8encode
(),
min_positional_args
,
max_positional_args
,
i
))
code
.
putln
(
code
.
error_goto
(
self
.
pos
))
code
.
putln
(
'}'
)
code
.
putln
(
'}'
)
i
+=
1
# --- optimised code when we do not receive any keyword arguments
# --- optimised code when we do not receive any keyword arguments
if
self
.
num_required_kw_args
:
if
self
.
num_required_kw_args
:
...
@@ -1839,8 +1825,9 @@ class DefNode(FuncDefNode):
...
@@ -1839,8 +1825,9 @@ class DefNode(FuncDefNode):
for
arg
in
self
.
args
:
for
arg
in
self
.
args
:
if
arg
.
kw_only
and
not
arg
.
default
:
if
arg
.
kw_only
and
not
arg
.
default
:
required_arg
=
arg
required_arg
=
arg
break
code
.
putln
(
'} else {'
)
code
.
putln
(
'} else {'
)
code
.
put
(
'PyErr_Format(PyExc_TypeError, "%s() needs keyword-only argument
%s");'
%
(
code
.
put
ln
(
'__Pyx_RaiseKeywordRequired("%s", "
%s");'
%
(
self
.
name
.
utf8encode
(),
required_arg
.
name
.
utf8encode
()))
self
.
name
.
utf8encode
(),
required_arg
.
name
.
utf8encode
()))
code
.
putln
(
code
.
error_goto
(
self
.
pos
))
code
.
putln
(
code
.
error_goto
(
self
.
pos
))
code
.
putln
(
'}'
)
code
.
putln
(
'}'
)
...
@@ -4282,7 +4269,7 @@ static int __Pyx_ArgTypeTest(PyObject *obj, PyTypeObject *type, int none_allowed
...
@@ -4282,7 +4269,7 @@ static int __Pyx_ArgTypeTest(PyObject *obj, PyTypeObject *type, int none_allowed
# new reference in *args2. Does not touch any of its arguments on
# new reference in *args2. Does not touch any of its arguments on
# failure.
# failure.
ge
t_stararg_utility_code
=
[
spli
t_stararg_utility_code
=
[
"""
"""
static INLINE int __Pyx_SplitStarArg(PyObject **args, Py_ssize_t nargs, PyObject **args2); /*proto*/
static INLINE int __Pyx_SplitStarArg(PyObject **args, Py_ssize_t nargs, PyObject **args2); /*proto*/
"""
,
"""
"""
,
"""
...
@@ -4313,7 +4300,7 @@ static INLINE int __Pyx_SplitStarArg(
...
@@ -4313,7 +4300,7 @@ static INLINE int __Pyx_SplitStarArg(
# many or too few positional arguments were found. This handles
# many or too few positional arguments were found. This handles
# Py_ssize_t formatting correctly.
# Py_ssize_t formatting correctly.
raise_argtuple_
too_long
_utility_code
=
[
raise_argtuple_
invalid
_utility_code
=
[
"""
"""
static INLINE void __Pyx_RaiseArgtupleInvalid(char* func_name,
static INLINE void __Pyx_RaiseArgtupleInvalid(char* func_name,
Py_ssize_t num_min, Py_ssize_t num_max, Py_ssize_t num_found); /*proto*/
Py_ssize_t num_min, Py_ssize_t num_max, Py_ssize_t num_found); /*proto*/
...
@@ -4324,27 +4311,59 @@ static INLINE void __Pyx_RaiseArgtupleInvalid(
...
@@ -4324,27 +4311,59 @@ static INLINE void __Pyx_RaiseArgtupleInvalid(
Py_ssize_t num_max,
Py_ssize_t num_max,
Py_ssize_t num_found)
Py_ssize_t num_found)
{
{
Py_ssize_t num_expected;
char* message;
if (num_found < num_min) {
if (num_found < num_min) {
PyErr_Format(PyExc_TypeError,
num_expected = num_min;
message =
#if PY_VERSION_HEX < 0x02050000
#if PY_VERSION_HEX < 0x02050000
"%s() takes at least %d positional arguments (%d given)"
,
"%s() takes at least %d positional arguments (%d given)"
;
#elif PY_MAJOR_VERSION >= 3
#elif PY_MAJOR_VERSION >= 3
"%U() takes at most %zd positional arguments (%zd given)"
,
"%U() takes at most %zd positional arguments (%zd given)"
;
#else
#else
"%s() takes at least %zd positional arguments (%zd given)"
,
"%s() takes at least %zd positional arguments (%zd given)"
;
#endif
#endif
func_name, num_min, num_found);
} else {
} else {
PyErr_Format(PyExc_TypeError,
num_expected = num_max;
message =
#if PY_VERSION_HEX < 0x02050000
#if PY_VERSION_HEX < 0x02050000
"%s() takes at most %d positional arguments (%d given)"
,
"%s() takes at most %d positional arguments (%d given)"
;
#elif PY_MAJOR_VERSION >= 3
#elif PY_MAJOR_VERSION >= 3
"%U() takes at most %zd positional arguments (%zd given)"
,
"%U() takes at most %zd positional arguments (%zd given)"
;
#else
#else
"%s() takes at most %zd positional arguments (%zd given)"
,
"%s() takes at most %zd positional arguments (%zd given)"
;
#endif
#endif
func_name, num_max, num_found);
}
}
PyErr_Format(PyExc_TypeError, message, func_name,
num_expected, num_found);
}
"""
]
raise_keyword_required_utility_code
=
[
"""
static INLINE void __Pyx_RaiseKeywordRequired(char* func_name, char* kw_name); /*proto*/
"""
,
"""
static INLINE void __Pyx_RaiseKeywordRequired(
char* func_name,
char* kw_name)
{
PyErr_Format(PyExc_TypeError,
"%s() needs keyword-only argument %s", func_name, kw_name);
}
"""
]
raise_double_keywords_utility_code
=
[
"""
static INLINE void __Pyx_RaiseDoubleKeywordsError(
const char* func_name, const char* kw_name); /*proto*/
"""
,
"""
static INLINE void __Pyx_RaiseDoubleKeywordsError(
const char* func_name,
const char* kw_name)
{
PyErr_Format(PyExc_TypeError,
"%s() got multiple values for keyword argument '%s'", func_name, kw_name);
}
}
"""
]
"""
]
...
@@ -4539,8 +4558,7 @@ arg_passed_twice:
...
@@ -4539,8 +4558,7 @@ arg_passed_twice:
"%s() got multiple values for keyword argument '%s'", func_name, *p);
"%s() got multiple values for keyword argument '%s'", func_name, *p);
goto bad;
goto bad;
missing_kwarg:
missing_kwarg:
PyErr_Format(PyExc_TypeError,
__Pyx_RaiseKeywordRequired(func_name, *p);
"%s() needs keyword-only argument %s", func_name, *p);
bad:
bad:
Py_XDECREF(s);
Py_XDECREF(s);
Py_XDECREF(kwds1);
Py_XDECREF(kwds1);
...
@@ -4549,7 +4567,7 @@ bad:
...
@@ -4549,7 +4567,7 @@ bad:
}
}
"""
]
"""
]
check_keywords_utility_code
=
[
check_
required_
keywords_utility_code
=
[
"""
"""
static INLINE int __Pyx_CheckRequiredKeywords(PyObject *kwds, char *kwd_list[],
static INLINE int __Pyx_CheckRequiredKeywords(PyObject *kwds, char *kwd_list[],
char rqd_kwds[], Py_ssize_t num_posargs, char* func_name); /*proto*/
char rqd_kwds[], Py_ssize_t num_posargs, char* func_name); /*proto*/
...
@@ -4588,38 +4606,7 @@ arg_passed_twice:
...
@@ -4588,38 +4606,7 @@ arg_passed_twice:
"%s() got multiple values for keyword argument '%s'", func_name, *p);
"%s() got multiple values for keyword argument '%s'", func_name, *p);
return -1;
return -1;
missing_kwarg:
missing_kwarg:
PyErr_Format(PyExc_TypeError,
__Pyx_RaiseKeywordRequired(func_name, *p);
"required keyword argument '%s' is missing", *p);
return -1;
}
"""
]
check_double_keywords_utility_code
=
[
"""
static INLINE int __Pyx_CheckDoubleKeywords(PyObject *kwds, char *kwd_list[],
Py_ssize_t num_posargs, const char* func_name); /*proto*/
static int __Pyx_RaiseDoubleKeywordsError(const char* func_name,
const char* kw_name); /*proto*/
"""
,
"""
static INLINE int __Pyx_CheckDoubleKeywords(
PyObject *kwds,
char *kwd_list[],
Py_ssize_t num_posargs,
const char* func_name)
{
int i;
char** p = kwd_list;
for (i=0; i < num_posargs && *p; i++, p++) {
if (PyDict_GetItemString(kwds, *p))
return __Pyx_RaiseDoubleKeywordsError(func_name, *p);
}
return 0;
}
static int __Pyx_RaiseDoubleKeywordsError(const char* func_name, const char* kw_name) {
PyErr_Format(PyExc_TypeError,
"%s() got multiple values for keyword argument '%s'", func_name, kw_name);
return -1;
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