macros/
helpers.rs

1// SPDX-License-Identifier: GPL-2.0
2
3use proc_macro::{token_stream, Group, Ident, TokenStream, TokenTree};
4
5pub(crate) fn try_ident(it: &mut token_stream::IntoIter) -> Option<String> {
6    if let Some(TokenTree::Ident(ident)) = it.next() {
7        Some(ident.to_string())
8    } else {
9        None
10    }
11}
12
13pub(crate) fn try_literal(it: &mut token_stream::IntoIter) -> Option<String> {
14    if let Some(TokenTree::Literal(literal)) = it.next() {
15        Some(literal.to_string())
16    } else {
17        None
18    }
19}
20
21pub(crate) fn try_string(it: &mut token_stream::IntoIter) -> Option<String> {
22    try_literal(it).and_then(|string| {
23        if string.starts_with('\"') && string.ends_with('\"') {
24            let content = &string[1..string.len() - 1];
25            if content.contains('\\') {
26                panic!("Escape sequences in string literals not yet handled");
27            }
28            Some(content.to_string())
29        } else if string.starts_with("r\"") {
30            panic!("Raw string literals are not yet handled");
31        } else {
32            None
33        }
34    })
35}
36
37pub(crate) fn expect_ident(it: &mut token_stream::IntoIter) -> String {
38    try_ident(it).expect("Expected Ident")
39}
40
41pub(crate) fn expect_punct(it: &mut token_stream::IntoIter) -> char {
42    if let TokenTree::Punct(punct) = it.next().expect("Reached end of token stream for Punct") {
43        punct.as_char()
44    } else {
45        panic!("Expected Punct");
46    }
47}
48
49pub(crate) fn expect_string(it: &mut token_stream::IntoIter) -> String {
50    try_string(it).expect("Expected string")
51}
52
53pub(crate) fn expect_string_ascii(it: &mut token_stream::IntoIter) -> String {
54    let string = try_string(it).expect("Expected string");
55    assert!(string.is_ascii(), "Expected ASCII string");
56    string
57}
58
59pub(crate) fn expect_group(it: &mut token_stream::IntoIter) -> Group {
60    if let TokenTree::Group(group) = it.next().expect("Reached end of token stream for Group") {
61        group
62    } else {
63        panic!("Expected Group");
64    }
65}
66
67pub(crate) fn expect_end(it: &mut token_stream::IntoIter) {
68    if it.next().is_some() {
69        panic!("Expected end");
70    }
71}
72
73/// Given a function declaration, finds the name of the function.
74pub(crate) fn function_name(input: TokenStream) -> Option<Ident> {
75    let mut input = input.into_iter();
76    while let Some(token) = input.next() {
77        match token {
78            TokenTree::Ident(i) if i.to_string() == "fn" => {
79                if let Some(TokenTree::Ident(i)) = input.next() {
80                    return Some(i);
81                }
82                return None;
83            }
84            _ => continue,
85        }
86    }
87    None
88}