You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
- This function takes a mutable reference to the `Providers` struct and sets the fields to point to the correct provider functions.
153
-
- You can also assign fields individually, e.g. `providers.type_of = type_of;`.
154
+
Note that this function accepts `query::Providers` not `util::Providers`. It is exceedingly rare to need a `provide` function that doesn't just accept `query::Providers`. If more than the `queries` field of `util::Providers` is being updated then `util::Providers` can be accepted instead:
Note that `util::Providers` implements `DerefMut` to `query::Providers` so callers of the `provide` functions can pass in a `util::Providers` and it will just work for provider functions that accept `query::Providers` too
169
+
170
+
- This function takes a mutable reference to the `query::Providers` struct and sets the fields to point to the correct provider functions.
171
+
- You can also assign queries individually, e.g. `providers.type_of = type_of;`.
172
+
- You can assign fields individually for each provider type (local, external, and hooks).
154
173
155
174
#### Adding a new provider
156
175
157
-
Suppose you want to add a new query called `fubar`. You would:
176
+
Suppose you want to add a new query called `fubar`. This section focuses on wiring up the providers; for how to declare the query itself in the big `rustc_queries!` macro, see [Adding a new query](#adding-a-new-query) below.
177
+
178
+
In practice you usually:
158
179
159
-
1. Implement the provider function:
180
+
1. Decide which crate "owns" the query (for example `rustc_hir_analysis`, `rustc_mir_build`, or another `rustc_*` crate).
181
+
2. In that crate, look for an existing `provide` function:
If it exists, you will extend it to set the field for your new query. If the crate does not yet have a `provide` function, add one and make sure it is included in `DEFAULT_QUERY_PROVIDERS` in the `rustc_interface` crate so that it actually gets called during initialization (see the discussion above).
### How queries interact with external crate metadata
203
+
204
+
When a query is made for an external crate (i.e., a dependency), the query system needs to load the information from that crate's metadata.
205
+
This is handled by the [`rustc_metadata` crate][rustc_metadata], which is responsible for decoding and providing the information stored in the `.rmeta` files.
206
+
207
+
The process works like this:
208
+
209
+
1. When a query is made, the query system first checks if the `DefId` refers to a local or external crate by checking if `def_id.krate == LOCAL_CRATE`.
210
+
This determines whether to use the local provider from [`Providers`][providers_struct] or the external provider from [`ExternProviders`][extern_providers_struct].
211
+
212
+
2. For external crates, the query system will look for a provider in the [`ExternProviders`][extern_providers_struct] struct.
213
+
The `rustc_metadata` crate registers these external providers through the `provide_extern` function in `rustc_metadata/src/rmeta/decoder/cstore_impl.rs`. Just like:
let cdata = CStore::from_tcx(tcx).get_crate_data(def_id.krate);
219
+
cdata.foo(def_id.index)
220
+
};
221
+
// Register other external providers...
222
+
}
223
+
```
224
+
225
+
3. The metadata is stored in a binary format in `.rmeta` files that contains pre-computed information about the external crate, such as types, function signatures, trait implementations, and other information needed by the compiler. When an external query is made, the `rustc_metadata` crate:
226
+
- Loads the `.rmeta` file for the external crate
227
+
- Decodes the metadata using the `Decodable` trait
228
+
- Returns the decoded information to the query system
229
+
230
+
This approach avoids recompiling external crates, allows for faster compilation of dependent crates, and enables incremental compilation to work across crate boundaries.
231
+
Here is a simplified example, when you call `tcx.type_of(def_id)` for a type defined in an external crate, the query system will:
232
+
1. Detect that the `def_id` refers to an external crate by checking `def_id.krate != LOCAL_CRATE`
233
+
2. Call the appropriate provider from `ExternProviders` which was registered by `rustc_metadata`
234
+
3. The provider will load and decode the type information from the external crate's metadata
235
+
4. Return the decoded type to the caller
236
+
237
+
This is why most `rustc_*` crates only need to provide local providers - the external providers are handled by the metadata system.
238
+
The only exception is when a crate needs to provide special handling for external queries, in which case it would implement both local and external providers.
239
+
240
+
When we define a new query that should work across crates, it does not automatically become cross-crate just because it is listed in `rustc_queries!`. You will typically need to:
241
+
242
+
- Add the query to `rustc_queries!` with appropriate modifiers (for example whether it is cached on disk).
243
+
- Implement a local provider in the owning crate and register it via that crate's `provide` function.
244
+
- Add an external provider in `rustc_metadata` via `provide_extern`, and ensure the query's result is encoded and decoded in the crate metadata.
245
+
246
+
An example of introducing such a cross-crate query can be found in commit [`996a185`](https://github.com/rust-lang/rust/commit/996a185) in the `rust-lang/rust` repository.
0 commit comments