Skip to content

Commit 8d11f90

Browse files
committed
Record expansion_that_defined into crate metadata
Fixes #77523 Now that hygiene serialization is implemented, we also need to record `expansion_that_defined` so that we properly handle a foreign `SyntaxContext`.
1 parent f317a93 commit 8d11f90

File tree

9 files changed

+66
-2
lines changed

9 files changed

+66
-2
lines changed

compiler/rustc_metadata/src/rmeta/decoder.rs

+4
Original file line numberDiff line numberDiff line change
@@ -1011,6 +1011,10 @@ impl<'a, 'tcx> CrateMetadataRef<'a> {
10111011
self.root.tables.impl_trait_ref.get(self, id).map(|tr| tr.decode((self, tcx)))
10121012
}
10131013

1014+
fn get_expn_that_defined(&self, id: DefIndex, sess: &Session) -> ExpnId {
1015+
self.root.tables.expn_that_defined.get(self, id).unwrap().decode((self, sess))
1016+
}
1017+
10141018
/// Iterates over all the stability attributes in the given crate.
10151019
fn get_lib_features(&self, tcx: TyCtxt<'tcx>) -> &'tcx [(Symbol, Option<Symbol>)] {
10161020
// FIXME: For a proc macro crate, not sure whether we should return the "host"

compiler/rustc_metadata/src/rmeta/decoder/cstore_impl.rs

+1
Original file line numberDiff line numberDiff line change
@@ -238,6 +238,7 @@ provide! { <'tcx> tcx, def_id, other, cdata,
238238
}
239239

240240
crate_extern_paths => { cdata.source().paths().cloned().collect() }
241+
expn_that_defined => { cdata.get_expn_that_defined(def_id.index, tcx.sess) }
241242
}
242243

243244
pub fn provide(providers: &mut Providers) {

compiler/rustc_metadata/src/rmeta/encoder.rs

+4
Original file line numberDiff line numberDiff line change
@@ -747,6 +747,7 @@ impl EncodeContext<'a, 'tcx> {
747747
ty::Visibility::from_hir(enum_vis, enum_id, self.tcx));
748748
record!(self.tables.span[def_id] <- self.tcx.def_span(def_id));
749749
record!(self.tables.attributes[def_id] <- &self.tcx.get_attrs(def_id)[..]);
750+
record!(self.tables.expn_that_defined[def_id] <- self.tcx.expansion_that_defined(def_id));
750751
record!(self.tables.children[def_id] <- variant.fields.iter().map(|f| {
751752
assert!(f.did.is_local());
752753
f.did.index
@@ -883,6 +884,7 @@ impl EncodeContext<'a, 'tcx> {
883884
record!(self.tables.visibility[def_id] <- field.vis);
884885
record!(self.tables.span[def_id] <- self.tcx.def_span(def_id));
885886
record!(self.tables.attributes[def_id] <- variant_data.fields()[field_index].attrs);
887+
record!(self.tables.expn_that_defined[def_id] <- self.tcx.expansion_that_defined(def_id));
886888
self.encode_ident_span(def_id, field.ident);
887889
self.encode_stability(def_id);
888890
self.encode_deprecation(def_id);
@@ -924,6 +926,7 @@ impl EncodeContext<'a, 'tcx> {
924926
record!(self.tables.kind[def_id] <- EntryKind::Struct(self.lazy(data), adt_def.repr));
925927
record!(self.tables.visibility[def_id] <- ctor_vis);
926928
record!(self.tables.span[def_id] <- self.tcx.def_span(def_id));
929+
record!(self.tables.expn_that_defined[def_id] <- self.tcx.expansion_that_defined(def_id));
927930
self.encode_stability(def_id);
928931
self.encode_deprecation(def_id);
929932
self.encode_item_type(def_id);
@@ -1339,6 +1342,7 @@ impl EncodeContext<'a, 'tcx> {
13391342
ty::Visibility::from_hir(&item.vis, item.hir_id, tcx));
13401343
record!(self.tables.span[def_id] <- self.tcx.def_span(def_id));
13411344
record!(self.tables.attributes[def_id] <- item.attrs);
1345+
record!(self.tables.expn_that_defined[def_id] <- self.tcx.expansion_that_defined(def_id));
13421346
// FIXME(eddyb) there should be a nicer way to do this.
13431347
match item.kind {
13441348
hir::ItemKind::ForeignMod(ref fm) => record!(self.tables.children[def_id] <-

compiler/rustc_metadata/src/rmeta/mod.rs

+1
Original file line numberDiff line numberDiff line change
@@ -294,6 +294,7 @@ define_tables! {
294294
variances: Table<DefIndex, Lazy<[ty::Variance]>>,
295295
generics: Table<DefIndex, Lazy<ty::Generics>>,
296296
explicit_predicates: Table<DefIndex, Lazy!(ty::GenericPredicates<'tcx>)>,
297+
expn_that_defined: Table<DefIndex, Lazy<ExpnId>>,
297298
// FIXME(eddyb) this would ideally be `Lazy<[...]>` but `ty::Predicate`
298299
// doesn't handle shorthands in its own (de)serialization impls,
299300
// as it's an `enum` for which we want to derive (de)serialization,

compiler/rustc_middle/src/query/mod.rs

+5
Original file line numberDiff line numberDiff line change
@@ -191,6 +191,11 @@ rustc_queries! {
191191
eval_always
192192
desc { |tcx| "parent module of `{}`", tcx.def_path_str(key.to_def_id()) }
193193
}
194+
195+
/// Internal helper query. Use `tcx.expansion_that_defined` instead
196+
query expn_that_defined(key: DefId) -> rustc_span::ExpnId {
197+
desc { |tcx| "expansion that defined `{}`", tcx.def_path_str(key) }
198+
}
194199
}
195200

196201
Codegen {

compiler/rustc_middle/src/ty/mod.rs

+4-2
Original file line numberDiff line numberDiff line change
@@ -3034,10 +3034,12 @@ impl<'tcx> TyCtxt<'tcx> {
30343034
.hygienic_eq(def_name.span.ctxt(), self.expansion_that_defined(def_parent_def_id))
30353035
}
30363036

3037-
fn expansion_that_defined(self, scope: DefId) -> ExpnId {
3037+
pub fn expansion_that_defined(self, scope: DefId) -> ExpnId {
30383038
match scope.as_local() {
3039+
// Parsing and expansion aren't incremental, so we don't
3040+
// need to go through a query for the same-crate case.
30393041
Some(scope) => self.hir().definitions().expansion_that_defined(scope),
3040-
None => ExpnId::root(),
3042+
None => self.expn_that_defined(scope),
30413043
}
30423044
}
30433045

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
// edition:2018
2+
3+
extern crate opaque_hygiene;
4+
5+
pub async fn serve() {
6+
opaque_hygiene::make_it!();
7+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
// force-host
2+
// no-prefer-dynamic
3+
4+
#![feature(proc_macro_quote)]
5+
#![crate_type = "proc-macro"]
6+
7+
extern crate proc_macro;
8+
use proc_macro::{TokenStream, quote};
9+
10+
#[proc_macro]
11+
pub fn make_it(input: TokenStream) -> TokenStream {
12+
// `quote!` applies def-site hygiene
13+
quote! {
14+
trait Foo {
15+
fn my_fn(&self) {}
16+
}
17+
18+
impl<T> Foo for T {}
19+
"a".my_fn();
20+
}
21+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
// build-pass
2+
// aux-build:opaque-hygiene.rs
3+
// aux-build:def-site-async-await.rs
4+
5+
// Regression test for issue #77523
6+
// Tests that we don't ICE when an unusual combination
7+
// of def-site hygiene and cross-crate monomorphization occurs.
8+
9+
extern crate def_site_async_await;
10+
11+
use std::future::Future;
12+
13+
fn mk_ctxt() -> std::task::Context<'static> {
14+
panic!()
15+
}
16+
17+
fn main() {
18+
Box::pin(def_site_async_await::serve()).as_mut().poll(&mut mk_ctxt());
19+
}

0 commit comments

Comments
 (0)