lib.rs 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352
  1. #![no_std]
  2. use core::{
  3. net::{IpAddr, Ipv4Addr, Ipv6Addr},
  4. num::{NonZeroUsize, TryFromIntError},
  5. };
  6. use num_enum::IntoPrimitive;
  7. pub const LOG_BUF_CAPACITY: usize = 8192;
  8. pub const LOG_FIELDS: usize = 6;
  9. pub type LogValueLength = u16;
  10. #[repr(u8)]
  11. #[derive(Copy, Clone, Eq, PartialEq, Debug, Hash, IntoPrimitive)]
  12. pub enum Level {
  13. /// The "error" level.
  14. ///
  15. /// Designates very serious errors.
  16. Error = 1,
  17. /// The "warn" level.
  18. ///
  19. /// Designates hazardous situations.
  20. Warn,
  21. /// The "info" level.
  22. ///
  23. /// Designates useful information.
  24. Info,
  25. /// The "debug" level.
  26. ///
  27. /// Designates lower priority information.
  28. Debug,
  29. /// The "trace" level.
  30. ///
  31. /// Designates very low priority, often extremely verbose, information.
  32. Trace,
  33. }
  34. macro_rules! impl_formatter_for_types {
  35. ($trait:path : { $($type:ty),*}) => {
  36. $(
  37. impl $trait for $type {}
  38. )*
  39. };
  40. }
  41. pub trait DefaultFormatter {}
  42. impl_formatter_for_types!(
  43. DefaultFormatter: {
  44. bool,
  45. i8, i16, i32, i64, isize,
  46. u8, u16, u32, u64, usize,
  47. f32, f64,
  48. char,
  49. str,
  50. &str,
  51. IpAddr, Ipv4Addr, Ipv6Addr
  52. }
  53. );
  54. pub trait LowerHexFormatter {}
  55. impl_formatter_for_types!(
  56. LowerHexFormatter: {
  57. i8, i16, i32, i64, isize,
  58. u8, u16, u32, u64, usize,
  59. &[u8]
  60. }
  61. );
  62. pub trait UpperHexFormatter {}
  63. impl_formatter_for_types!(
  64. UpperHexFormatter: {
  65. i8, i16, i32, i64, isize,
  66. u8, u16, u32, u64, usize,
  67. &[u8]
  68. }
  69. );
  70. pub trait IpFormatter {}
  71. impl IpFormatter for IpAddr {}
  72. impl IpFormatter for Ipv4Addr {}
  73. impl IpFormatter for Ipv6Addr {}
  74. impl IpFormatter for u32 {}
  75. impl IpFormatter for [u8; 4] {}
  76. impl IpFormatter for [u8; 16] {}
  77. impl IpFormatter for [u16; 8] {}
  78. pub trait LowerMacFormatter {}
  79. impl LowerMacFormatter for [u8; 6] {}
  80. pub trait UpperMacFormatter {}
  81. impl UpperMacFormatter for [u8; 6] {}
  82. #[repr(u8)]
  83. #[derive(Copy, Clone, Debug, IntoPrimitive)]
  84. pub enum RecordField {
  85. Target = 1,
  86. Level,
  87. Module,
  88. File,
  89. Line,
  90. NumArgs,
  91. }
  92. /// Types which are supported by aya-log and can be safely sent from eBPF
  93. /// programs to userspace.
  94. #[repr(u8)]
  95. #[derive(Copy, Clone, Debug, IntoPrimitive)]
  96. pub enum Argument {
  97. DisplayHint,
  98. I8,
  99. I16,
  100. I32,
  101. I64,
  102. Isize,
  103. U8,
  104. U16,
  105. U32,
  106. U64,
  107. Usize,
  108. F32,
  109. F64,
  110. Ipv4Addr,
  111. Ipv6Addr,
  112. /// `[u8; 4]` array which represents an IPv4 address.
  113. ArrU8Len4,
  114. /// `[u8; 6]` array which represents a MAC address.
  115. ArrU8Len6,
  116. /// `[u8; 16]` array which represents an IPv6 address.
  117. ArrU8Len16,
  118. /// `[u16; 8]` array which represents an IPv6 address.
  119. ArrU16Len8,
  120. Bytes,
  121. Str,
  122. }
  123. /// All display hints
  124. #[repr(u8)]
  125. #[derive(Copy, Clone, Debug, PartialEq, Eq, IntoPrimitive)]
  126. pub enum DisplayHint {
  127. /// Default string representation.
  128. Default = 1,
  129. /// `:x`
  130. LowerHex,
  131. /// `:X`
  132. UpperHex,
  133. /// `:i`
  134. Ip,
  135. /// `:mac`
  136. LowerMac,
  137. /// `:MAC`
  138. UpperMac,
  139. }
  140. // Must be inlined, else the BPF backend emits:
  141. //
  142. // llvm: <unknown>:0:0: in function _ZN14aya_log_common5write17hc9ed05433e23a663E { i64, i64 } (i8, ptr, i64, ptr, i64): only integer returns supported
  143. #[inline(always)]
  144. pub(crate) fn write(tag: u8, value: &[u8], buf: &mut [u8]) -> Option<NonZeroUsize> {
  145. // TODO(https://github.com/rust-lang/rust-clippy/issues/14112): Remove this allowance when the
  146. // lint behaves more sensibly.
  147. #[allow(clippy::manual_ok_err)]
  148. let wire_len: LogValueLength = match value.len().try_into() {
  149. Ok(wire_len) => Some(wire_len),
  150. Err(TryFromIntError { .. }) => None,
  151. }?;
  152. let mut size = 0;
  153. macro_rules! copy_from_slice {
  154. ($value:expr) => {{
  155. let buf = buf.get_mut(size..)?;
  156. let buf = buf.get_mut(..$value.len())?;
  157. buf.copy_from_slice($value);
  158. size += $value.len();
  159. }};
  160. }
  161. copy_from_slice!(&[tag]);
  162. copy_from_slice!(&wire_len.to_ne_bytes());
  163. copy_from_slice!(value);
  164. NonZeroUsize::new(size)
  165. }
  166. pub trait WriteToBuf {
  167. fn write(self, buf: &mut [u8]) -> Option<NonZeroUsize>;
  168. }
  169. macro_rules! impl_write_to_buf {
  170. ($type:ident, $arg_type:expr) => {
  171. impl WriteToBuf for $type {
  172. // This need not be inlined because the return value is Option<N> where N is
  173. // mem::size_of<$type>, which is a compile-time constant.
  174. #[inline(never)]
  175. fn write(self, buf: &mut [u8]) -> Option<NonZeroUsize> {
  176. write($arg_type.into(), &self.to_ne_bytes(), buf)
  177. }
  178. }
  179. };
  180. }
  181. impl_write_to_buf!(i8, Argument::I8);
  182. impl_write_to_buf!(i16, Argument::I16);
  183. impl_write_to_buf!(i32, Argument::I32);
  184. impl_write_to_buf!(i64, Argument::I64);
  185. impl_write_to_buf!(isize, Argument::Isize);
  186. impl_write_to_buf!(u8, Argument::U8);
  187. impl_write_to_buf!(u16, Argument::U16);
  188. impl_write_to_buf!(u32, Argument::U32);
  189. impl_write_to_buf!(u64, Argument::U64);
  190. impl_write_to_buf!(usize, Argument::Usize);
  191. impl_write_to_buf!(f32, Argument::F32);
  192. impl_write_to_buf!(f64, Argument::F64);
  193. impl WriteToBuf for IpAddr {
  194. fn write(self, buf: &mut [u8]) -> Option<NonZeroUsize> {
  195. match self {
  196. IpAddr::V4(ipv4_addr) => write(Argument::Ipv4Addr.into(), &ipv4_addr.octets(), buf),
  197. IpAddr::V6(ipv6_addr) => write(Argument::Ipv6Addr.into(), &ipv6_addr.octets(), buf),
  198. }
  199. }
  200. }
  201. impl WriteToBuf for Ipv4Addr {
  202. fn write(self, buf: &mut [u8]) -> Option<NonZeroUsize> {
  203. write(Argument::Ipv4Addr.into(), &self.octets(), buf)
  204. }
  205. }
  206. impl WriteToBuf for [u8; 4] {
  207. // This need not be inlined because the return value is Option<N> where N is 16, which is a
  208. // compile-time constant.
  209. #[inline(never)]
  210. fn write(self, buf: &mut [u8]) -> Option<NonZeroUsize> {
  211. write(Argument::ArrU8Len4.into(), &self, buf)
  212. }
  213. }
  214. impl WriteToBuf for Ipv6Addr {
  215. fn write(self, buf: &mut [u8]) -> Option<NonZeroUsize> {
  216. write(Argument::Ipv6Addr.into(), &self.octets(), buf)
  217. }
  218. }
  219. impl WriteToBuf for [u8; 16] {
  220. // This need not be inlined because the return value is Option<N> where N is 16, which is a
  221. // compile-time constant.
  222. #[inline(never)]
  223. fn write(self, buf: &mut [u8]) -> Option<NonZeroUsize> {
  224. write(Argument::ArrU8Len16.into(), &self, buf)
  225. }
  226. }
  227. impl WriteToBuf for [u16; 8] {
  228. // This need not be inlined because the return value is Option<N> where N is 16, which is a
  229. // compile-time constant.
  230. #[inline(never)]
  231. fn write(self, buf: &mut [u8]) -> Option<NonZeroUsize> {
  232. let bytes = unsafe { core::mem::transmute::<[u16; 8], [u8; 16]>(self) };
  233. write(Argument::ArrU16Len8.into(), &bytes, buf)
  234. }
  235. }
  236. impl WriteToBuf for [u8; 6] {
  237. // This need not be inlined because the return value is Option<N> where N is 6, which is a
  238. // compile-time constant.
  239. #[inline(never)]
  240. fn write(self, buf: &mut [u8]) -> Option<NonZeroUsize> {
  241. write(Argument::ArrU8Len6.into(), &self, buf)
  242. }
  243. }
  244. impl WriteToBuf for &[u8] {
  245. // Must be inlined, else the BPF backend emits:
  246. //
  247. // llvm: <unknown>:0:0: in function _ZN63_$LT$$RF$$u5b$u8$u5d$$u20$as$u20$aya_log_common..WriteToBuf$GT$5write17h08f30a45f7b9f09dE { i64, i64 } (ptr, i64, ptr, i64): only integer returns supported
  248. #[inline(always)]
  249. fn write(self, buf: &mut [u8]) -> Option<NonZeroUsize> {
  250. write(Argument::Bytes.into(), self, buf)
  251. }
  252. }
  253. impl WriteToBuf for &str {
  254. // Must be inlined, else the BPF backend emits:
  255. //
  256. // llvm: <unknown>:0:0: in function _ZN54_$LT$$RF$str$u20$as$u20$aya_log_common..WriteToBuf$GT$5write17h7e2d1ccaa758e2b5E { i64, i64 } (ptr, i64, ptr, i64): only integer returns supported
  257. #[inline(always)]
  258. fn write(self, buf: &mut [u8]) -> Option<NonZeroUsize> {
  259. write(Argument::Str.into(), self.as_bytes(), buf)
  260. }
  261. }
  262. impl WriteToBuf for DisplayHint {
  263. // This need not be inlined because the return value is Option<N> where N is 1, which is a
  264. // compile-time constant.
  265. #[inline(never)]
  266. fn write(self, buf: &mut [u8]) -> Option<NonZeroUsize> {
  267. let v: u8 = self.into();
  268. write(Argument::DisplayHint.into(), &v.to_ne_bytes(), buf)
  269. }
  270. }
  271. #[doc(hidden)]
  272. #[inline(always)] // This function takes too many arguments to not be inlined.
  273. pub fn write_record_header(
  274. buf: &mut [u8],
  275. target: &str,
  276. level: Level,
  277. module: &str,
  278. file: &str,
  279. line: u32,
  280. num_args: usize,
  281. ) -> Option<NonZeroUsize> {
  282. let level: u8 = level.into();
  283. let mut size = 0;
  284. macro_rules! write {
  285. ($tag:expr, $value:expr) => {{
  286. let buf = buf.get_mut(size..)?;
  287. let len = write($tag.into(), $value, buf)?;
  288. size += len.get();
  289. }};
  290. }
  291. write!(RecordField::Target, target.as_bytes());
  292. write!(RecordField::Level, &level.to_ne_bytes());
  293. write!(RecordField::Module, module.as_bytes());
  294. write!(RecordField::File, file.as_bytes());
  295. write!(RecordField::Line, &line.to_ne_bytes());
  296. write!(RecordField::NumArgs, &num_args.to_ne_bytes());
  297. NonZeroUsize::new(size)
  298. }
  299. #[cfg(test)]
  300. mod test {
  301. use super::*;
  302. #[test]
  303. fn log_value_length_sufficient() {
  304. assert!(
  305. LOG_BUF_CAPACITY <= LogValueLength::MAX.into(),
  306. "{} > {}",
  307. LOG_BUF_CAPACITY,
  308. LogValueLength::MAX
  309. );
  310. }
  311. }