Skip to content

Commit abd78f2

Browse files
committed
rustc: Create stub glue for structural comparison
1 parent ef62fae commit abd78f2

File tree

2 files changed

+80
-23
lines changed

2 files changed

+80
-23
lines changed

src/comp/back/abi.rs

+2
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,8 @@ const int tydesc_field_sever_glue = 6;
3939
const int tydesc_field_mark_glue = 7;
4040
const int tydesc_field_obj_drop_glue = 8;
4141
const int tydesc_field_is_stateful = 9;
42+
const int tydesc_field_cmp_glue = 10;
43+
const int n_tydesc_fields = 11;
4244

4345

4446
const int obj_field_vtbl = 0;

src/comp/middle/trans.rs

+78-23
Original file line numberDiff line numberDiff line change
@@ -66,7 +66,8 @@ type glue_fns = rec(ValueRef activate_glue,
6666

6767
type tydesc_info = rec(ValueRef tydesc,
6868
ValueRef take_glue,
69-
ValueRef drop_glue);
69+
ValueRef drop_glue,
70+
ValueRef cmp_glue);
7071

7172
/*
7273
* A note on nomenclature of linking: "upcall", "extern" and "native".
@@ -325,19 +326,34 @@ fn T_task(type_names tn) -> TypeRef {
325326
ret t;
326327
}
327328

329+
fn T_tydesc_field(type_names tn, int field) -> TypeRef {
330+
// Bit of a kludge: pick the fn typeref out of the tydesc..
331+
let vec[TypeRef] tydesc_elts =
332+
_vec.init_elt[TypeRef](T_nil(), abi.n_tydesc_fields as uint);
333+
llvm.LLVMGetStructElementTypes(T_tydesc(tn),
334+
_vec.buf[TypeRef](tydesc_elts));
335+
auto t = llvm.LLVMGetElementType(tydesc_elts.(field));
336+
ret t;
337+
}
338+
328339
fn T_glue_fn(type_names tn) -> TypeRef {
329340
auto s = "glue_fn";
330341
if (tn.name_has_type(s)) {
331342
ret tn.get_type(s);
332343
}
333344

334-
// Bit of a kludge: pick the fn typeref out of the tydesc..
335-
let vec[TypeRef] tydesc_elts = _vec.init_elt[TypeRef](T_nil(), 10u);
336-
llvm.LLVMGetStructElementTypes(T_tydesc(tn),
337-
_vec.buf[TypeRef](tydesc_elts));
338-
auto t =
339-
llvm.LLVMGetElementType
340-
(tydesc_elts.(abi.tydesc_field_drop_glue));
345+
auto t = T_tydesc_field(tn, abi.tydesc_field_drop_glue);
346+
tn.associate(s, t);
347+
ret t;
348+
}
349+
350+
fn T_cmp_glue_fn(type_names tn) -> TypeRef {
351+
auto s = "cmp_glue_fn";
352+
if (tn.name_has_type(s)) {
353+
ret tn.get_type(s);
354+
}
355+
356+
auto t = T_tydesc_field(tn, abi.tydesc_field_cmp_glue);
341357
tn.associate(s, t);
342358
ret t;
343359
}
@@ -358,6 +374,12 @@ fn T_tydesc(type_names tn) -> TypeRef {
358374
T_ptr(T_nil()),
359375
tydescpp,
360376
pvoid), T_void()));
377+
auto cmp_glue_fn_ty = T_ptr(T_fn(vec(T_ptr(T_nil()),
378+
T_taskptr(tn),
379+
T_ptr(T_nil()),
380+
tydescpp,
381+
pvoid,
382+
pvoid), T_void()));
361383
auto tydesc = T_struct(vec(tydescpp, // first_param
362384
T_int(), // size
363385
T_int(), // align
@@ -367,7 +389,8 @@ fn T_tydesc(type_names tn) -> TypeRef {
367389
glue_fn_ty, // sever_glue
368390
glue_fn_ty, // mark_glue
369391
glue_fn_ty, // obj_drop_glue
370-
glue_fn_ty)); // is_stateful
392+
glue_fn_ty, // is_stateful
393+
cmp_glue_fn_ty)); // cmp_glue
371394

372395
llvm.LLVMRefineType(abs_tydesc, tydesc);
373396
auto t = llvm.LLVMResolveTypeHandle(th.llth);
@@ -1580,8 +1603,12 @@ fn get_tydesc(&@block_ctxt cx, @ty.t t) -> result {
15801603
// needs to be separate from make_tydesc() below, because sometimes type glue
15811604
// functions needs to refer to their own type descriptors.
15821605
fn declare_tydesc(@local_ctxt cx, @ty.t t) {
1583-
auto take_glue = declare_generic_glue(cx, t, "take");
1584-
auto drop_glue = declare_generic_glue(cx, t, "drop");
1606+
auto take_glue = declare_generic_glue(cx, t, T_glue_fn(cx.ccx.tn),
1607+
"take");
1608+
auto drop_glue = declare_generic_glue(cx, t, T_glue_fn(cx.ccx.tn),
1609+
"drop");
1610+
auto cmp_glue = declare_generic_glue(cx, t, T_cmp_glue_fn(cx.ccx.tn),
1611+
"cmp");
15851612
auto ccx = cx.ccx;
15861613

15871614
auto llsize;
@@ -1611,7 +1638,8 @@ fn declare_tydesc(@local_ctxt cx, @ty.t t) {
16111638
C_null(glue_fn_ty), // sever_glue
16121639
C_null(glue_fn_ty), // mark_glue
16131640
C_null(glue_fn_ty), // obj_drop_glue
1614-
C_null(glue_fn_ty))); // is_stateful
1641+
C_null(glue_fn_ty), // is_stateful
1642+
cmp_glue)); // cmp_glue
16151643

16161644
llvm.LLVMSetInitializer(gvar, tydesc);
16171645
llvm.LLVMSetGlobalConstant(gvar, True);
@@ -1621,35 +1649,47 @@ fn declare_tydesc(@local_ctxt cx, @ty.t t) {
16211649
auto info = rec(
16221650
tydesc=gvar,
16231651
take_glue=take_glue,
1624-
drop_glue=drop_glue
1652+
drop_glue=drop_glue,
1653+
cmp_glue=cmp_glue
16251654
);
16261655

16271656
ccx.tydescs.insert(t, @info);
16281657
}
16291658

1659+
tag make_generic_glue_helper_fn {
1660+
mgghf_single(val_and_ty_fn);
1661+
mgghf_pair(val_pair_and_ty_fn);
1662+
}
1663+
16301664
// declare_tydesc() above must have been called first.
16311665
fn define_tydesc(@local_ctxt cx, @ty.t t, vec[uint] ty_params) {
16321666
auto info = cx.ccx.tydescs.get(t);
16331667
auto gvar = info.tydesc;
16341668

16351669
auto tg = make_take_glue;
1636-
auto take_glue = make_generic_glue(cx, t, info.take_glue, tg, ty_params);
1670+
make_generic_glue(cx, t, info.take_glue, mgghf_single(tg), ty_params);
16371671
auto dg = make_drop_glue;
1638-
auto drop_glue = make_generic_glue(cx, t, info.drop_glue, dg, ty_params);
1672+
make_generic_glue(cx, t, info.drop_glue, mgghf_single(dg), ty_params);
1673+
auto cg = make_cmp_glue;
1674+
make_generic_glue(cx, t, info.cmp_glue, mgghf_pair(cg), ty_params);
16391675
}
16401676

1641-
fn declare_generic_glue(@local_ctxt cx, @ty.t t, str name) -> ValueRef {
1642-
auto llfnty = T_glue_fn(cx.ccx.tn);
1643-
1677+
fn declare_generic_glue(@local_ctxt cx,
1678+
@ty.t t,
1679+
TypeRef llfnty,
1680+
str name) -> ValueRef {
16441681
auto gcx = @rec(path=vec("glue", name) with *cx);
16451682
auto fn_name = mangle_name_by_type(gcx, t);
16461683
fn_name = sanitize(fn_name);
16471684
auto llfn = decl_internal_fastcall_fn(cx.ccx.llmod, fn_name, llfnty);
16481685
ret llfn;
16491686
}
16501687

1651-
fn make_generic_glue(@local_ctxt cx, @ty.t t, ValueRef llfn,
1652-
val_and_ty_fn helper, vec[uint] ty_params) -> ValueRef {
1688+
fn make_generic_glue(@local_ctxt cx,
1689+
@ty.t t,
1690+
ValueRef llfn,
1691+
make_generic_glue_helper_fn helper,
1692+
vec[uint] ty_params) -> ValueRef {
16531693
auto fcx = new_fn_ctxt(cx, llfn);
16541694
auto bcx = new_top_block_ctxt(fcx);
16551695
auto lltop = bcx.llbb;
@@ -1684,10 +1724,19 @@ fn make_generic_glue(@local_ctxt cx, @ty.t t, ValueRef llfn,
16841724
}
16851725
bcx.fcx.lltydescs = _vec.freeze[ValueRef](lltydescs);
16861726

1687-
auto llrawptr = llvm.LLVMGetParam(llfn, 4u);
1688-
auto llval = bcx.build.BitCast(llrawptr, llty);
1727+
auto llrawptr0 = llvm.LLVMGetParam(llfn, 4u);
1728+
auto llval0 = bcx.build.BitCast(llrawptr0, llty);
16891729

1690-
re = helper(bcx, llval, t);
1730+
alt (helper) {
1731+
case (mgghf_single(?single_fn)) {
1732+
re = single_fn(bcx, llval0, t);
1733+
}
1734+
case (mgghf_pair(?pair_fn)) {
1735+
auto llrawptr1 = llvm.LLVMGetParam(llfn, 5u);
1736+
auto llval1 = bcx.build.BitCast(llrawptr0, llty);
1737+
re = pair_fn(bcx, llval0, llval1, t);
1738+
}
1739+
}
16911740
} else {
16921741
re = res(bcx, C_nil());
16931742
}
@@ -1933,6 +1982,12 @@ fn decr_refcnt_and_if_zero(@block_ctxt cx,
19331982
ret res(next_cx, phi);
19341983
}
19351984

1985+
fn make_cmp_glue(@block_ctxt cx, ValueRef v0, ValueRef v1, @ty.t t)
1986+
-> result {
1987+
ret res(cx, C_nil()); // TODO
1988+
}
1989+
1990+
19361991
// Tag information
19371992

19381993
type variant_info = rec(vec[@ty.t] args, @ty.t ctor_ty, ast.def_id id);

0 commit comments

Comments
 (0)