mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-16 23:24:03 +00:00
Auto merge of #14572 - Veykril:proc-macro-close-span, r=Veykril
Encode closing delimiter span in FlatTrees Mainly serves as a test for the api versioning, as I don't think we make use of the closing span yet.
This commit is contained in:
commit
c479804d79
5 changed files with 68 additions and 26 deletions
|
@ -146,15 +146,16 @@ impl ProcMacro {
|
||||||
attr: Option<&tt::Subtree>,
|
attr: Option<&tt::Subtree>,
|
||||||
env: Vec<(String, String)>,
|
env: Vec<(String, String)>,
|
||||||
) -> Result<Result<tt::Subtree, PanicMessage>, ServerError> {
|
) -> Result<Result<tt::Subtree, PanicMessage>, ServerError> {
|
||||||
|
let version = self.process.lock().unwrap_or_else(|e| e.into_inner()).version();
|
||||||
let current_dir = env
|
let current_dir = env
|
||||||
.iter()
|
.iter()
|
||||||
.find(|(name, _)| name == "CARGO_MANIFEST_DIR")
|
.find(|(name, _)| name == "CARGO_MANIFEST_DIR")
|
||||||
.map(|(_, value)| value.clone());
|
.map(|(_, value)| value.clone());
|
||||||
|
|
||||||
let task = ExpandMacro {
|
let task = ExpandMacro {
|
||||||
macro_body: FlatTree::new(subtree),
|
macro_body: FlatTree::new(subtree, version),
|
||||||
macro_name: self.name.to_string(),
|
macro_name: self.name.to_string(),
|
||||||
attributes: attr.map(FlatTree::new),
|
attributes: attr.map(|subtree| FlatTree::new(subtree, version)),
|
||||||
lib: self.dylib_path.to_path_buf().into(),
|
lib: self.dylib_path.to_path_buf().into(),
|
||||||
env,
|
env,
|
||||||
current_dir,
|
current_dir,
|
||||||
|
@ -163,7 +164,9 @@ impl ProcMacro {
|
||||||
let request = msg::Request::ExpandMacro(task);
|
let request = msg::Request::ExpandMacro(task);
|
||||||
let response = self.process.lock().unwrap_or_else(|e| e.into_inner()).send_task(request)?;
|
let response = self.process.lock().unwrap_or_else(|e| e.into_inner()).send_task(request)?;
|
||||||
match response {
|
match response {
|
||||||
msg::Response::ExpandMacro(it) => Ok(it.map(FlatTree::to_subtree)),
|
msg::Response::ExpandMacro(it) => {
|
||||||
|
Ok(it.map(|tree| FlatTree::to_subtree(tree, version)))
|
||||||
|
}
|
||||||
msg::Response::ListMacros(..) | msg::Response::ApiVersionCheck(..) => {
|
msg::Response::ListMacros(..) | msg::Response::ApiVersionCheck(..) => {
|
||||||
Err(ServerError { message: "unexpected response".to_string(), io: None })
|
Err(ServerError { message: "unexpected response".to_string(), io: None })
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,8 +12,12 @@ use crate::ProcMacroKind;
|
||||||
|
|
||||||
pub use crate::msg::flat::FlatTree;
|
pub use crate::msg::flat::FlatTree;
|
||||||
|
|
||||||
|
// The versions of the server protocol
|
||||||
pub const NO_VERSION_CHECK_VERSION: u32 = 0;
|
pub const NO_VERSION_CHECK_VERSION: u32 = 0;
|
||||||
pub const CURRENT_API_VERSION: u32 = 1;
|
pub const VERSION_CHECK_VERSION: u32 = 1;
|
||||||
|
pub const ENCODE_CLOSE_SPAN_VERSION: u32 = 2;
|
||||||
|
|
||||||
|
pub const CURRENT_API_VERSION: u32 = ENCODE_CLOSE_SPAN_VERSION;
|
||||||
|
|
||||||
#[derive(Debug, Serialize, Deserialize)]
|
#[derive(Debug, Serialize, Deserialize)]
|
||||||
pub enum Request {
|
pub enum Request {
|
||||||
|
@ -146,7 +150,7 @@ mod tests {
|
||||||
fn test_proc_macro_rpc_works() {
|
fn test_proc_macro_rpc_works() {
|
||||||
let tt = fixture_token_tree();
|
let tt = fixture_token_tree();
|
||||||
let task = ExpandMacro {
|
let task = ExpandMacro {
|
||||||
macro_body: FlatTree::new(&tt),
|
macro_body: FlatTree::new(&tt, CURRENT_API_VERSION),
|
||||||
macro_name: Default::default(),
|
macro_name: Default::default(),
|
||||||
attributes: None,
|
attributes: None,
|
||||||
lib: std::env::current_dir().unwrap(),
|
lib: std::env::current_dir().unwrap(),
|
||||||
|
@ -158,6 +162,6 @@ mod tests {
|
||||||
// println!("{}", json);
|
// println!("{}", json);
|
||||||
let back: ExpandMacro = serde_json::from_str(&json).unwrap();
|
let back: ExpandMacro = serde_json::from_str(&json).unwrap();
|
||||||
|
|
||||||
assert_eq!(tt, back.macro_body.to_subtree());
|
assert_eq!(tt, back.macro_body.to_subtree(CURRENT_API_VERSION));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -39,7 +39,10 @@ use std::collections::{HashMap, VecDeque};
|
||||||
|
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
use crate::tt::{self, TokenId};
|
use crate::{
|
||||||
|
msg::ENCODE_CLOSE_SPAN_VERSION,
|
||||||
|
tt::{self, TokenId},
|
||||||
|
};
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize, Debug)]
|
#[derive(Serialize, Deserialize, Debug)]
|
||||||
pub struct FlatTree {
|
pub struct FlatTree {
|
||||||
|
@ -52,7 +55,8 @@ pub struct FlatTree {
|
||||||
}
|
}
|
||||||
|
|
||||||
struct SubtreeRepr {
|
struct SubtreeRepr {
|
||||||
id: tt::TokenId,
|
open: tt::TokenId,
|
||||||
|
close: tt::TokenId,
|
||||||
kind: tt::DelimiterKind,
|
kind: tt::DelimiterKind,
|
||||||
tt: [u32; 2],
|
tt: [u32; 2],
|
||||||
}
|
}
|
||||||
|
@ -74,7 +78,7 @@ struct IdentRepr {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl FlatTree {
|
impl FlatTree {
|
||||||
pub fn new(subtree: &tt::Subtree) -> FlatTree {
|
pub fn new(subtree: &tt::Subtree, version: u32) -> FlatTree {
|
||||||
let mut w = Writer {
|
let mut w = Writer {
|
||||||
string_table: HashMap::new(),
|
string_table: HashMap::new(),
|
||||||
work: VecDeque::new(),
|
work: VecDeque::new(),
|
||||||
|
@ -89,7 +93,11 @@ impl FlatTree {
|
||||||
w.write(subtree);
|
w.write(subtree);
|
||||||
|
|
||||||
return FlatTree {
|
return FlatTree {
|
||||||
subtree: write_vec(w.subtree, SubtreeRepr::write),
|
subtree: if version >= ENCODE_CLOSE_SPAN_VERSION {
|
||||||
|
write_vec(w.subtree, SubtreeRepr::write_with_close_span)
|
||||||
|
} else {
|
||||||
|
write_vec(w.subtree, SubtreeRepr::write)
|
||||||
|
},
|
||||||
literal: write_vec(w.literal, LiteralRepr::write),
|
literal: write_vec(w.literal, LiteralRepr::write),
|
||||||
punct: write_vec(w.punct, PunctRepr::write),
|
punct: write_vec(w.punct, PunctRepr::write),
|
||||||
ident: write_vec(w.ident, IdentRepr::write),
|
ident: write_vec(w.ident, IdentRepr::write),
|
||||||
|
@ -102,9 +110,13 @@ impl FlatTree {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn to_subtree(self) -> tt::Subtree {
|
pub fn to_subtree(self, version: u32) -> tt::Subtree {
|
||||||
return Reader {
|
return Reader {
|
||||||
subtree: read_vec(self.subtree, SubtreeRepr::read),
|
subtree: if version >= ENCODE_CLOSE_SPAN_VERSION {
|
||||||
|
read_vec(self.subtree, SubtreeRepr::read_with_close_span)
|
||||||
|
} else {
|
||||||
|
read_vec(self.subtree, SubtreeRepr::read)
|
||||||
|
},
|
||||||
literal: read_vec(self.literal, LiteralRepr::read),
|
literal: read_vec(self.literal, LiteralRepr::read),
|
||||||
punct: read_vec(self.punct, PunctRepr::read),
|
punct: read_vec(self.punct, PunctRepr::read),
|
||||||
ident: read_vec(self.ident, IdentRepr::read),
|
ident: read_vec(self.ident, IdentRepr::read),
|
||||||
|
@ -130,9 +142,9 @@ impl SubtreeRepr {
|
||||||
tt::DelimiterKind::Brace => 2,
|
tt::DelimiterKind::Brace => 2,
|
||||||
tt::DelimiterKind::Bracket => 3,
|
tt::DelimiterKind::Bracket => 3,
|
||||||
};
|
};
|
||||||
[self.id.0, kind, self.tt[0], self.tt[1]]
|
[self.open.0, kind, self.tt[0], self.tt[1]]
|
||||||
}
|
}
|
||||||
fn read([id, kind, lo, len]: [u32; 4]) -> SubtreeRepr {
|
fn read([open, kind, lo, len]: [u32; 4]) -> SubtreeRepr {
|
||||||
let kind = match kind {
|
let kind = match kind {
|
||||||
0 => tt::DelimiterKind::Invisible,
|
0 => tt::DelimiterKind::Invisible,
|
||||||
1 => tt::DelimiterKind::Parenthesis,
|
1 => tt::DelimiterKind::Parenthesis,
|
||||||
|
@ -140,7 +152,26 @@ impl SubtreeRepr {
|
||||||
3 => tt::DelimiterKind::Bracket,
|
3 => tt::DelimiterKind::Bracket,
|
||||||
other => panic!("bad kind {other}"),
|
other => panic!("bad kind {other}"),
|
||||||
};
|
};
|
||||||
SubtreeRepr { id: TokenId(id), kind, tt: [lo, len] }
|
SubtreeRepr { open: TokenId(open), close: TokenId::UNSPECIFIED, kind, tt: [lo, len] }
|
||||||
|
}
|
||||||
|
fn write_with_close_span(self) -> [u32; 5] {
|
||||||
|
let kind = match self.kind {
|
||||||
|
tt::DelimiterKind::Invisible => 0,
|
||||||
|
tt::DelimiterKind::Parenthesis => 1,
|
||||||
|
tt::DelimiterKind::Brace => 2,
|
||||||
|
tt::DelimiterKind::Bracket => 3,
|
||||||
|
};
|
||||||
|
[self.open.0, self.close.0, kind, self.tt[0], self.tt[1]]
|
||||||
|
}
|
||||||
|
fn read_with_close_span([open, close, kind, lo, len]: [u32; 5]) -> SubtreeRepr {
|
||||||
|
let kind = match kind {
|
||||||
|
0 => tt::DelimiterKind::Invisible,
|
||||||
|
1 => tt::DelimiterKind::Parenthesis,
|
||||||
|
2 => tt::DelimiterKind::Brace,
|
||||||
|
3 => tt::DelimiterKind::Bracket,
|
||||||
|
other => panic!("bad kind {other}"),
|
||||||
|
};
|
||||||
|
SubtreeRepr { open: TokenId(open), close: TokenId(close), kind, tt: [lo, len] }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -244,9 +275,10 @@ impl<'a> Writer<'a> {
|
||||||
|
|
||||||
fn enqueue(&mut self, subtree: &'a tt::Subtree) -> u32 {
|
fn enqueue(&mut self, subtree: &'a tt::Subtree) -> u32 {
|
||||||
let idx = self.subtree.len();
|
let idx = self.subtree.len();
|
||||||
let delimiter_id = subtree.delimiter.open;
|
let open = subtree.delimiter.open;
|
||||||
|
let close = subtree.delimiter.close;
|
||||||
let delimiter_kind = subtree.delimiter.kind;
|
let delimiter_kind = subtree.delimiter.kind;
|
||||||
self.subtree.push(SubtreeRepr { id: delimiter_id, kind: delimiter_kind, tt: [!0, !0] });
|
self.subtree.push(SubtreeRepr { open, close, kind: delimiter_kind, tt: [!0, !0] });
|
||||||
self.work.push_back((idx, subtree));
|
self.work.push_back((idx, subtree));
|
||||||
idx as u32
|
idx as u32
|
||||||
}
|
}
|
||||||
|
@ -277,11 +309,7 @@ impl Reader {
|
||||||
let repr = &self.subtree[i];
|
let repr = &self.subtree[i];
|
||||||
let token_trees = &self.token_tree[repr.tt[0] as usize..repr.tt[1] as usize];
|
let token_trees = &self.token_tree[repr.tt[0] as usize..repr.tt[1] as usize];
|
||||||
let s = tt::Subtree {
|
let s = tt::Subtree {
|
||||||
delimiter: tt::Delimiter {
|
delimiter: tt::Delimiter { open: repr.open, close: repr.close, kind: repr.kind },
|
||||||
open: repr.id,
|
|
||||||
close: TokenId::UNSPECIFIED,
|
|
||||||
kind: repr.kind,
|
|
||||||
},
|
|
||||||
token_trees: token_trees
|
token_trees: token_trees
|
||||||
.iter()
|
.iter()
|
||||||
.copied()
|
.copied()
|
||||||
|
|
|
@ -56,6 +56,10 @@ impl ProcMacroProcessSrv {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(crate) fn version(&self) -> u32 {
|
||||||
|
self.version
|
||||||
|
}
|
||||||
|
|
||||||
pub(crate) fn version_check(&mut self) -> Result<u32, ServerError> {
|
pub(crate) fn version_check(&mut self) -> Result<u32, ServerError> {
|
||||||
let request = Request::ApiVersionCheck {};
|
let request = Request::ApiVersionCheck {};
|
||||||
let response = self.send_task(request)?;
|
let response = self.send_task(request)?;
|
||||||
|
|
|
@ -31,7 +31,10 @@ use std::{
|
||||||
time::SystemTime,
|
time::SystemTime,
|
||||||
};
|
};
|
||||||
|
|
||||||
use proc_macro_api::{msg, ProcMacroKind};
|
use proc_macro_api::{
|
||||||
|
msg::{self, CURRENT_API_VERSION},
|
||||||
|
ProcMacroKind,
|
||||||
|
};
|
||||||
|
|
||||||
use ::tt::token_id as tt;
|
use ::tt::token_id as tt;
|
||||||
|
|
||||||
|
@ -67,8 +70,8 @@ impl ProcMacroSrv {
|
||||||
None => None,
|
None => None,
|
||||||
};
|
};
|
||||||
|
|
||||||
let macro_body = task.macro_body.to_subtree();
|
let macro_body = task.macro_body.to_subtree(CURRENT_API_VERSION);
|
||||||
let attributes = task.attributes.map(|it| it.to_subtree());
|
let attributes = task.attributes.map(|it| it.to_subtree(CURRENT_API_VERSION));
|
||||||
let result = thread::scope(|s| {
|
let result = thread::scope(|s| {
|
||||||
let thread = thread::Builder::new()
|
let thread = thread::Builder::new()
|
||||||
.stack_size(EXPANDER_STACK_SIZE)
|
.stack_size(EXPANDER_STACK_SIZE)
|
||||||
|
@ -76,7 +79,7 @@ impl ProcMacroSrv {
|
||||||
.spawn_scoped(s, || {
|
.spawn_scoped(s, || {
|
||||||
expander
|
expander
|
||||||
.expand(&task.macro_name, ¯o_body, attributes.as_ref())
|
.expand(&task.macro_name, ¯o_body, attributes.as_ref())
|
||||||
.map(|it| msg::FlatTree::new(&it))
|
.map(|it| msg::FlatTree::new(&it, CURRENT_API_VERSION))
|
||||||
});
|
});
|
||||||
let res = match thread {
|
let res = match thread {
|
||||||
Ok(handle) => handle.join(),
|
Ok(handle) => handle.join(),
|
||||||
|
|
Loading…
Reference in a new issue