Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Tuples made me sad (so I fixed them) #4042

Merged
merged 28 commits into from
Dec 7, 2013
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
28 commits
Select commit Hold shift + click to select a range
9f2002c
Pass tuples unboxed if possible
Keno Aug 11, 2013
793562b
All tests pass
Keno Aug 12, 2013
9bfd041
Specialize more functions
Keno Aug 12, 2013
4917312
bugfixes
Keno Aug 13, 2013
2ffa794
Remove unnessecary pto argument from emit_unbox
Keno Aug 13, 2013
c802070
Adjust llvmcall for new tuple infrastructure
Keno Aug 13, 2013
91f8ae7
Some llvmcall bugfixes
Keno Aug 13, 2013
c343b5a
Address compiler warnings
Keno Aug 13, 2013
49217ff
Also remove PM initialization
Keno Aug 13, 2013
9cc2819
Handle nothingness correctly
Keno Aug 13, 2013
aec2612
Fix a performance regression
Keno Aug 14, 2013
6b567e1
tpl -> &tpl and don't use return value from unreachable
Keno Aug 15, 2013
1933ea7
Allow unboxed storing tupl purebits tuples
Keno Aug 15, 2013
c58e1ff
remove invalid check from store_unboxed_p
Keno Aug 16, 2013
17690a0
is_stable_expr for tuples
Keno Aug 16, 2013
10330fa
Fix tuples branch for some changes since it was introduced
Keno Oct 2, 2013
a93b442
Tuples branch bug-fixes
Keno Oct 5, 2013
0e2c3e6
Fix a missing JL_GC_POP
Keno Oct 25, 2013
f8b79c4
Fix ifelse on the tuple branch
Keno Oct 26, 2013
8217d70
Merge some bugfixes from MCJIT branch
Keno Nov 16, 2013
71a5def
Fix build failure after merge
Keno Nov 17, 2013
665ffe5
Cleanup
Keno Nov 19, 2013
e02db98
Separate out llvmcall
Keno Nov 19, 2013
9deb3a1
Simplify emit_tupleset, but disallowing non-constant ones
Keno Nov 20, 2013
ee26d21
Add a missing GC root
Keno Nov 21, 2013
f9999b1
Fix an improperly adjusted gc frame
Keno Nov 23, 2013
a7d99e6
Fix vararg root counting
Keno Nov 27, 2013
d5ad8e6
Fix a rebase conflict
Keno Nov 27, 2013
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
Allow unboxed storing tupl purebits tuples
  • Loading branch information
Keno committed Nov 29, 2013
commit 1933ea7ecb07817c4c7b929db0b74f58792fcc71
2 changes: 1 addition & 1 deletion base/inference.jl
Original file line number Diff line number Diff line change
Expand Up @@ -2070,7 +2070,7 @@ function inlining_pass(e::Expr, sv, ast)
end

