mirror of
https://github.com/servo/servo.git
synced 2025-08-02 20:20:14 +01:00
* Upgrade rustc to 1.83 Signed-off-by: Nico Burns <nico@nicoburns.com> * Fix crown (change copied from linked clippy function) Signed-off-by: Nico Burns <nico@nicoburns.com> * Fix named lifetime lint Signed-off-by: Nico Burns <nico@nicoburns.com> * Bump shell.nix Signed-off-by: Nico Burns <nico@nicoburns.com> * Fix non-local impl warnings Signed-off-by: Nico Burns <nico@nicoburns.com> * Format with 1.83 formatting changes Signed-off-by: Nico Burns <nico@nicoburns.com> * Fix manual non-local impl Signed-off-by: Nico Burns <nico@nicoburns.com> * More fixes for crown Signed-off-by: Nico Burns <nico@nicoburns.com> * Fix tidy Signed-off-by: Nico Burns <nico@nicoburns.com> * Fix needless_return lints Signed-off-by: Nico Burns <nico@nicoburns.com> * Fix doc comment lint Signed-off-by: Nico Burns <nico@nicoburns.com> * Fix missing wait lint Signed-off-by: Nico Burns <nico@nicoburns.com> * Allow needless_lifetimes lint Signed-off-by: Nico Burns <nico@nicoburns.com> * more doc comments Signed-off-by: Nico Burns <nico@nicoburns.com> * More needless_returns Signed-off-by: Nico Burns <nico@nicoburns.com> * is_empty lint Signed-off-by: Nico Burns <nico@nicoburns.com> * Fix needless_lifetime lints Signed-off-by: Nico Burns <nico@nicoburns.com> * fix div_ceil lint Signed-off-by: Nico Burns <nico@nicoburns.com> * Allow non-minimal bool Signed-off-by: Nico Burns <nico@nicoburns.com> * Non-local impl in constellation Signed-off-by: Nico Burns <nico@nicoburns.com> * Missing wait in constellation Signed-off-by: Nico Burns <nico@nicoburns.com> * fmt Signed-off-by: Nico Burns <nico@nicoburns.com> * remove useless lints table Signed-off-by: Nico Burns <nico@nicoburns.com> * Fixup comments Signed-off-by: Nico Burns <nico@nicoburns.com> * Allow non-local definition in sandboxing code to simplify feature flagging Signed-off-by: Nico Burns <nico@nicoburns.com> * Remove wait calls and allow zombie_processes lint Signed-off-by: Nico Burns <nico@nicoburns.com> --------- Signed-off-by: Nico Burns <nico@nicoburns.com>
121 lines
4.3 KiB
Rust
121 lines
4.3 KiB
Rust
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at https://mozilla.org/MPL/2.0/. */
|
|
|
|
#![recursion_limit = "128"]
|
|
|
|
use quote::{quote, TokenStreamExt};
|
|
|
|
/// First field of DomObject must be either reflector or another dom_struct,
|
|
/// all other fields must not implement DomObject
|
|
#[proc_macro_derive(DomObject)]
|
|
pub fn expand_token_stream(input: proc_macro::TokenStream) -> proc_macro::TokenStream {
|
|
let input = syn::parse(input).unwrap();
|
|
expand_dom_object(input).into()
|
|
}
|
|
|
|
fn expand_dom_object(input: syn::DeriveInput) -> proc_macro2::TokenStream {
|
|
let fields = if let syn::Data::Struct(syn::DataStruct { ref fields, .. }) = input.data {
|
|
fields.iter().collect::<Vec<&syn::Field>>()
|
|
} else {
|
|
panic!("#[derive(DomObject)] should only be applied on proper structs")
|
|
};
|
|
|
|
let (first_field, fields) = fields
|
|
.split_first()
|
|
.expect("#[derive(DomObject)] should not be applied on empty structs");
|
|
|
|
let first_field_name = first_field.ident.as_ref().unwrap();
|
|
let mut field_types = vec![];
|
|
for field in fields {
|
|
if !field_types.contains(&&field.ty) {
|
|
field_types.push(&field.ty);
|
|
}
|
|
}
|
|
|
|
let name = &input.ident;
|
|
let (impl_generics, ty_generics, where_clause) = input.generics.split_for_impl();
|
|
let items = quote! {
|
|
impl #impl_generics ::js::conversions::ToJSValConvertible for #name #ty_generics #where_clause {
|
|
#[allow(unsafe_code)]
|
|
unsafe fn to_jsval(&self,
|
|
cx: *mut js::jsapi::JSContext,
|
|
rval: js::rust::MutableHandleValue) {
|
|
let object = crate::DomObject::reflector(self).get_jsobject();
|
|
object.to_jsval(cx, rval)
|
|
}
|
|
}
|
|
|
|
impl #impl_generics crate::DomObject for #name #ty_generics #where_clause {
|
|
#[inline]
|
|
fn reflector(&self) -> &crate::Reflector {
|
|
self.#first_field_name.reflector()
|
|
}
|
|
}
|
|
|
|
impl #impl_generics crate::MutDomObject for #name #ty_generics #where_clause {
|
|
unsafe fn init_reflector(&self, obj: *mut js::jsapi::JSObject) {
|
|
self.#first_field_name.init_reflector(obj);
|
|
}
|
|
}
|
|
|
|
impl #impl_generics Eq for #name #ty_generics #where_clause {}
|
|
|
|
impl #impl_generics PartialEq for #name #ty_generics #where_clause {
|
|
fn eq(&self, other: &Self) -> bool {
|
|
crate::DomObject::reflector(self) == crate::DomObject::reflector(other)
|
|
}
|
|
}
|
|
};
|
|
|
|
let mut params = proc_macro2::TokenStream::new();
|
|
params.append_separated(
|
|
input.generics.type_params().map(|param| ¶m.ident),
|
|
quote! {,},
|
|
);
|
|
|
|
let mut dummy_items = quote! {
|
|
// Generic trait with a blanket impl over `()` for all types.
|
|
// becomes ambiguous if impl
|
|
trait NoDomObjectInDomObject<A> {
|
|
// Required for actually being able to reference the trait.
|
|
fn some_item() {}
|
|
}
|
|
|
|
impl<T: ?Sized> NoDomObjectInDomObject<()> for T {}
|
|
|
|
// Used for the specialized impl when DomObject is implemented.
|
|
#[allow(dead_code)]
|
|
struct Invalid;
|
|
// forbids DomObject
|
|
impl<T> NoDomObjectInDomObject<Invalid> for T where T: ?Sized + crate::DomObject {}
|
|
};
|
|
|
|
dummy_items.append_all(field_types.iter().enumerate().map(|(i, ty)| {
|
|
let s = syn::Ident::new(&format!("S{i}"), proc_macro2::Span::call_site());
|
|
quote! {
|
|
struct #s<#params>(#params);
|
|
|
|
impl #impl_generics #s<#params> #where_clause {
|
|
fn f() {
|
|
// If there is only one specialized trait impl, type inference with
|
|
// `_` can be resolved and this can compile. Fails to compile if
|
|
// ty implements `NoDomObjectInDomObject<Invalid>`.
|
|
let _ = <#ty as NoDomObjectInDomObject<_>>::some_item;
|
|
}
|
|
}
|
|
}
|
|
}));
|
|
|
|
let dummy_const = syn::Ident::new(
|
|
&format!("_IMPL_DOMOBJECT_FOR_{}", name),
|
|
proc_macro2::Span::call_site(),
|
|
);
|
|
let tokens = quote! {
|
|
#[allow(non_upper_case_globals)]
|
|
const #dummy_const: () = { #dummy_items };
|
|
#items
|
|
};
|
|
|
|
tokens
|
|
}
|