Commit 67973cf6 authored by Marius Wachtler's avatar Marius Wachtler

Merge pull request #956 from undingen/keyword_args

Add rewrite support for calls which use keyword args
parents e66962b4 72a28355
...@@ -633,7 +633,7 @@ void dictMerge(BoxedDict* self, Box* other) { ...@@ -633,7 +633,7 @@ void dictMerge(BoxedDict* self, Box* other) {
assert(keys); assert(keys);
for (Box* k : keys->pyElements()) { for (Box* k : keys->pyElements()) {
self->d[k] = getitem(other, k); self->d[k] = getitemInternal<CXX>(other, k);
} }
} }
......
...@@ -3229,14 +3229,10 @@ static llvm::StringRef getFunctionName(CLFunction* f) { ...@@ -3229,14 +3229,10 @@ static llvm::StringRef getFunctionName(CLFunction* f) {
return "<unknown function>"; return "<unknown function>";
} }
enum class KeywordDest {
POSITIONAL,
KWARGS,
};
template <typename FuncNameCB> template <typename FuncNameCB>
static KeywordDest placeKeyword(const ParamNames* param_names, llvm::SmallVector<bool, 8>& params_filled, static int placeKeyword(const ParamNames* param_names, llvm::SmallVector<bool, 8>& params_filled, BoxedString* kw_name,
BoxedString* kw_name, Box* kw_val, Box*& oarg1, Box*& oarg2, Box*& oarg3, Box** oargs, Box* kw_val, Box*& oarg1, Box*& oarg2, Box*& oarg3, Box** oargs, BoxedDict* okwargs,
BoxedDict* okwargs, FuncNameCB func_name_cb) { FuncNameCB func_name_cb) {
assert(kw_val); assert(kw_val);
assert(gc::isValidGCObject(kw_val)); assert(gc::isValidGCObject(kw_val));
assert(kw_name); assert(kw_name);
...@@ -3248,11 +3244,9 @@ static KeywordDest placeKeyword(const ParamNames* param_names, llvm::SmallVector ...@@ -3248,11 +3244,9 @@ static KeywordDest placeKeyword(const ParamNames* param_names, llvm::SmallVector
raiseExcHelper(TypeError, "%.200s() got multiple values for keyword argument '%s'", func_name_cb(), raiseExcHelper(TypeError, "%.200s() got multiple values for keyword argument '%s'", func_name_cb(),
kw_name->c_str()); kw_name->c_str());
} }
getArg(j, oarg1, oarg2, oarg3, oargs) = kw_val; getArg(j, oarg1, oarg2, oarg3, oargs) = kw_val;
params_filled[j] = true; params_filled[j] = true;
return j;
return KeywordDest::POSITIONAL;
} }
} }
...@@ -3263,7 +3257,7 @@ static KeywordDest placeKeyword(const ParamNames* param_names, llvm::SmallVector ...@@ -3263,7 +3257,7 @@ static KeywordDest placeKeyword(const ParamNames* param_names, llvm::SmallVector
kw_name->c_str()); kw_name->c_str());
} }
v = kw_val; v = kw_val;
return KeywordDest::KWARGS; return -1;
} else { } else {
raiseExcHelper(TypeError, "%.200s() got an unexpected keyword argument '%s'", func_name_cb(), kw_name->c_str()); raiseExcHelper(TypeError, "%.200s() got an unexpected keyword argument '%s'", func_name_cb(), kw_name->c_str());
} }
...@@ -3274,7 +3268,7 @@ static Box* _callFuncHelper(BoxedFunctionBase* func, ArgPassSpec argspec, Box* a ...@@ -3274,7 +3268,7 @@ static Box* _callFuncHelper(BoxedFunctionBase* func, ArgPassSpec argspec, Box* a
void** extra_args) { void** extra_args) {
Box** args = (Box**)extra_args[0]; Box** args = (Box**)extra_args[0];
auto keyword_names = (const std::vector<BoxedString*>*)extra_args[1]; auto keyword_names = (const std::vector<BoxedString*>*)extra_args[1];
return callFunc<S>(func, NULL, argspec, arg1, arg2, arg3, args, keyword_names); return callFunc<S, NOT_REWRITABLE>(func, NULL, argspec, arg1, arg2, arg3, args, keyword_names);
} }
typedef std::function<Box*(int, int, RewriterVar*&)> GetDefaultFunc; typedef std::function<Box*(int, int, RewriterVar*&)> GetDefaultFunc;
...@@ -3411,7 +3405,7 @@ void rearrangeArgumentsInternal(ParamReceiveSpec paramspec, const ParamNames* pa ...@@ -3411,7 +3405,7 @@ void rearrangeArgumentsInternal(ParamReceiveSpec paramspec, const ParamNames* pa
static StatCounter slowpath_rearrangeargs_slowpath("slowpath_rearrangeargs_slowpath"); static StatCounter slowpath_rearrangeargs_slowpath("slowpath_rearrangeargs_slowpath");
slowpath_rearrangeargs_slowpath.log(); slowpath_rearrangeargs_slowpath.log();
if (argspec.has_starargs || argspec.has_kwargs || argspec.num_keywords) { if (argspec.has_starargs || argspec.has_kwargs || (paramspec.takes_kwargs && argspec.num_keywords)) {
rewrite_args = NULL; rewrite_args = NULL;
} }
...@@ -3589,9 +3583,24 @@ void rearrangeArgumentsInternal(ParamReceiveSpec paramspec, const ParamNames* pa ...@@ -3589,9 +3583,24 @@ void rearrangeArgumentsInternal(ParamReceiveSpec paramspec, const ParamNames* pa
if (argspec.num_keywords) { if (argspec.num_keywords) {
assert(argspec.num_keywords == keyword_names->size()); assert(argspec.num_keywords == keyword_names->size());
RewriterVar::SmallVector r_vars;
if (rewrite_args) {
for (int i = argspec.num_args; i < argspec.num_args + argspec.num_keywords; i++) {
if (i == 0)
r_vars.push_back(rewrite_args->arg1);
if (i == 1)
r_vars.push_back(rewrite_args->arg2);
if (i == 2)
r_vars.push_back(rewrite_args->arg3);
if (i >= 3)
r_vars.push_back(rewrite_args->args->getAttr((i - 3) * sizeof(Box*)));
}
}
BoxedDict* okwargs = get_okwargs(); BoxedDict* okwargs = get_okwargs();
for (int i = 0; i < argspec.num_keywords; i++) { for (int i = 0; i < argspec.num_keywords; i++) {
assert(!rewrite_args && "would need to be handled here"); if (rewrite_args)
assert(!okwargs && "would need to be handled here");
int arg_idx = i + argspec.num_args; int arg_idx = i + argspec.num_args;
Box* kw_val = getArg(arg_idx, arg1, arg2, arg3, args); Box* kw_val = getArg(arg_idx, arg1, arg2, arg3, args);
...@@ -3602,9 +3611,19 @@ void rearrangeArgumentsInternal(ParamReceiveSpec paramspec, const ParamNames* pa ...@@ -3602,9 +3611,19 @@ void rearrangeArgumentsInternal(ParamReceiveSpec paramspec, const ParamNames* pa
continue; continue;
} }
auto dest = placeKeyword(param_names, params_filled, (*keyword_names)[i], kw_val, oarg1, oarg2, oarg3, int dest = placeKeyword(param_names, params_filled, (*keyword_names)[i], kw_val, oarg1, oarg2, oarg3, oargs,
oargs, okwargs, func_name_cb); okwargs, func_name_cb);
assert(!rewrite_args); if (rewrite_args) {
assert(dest != -1);
if (dest == 0)
rewrite_args->arg1 = r_vars[i];
else if (dest == 1)
rewrite_args->arg2 = r_vars[i];
else if (dest == 2)
rewrite_args->arg3 = r_vars[i];
else
rewrite_args->args->setAttr((dest - 3) * sizeof(Box*), r_vars[i]);
}
} }
} }
......
# expected: statfail # statcheck: noninit_count('slowpath_runtimecall') <= 500
# - rewriter bails on keywords for now # statcheck: noninit_count('slowpath_callfunc') <= 500
# run_args: -n
# statcheck: stats['slowpath_runtimecall'] <= 20 def f(a=-1, b=-2):
# statcheck: stats.get("slowpath_callclfunc", 0) <= 20 return a + b
# statcheck: stats['rewriter_nopatch'] <= 20
def f(a, b):
print a, b
for i in xrange(10000): s = 0
f(a=1, b=2) for i in xrange(20000):
f(b=1, a=2) s += f(a=1, b=2)
s += f(b=3, a=4)
s += f(b=5)
print s
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