Commit 92fe0278 authored by Robert Bradshaw's avatar Robert Bradshaw

...

parents 7c69d631 cb6cebbd
...@@ -577,7 +577,7 @@ class BoolNode(PyConstNode): ...@@ -577,7 +577,7 @@ class BoolNode(PyConstNode):
# The constant value True or False # The constant value True or False
def compile_time_value(self, denv): def compile_time_value(self, denv):
return None return self.value
def calculate_result_code(self): def calculate_result_code(self):
if self.value: if self.value:
......
...@@ -710,7 +710,7 @@ class FuncDefNode(StatNode, BlockNode): ...@@ -710,7 +710,7 @@ class FuncDefNode(StatNode, BlockNode):
if acquire_gil: if acquire_gil:
code.putln("PyGILState_STATE _save = PyGILState_Ensure();") code.putln("PyGILState_STATE _save = PyGILState_Ensure();")
# ----- Fetch arguments # ----- Fetch arguments
self.generate_argument_parsing_code(code) self.generate_argument_parsing_code(env, code)
self.generate_argument_increfs(lenv, code) self.generate_argument_increfs(lenv, code)
# ----- Initialise local variables # ----- Initialise local variables
for entry in lenv.var_entries: for entry in lenv.var_entries:
...@@ -952,7 +952,7 @@ class CFuncDefNode(FuncDefNode): ...@@ -952,7 +952,7 @@ class CFuncDefNode(FuncDefNode):
def generate_keyword_list(self, code): def generate_keyword_list(self, code):
pass pass
def generate_argument_parsing_code(self, code): def generate_argument_parsing_code(self, env, code):
i = 0 i = 0
if self.type.optional_arg_count: if self.type.optional_arg_count:
code.putln('if (%s) {' % Naming.optional_args_cname) code.putln('if (%s) {' % Naming.optional_args_cname)
...@@ -1082,10 +1082,13 @@ class DefNode(FuncDefNode): ...@@ -1082,10 +1082,13 @@ 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()
if self.star_arg: if self.signature_has_generic_args():
env.use_utility_code(get_stararg_utility_code) if self.star_arg:
if self.starstar_arg: env.use_utility_code(get_stararg_utility_code)
env.use_utility_code(get_splitkeywords_utility_code) if not self.signature_has_nongeneric_args():
env.use_utility_code(get_keyword_string_check_utility_code)
elif self.starstar_arg:
env.use_utility_code(get_splitkeywords_utility_code)
if self.num_required_kw_args: if self.num_required_kw_args:
env.use_utility_code(get_checkkeywords_utility_code) env.use_utility_code(get_checkkeywords_utility_code)
...@@ -1162,6 +1165,15 @@ class DefNode(FuncDefNode): ...@@ -1162,6 +1165,15 @@ class DefNode(FuncDefNode):
"%s %s has wrong number of arguments " "%s %s has wrong number of arguments "
"(%d declared, %s expected)" % ( "(%d declared, %s expected)" % (
desc, self.name, len(self.args), expected_str)) desc, self.name, len(self.args), expected_str))
def signature_has_nongeneric_args(self):
argcount = len(self.args)
if argcount == 0 or (argcount == 1 and self.args[0].is_self_arg):
return 0
return 1
def signature_has_generic_args(self):
return self.entry.signature.has_generic_args
def declare_pyfunction(self, env): def declare_pyfunction(self, env):
#print "DefNode.declare_pyfunction:", self.name, "in", env ### #print "DefNode.declare_pyfunction:", self.name, "in", env ###
...@@ -1277,7 +1289,8 @@ class DefNode(FuncDefNode): ...@@ -1277,7 +1289,8 @@ class DefNode(FuncDefNode):
code.put_var_declaration(arg.entry) code.put_var_declaration(arg.entry)
def generate_keyword_list(self, code): def generate_keyword_list(self, code):
if self.entry.signature.has_generic_args: if self.signature_has_generic_args() and \
self.signature_has_nongeneric_args():
reqd_kw_flags = [] reqd_kw_flags = []
has_reqd_kwds = False has_reqd_kwds = False
code.put( code.put(
...@@ -1304,16 +1317,19 @@ class DefNode(FuncDefNode): ...@@ -1304,16 +1317,19 @@ class DefNode(FuncDefNode):
flags_name, flags_name,
",".join(reqd_kw_flags))) ",".join(reqd_kw_flags)))
def generate_argument_parsing_code(self, code): def generate_argument_parsing_code(self, env, code):
# Generate PyArg_ParseTuple call for generic # Generate PyArg_ParseTuple call for generic
# arguments, if any. # arguments, if any.
has_kwonly_args = self.num_kwonly_args > 0 has_kwonly_args = self.num_kwonly_args > 0
has_star_or_kw_args = self.star_arg is not None \ has_star_or_kw_args = self.star_arg is not None \
or self.starstar_arg is not None or has_kwonly_args or self.starstar_arg is not None or has_kwonly_args
if not self.entry.signature.has_generic_args: if not self.signature_has_generic_args():
if has_star_or_kw_args: if has_star_or_kw_args:
error(self.pos, "This method cannot have * or keyword arguments") error(self.pos, "This method cannot have * or keyword arguments")
self.generate_argument_conversion_code(code) self.generate_argument_conversion_code(code)
elif not self.signature_has_nongeneric_args():
# func(*args) or func(**kw) or func(*args, **kw)
self.generate_stararg_copy_code(code)
else: else:
arg_addrs = [] arg_addrs = []
arg_formats = [] arg_formats = []
...@@ -1352,27 +1368,47 @@ class DefNode(FuncDefNode): ...@@ -1352,27 +1368,47 @@ class DefNode(FuncDefNode):
error(arg.pos, error(arg.pos,
"Cannot convert Python object argument to type '%s' (when parsing input arguments)" "Cannot convert Python object argument to type '%s' (when parsing input arguments)"
% arg.type) % arg.type)
error_return_code = "return %s;" % self.error_value()
argformat = '"%s"' % string.join(arg_formats, "")
if has_star_or_kw_args: if has_star_or_kw_args:
self.generate_stararg_getting_code(code) self.generate_stararg_getting_code(code)
pt_arglist = [Naming.args_cname, Naming.kwds_cname, argformat,
Naming.kwdlist_cname] + arg_addrs
pt_argstring = string.join(pt_arglist, ", ")
old_error_label = code.new_error_label() old_error_label = code.new_error_label()
our_error_label = code.error_label our_error_label = code.error_label
end_label = code.new_label() end_label = code.new_label()
# Unpack inplace if it's simple
has_self_arg = len(self.args) > 0 and self.args[0].is_self_arg self.generate_argument_tuple_parsing_code(
if self.num_required_args == len(positional_args) + has_self_arg: positional_args, arg_formats, arg_addrs, code)
code.error_label = old_error_label
code.put_goto(end_label)
code.put_label(our_error_label)
if has_star_or_kw_args:
self.put_stararg_decrefs(code)
self.generate_arg_decref(self.star_arg, code)
if self.starstar_arg:
if self.starstar_arg.entry.xdecref_cleanup:
code.put_var_xdecref(self.starstar_arg.entry)
else:
code.put_var_decref(self.starstar_arg.entry)
code.putln("return %s;" % self.error_value())
code.put_label(end_label)
def generate_argument_tuple_parsing_code(self, positional_args,
arg_formats, arg_addrs, code):
# Unpack inplace if it's simple
if not self.num_required_kw_args:
min_positional_args = self.num_required_args - self.num_required_kw_args
max_positional_args = len(positional_args)
if len(self.args) > 0 and self.args[0].is_self_arg:
min_positional_args -= 1
if max_positional_args == min_positional_args:
count_cond = "likely(PyTuple_GET_SIZE(%s) == %s)" % ( count_cond = "likely(PyTuple_GET_SIZE(%s) == %s)" % (
Naming.args_cname, len(positional_args)) Naming.args_cname, max_positional_args)
else: else:
count_cond = "likely(%s <= PyTuple_GET_SIZE(%s)) && likely(PyTuple_GET_SIZE(%s) <= %s)" % ( count_cond = "likely(%s <= PyTuple_GET_SIZE(%s)) && likely(PyTuple_GET_SIZE(%s) <= %s)" % (
self.num_required_args - has_self_arg, min_positional_args,
Naming.args_cname, Naming.args_cname,
Naming.args_cname, Naming.args_cname,
len(positional_args)) max_positional_args)
code.putln( code.putln(
'if (likely(!%s) && %s) {' % (Naming.kwds_cname, count_cond)) 'if (likely(!%s) && %s) {' % (Naming.kwds_cname, count_cond))
i = 0 i = 0
...@@ -1401,28 +1437,20 @@ class DefNode(FuncDefNode): ...@@ -1401,28 +1437,20 @@ class DefNode(FuncDefNode):
code.putln('}') code.putln('}')
code.putln( code.putln(
'}') '}')
code.putln( code.putln('else {')
'else {')
code.putln( argformat = '"%s"' % string.join(arg_formats, "")
'if (unlikely(!PyArg_ParseTupleAndKeywords(%s))) %s' % ( pt_arglist = [Naming.args_cname, Naming.kwds_cname, argformat,
pt_argstring, Naming.kwdlist_cname] + arg_addrs
code.error_goto(self.pos))) pt_argstring = string.join(pt_arglist, ", ")
self.generate_argument_conversion_code(code) code.putln(
code.putln( 'if (unlikely(!PyArg_ParseTupleAndKeywords(%s))) %s' % (
'}') pt_argstring,
code.error_label = old_error_label code.error_goto(self.pos)))
code.put_goto(end_label) self.generate_argument_conversion_code(code)
code.put_label(our_error_label)
if has_star_or_kw_args: if not self.num_required_kw_args:
self.put_stararg_decrefs(code) code.putln('}')
self.generate_arg_decref(self.star_arg, code)
if self.starstar_arg:
if self.starstar_arg.entry.xdecref_cleanup:
code.put_var_xdecref(self.starstar_arg.entry)
else:
code.put_var_decref(self.starstar_arg.entry)
code.putln(error_return_code)
code.put_label(end_label)
def put_stararg_decrefs(self, code): def put_stararg_decrefs(self, code):
if self.star_arg: if self.star_arg:
...@@ -1444,6 +1472,29 @@ class DefNode(FuncDefNode): ...@@ -1444,6 +1472,29 @@ class DefNode(FuncDefNode):
else: else:
return 0 return 0
def generate_stararg_copy_code(self, code):
if not self.star_arg:
self.generate_positional_args_check(code, 0)
self.generate_keyword_args_check(code)
if self.starstar_arg:
code.putln("%s = (%s) ? PyDict_Copy(%s) : PyDict_New();" % (
self.starstar_arg.entry.cname,
Naming.kwds_cname,
Naming.kwds_cname))
code.putln("if (unlikely(!%s)) return %s;" % (
self.starstar_arg.entry.cname, self.error_value()))
self.starstar_arg.entry.xdecref_cleanup = 0
self.starstar_arg = None
if self.star_arg:
code.put_incref(Naming.args_cname, py_object_type)
code.putln("%s = %s;" % (
self.star_arg.entry.cname,
Naming.args_cname))
self.star_arg.entry.xdecref_cleanup = 0
self.star_arg = None
def generate_stararg_getting_code(self, code): def generate_stararg_getting_code(self, code):
num_kwonly = self.num_kwonly_args num_kwonly = self.num_kwonly_args
fixed_args = self.entry.signature.num_fixed_args() fixed_args = self.entry.signature.num_fixed_args()
...@@ -1467,17 +1518,11 @@ class DefNode(FuncDefNode): ...@@ -1467,17 +1518,11 @@ class DefNode(FuncDefNode):
self.error_value())) self.error_value()))
code.putln("}") code.putln("}")
self.star_arg.entry.xdecref_cleanup = 0 self.star_arg.entry.xdecref_cleanup = 0
elif self.entry.signature.has_generic_args: elif self.signature_has_generic_args():
# make sure supernumerous positional arguments do not run # make sure supernumerous positional arguments do not run
# into keyword-only arguments and provide a more helpful # into keyword-only arguments and provide a more helpful
# message than PyArg_ParseTupelAndKeywords() # message than PyArg_ParseTupelAndKeywords()
code.putln("if (unlikely(PyTuple_GET_SIZE(%s) > %d)) {" % ( self.generate_positional_args_check(code, nargs)
Naming.args_cname, nargs))
error_message = "function takes at most %d positional arguments (%d given)"
code.putln("PyErr_Format(PyExc_TypeError, \"%s\", %d, PyTuple_GET_SIZE(%s));" % (
error_message, nargs, Naming.args_cname))
code.putln(error_return)
code.putln("}")
handle_error = 0 handle_error = 0
if self.starstar_arg: if self.starstar_arg:
...@@ -1506,6 +1551,22 @@ class DefNode(FuncDefNode): ...@@ -1506,6 +1551,22 @@ class DefNode(FuncDefNode):
else: else:
code.putln(error_return) code.putln(error_return)
def generate_positional_args_check(self, code, nargs):
code.putln("if (unlikely(PyTuple_GET_SIZE(%s) > %d)) {" % (
Naming.args_cname, nargs))
error_message = "function takes at most %d positional arguments (%d given)"
code.putln("PyErr_Format(PyExc_TypeError, \"%s\", %d, PyTuple_GET_SIZE(%s));" % (
error_message, nargs, Naming.args_cname))
code.putln("return %s;" % self.error_value())
code.putln("}")
def generate_keyword_args_check(self, code):
code.putln("if (unlikely(%s)) {" % Naming.kwds_cname)
code.putln("if (unlikely(!__Pyx_CheckKeywordStrings(%s, \"%s\", %d))) return %s;" % (
Naming.kwds_cname, self.name,
bool(self.starstar_arg), self.error_value()))
code.putln("}")
def generate_argument_conversion_code(self, code): def generate_argument_conversion_code(self, code):
# Generate code to convert arguments from # Generate code to convert arguments from
# signature type to declared type, if needed. # signature type to declared type, if needed.
...@@ -3557,6 +3618,40 @@ static INLINE int __Pyx_SplitStarArg( ...@@ -3557,6 +3618,40 @@ static INLINE int __Pyx_SplitStarArg(
} }
"""] """]
#------------------------------------------------------------------------------------
#
# __Pyx_CheckKeywordStrings raises an error if non-string keywords
# were passed to a function, or if any keywords were passed to a
# function that does not accept them.
get_keyword_string_check_utility_code = [
"""
static int __Pyx_CheckKeywordStrings(PyObject *kwdict, const char* function_name, int kw_allowed); /*proto*/
""","""
static int __Pyx_CheckKeywordStrings(
PyObject *kwdict,
const char* function_name,
int kw_allowed)
{
PyObject* key = 0;
Py_ssize_t pos = 0;
while (PyDict_Next(kwdict, &pos, &key, 0)) {
if (unlikely(!PyString_Check(key))) {
PyErr_Format(PyExc_TypeError,
"%s() keywords must be strings", function_name);
return 0;
}
}
if (unlikely(!kw_allowed) && unlikely(key)) {
PyErr_Format(PyExc_TypeError,
"'%s' is an invalid keyword argument for this function",
PyString_AsString(key));
return 0;
}
return 1;
}
"""]
#------------------------------------------------------------------------------------ #------------------------------------------------------------------------------------
# #
# __Pyx_SplitKeywords splits the kwds dict into two parts one part # __Pyx_SplitKeywords splits the kwds dict into two parts one part
......
...@@ -1302,7 +1302,7 @@ def p_IF_statement(s, level, cdef_flag, visibility, api): ...@@ -1302,7 +1302,7 @@ def p_IF_statement(s, level, cdef_flag, visibility, api):
if current_eval: if current_eval:
result = body result = body
if not result: if not result:
result = PassStatNode(pos) result = Nodes.PassStatNode(pos)
s.compile_time_eval = saved_eval s.compile_time_eval = saved_eval
return result return result
......
...@@ -1129,8 +1129,9 @@ static INLINE PY_LONG_LONG __pyx_PyInt_AsLongLong(PyObject* x) { ...@@ -1129,8 +1129,9 @@ static INLINE PY_LONG_LONG __pyx_PyInt_AsLongLong(PyObject* x) {
return PyLong_AsLongLong(x); return PyLong_AsLongLong(x);
} }
else { else {
PY_LONG_LONG val;
PyObject* tmp = PyNumber_Int(x); if (!tmp) return (PY_LONG_LONG)-1; PyObject* tmp = PyNumber_Int(x); if (!tmp) return (PY_LONG_LONG)-1;
PY_LONG_LONG val = __pyx_PyInt_AsLongLong(tmp); val = __pyx_PyInt_AsLongLong(tmp);
Py_DECREF(tmp); Py_DECREF(tmp);
return val; return val;
} }
...@@ -1149,8 +1150,9 @@ static INLINE unsigned PY_LONG_LONG __pyx_PyInt_AsUnsignedLongLong(PyObject* x) ...@@ -1149,8 +1150,9 @@ static INLINE unsigned PY_LONG_LONG __pyx_PyInt_AsUnsignedLongLong(PyObject* x)
return PyLong_AsUnsignedLongLong(x); return PyLong_AsUnsignedLongLong(x);
} }
else { else {
PY_LONG_LONG val;
PyObject* tmp = PyNumber_Int(x); if (!tmp) return (PY_LONG_LONG)-1; PyObject* tmp = PyNumber_Int(x); if (!tmp) return (PY_LONG_LONG)-1;
PY_LONG_LONG val = __pyx_PyInt_AsUnsignedLongLong(tmp); val = __pyx_PyInt_AsUnsignedLongLong(tmp);
Py_DECREF(tmp); Py_DECREF(tmp);
return val; return val;
} }
......
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