1use rustc_middle::ty;
2
3use super::translate_ctx::*;
4use crate::hax;
5use crate::hax::FullDefKind;
6use crate::translate::translate_crate::TransItemSourceKind;
7use charon_lib::ast::*;
8
9impl<'tcx> ItemTransCtx<'tcx, '_> {
10 pub fn translate_drop_glue_method_id(
11 &mut self,
12 destruct_trait_def_id: &hax::DefId,
13 destruct_trait_id: TraitDeclId,
14 ) -> Result<TraitMethodId, Error> {
15 self.register_assoc_items(destruct_trait_def_id, destruct_trait_id)?;
16 let method_id = TraitMethodId::ZERO; Ok(method_id)
18 }
19
20 pub fn translate_drop_glue_method_call(
22 &mut self,
23 span: Span,
24 ty: ty::Ty<'tcx>,
25 ) -> Result<FnPtr, Error> {
26 let trait_proof = hax::solve_destruct(self.hax_state_with_id(), ty);
27 let tref = self.translate_trait_proof(span, &trait_proof)?;
28 let method_id = self.translate_drop_glue_method_id(
29 &trait_proof.pred.hax_skip_binder_ref().def_id,
30 tref.trait_id(),
31 )?;
32 let fn_ptr = FnPtr::new(
33 FnPtrKind::Trait(tref, method_id),
34 self.drop_glue_generic_args(),
35 );
36 Ok(fn_ptr)
37 }
38
39 fn translate_drop_glue_method_body(
40 &mut self,
41 span: Span,
42 def: &hax::FullDef<'tcx>,
43 ) -> Result<Body, Error> {
44 let (hax::FullDefKind::Adt { .. } | hax::FullDefKind::Closure { .. }) = def.kind() else {
45 return Ok(Body::Missing);
46 };
47
48 let body = def.this().drop_glue_shim(self.hax_state());
49 Ok(self.translate_body(span, body, &def.source_text))
50 }
51
52 #[tracing::instrument(skip(self, item_meta, def))]
55 pub fn translate_drop_glue_method(
56 mut self,
57 def_id: FunDeclId,
58 item_meta: ItemMeta,
59 def: &hax::FullDef<'tcx>,
60 impl_kind: TraitImplSource,
61 ) -> Result<FunDecl, Error> {
62 let span = item_meta.span;
63 let borrow_region = self.drop_glue_region();
64
65 let trait_pred = match def.kind() {
66 FullDefKind::Adt { destruct_impl, .. } | FullDefKind::Closure { destruct_impl, .. } => {
68 assert_eq!(impl_kind, TraitImplSource::ImplicitDestruct);
69 &destruct_impl.trait_pred
70 }
71 _ => unreachable!(),
72 };
73
74 let implemented_trait = self.translate_trait_predicate(span, trait_pred)?;
75 let item_id: AssocItemId = self
76 .translate_drop_glue_method_id(&trait_pred.trait_ref.def_id, implemented_trait.id)?
77 .into();
78 let self_ty = implemented_trait
79 .self_ty(&self.t_ctx.translated)
80 .unwrap()
81 .clone();
82
83 let signature = self.drop_glue_method_sig(self_ty.clone(), borrow_region);
84 let src = {
85 let mut impl_generics = self.the_only_binder().params.identity_args();
86 impl_generics
87 .regions
88 .pop()
89 .expect("drop glue method should have a borrow lifetime");
90 let destruct_impl_id =
91 self.register_item(span, def.this(), TransItemSourceKind::TraitImpl(impl_kind));
92 let impl_ref = TraitImplRef {
93 id: destruct_impl_id,
94 generics: Box::new(impl_generics),
95 };
96 ItemSource::TraitImpl {
97 impl_ref,
98 trait_ref: implemented_trait,
99 item_id,
100 reuses_default: false,
101 }
102 };
103
104 let body = if item_meta.opacity.with_private_contents().is_opaque() {
105 Body::Opaque
106 } else {
107 self.translate_drop_glue_method_body(span, def)?
108 };
109
110 Ok(FunDecl {
111 def_id,
112 item_meta,
113 generics: self.into_generics(),
114 signature: Box::new(signature),
115 src,
116 is_global_initializer: None,
117 body,
118 })
119 }
120
121 pub(crate) fn drop_glue_region(&self) -> Region {
122 Region::Var(DeBruijnVar::new_at_zero(
123 self.the_only_binder()
124 .drop_glue_region
125 .expect("drop glue item should have a borrow lifetime"),
126 ))
127 }
128
129 pub(crate) fn drop_glue_generic_args(&mut self) -> GenericArgs {
130 let mut generics = GenericArgs::empty();
131 generics.regions.push(self.translate_erased_region());
132 generics
133 }
134
135 pub(crate) fn drop_glue_params() -> GenericParams {
136 let mut params = GenericParams::empty();
137 params
138 .regions
139 .push_with(|index| RegionParam::new(index, None));
140 params
141 }
142
143 pub fn drop_glue_method_sig(&self, self_ty: Ty, region: Region) -> FunSig {
145 let self_ref = TyKind::Ref(region, self_ty, RefKind::Mut).into_ty();
146 FunSig {
147 is_unsafe: true,
148 abi: Abi::rust(),
149 inputs: [self_ref].into(),
150 output: Ty::mk_unit(),
151 }
152 }
153
154 pub fn drop_glue_fn_ptr_sig(&self, self_ty: Ty) -> RegionBinder<FunSig> {
155 let params = Self::drop_glue_params();
156 let region = Region::Var(DeBruijnVar::new_at_zero(RegionId::ZERO));
157 let self_ty = self_ty.move_under_binder();
158 RegionBinder {
159 regions: params.regions,
160 skip_binder: self.drop_glue_method_sig(self_ty, region),
161 }
162 }
163
164 #[tracing::instrument(skip(self, item_meta, def))]
165 pub fn translate_implicit_destruct_impl(
166 mut self,
167 impl_id: TraitImplId,
168 item_meta: ItemMeta,
169 def: &hax::FullDef<'tcx>,
170 ) -> Result<TraitImpl, Error> {
171 let span = item_meta.span;
172
173 let (FullDefKind::Adt { destruct_impl, .. } | FullDefKind::Closure { destruct_impl, .. }) =
174 def.kind()
175 else {
176 unreachable!("{:?}", def.def_id())
177 };
178 let mut timpl = self.translate_virtual_trait_impl(impl_id, item_meta, destruct_impl)?;
179
180 let destruct_trait_id = timpl.impl_trait.id;
182 let destruct_trait_def_id: &hax::DefId = &destruct_impl.trait_pred.trait_ref.def_id;
183 let method_id =
184 self.translate_drop_glue_method_id(destruct_trait_def_id, destruct_trait_id)?;
185 let method_binder = {
186 let fun_id = self.register_item(
187 span,
188 def.this(),
189 TransItemSourceKind::DropGlueMethod(TraitImplSource::ImplicitDestruct),
190 );
191 let method_params = Self::drop_glue_params();
192 let generics = self
193 .outermost_binder()
194 .params
195 .identity_args_at_depth(DeBruijnId::one())
196 .concat(&method_params.identity_args());
197 Binder::new(
198 BinderKind::TraitMethod(destruct_trait_id, method_id),
199 method_params,
200 FunDeclRef {
201 id: fun_id,
202 generics: Box::new(generics),
203 },
204 )
205 };
206 timpl.methods.set_slot_extend(method_id, method_binder);
207
208 Ok(timpl)
209 }
210}