Merge pull request #483 from nikomatsakis/salsa-2022-spans

improve spans for getters, constructors
This commit is contained in:
Niko Matsakis 2024-04-04 10:34:23 +00:00 committed by GitHub
commit 45fd94120b
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
11 changed files with 123 additions and 27 deletions

View file

@ -1,4 +1,4 @@
use syn::ItemStruct; use syn::{spanned::Spanned, ItemStruct};
// #[salsa::accumulator(jar = Jar0)] // #[salsa::accumulator(jar = Jar0)]
// struct Accumulator(DataType); // struct Accumulator(DataType);
@ -86,7 +86,7 @@ fn struct_item_out(
data_ty: &syn::Type, data_ty: &syn::Type,
) -> syn::ItemStruct { ) -> syn::ItemStruct {
let mut struct_item_out = struct_item.clone(); let mut struct_item_out = struct_item.clone();
struct_item_out.fields = syn::Fields::Unnamed(parse_quote! { struct_item_out.fields = syn::Fields::Unnamed(parse_quote_spanned! { data_ty.span() =>
(std::marker::PhantomData<#data_ty>) (std::marker::PhantomData<#data_ty>)
}); });
struct_item_out struct_item_out
@ -94,7 +94,7 @@ fn struct_item_out(
fn inherent_impl(args: &Args, struct_ty: &syn::Type, data_ty: &syn::Type) -> syn::ItemImpl { fn inherent_impl(args: &Args, struct_ty: &syn::Type, data_ty: &syn::Type) -> syn::ItemImpl {
let jar_ty = args.jar_ty(); let jar_ty = args.jar_ty();
parse_quote! { parse_quote_spanned! { struct_ty.span() =>
impl #struct_ty { impl #struct_ty {
pub fn push<DB: ?Sized>(db: &DB, data: #data_ty) pub fn push<DB: ?Sized>(db: &DB, data: #data_ty)
where where
@ -115,7 +115,7 @@ fn ingredients_for_impl(
) -> syn::ItemImpl { ) -> syn::ItemImpl {
let jar_ty = args.jar_ty(); let jar_ty = args.jar_ty();
let debug_name = crate::literal(struct_name); let debug_name = crate::literal(struct_name);
parse_quote! { parse_quote_spanned! { struct_name.span() =>
impl salsa::storage::IngredientsFor for #struct_name { impl salsa::storage::IngredientsFor for #struct_name {
type Ingredients = salsa::accumulator::AccumulatorIngredient<#data_ty>; type Ingredients = salsa::accumulator::AccumulatorIngredient<#data_ty>;
type Jar = #jar_ty; type Jar = #jar_ty;
@ -142,7 +142,7 @@ fn ingredients_for_impl(
fn accumulator_impl(args: &Args, struct_ty: &syn::Type, data_ty: &syn::Type) -> syn::ItemImpl { fn accumulator_impl(args: &Args, struct_ty: &syn::Type, data_ty: &syn::Type) -> syn::ItemImpl {
let jar_ty = args.jar_ty(); let jar_ty = args.jar_ty();
parse_quote! { parse_quote_spanned! { struct_ty.span() =>
impl salsa::accumulator::Accumulator for #struct_ty { impl salsa::accumulator::Accumulator for #struct_ty {
type Data = #data_ty; type Data = #data_ty;
type Jar = #jar_ty; type Jar = #jar_ty;

View file

@ -81,7 +81,7 @@ impl InputStruct {
let get_field_names: Vec<_> = self.all_get_field_names(); let get_field_names: Vec<_> = self.all_get_field_names();
let field_getters: Vec<syn::ImplItemMethod> = field_indices.iter().zip(&get_field_names).zip(&field_vises).zip(&field_tys).zip(&field_clones).map(|((((field_index, get_field_name), field_vis), field_ty), is_clone_field)| let field_getters: Vec<syn::ImplItemMethod> = field_indices.iter().zip(&get_field_names).zip(&field_vises).zip(&field_tys).zip(&field_clones).map(|((((field_index, get_field_name), field_vis), field_ty), is_clone_field)|
if !*is_clone_field { if !*is_clone_field {
parse_quote! { parse_quote_spanned! { get_field_name.span() =>
#field_vis fn #get_field_name<'db>(self, __db: &'db #db_dyn_ty) -> &'db #field_ty #field_vis fn #get_field_name<'db>(self, __db: &'db #db_dyn_ty) -> &'db #field_ty
{ {
let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(__db); let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(__db);
@ -90,7 +90,7 @@ impl InputStruct {
} }
} }
} else { } else {
parse_quote! { parse_quote_spanned! { get_field_name.span() =>
#field_vis fn #get_field_name<'db>(self, __db: &'db #db_dyn_ty) -> #field_ty #field_vis fn #get_field_name<'db>(self, __db: &'db #db_dyn_ty) -> #field_ty
{ {
let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(__db); let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(__db);
@ -110,7 +110,7 @@ impl InputStruct {
.zip(&field_tys) .zip(&field_tys)
.filter_map(|(((field_index, &set_field_name), field_vis), field_ty)| { .filter_map(|(((field_index, &set_field_name), field_vis), field_ty)| {
let set_field_name = set_field_name?; let set_field_name = set_field_name?;
Some(parse_quote! { Some(parse_quote_spanned! { set_field_name.span() =>
#field_vis fn #set_field_name<'db>(self, __db: &'db mut #db_dyn_ty) -> salsa::setter::Setter<'db, #ident, #field_ty> #field_vis fn #set_field_name<'db>(self, __db: &'db mut #db_dyn_ty) -> salsa::setter::Setter<'db, #ident, #field_ty>
{ {
let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar_mut(__db); let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar_mut(__db);
@ -125,7 +125,7 @@ impl InputStruct {
let singleton = self.0.is_isingleton(); let singleton = self.0.is_isingleton();
let constructor: syn::ImplItemMethod = if singleton { let constructor: syn::ImplItemMethod = if singleton {
parse_quote! { parse_quote_spanned! { constructor_name.span() =>
/// Creates a new singleton input /// Creates a new singleton input
/// ///
/// # Panics /// # Panics
@ -143,7 +143,7 @@ impl InputStruct {
} }
} }
} else { } else {
parse_quote! { parse_quote_spanned! { constructor_name.span() =>
pub fn #constructor_name(__db: &#db_dyn_ty, #(#field_names: #field_tys,)*) -> Self pub fn #constructor_name(__db: &#db_dyn_ty, #(#field_names: #field_tys,)*) -> Self
{ {
let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(__db); let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(__db);
@ -177,6 +177,7 @@ impl InputStruct {
}; };
parse_quote! { parse_quote! {
#[allow(dead_code)]
impl #ident { impl #ident {
#constructor #constructor
@ -191,6 +192,7 @@ impl InputStruct {
} }
} else { } else {
parse_quote! { parse_quote! {
#[allow(dead_code)]
impl #ident { impl #ident {
#constructor #constructor

View file

@ -94,7 +94,7 @@ impl InternedStruct {
let field_vis = field.vis(); let field_vis = field.vis();
let field_get_name = field.get_name(); let field_get_name = field.get_name();
if field.is_clone_field() { if field.is_clone_field() {
parse_quote! { parse_quote_spanned! { field_get_name.span() =>
#field_vis fn #field_get_name(self, db: &#db_dyn_ty) -> #field_ty { #field_vis fn #field_get_name(self, db: &#db_dyn_ty) -> #field_ty {
let (jar, runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(db); let (jar, runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(db);
let ingredients = <#jar_ty as salsa::storage::HasIngredientsFor< #id_ident >>::ingredient(jar); let ingredients = <#jar_ty as salsa::storage::HasIngredientsFor< #id_ident >>::ingredient(jar);
@ -102,7 +102,7 @@ impl InternedStruct {
} }
} }
} else { } else {
parse_quote! { parse_quote_spanned! { field_get_name.span() =>
#field_vis fn #field_get_name<'db>(self, db: &'db #db_dyn_ty) -> &'db #field_ty { #field_vis fn #field_get_name<'db>(self, db: &'db #db_dyn_ty) -> &'db #field_ty {
let (jar, runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(db); let (jar, runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(db);
let ingredients = <#jar_ty as salsa::storage::HasIngredientsFor< #id_ident >>::ingredient(jar); let ingredients = <#jar_ty as salsa::storage::HasIngredientsFor< #id_ident >>::ingredient(jar);
@ -117,7 +117,7 @@ impl InternedStruct {
let field_tys = self.all_field_tys(); let field_tys = self.all_field_tys();
let data_ident = self.data_ident(); let data_ident = self.data_ident();
let constructor_name = self.constructor_name(); let constructor_name = self.constructor_name();
let new_method: syn::ImplItemMethod = parse_quote! { let new_method: syn::ImplItemMethod = parse_quote_spanned! { constructor_name.span() =>
#vis fn #constructor_name( #vis fn #constructor_name(
db: &#db_dyn_ty, db: &#db_dyn_ty,
#(#field_names: #field_tys,)* #(#field_names: #field_tys,)*
@ -131,6 +131,7 @@ impl InternedStruct {
}; };
parse_quote! { parse_quote! {
#[allow(dead_code)]
impl #id_ident { impl #id_ident {
#(#field_getters)* #(#field_getters)*

View file

@ -188,7 +188,7 @@ impl<A: AllowedOptions> SalsaStruct<A> {
.filter(|attr| !attr.path.is_ident("customize")) .filter(|attr| !attr.path.is_ident("customize"))
.collect(); .collect();
parse_quote! { parse_quote_spanned! { ident.span() =>
#(#attrs)* #(#attrs)*
#[derive(Copy, Clone, PartialEq, PartialOrd, Eq, Ord, Hash, Debug)] #[derive(Copy, Clone, PartialEq, PartialOrd, Eq, Ord, Hash, Debug)]
#visibility struct #ident(salsa::Id); #visibility struct #ident(salsa::Id);
@ -206,7 +206,7 @@ impl<A: AllowedOptions> SalsaStruct<A> {
let visibility = self.visibility(); let visibility = self.visibility();
let all_field_names = self.all_field_names(); let all_field_names = self.all_field_names();
let all_field_tys = self.all_field_tys(); let all_field_tys = self.all_field_tys();
parse_quote! { parse_quote_spanned! { ident.span() =>
/// Internal struct used for interned item /// Internal struct used for interned item
#[derive(Eq, PartialEq, Hash, Clone)] #[derive(Eq, PartialEq, Hash, Clone)]
#visibility struct #ident { #visibility struct #ident {
@ -226,14 +226,14 @@ impl<A: AllowedOptions> SalsaStruct<A> {
pub(crate) fn constructor_name(&self) -> syn::Ident { pub(crate) fn constructor_name(&self) -> syn::Ident {
match self.args.constructor_name.clone() { match self.args.constructor_name.clone() {
Some(name) => name, Some(name) => name,
None => Ident::new("new", Span::call_site()), None => Ident::new("new", self.id_ident().span()),
} }
} }
/// Generate `impl salsa::AsId for Foo` /// Generate `impl salsa::AsId for Foo`
pub(crate) fn as_id_impl(&self) -> syn::ItemImpl { pub(crate) fn as_id_impl(&self) -> syn::ItemImpl {
let ident = self.id_ident(); let ident = self.id_ident();
parse_quote! { parse_quote_spanned! { ident.span() =>
impl salsa::AsId for #ident { impl salsa::AsId for #ident {
fn as_id(self) -> salsa::Id { fn as_id(self) -> salsa::Id {
self.0 self.0
@ -352,8 +352,8 @@ impl SalsaField {
)); ));
} }
let get_name = Ident::new(&field_name_str, Span::call_site()); let get_name = Ident::new(&field_name_str, field_name.span());
let set_name = Ident::new(&format!("set_{}", field_name_str), Span::call_site()); let set_name = Ident::new(&format!("set_{}", field_name_str), field_name.span());
let mut result = SalsaField { let mut result = SalsaField {
field: field.clone(), field: field.clone(),
has_id_attr: false, has_id_attr: false,

View file

@ -168,7 +168,7 @@ impl TrackedStruct {
let field_clones: Vec<_> = self.all_fields().map(SalsaField::is_clone_field).collect(); let field_clones: Vec<_> = self.all_fields().map(SalsaField::is_clone_field).collect();
let field_getters: Vec<syn::ImplItemMethod> = field_indices.iter().zip(&field_get_names).zip(&field_tys).zip(&field_vises).zip(&field_clones).map(|((((field_index, field_get_name), field_ty), field_vis), is_clone_field)| let field_getters: Vec<syn::ImplItemMethod> = field_indices.iter().zip(&field_get_names).zip(&field_tys).zip(&field_vises).zip(&field_clones).map(|((((field_index, field_get_name), field_ty), field_vis), is_clone_field)|
if !*is_clone_field { if !*is_clone_field {
parse_quote! { parse_quote_spanned! { field_get_name.span() =>
#field_vis fn #field_get_name<'db>(self, __db: &'db #db_dyn_ty) -> &'db #field_ty #field_vis fn #field_get_name<'db>(self, __db: &'db #db_dyn_ty) -> &'db #field_ty
{ {
let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(__db); let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(__db);
@ -177,7 +177,7 @@ impl TrackedStruct {
} }
} }
} else { } else {
parse_quote! { parse_quote_spanned! { field_get_name.span() =>
#field_vis fn #field_get_name<'db>(self, __db: &'db #db_dyn_ty) -> #field_ty #field_vis fn #field_get_name<'db>(self, __db: &'db #db_dyn_ty) -> #field_ty
{ {
let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(__db); let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(__db);

View file

@ -1,8 +1,8 @@
error[E0624]: method `field` is private error[E0624]: method `field` is private
--> tests/compile-fail/get-set-on-private-field.rs:29:11 --> tests/compile-fail/get-set-on-private-field.rs:29:11
| |
7 | #[salsa::input(jar = Jar)] 9 | field: u32,
| -------------------------- private method defined here | ---------- private method defined here
... ...
29 | input.field(&db); 29 | input.field(&db);
| ^^^^^ private method | ^^^^^ private method
@ -10,8 +10,8 @@ error[E0624]: method `field` is private
error[E0624]: method `set_field` is private error[E0624]: method `set_field` is private
--> tests/compile-fail/get-set-on-private-field.rs:30:11 --> tests/compile-fail/get-set-on-private-field.rs:30:11
| |
7 | #[salsa::input(jar = Jar)] 9 | field: u32,
| -------------------------- private method defined here | ----- private method defined here
... ...
30 | input.set_field(&mut db).to(23); 30 | input.set_field(&mut db).to(23);
| ^^^^^^^^^ private method | ^^^^^^^^^ private method

View file

@ -1,8 +1,8 @@
error[E0599]: no method named `set_id_one` found for struct `MyInput` in the current scope error[E0599]: no method named `set_id_one` found for struct `MyInput` in the current scope
--> tests/compile-fail/input_struct_id_fields_no_setters.rs:30:11 --> tests/compile-fail/input_struct_id_fields_no_setters.rs:30:11
| |
7 | #[salsa::input(jar = Jar)] 8 | struct MyInput {
| -------------------------- method `set_id_one` not found for this struct | ------- method `set_id_one` not found for this struct
... ...
30 | input.set_id_one(1); 30 | input.set_id_one(1);
| ^^^^^^^^^^ help: there is a method with a similar name: `id_one` | ^^^^^^^^^^ help: there is a method with a similar name: `id_one`

View file

@ -0,0 +1,27 @@
#[salsa::jar(db = Db)]
pub struct Jar(MyInput);
pub trait Db: salsa::DbWithJar<Jar> {}
#[salsa::db(Jar)]
#[derive(Default)]
struct Database {
storage: salsa::Storage<Self>,
}
impl salsa::Database for Database {}
impl Db for Database {}
#[salsa::input]
pub struct MyInput {
field: u32,
}
fn main() {
let mut db = Database::default();
let input = MyInput::new(&mut db, 22);
input.field(&db);
input.set_field(22);
}

View file

@ -0,0 +1,18 @@
error[E0308]: mismatched types
--> tests/compile-fail/span-input-setter.rs:26:21
|
26 | input.set_field(22);
| --------- ^^ expected `&mut dyn Db`, found integer
| |
| arguments to this method are incorrect
|
= note: expected mutable reference `&mut dyn Db`
found type `{integer}`
note: method defined here
--> tests/compile-fail/span-input-setter.rs:18:5
|
16 | #[salsa::input]
| ---------------
17 | pub struct MyInput {
18 | field: u32,
| ^^^^^

View file

@ -0,0 +1,30 @@
#[salsa::jar(db = Db)]
pub struct Jar(MyTracked, my_fn);
pub trait Db: salsa::DbWithJar<Jar> {}
#[salsa::db(Jar)]
#[derive(Default)]
struct Database {
storage: salsa::Storage<Self>,
}
impl salsa::Database for Database {}
impl Db for Database {}
#[salsa::tracked]
pub struct MyTracked {
field: u32,
}
#[salsa::tracked]
fn my_fn(db: &dyn crate::Db) {
let x = MyTracked::new(db, 22);
x.field(22);
}
fn main() {
let mut db = Database::default();
my_fn(&db);
}

View file

@ -0,0 +1,18 @@
error[E0308]: mismatched types
--> tests/compile-fail/span-tracked-getter.rs:24:13
|
24 | x.field(22);
| ----- ^^ expected `&dyn Db`, found integer
| |
| arguments to this method are incorrect
|
= note: expected reference `&dyn Db`
found type `{integer}`
note: method defined here
--> tests/compile-fail/span-tracked-getter.rs:18:5
|
16 | #[salsa::tracked]
| -----------------
17 | pub struct MyTracked {
18 | field: u32,
| ^^^^^