rust-analyzer/crates/hir_expand/src/proc_macro.rs

144 lines
3.5 KiB
Rust
Raw Normal View History

2020-03-18 09:47:59 +00:00
//! Proc Macro Expander stub
2020-03-18 12:56:46 +00:00
use crate::{db::AstDatabase, LazyMacroId};
2020-08-13 14:25:38 +00:00
use base_db::{CrateId, ProcMacroId};
2020-04-21 17:44:21 +00:00
use tt::buffer::{Cursor, TokenBuffer};
2020-03-18 09:47:59 +00:00
#[derive(Debug, Clone, Copy, Eq, PartialEq, Hash)]
pub struct ProcMacroExpander {
krate: CrateId,
2020-03-18 12:56:46 +00:00
proc_macro_id: ProcMacroId,
2020-03-18 09:47:59 +00:00
}
2020-03-26 20:26:34 +00:00
macro_rules! err {
($fmt:literal, $($tt:tt),*) => {
mbe::ExpandError::ProcMacroError(tt::ExpansionError::Unknown(format!($fmt, $($tt),*)))
};
($fmt:literal) => {
mbe::ExpandError::ProcMacroError(tt::ExpansionError::Unknown($fmt.to_string()))
}
}
2020-03-18 09:47:59 +00:00
impl ProcMacroExpander {
2020-03-18 12:56:46 +00:00
pub fn new(krate: CrateId, proc_macro_id: ProcMacroId) -> ProcMacroExpander {
ProcMacroExpander { krate, proc_macro_id }
2020-03-18 09:47:59 +00:00
}
pub fn expand(
2020-05-26 18:12:13 +00:00
self,
2020-03-18 09:47:59 +00:00
db: &dyn AstDatabase,
2020-03-18 12:56:46 +00:00
_id: LazyMacroId,
tt: &tt::Subtree,
2020-03-18 09:47:59 +00:00
) -> Result<tt::Subtree, mbe::ExpandError> {
2020-03-18 12:56:46 +00:00
let krate_graph = db.crate_graph();
let proc_macro = krate_graph[self.krate]
.proc_macro
2020-03-26 16:41:44 +00:00
.get(self.proc_macro_id.0 as usize)
2020-03-18 12:56:46 +00:00
.clone()
2020-03-26 20:26:34 +00:00
.ok_or_else(|| err!("No derive macro found."))?;
2020-04-21 17:44:21 +00:00
let tt = remove_derive_attrs(tt)
2020-03-26 20:26:34 +00:00
.ok_or_else(|| err!("Fail to remove derive for custom derive"))?;
2020-03-26 16:41:44 +00:00
proc_macro.expander.expand(&tt, None).map_err(mbe::ExpandError::from)
2020-03-18 09:47:59 +00:00
}
}
2020-03-26 20:26:34 +00:00
2020-04-21 17:44:21 +00:00
fn eat_punct(cursor: &mut Cursor, c: char) -> bool {
if let Some(tt::TokenTree::Leaf(tt::Leaf::Punct(punct))) = cursor.token_tree() {
if punct.char == c {
*cursor = cursor.bump();
return true;
}
2020-03-26 20:26:34 +00:00
}
2020-04-21 17:44:21 +00:00
false
}
fn eat_subtree(cursor: &mut Cursor, kind: tt::DelimiterKind) -> bool {
if let Some(tt::TokenTree::Subtree(subtree)) = cursor.token_tree() {
if Some(kind) == subtree.delimiter_kind() {
*cursor = cursor.bump_subtree();
return true;
}
}
false
}
fn eat_ident(cursor: &mut Cursor, t: &str) -> bool {
if let Some(tt::TokenTree::Leaf(tt::Leaf::Ident(ident))) = cursor.token_tree() {
if t == ident.text.as_str() {
*cursor = cursor.bump();
return true;
}
}
false
}
fn remove_derive_attrs(tt: &tt::Subtree) -> Option<tt::Subtree> {
let buffer = TokenBuffer::new(&tt.token_trees);
let mut p = buffer.begin();
let mut result = tt::Subtree::default();
while !p.eof() {
let curr = p;
2020-03-26 20:26:34 +00:00
2020-04-21 17:44:21 +00:00
if eat_punct(&mut p, '#') {
eat_punct(&mut p, '!');
let parent = p;
if eat_subtree(&mut p, tt::DelimiterKind::Bracket) {
if eat_ident(&mut p, "derive") {
p = parent.bump();
continue;
}
}
}
result.token_trees.push(curr.token_tree()?.clone());
p = curr.bump();
}
Some(result)
}
#[cfg(test)]
2020-07-31 16:30:37 +00:00
mod tests {
2020-04-21 17:44:21 +00:00
use super::*;
use test_utils::assert_eq_text;
#[test]
fn test_remove_derive_attrs() {
let tt = mbe::parse_to_token_tree(
r#"
#[allow(unused)]
#[derive(Copy)]
#[derive(Hello)]
struct A {
bar: u32
}
"#,
)
.unwrap()
.0;
let result = format!("{:#?}", remove_derive_attrs(&tt).unwrap());
assert_eq_text!(
&result,
r#"
SUBTREE $
PUNCH # [alone] 0
SUBTREE [] 1
IDENT allow 2
SUBTREE () 3
IDENT unused 4
IDENT struct 15
IDENT A 16
SUBTREE {} 17
IDENT bar 18
PUNCH : [alone] 19
IDENT u32 20
"#
.trim()
);
}
2020-03-26 20:26:34 +00:00
}