1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
use proc_macro::TokenStream;
use quote::format_ident;
use quote::quote;
use sh_builtin_common_util::attr_doc::parse_doc_comments;
use sh_builtin_common_util::attr_shell_builtin::parse_named_function;
use sh_builtin_common_util::common::ShellFunctionInfo;
use sh_builtin_common_util::tokens::CString;
use syn::parse::{Parse, ParseStream};
use syn::{parse_macro_input, AttributeArgs, Ident, ItemFn, Result};
#[derive(Clone, Debug)]
struct BashFunction {
fn_ident: Ident,
doc: Vec<String>,
inner: ItemFn,
}
impl ShellFunctionInfo for BashFunction {
fn doc_comments(&self) -> &Vec<String> {
&self.doc
}
fn name(&self) -> String {
self.fn_ident.to_string()
}
}
impl Parse for BashFunction {
fn parse(input: ParseStream) -> Result<Self> {
let inner = ItemFn::parse(input)?;
Ok(BashFunction {
fn_ident: inner.sig.ident.clone(),
doc: parse_doc_comments(inner.attrs.iter()),
inner,
})
}
}
#[proc_macro_attribute]
pub fn bash_builtin(args: TokenStream, input: TokenStream) -> TokenStream {
let args = parse_macro_input!(args as AttributeArgs);
let input = parse_macro_input!(input as BashFunction);
let inner = &input.inner;
let export_symbol_name = parse_named_function(&args).unwrap_or_else(|| input.name());
if !export_symbol_name
.chars()
.all(|c| c == '_' || c == ':' || c == '-' || c.is_ascii_alphanumeric())
{
let error = format!("'{}' is not a valid bash function name", export_symbol_name);
return TokenStream::from(quote! {
compile_error!(#error)
});
}
let func_wrapper = format_ident!("__bashexport__{}__wrapper", input.fn_ident);
let func = &input.inner.sig.ident;
let descriptor = format_ident!("__bashexport__{}__descriptor", input.fn_ident);
let descriptor_symbol = format!("{}_struct", export_symbol_name);
let descriptor_field_usage = CString(input.usage());
let descriptor_field_name = CString(export_symbol_name.clone());
TokenStream::from(quote! {
#[allow(non_upper_case_globals, non_snake_case)]
#[export_name = #descriptor_symbol]
#[link_section = ".data,shell_builtin"]
pub static #descriptor: ::sh_builtin_bash::__internal::builtin_descriptor = ::sh_builtin_bash::__internal::builtin_descriptor {
name: (#descriptor_field_name) as _,
function: Some(#func_wrapper as _),
flags: ::sh_builtin_bash::__internal::BUILTIN_ENABLED as i32,
long_doc: 0 as _,
short_doc: (#descriptor_field_usage) as _,
handle: 0 as *mut i8,
};
#inner
#[allow(non_snake_case)]
pub unsafe extern "C" fn #func_wrapper(words: *mut ::sh_builtin_bash::__internal::WORD_LIST) -> ::std::os::raw::c_int {
let args: Vec<String> = if words.is_null() {
vec![]
} else {
(&unsafe { *words }).into()
};
match #func(&args) {
Ok(()) => ::sh_builtin_bash::__internal::EXECUTION_SUCCESS as _,
Err(err) => {
eprintln!("{}", err);
::sh_builtin_bash::__internal::EXECUTION_FAILURE as _
}
}
}
})
}