Skip to content

Commit

Permalink
feat(exporter): thir: call: separate trait VS method generic args
Browse files Browse the repository at this point in the history
This PR separates method generic args and trait generic args of the
`call` nodes of THIR expressions.

Consider the following example:
```rust
trait MyTrait<TraitType> {
  fn meth<MethType>(...) {...}
}
fn example_call<TraitType, SelfType: MyTrait<TraitType>>(x: SelfType) {
  x.meth::<String>(...)
}
```

Before this commit, the call `x.meth::<String>` produced a `Call` node
whose field `generic_args` was `[SelfType, TraitType, String]`.

Now, that `Call` node has a trait-specific generic arguments field
`[SelfType, TraitType]` and a method/function-specific generic
arguments field `[String].`

This commit does no change to the engine, that will be for a
subsequent PR.

This PR is related to #719.
  • Loading branch information
W95Psp committed Jul 8, 2024
1 parent bc51591 commit 4c2efd6
Show file tree
Hide file tree
Showing 2 changed files with 45 additions and 16 deletions.
12 changes: 8 additions & 4 deletions engine/lib/import_thir.ml
Original file line number Diff line number Diff line change
Expand Up @@ -426,7 +426,7 @@ end) : EXPR = struct
{
args;
fn_span = _;
impl;
trait;
from_hir_call = _;
fun';
ty = _;
Expand All @@ -435,12 +435,16 @@ end) : EXPR = struct
} ->
let args = List.map ~f:c_expr args in
let bounds_impls = List.map ~f:(c_impl_expr e.span) bounds_impls in
let trait_generic_args =
Option.map ~f:snd trait |> Option.value ~default:[]
in
let generic_args =
List.map ~f:(c_generic_value e.span) generic_args
List.map ~f:(c_generic_value e.span)
(trait_generic_args @ generic_args)
in
let f =
let f = c_expr fun' in
match (impl, fun'.contents) with
match (trait, fun'.contents) with
| Some _, GlobalName { id } ->
{ f with e = GlobalVar (def_id (AssociatedItem Value) id) }
| _ -> f
Expand All @@ -452,7 +456,7 @@ end) : EXPR = struct
args;
generic_args;
bounds_impls;
impl = Option.map ~f:(c_impl_expr e.span) impl;
impl = Option.map ~f:(fst >> c_impl_expr e.span) trait;
}
| Box { value } ->
(U.call Rust_primitives__hax__box_new [ c_expr value ] span typ).e
Expand Down
49 changes: 37 additions & 12 deletions frontend/exporter/src/types/copied.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2414,7 +2414,7 @@ pub enum ExprKind {
},
#[map({
let e = gstate.thir().exprs[*fun].unroll_scope(gstate);
let (generic_args, r#impl, bounds_impls);
let (generic_args, r#trait, bounds_impls);
let fun = match &e.kind {
/* TODO: see whether [user_ty] below is relevant or not */
rustc_middle::thir::ExprKind::ZstLiteral {user_ty: _ } => {
Expand All @@ -2425,11 +2425,22 @@ pub enum ExprKind {
let contents = Box::new(ExprKind::GlobalName {
id: def_id.sinto(gstate)
});
let mut translated_generics = generics.sinto(gstate);
let tcx = gstate.base().tcx;
r#impl = tcx.opt_associated_item(*def_id).as_ref().and_then(|assoc| {
poly_trait_ref(gstate, assoc, generics)
}).map(|poly_trait_ref| poly_trait_ref.impl_expr(gstate, gstate.param_env()));
generic_args = generics.sinto(gstate);
r#trait = (|| {
let assoc_item = tcx.opt_associated_item(*def_id)?;
let assoc_trait = tcx.trait_of_item(assoc_item.def_id)?;
let trait_ref = ty::TraitRef::new(tcx, assoc_trait, generics.iter());
let impl_expr = {
// TODO: we should not wrap into a dummy binder
let poly_trait_ref = ty::Binder::dummy(trait_ref);
poly_trait_ref.impl_expr(gstate, gstate.param_env())
};
let assoc_generics = tcx.generics_of(assoc_item.def_id);
let assoc_generics = translated_generics.drain(0..assoc_generics.parent_count);
Some((impl_expr, assoc_generics.collect()))
})();
generic_args = translated_generics;
bounds_impls = solve_item_traits(gstate, gstate.param_env(), *def_id, generics, None);
Expr {
contents,
Expand All @@ -2451,7 +2462,7 @@ pub enum ExprKind {
rustc_middle::ty::TyKind::FnPtr(..) => {
generic_args = vec![]; // A function pointer has no generics
bounds_impls = vec![]; // A function pointer has no bounds
r#impl = None; // A function pointer is not a method
r#trait = None; // A function pointer is not a method
e.sinto(gstate)
},
ty_kind => {
Expand All @@ -2472,7 +2483,7 @@ pub enum ExprKind {
from_hir_call: from_hir_call.sinto(gstate),
fn_span: fn_span.sinto(gstate),
bounds_impls,
r#impl,
r#trait,
fun,
}
})]
Expand Down Expand Up @@ -2507,15 +2518,29 @@ pub enum ExprKind {
/// one for the implicit `i8: Sized`.
#[not_in_source]
bounds_impls: Vec<ImplExpr>,
/// `impl` is `None` if this is a function call or a method to
/// an inherent trait. If this is a method call from a trait
/// `Trait`, then it contains the concrete implementation of
/// `Trait` it is called on.
/// `trait` is `None` if this is a function call or a method
/// to an inherent trait. If this is a method call from a
/// trait `Trait`, then it contains the concrete
/// implementation of `Trait` it is called on, and the generic
/// arguments that comes from the trait declaration.
///
/// Example: `f(0i8)` is a function call, hence the field
/// `impl` is `None`.
///
/// Example:
/// ```ignore
/// trait MyTrait<TraitType, const TraitConst: usize> {
/// fn meth<MethType>(...) {...}
/// }
/// fn example_call<TraitType, SelfType: MyTrait<TraitType, 12>>(x: SelfType) {
/// x.meth::<String>(...)
/// }
/// ```
/// Here, in the call `x.meth::<String>(...)`, `r#trait` will
/// be `Some((..., [SelfType, TraitType, 12]))`, and `generic_args`
/// will be `[String]`.
#[not_in_source]
r#impl: Option<ImplExpr>,
r#trait: Option<(ImplExpr, Vec<GenericArg>)>,
},
Deref {
arg: Expr,
Expand Down

0 comments on commit 4c2efd6

Please sign in to comment.