function add_variable(ast, name, typ)
vinf = {name,typ,2}
vinf = {name,typ,18}
locllist = ast.args[2][1]::Array{Any,1}
vinflist = ast.args[2][2]::Array{Any,1}
push!(locllist, name)
Expand Down
63 changes: 38 additions & 25 deletions src/codegen.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -2414,13 +2414,28 @@ static Value *emit_expr(jl_value_t *expr, jl_codectx_t *ctx, bool isboxed,

// --- allocating local variables ---

static bool jltupleisbits(jl_value_t *jt, bool allow_unsized = true)
{
if (!jl_is_tuple(jt))
return jl_isbits(jt) && (allow_unsized ||
((jl_is_bitstype(jt) && jl_datatype_size(jt) > 0) ||
(jl_is_datatype(jt) && jl_tuple_len(((jl_datatype_t*)jt)->names)>0)));
size_t ntypes = jl_tuple_len(jt);
if (ntypes == 0)
return allow_unsized;
for (size_t i = 0; i < ntypes; ++i)
if (!jltupleisbits(jl_tupleref(jt,i),allow_unsized))
return false;
return true;
}

static bool store_unboxed_p(jl_sym_t *s, jl_codectx_t *ctx)
{
jl_varinfo_t &vi = ctx->vars[s];
jl_value_t *jt = vi.declType;
// only store a variable unboxed if type inference has run, which
// checks that the variable is not referenced undefined.
return (ctx->linfo->inferred && jl_isbits(jt) &&
return (ctx->linfo->inferred && jltupleisbits(jt,false) &&
((jl_datatype_t*)jt)->size > 0 &&
// don't unbox intrinsics, since inference depends on their having
// stable addresses for table lookup.
Expand All @@ -2432,8 +2447,8 @@ static Value *alloc_local(jl_sym_t *s, jl_codectx_t *ctx)
jl_varinfo_t &vi = ctx->vars[s];
jl_value_t *jt = vi.declType;
Value *lv = NULL;
Type *vtype=NULL;
if (store_unboxed_p(s, ctx))
Type *vtype = NULL;
if(store_unboxed_p(s,ctx) && s != ctx->vaName)
vtype = julia_type_to_llvm(jt);
if (vtype != T_void)
{
Expand Down Expand Up @@ -2580,17 +2595,6 @@ static void finalize_gc_frame(jl_codectx_t *ctx)
}
}

static bool jltupleisbits(jl_value_t *jt)
{
if (!jl_is_tuple(jt))
return jl_isbits(jt);
size_t ntypes = jl_tuple_len(jt);
for (size_t i = 0; i < ntypes; ++i)
if (!jltupleisbits(jl_tupleref(jt,i)))
return false;
return true;
}

// generate a julia-callable function that calls f (AKA lam)
static Function *gen_jlcall_wrapper(jl_lambda_info_t *lam, Function *f)
{
Expand Down Expand Up @@ -3111,6 +3115,8 @@ static Function *emit_function(jl_lambda_info_t *lam, bool cstyle)
}
else if (dyn_cast<GetElementPtrInst>(lv) != NULL)
builder.CreateStore(boxed(theArg,&ctx), lv);
else if (dyn_cast<AllocaInst>(lv)->getAllocatedType() == jl_pvalue_llvmt)
builder.CreateStore(theArg,lv);
else
builder.CreateStore(emit_unbox(dyn_cast<AllocaInst>(lv)->getAllocatedType(),
theArg,
Expand All @@ -3132,19 +3138,26 @@ static Function *emit_function(jl_lambda_info_t *lam, bool cstyle)
if (!vi.escapes && !vi.isAssigned) {
ctx.vaStack = true;
}
else {
else if(!vi.isGhost) {
// restarg = jl_f_tuple(NULL, &args[nreq], nargs-nreq)
Value *restTuple =
builder.CreateCall3(jltuple_func, V_null,
builder.CreateGEP(argArray,
ConstantInt::get(T_size,nreq)),
builder.CreateSub(argCount,
ConstantInt::get(T_int32,nreq)));
Value *lv = vi.memvalue;
if (isBoxed(argname, &ctx))
builder.CreateStore(builder.CreateCall(jlbox_func, restTuple), lv);
else
builder.CreateStore(restTuple, lv);
if (dyn_cast<GetElementPtrInst>(lv) != NULL || dyn_cast<AllocaInst>(lv)->getAllocatedType() == jl_pvalue_llvmt)
{
Value *restTuple =
builder.CreateCall3(jltuple_func, V_null,
builder.CreateGEP(argArray,
ConstantInt::get(T_size,nreq)),
builder.CreateSub(argCount,
ConstantInt::get(T_int32,nreq)));
if (isBoxed(argname, &ctx))
builder.CreateStore(builder.CreateCall(jlbox_func, restTuple), lv);
else
builder.CreateStore(restTuple, lv);
} else {
// TODO: Perhaps allow this in the future, but for now sice varargs are always unspecialized
// we don't
assert(false);
}
}
}

Expand Down
25 changes: 25 additions & 0 deletions src/intrinsics.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -177,9 +177,34 @@ static Type *jl_llvmtuple_eltype(Type *tuple, size_t i)
assert(false);
return ety;
}

static size_t jl_llvmtuple_nargs(Type *tuple)
{
size_t n = 0;
if (tuple->isStructTy())
n = dyn_cast<StructType>(tuple)->getNumElements();
else if(tuple->isArrayTy())
n = dyn_cast<ArrayType>(tuple)->getNumElements();
else if(tuple->isVectorTy())
n = dyn_cast<VectorType>(tuple)->getNumElements();
else
assert(false);
return n;
}

static Value *ghostValue(jl_value_t *ty);

// emit code to unpack a raw value from a box
static Value *emit_unbox(Type *to, Value *x, jl_value_t *jt)
{
if (x == NULL) {
if (to == T_void) {
if (jt != NULL)
return ghostValue(jt);
return NULL;
}
return UndefValue::get(to);
}
Type *ty = x->getType();
if (ty != jl_pvalue_llvmt) {
// bools are stored internally as int8 (for now)
Expand Down