lib.rs 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263
  1. #![deny(warnings)]
  2. extern crate proc_macro;
  3. #[macro_use]
  4. extern crate quote;
  5. extern crate core;
  6. extern crate proc_macro2;
  7. #[macro_use]
  8. extern crate syn;
  9. use proc_macro2::Span;
  10. use syn::{
  11. parse::{self, Parse},
  12. spanned::Spanned,
  13. FnArg, ItemFn, LitInt, LitStr, PathArguments, ReturnType, Type, Visibility,
  14. };
  15. use proc_macro::TokenStream;
  16. /// Attribute to declare the entry point of the program
  17. ///
  18. /// **IMPORTANT**: This attribute must appear exactly *once* in the dependency graph. Also, if you
  19. /// are using Rust 1.30 the attribute must be used on a reachable item (i.e. there must be no
  20. /// private modules between the item and the root of the crate); if the item is in the root of the
  21. /// crate you'll be fine. This reachability restriction doesn't apply to Rust 1.31 and newer releases.
  22. ///
  23. /// The specified function will be called by the reset handler *after* RAM has been initialized.
  24. /// If present, the FPU will also be enabled before the function is called.
  25. ///
  26. /// The type of the specified function must be `[unsafe] fn() -> !` (never ending function)
  27. ///
  28. /// # Properties
  29. ///
  30. /// The entry point will be called by the reset handler. The program can't reference to the entry
  31. /// point, much less invoke it.
  32. ///
  33. /// # Examples
  34. ///
  35. /// - Simple entry point
  36. ///
  37. /// ``` no_run
  38. /// # #![no_main]
  39. /// # use riscv_rt_macros::entry;
  40. /// #[entry]
  41. /// fn main() -> ! {
  42. /// loop {
  43. /// /* .. */
  44. /// }
  45. /// }
  46. /// ```
  47. #[proc_macro_attribute]
  48. pub fn entry(args: TokenStream, input: TokenStream) -> TokenStream {
  49. let f = parse_macro_input!(input as ItemFn);
  50. // check the function arguments
  51. if f.sig.inputs.len() > 3 {
  52. return parse::Error::new(
  53. f.sig.inputs.last().unwrap().span(),
  54. "`#[entry]` function has too many arguments",
  55. )
  56. .to_compile_error()
  57. .into();
  58. }
  59. for arg in &f.sig.inputs {
  60. match arg {
  61. FnArg::Receiver(_) => {
  62. return parse::Error::new(arg.span(), "invalid argument")
  63. .to_compile_error()
  64. .into();
  65. }
  66. FnArg::Typed(t) => {
  67. if !is_simple_type(&t.ty, "usize") {
  68. return parse::Error::new(t.ty.span(), "argument type must be usize")
  69. .to_compile_error()
  70. .into();
  71. }
  72. }
  73. }
  74. }
  75. // check the function signature
  76. let valid_signature = f.sig.constness.is_none()
  77. && f.sig.asyncness.is_none()
  78. && f.vis == Visibility::Inherited
  79. && f.sig.abi.is_none()
  80. && f.sig.generics.params.is_empty()
  81. && f.sig.generics.where_clause.is_none()
  82. && f.sig.variadic.is_none()
  83. && match f.sig.output {
  84. ReturnType::Default => false,
  85. ReturnType::Type(_, ref ty) => matches!(**ty, Type::Never(_)),
  86. };
  87. if !valid_signature {
  88. return parse::Error::new(
  89. f.span(),
  90. "`#[entry]` function must have signature `[unsafe] fn([arg0: usize, ...]) -> !`",
  91. )
  92. .to_compile_error()
  93. .into();
  94. }
  95. if !args.is_empty() {
  96. return parse::Error::new(Span::call_site(), "This attribute accepts no arguments")
  97. .to_compile_error()
  98. .into();
  99. }
  100. // XXX should we blacklist other attributes?
  101. let attrs = f.attrs;
  102. let unsafety = f.sig.unsafety;
  103. let args = f.sig.inputs;
  104. let stmts = f.block.stmts;
  105. quote!(
  106. #[allow(non_snake_case)]
  107. #[export_name = "main"]
  108. #(#attrs)*
  109. pub #unsafety fn __risc_v_rt__main(#args) -> ! {
  110. #(#stmts)*
  111. }
  112. )
  113. .into()
  114. }
  115. #[allow(unused)]
  116. fn is_simple_type(ty: &Type, name: &str) -> bool {
  117. if let Type::Path(p) = ty {
  118. if p.qself.is_none() && p.path.leading_colon.is_none() && p.path.segments.len() == 1 {
  119. let segment = p.path.segments.first().unwrap();
  120. if segment.ident == name && segment.arguments == PathArguments::None {
  121. return true;
  122. }
  123. }
  124. }
  125. false
  126. }
  127. /// Attribute to mark which function will be called at the beginning of the reset handler.
  128. ///
  129. /// **IMPORTANT**: This attribute can appear at most *once* in the dependency graph. Also, if you
  130. /// are using Rust 1.30 the attribute must be used on a reachable item (i.e. there must be no
  131. /// private modules between the item and the root of the crate); if the item is in the root of the
  132. /// crate you'll be fine. This reachability restriction doesn't apply to Rust 1.31 and newer
  133. /// releases.
  134. ///
  135. /// The function must have the signature of `unsafe fn()`.
  136. ///
  137. /// The function passed will be called before static variables are initialized. Any access of static
  138. /// variables will result in undefined behavior.
  139. ///
  140. /// # Examples
  141. ///
  142. /// ```
  143. /// # use riscv_rt_macros::pre_init;
  144. /// #[pre_init]
  145. /// unsafe fn before_main() {
  146. /// // do something here
  147. /// }
  148. ///
  149. /// # fn main() {}
  150. /// ```
  151. #[proc_macro_attribute]
  152. pub fn pre_init(args: TokenStream, input: TokenStream) -> TokenStream {
  153. let f = parse_macro_input!(input as ItemFn);
  154. // check the function signature
  155. let valid_signature = f.sig.constness.is_none()
  156. && f.sig.asyncness.is_none()
  157. && f.vis == Visibility::Inherited
  158. && f.sig.unsafety.is_some()
  159. && f.sig.abi.is_none()
  160. && f.sig.inputs.is_empty()
  161. && f.sig.generics.params.is_empty()
  162. && f.sig.generics.where_clause.is_none()
  163. && f.sig.variadic.is_none()
  164. && match f.sig.output {
  165. ReturnType::Default => true,
  166. ReturnType::Type(_, ref ty) => match **ty {
  167. Type::Tuple(ref tuple) => tuple.elems.is_empty(),
  168. _ => false,
  169. },
  170. };
  171. if !valid_signature {
  172. return parse::Error::new(
  173. f.span(),
  174. "`#[pre_init]` function must have signature `unsafe fn()`",
  175. )
  176. .to_compile_error()
  177. .into();
  178. }
  179. if !args.is_empty() {
  180. return parse::Error::new(Span::call_site(), "This attribute accepts no arguments")
  181. .to_compile_error()
  182. .into();
  183. }
  184. // XXX should we blacklist other attributes?
  185. let attrs = f.attrs;
  186. let ident = f.sig.ident;
  187. let block = f.block;
  188. quote!(
  189. #[export_name = "__pre_init"]
  190. #(#attrs)*
  191. pub unsafe fn #ident() #block
  192. )
  193. .into()
  194. }
  195. struct AsmLoopArgs {
  196. asm_template: String,
  197. count: usize,
  198. }
  199. impl Parse for AsmLoopArgs {
  200. fn parse(input: syn::parse::ParseStream) -> syn::Result<Self> {
  201. let template: LitStr = input.parse().unwrap();
  202. _ = input.parse::<Token![,]>().unwrap();
  203. let count: LitInt = input.parse().unwrap();
  204. Ok(Self {
  205. asm_template: template.value(),
  206. count: count.base10_parse().unwrap(),
  207. })
  208. }
  209. }
  210. /// Loops an asm expression n times.
  211. ///
  212. /// `loop_asm!` takes 2 arguments, the first is a string literal and the second is a number literal
  213. /// See [the formatting syntax documentation in `std::fmt`](../std/fmt/index.html)
  214. /// for details.
  215. ///
  216. /// Argument 1 is an assembly expression, all "{}" in this assembly expression will be replaced with the
  217. /// current loop index.
  218. ///
  219. /// Argument 2 is the number of loops to do with the provided expression.
  220. ///
  221. /// # Examples
  222. ///
  223. /// ```
  224. /// # use riscv_rt_macros::loop_asm;
  225. /// unsafe {
  226. /// loop_asm!("fmv.w.x f{}, x0", 32); // => core::arch::asm!("fmv.w.x f0, x0") ... core::arch::asm!("fmv.w.x f31, x0")
  227. /// }
  228. /// ```
  229. #[proc_macro]
  230. pub fn loop_asm(input: TokenStream) -> TokenStream {
  231. let args = parse_macro_input!(input as AsmLoopArgs);
  232. let tokens = (0..args.count)
  233. .map(|i| {
  234. let i = i.to_string();
  235. let asm = args.asm_template.replace("{}", &i);
  236. format!("core::arch::asm!(\"{}\");", asm)
  237. })
  238. .collect::<Vec<String>>()
  239. .join("\n");
  240. tokens.parse().unwrap()
  241. }