@@ -66,7 +66,8 @@ type glue_fns = rec(ValueRef activate_glue,
66
66
67
67
type tydesc_info = rec ( ValueRef tydesc,
68
68
ValueRef take_glue ,
69
- ValueRef drop_glue ) ;
69
+ ValueRef drop_glue ,
70
+ ValueRef cmp_glue) ;
70
71
71
72
/*
72
73
* A note on nomenclature of linking: "upcall", "extern" and "native".
@@ -325,19 +326,34 @@ fn T_task(type_names tn) -> TypeRef {
325
326
ret t;
326
327
}
327
328
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
+
328
339
fn T_glue_fn ( type_names tn) -> TypeRef {
329
340
auto s = "glue_fn" ;
330
341
if ( tn. name_has_type ( s) ) {
331
342
ret tn. get_type ( s) ;
332
343
}
333
344
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 ( ) , 10 u) ;
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 ) ;
341
357
tn. associate ( s, t) ;
342
358
ret t;
343
359
}
@@ -358,6 +374,12 @@ fn T_tydesc(type_names tn) -> TypeRef {
358
374
T_ptr ( T_nil ( ) ) ,
359
375
tydescpp,
360
376
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 ( ) ) ) ;
361
383
auto tydesc = T_struct ( vec ( tydescpp, // first_param
362
384
T_int ( ) , // size
363
385
T_int ( ) , // align
@@ -367,7 +389,8 @@ fn T_tydesc(type_names tn) -> TypeRef {
367
389
glue_fn_ty, // sever_glue
368
390
glue_fn_ty, // mark_glue
369
391
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
371
394
372
395
llvm. LLVMRefineType ( abs_tydesc, tydesc) ;
373
396
auto t = llvm. LLVMResolveTypeHandle ( th. llth ) ;
@@ -1580,8 +1603,12 @@ fn get_tydesc(&@block_ctxt cx, @ty.t t) -> result {
1580
1603
// needs to be separate from make_tydesc() below, because sometimes type glue
1581
1604
// functions needs to refer to their own type descriptors.
1582
1605
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" ) ;
1585
1612
auto ccx = cx. ccx;
1586
1613
1587
1614
auto llsize;
@@ -1611,7 +1638,8 @@ fn declare_tydesc(@local_ctxt cx, @ty.t t) {
1611
1638
C_null ( glue_fn_ty) , // sever_glue
1612
1639
C_null ( glue_fn_ty) , // mark_glue
1613
1640
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
1615
1643
1616
1644
llvm. LLVMSetInitializer ( gvar, tydesc) ;
1617
1645
llvm. LLVMSetGlobalConstant ( gvar, True ) ;
@@ -1621,35 +1649,47 @@ fn declare_tydesc(@local_ctxt cx, @ty.t t) {
1621
1649
auto info = rec(
1622
1650
tydesc=gvar,
1623
1651
take_glue=take_glue,
1624
- drop_glue=drop_glue
1652
+ drop_glue=drop_glue,
1653
+ cmp_glue=cmp_glue
1625
1654
) ;
1626
1655
1627
1656
ccx. tydescs. insert( t, @info) ;
1628
1657
}
1629
1658
1659
+ tag make_generic_glue_helper_fn {
1660
+ mgghf_single( val_and_ty_fn) ;
1661
+ mgghf_pair( val_pair_and_ty_fn) ;
1662
+ }
1663
+
1630
1664
// declare_tydesc() above must have been called first.
1631
1665
fn define_tydesc( @local_ctxt cx, @ty. t t, vec[ uint] ty_params) {
1632
1666
auto info = cx. ccx. tydescs. get( t) ;
1633
1667
auto gvar = info. tydesc;
1634
1668
1635
1669
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) ;
1637
1671
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) ;
1639
1675
}
1640
1676
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 {
1644
1681
auto gcx = @rec( path=vec( "glue" , name) with * cx) ;
1645
1682
auto fn_name = mangle_name_by_type( gcx, t) ;
1646
1683
fn_name = sanitize( fn_name) ;
1647
1684
auto llfn = decl_internal_fastcall_fn( cx. ccx. llmod, fn_name, llfnty) ;
1648
1685
ret llfn;
1649
1686
}
1650
1687
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 {
1653
1693
auto fcx = new_fn_ctxt( cx, llfn) ;
1654
1694
auto bcx = new_top_block_ctxt( fcx) ;
1655
1695
auto lltop = bcx. llbb;
@@ -1684,10 +1724,19 @@ fn make_generic_glue(@local_ctxt cx, @ty.t t, ValueRef llfn,
1684
1724
}
1685
1725
bcx. fcx. lltydescs = _vec. freeze[ ValueRef ] ( lltydescs) ;
1686
1726
1687
- auto llrawptr = llvm. LLVMGetParam ( llfn, 4 u) ;
1688
- auto llval = bcx. build. BitCast ( llrawptr , llty) ;
1727
+ auto llrawptr0 = llvm. LLVMGetParam ( llfn, 4 u) ;
1728
+ auto llval0 = bcx. build. BitCast ( llrawptr0 , llty) ;
1689
1729
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, 5 u) ;
1736
+ auto llval1 = bcx. build. BitCast ( llrawptr0, llty) ;
1737
+ re = pair_fn( bcx, llval0, llval1, t) ;
1738
+ }
1739
+ }
1691
1740
} else {
1692
1741
re = res( bcx, C_nil ( ) ) ;
1693
1742
}
@@ -1933,6 +1982,12 @@ fn decr_refcnt_and_if_zero(@block_ctxt cx,
1933
1982
ret res( next_cx, phi) ;
1934
1983
}
1935
1984
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
+
1936
1991
// Tag information
1937
1992
1938
1993
type variant_info = rec( vec[ @ty. t] args, @ty. t ctor_ty, ast. def_id id) ;
0 commit comments