sbi_ret.rs 39 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146
  1. /// SBI functions return type.
  2. ///
  3. /// > SBI functions must return a pair of values in a0 and a1,
  4. /// > with a0 returning an error code.
  5. /// > This is analogous to returning the C structure `SbiRet`.
  6. ///
  7. /// Note: if this structure is used in function return on conventional
  8. /// Rust code, it would not require pinning memory representation as
  9. /// extern C. The `repr(C)` is set in case that some users want to use
  10. /// this structure in FFI code.
  11. #[derive(Clone, Copy, PartialEq, Eq)]
  12. #[repr(C)]
  13. pub struct SbiRet<T = usize> {
  14. /// Error number.
  15. pub error: T,
  16. /// Result value.
  17. pub value: T,
  18. }
  19. /// Standard RISC-V SBI error IDs in `usize`.
  20. pub mod id {
  21. use super::SbiRegister;
  22. /// SBI success state return value.
  23. #[doc(alias = "SBI_SUCCESS")]
  24. pub const RET_SUCCESS: usize = <usize as SbiRegister>::RET_SUCCESS;
  25. /// Error for SBI call failed for unknown reasons.
  26. #[doc(alias = "SBI_ERR_FAILED")]
  27. pub const RET_ERR_FAILED: usize = <usize as SbiRegister>::RET_ERR_FAILED;
  28. /// Error for target operation not supported.
  29. #[doc(alias = "SBI_ERR_NOT_SUPPORTED")]
  30. pub const RET_ERR_NOT_SUPPORTED: usize = <usize as SbiRegister>::RET_ERR_NOT_SUPPORTED;
  31. /// Error for invalid parameter.
  32. #[doc(alias = "SBI_ERR_INVALID_PARAM")]
  33. pub const RET_ERR_INVALID_PARAM: usize = <usize as SbiRegister>::RET_ERR_INVALID_PARAM;
  34. /// Error for denied.
  35. #[doc(alias = "SBI_ERR_DENIED")]
  36. pub const RET_ERR_DENIED: usize = <usize as SbiRegister>::RET_ERR_DENIED;
  37. /// Error for invalid address.
  38. #[doc(alias = "SBI_ERR_INVALID_ADDRESS")]
  39. pub const RET_ERR_INVALID_ADDRESS: usize = <usize as SbiRegister>::RET_ERR_INVALID_ADDRESS;
  40. /// Error for resource already available.
  41. #[doc(alias = "SBI_ERR_ALREADY_AVAILABLE")]
  42. pub const RET_ERR_ALREADY_AVAILABLE: usize = <usize as SbiRegister>::RET_ERR_ALREADY_AVAILABLE;
  43. /// Error for resource already started.
  44. #[doc(alias = "SBI_ERR_ALREADY_STARTED")]
  45. pub const RET_ERR_ALREADY_STARTED: usize = <usize as SbiRegister>::RET_ERR_ALREADY_STARTED;
  46. /// Error for resource already stopped.
  47. #[doc(alias = "SBI_ERR_ALREADY_STOPPED")]
  48. pub const RET_ERR_ALREADY_STOPPED: usize = <usize as SbiRegister>::RET_ERR_ALREADY_STOPPED;
  49. /// Error for shared memory not available.
  50. #[doc(alias = "SBI_ERR_NO_SHMEM")]
  51. pub const RET_ERR_NO_SHMEM: usize = <usize as SbiRegister>::RET_ERR_NO_SHMEM;
  52. /// Error for invalid state.
  53. #[doc(alias = "SBI_ERR_INVALID_STATE")]
  54. pub const RET_ERR_INVALID_STATE: usize = <usize as SbiRegister>::RET_ERR_INVALID_STATE;
  55. /// Error for bad or invalid range.
  56. #[doc(alias = "SBI_ERR_BAD_RANGE")]
  57. pub const RET_ERR_BAD_RANGE: usize = <usize as SbiRegister>::RET_ERR_BAD_RANGE;
  58. /// Error for failed due to timeout.
  59. #[doc(alias = "SBI_ERR_TIMEOUT")]
  60. pub const RET_ERR_TIMEOUT: usize = <usize as SbiRegister>::RET_ERR_TIMEOUT;
  61. /// Error for input or output error.
  62. #[doc(alias = "SBI_ERR_IO")]
  63. pub const RET_ERR_IO: usize = <usize as SbiRegister>::RET_ERR_IO;
  64. /// Error for denied or not allowed due to lock status.
  65. #[doc(alias = "SBI_ERR_DENIED_LOCKED")]
  66. pub const RET_ERR_DENIED_LOCKED: usize = <usize as SbiRegister>::RET_ERR_DENIED_LOCKED;
  67. // ^^ Note: remember to add a test case in `rustsbi_sbi_ret_constructors` in this file,
  68. // and `test_binary` in lib.rs after adding an error number!
  69. }
  70. // Use each constants in `id` module, so that any `match` operations will not treat constant
  71. // names (`RET_ERR_*`) as newly defined variable names.
  72. use id::*;
  73. /// Data type of register that can be passed to the RISC-V SBI ABI.
  74. ///
  75. /// This trait defines the requirements for types that are used as the underlying
  76. /// representation for both the `value` and `error` fields in the `SbiRet` structure.
  77. /// In most cases, this trait is implemented for primitive integer types (e.g., `usize`),
  78. /// but it can also be implemented for other types that satisfy the constraints.
  79. ///
  80. /// # Examples
  81. ///
  82. /// Implemented automatically for all types that satisfy `Copy`, `Eq`, and `Debug`.
  83. pub trait SbiRegister: Copy + Eq + Ord + core::fmt::Debug {
  84. /// SBI success state return value.
  85. const RET_SUCCESS: Self;
  86. /// Error for SBI call failed for unknown reasons.
  87. const RET_ERR_FAILED: Self;
  88. /// Error for target operation not supported.
  89. const RET_ERR_NOT_SUPPORTED: Self;
  90. /// Error for invalid parameter.
  91. const RET_ERR_INVALID_PARAM: Self;
  92. /// Error for denied.
  93. const RET_ERR_DENIED: Self;
  94. /// Error for invalid address.
  95. const RET_ERR_INVALID_ADDRESS: Self;
  96. /// Error for resource already available.
  97. const RET_ERR_ALREADY_AVAILABLE: Self;
  98. /// Error for resource already started.
  99. const RET_ERR_ALREADY_STARTED: Self;
  100. /// Error for resource already stopped.
  101. const RET_ERR_ALREADY_STOPPED: Self;
  102. /// Error for shared memory not available.
  103. const RET_ERR_NO_SHMEM: Self;
  104. /// Error for invalid state.
  105. const RET_ERR_INVALID_STATE: Self;
  106. /// Error for bad or invalid range.
  107. const RET_ERR_BAD_RANGE: Self;
  108. /// Error for failed due to timeout.
  109. const RET_ERR_TIMEOUT: Self;
  110. /// Error for input or output error.
  111. const RET_ERR_IO: Self;
  112. /// Error for denied or not allowed due to lock status.
  113. const RET_ERR_DENIED_LOCKED: Self;
  114. /// Zero value for this type; this is used on `value` fields once `SbiRet` returns an error.
  115. const ZERO: Self;
  116. /// Full-ones value for this type; this is used on SBI mask structures like `CounterMask`
  117. /// and `HartMask`.
  118. const FULL_MASK: Self;
  119. /// Converts an `SbiRet` of this type to a `Result` of self and `Error`.
  120. fn into_result(ret: SbiRet<Self>) -> Result<Self, Error<Self>>;
  121. }
  122. macro_rules! impl_sbi_register {
  123. ($ty:ty, $signed:ty) => {
  124. impl SbiRegister for $ty {
  125. const RET_SUCCESS: Self = 0;
  126. const RET_ERR_FAILED: Self = -1 as $signed as $ty;
  127. const RET_ERR_NOT_SUPPORTED: Self = -2 as $signed as $ty;
  128. const RET_ERR_INVALID_PARAM: Self = -3 as $signed as $ty;
  129. const RET_ERR_DENIED: Self = -4 as $signed as $ty;
  130. const RET_ERR_INVALID_ADDRESS: Self = -5 as $signed as $ty;
  131. const RET_ERR_ALREADY_AVAILABLE: Self = -6 as $signed as $ty;
  132. const RET_ERR_ALREADY_STARTED: Self = -7 as $signed as $ty;
  133. const RET_ERR_ALREADY_STOPPED: Self = -8 as $signed as $ty;
  134. const RET_ERR_NO_SHMEM: Self = -9 as $signed as $ty;
  135. const RET_ERR_INVALID_STATE: Self = -10 as $signed as $ty;
  136. const RET_ERR_BAD_RANGE: Self = -11 as $signed as $ty;
  137. const RET_ERR_TIMEOUT: Self = -12 as $signed as $ty;
  138. const RET_ERR_IO: Self = -13 as $signed as $ty;
  139. const RET_ERR_DENIED_LOCKED: Self = -14 as $signed as $ty;
  140. const ZERO: Self = 0;
  141. const FULL_MASK: Self = !0;
  142. fn into_result(ret: SbiRet<Self>) -> Result<Self, Error<Self>> {
  143. match ret.error {
  144. Self::RET_SUCCESS => Ok(ret.value),
  145. Self::RET_ERR_FAILED => Err(Error::Failed),
  146. Self::RET_ERR_NOT_SUPPORTED => Err(Error::NotSupported),
  147. Self::RET_ERR_INVALID_PARAM => Err(Error::InvalidParam),
  148. Self::RET_ERR_DENIED => Err(Error::Denied),
  149. Self::RET_ERR_INVALID_ADDRESS => Err(Error::InvalidAddress),
  150. Self::RET_ERR_ALREADY_AVAILABLE => Err(Error::AlreadyAvailable),
  151. Self::RET_ERR_ALREADY_STARTED => Err(Error::AlreadyStarted),
  152. Self::RET_ERR_ALREADY_STOPPED => Err(Error::AlreadyStopped),
  153. Self::RET_ERR_NO_SHMEM => Err(Error::NoShmem),
  154. Self::RET_ERR_INVALID_STATE => Err(Error::InvalidState),
  155. Self::RET_ERR_BAD_RANGE => Err(Error::BadRange),
  156. Self::RET_ERR_TIMEOUT => Err(Error::Timeout),
  157. Self::RET_ERR_IO => Err(Error::Io),
  158. Self::RET_ERR_DENIED_LOCKED => Err(Error::DeniedLocked),
  159. unknown => Err(Error::Custom(unknown as _)),
  160. }
  161. }
  162. }
  163. };
  164. }
  165. impl_sbi_register!(usize, isize);
  166. impl_sbi_register!(isize, isize);
  167. impl_sbi_register!(u32, i32);
  168. impl_sbi_register!(i32, i32);
  169. impl_sbi_register!(u64, i64);
  170. impl_sbi_register!(i64, i64);
  171. impl_sbi_register!(u128, i128);
  172. impl_sbi_register!(i128, i128);
  173. impl<T: SbiRegister + core::fmt::LowerHex> core::fmt::Debug for SbiRet<T> {
  174. fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
  175. match T::into_result(*self) {
  176. Ok(value) => write!(f, "{:?}", value),
  177. Err(err) => match err {
  178. Error::Failed => write!(f, "<SBI call failed>"),
  179. Error::NotSupported => write!(f, "<SBI feature not supported>"),
  180. Error::InvalidParam => write!(f, "<SBI invalid parameter>"),
  181. Error::Denied => write!(f, "<SBI denied>"),
  182. Error::InvalidAddress => write!(f, "<SBI invalid address>"),
  183. Error::AlreadyAvailable => write!(f, "<SBI already available>"),
  184. Error::AlreadyStarted => write!(f, "<SBI already started>"),
  185. Error::AlreadyStopped => write!(f, "<SBI already stopped>"),
  186. Error::NoShmem => write!(f, "<SBI shared memory not available>"),
  187. Error::InvalidState => write!(f, "<SBI invalid state>"),
  188. Error::BadRange => write!(f, "<SBI bad range>"),
  189. Error::Timeout => write!(f, "<SBI timeout>"),
  190. Error::Io => write!(f, "<SBI input/output error>"),
  191. Error::DeniedLocked => write!(f, "<SBI denied due to locked status>"),
  192. Error::Custom(unknown) => write!(f, "[SBI Unknown error: {:#x}]", unknown),
  193. },
  194. }
  195. }
  196. }
  197. /// RISC-V SBI error in enumeration.
  198. #[derive(Debug, Clone, Copy, PartialEq, Eq)]
  199. pub enum Error<T = usize> {
  200. /// Error for SBI call failed for unknown reasons.
  201. Failed,
  202. /// Error for target operation not supported.
  203. NotSupported,
  204. /// Error for invalid parameter.
  205. InvalidParam,
  206. /// Error for denied.
  207. Denied,
  208. /// Error for invalid address.
  209. InvalidAddress,
  210. /// Error for resource already available.
  211. AlreadyAvailable,
  212. /// Error for resource already started.
  213. AlreadyStarted,
  214. /// Error for resource already stopped.
  215. AlreadyStopped,
  216. /// Error for shared memory not available.
  217. NoShmem,
  218. /// Error for invalid state.
  219. InvalidState,
  220. /// Error for bad or invalid range.
  221. BadRange,
  222. /// Error for failed due to timeout.
  223. Timeout,
  224. /// Error for input or output error.
  225. Io,
  226. /// Error for denied or not allowed due to lock status.
  227. DeniedLocked,
  228. /// Custom error code.
  229. Custom(T),
  230. }
  231. impl<T: SbiRegister> SbiRet<T> {
  232. /// Returns success SBI state with given `value`.
  233. #[inline]
  234. pub const fn success(value: T) -> Self {
  235. Self {
  236. error: T::RET_SUCCESS,
  237. value,
  238. }
  239. }
  240. /// The SBI call request failed for unknown reasons.
  241. #[inline]
  242. pub const fn failed() -> Self {
  243. Self {
  244. error: T::RET_ERR_FAILED,
  245. value: T::ZERO,
  246. }
  247. }
  248. /// SBI call failed due to not supported by target ISA,
  249. /// operation type not supported,
  250. /// or target operation type not implemented on purpose.
  251. #[inline]
  252. pub const fn not_supported() -> Self {
  253. Self {
  254. error: T::RET_ERR_NOT_SUPPORTED,
  255. value: T::ZERO,
  256. }
  257. }
  258. /// SBI call failed due to invalid hart mask parameter,
  259. /// invalid target hart id,
  260. /// invalid operation type,
  261. /// or invalid resource index.
  262. #[inline]
  263. pub const fn invalid_param() -> Self {
  264. Self {
  265. error: T::RET_ERR_INVALID_PARAM,
  266. value: T::ZERO,
  267. }
  268. }
  269. /// SBI call denied for unsatisfied entry criteria, or insufficient access
  270. /// permission to debug console or CPPC register.
  271. #[inline]
  272. pub const fn denied() -> Self {
  273. Self {
  274. error: T::RET_ERR_DENIED,
  275. value: T::ZERO,
  276. }
  277. }
  278. /// SBI call failed for invalid mask start address,
  279. /// not a valid physical address parameter,
  280. /// or the target address is prohibited by PMP to run in supervisor mode.
  281. #[inline]
  282. pub const fn invalid_address() -> Self {
  283. Self {
  284. error: T::RET_ERR_INVALID_ADDRESS,
  285. value: T::ZERO,
  286. }
  287. }
  288. /// SBI call failed for the target resource is already available,
  289. /// e.g., the target hart is already started when caller still requests it to start.
  290. #[inline]
  291. pub const fn already_available() -> Self {
  292. Self {
  293. error: T::RET_ERR_ALREADY_AVAILABLE,
  294. value: T::ZERO,
  295. }
  296. }
  297. /// SBI call failed for the target resource is already started,
  298. /// e.g., target performance counter is started.
  299. #[inline]
  300. pub const fn already_started() -> Self {
  301. Self {
  302. error: T::RET_ERR_ALREADY_STARTED,
  303. value: T::ZERO,
  304. }
  305. }
  306. /// SBI call failed for the target resource is already stopped,
  307. /// e.g., target performance counter is stopped.
  308. #[inline]
  309. pub const fn already_stopped() -> Self {
  310. Self {
  311. error: T::RET_ERR_ALREADY_STOPPED,
  312. value: T::ZERO,
  313. }
  314. }
  315. /// SBI call failed for shared memory is not available,
  316. /// e.g. nested acceleration shared memory is not available.
  317. #[inline]
  318. pub const fn no_shmem() -> Self {
  319. Self {
  320. error: T::RET_ERR_NO_SHMEM,
  321. value: T::ZERO,
  322. }
  323. }
  324. /// SBI call failed for invalid state,
  325. /// e.g. register a software event but the event is not in unused state.
  326. #[inline]
  327. pub const fn invalid_state() -> Self {
  328. Self {
  329. error: T::RET_ERR_INVALID_STATE,
  330. value: T::ZERO,
  331. }
  332. }
  333. /// SBI call failed for bad or invalid range,
  334. /// e.g. the software event is not exist in the specified range.
  335. #[inline]
  336. pub const fn bad_range() -> Self {
  337. Self {
  338. error: T::RET_ERR_BAD_RANGE,
  339. value: T::ZERO,
  340. }
  341. }
  342. /// SBI call failed for timeout,
  343. /// e.g. message send timeout.
  344. #[inline]
  345. pub const fn timeout() -> Self {
  346. Self {
  347. error: T::RET_ERR_TIMEOUT,
  348. value: T::ZERO,
  349. }
  350. }
  351. /// SBI call failed for input or output error.
  352. #[inline]
  353. pub const fn io() -> Self {
  354. Self {
  355. error: T::RET_ERR_IO,
  356. value: T::ZERO,
  357. }
  358. }
  359. /// SBI call failed for denied or not allowed due to lock status.
  360. #[inline]
  361. pub const fn denied_locked() -> Self {
  362. Self {
  363. error: T::RET_ERR_DENIED_LOCKED,
  364. value: T::ZERO,
  365. }
  366. }
  367. }
  368. impl<T: SbiRegister> From<Error<T>> for SbiRet<T> {
  369. #[inline]
  370. fn from(value: Error<T>) -> Self {
  371. match value {
  372. Error::Failed => SbiRet::failed(),
  373. Error::NotSupported => SbiRet::not_supported(),
  374. Error::InvalidParam => SbiRet::invalid_param(),
  375. Error::Denied => SbiRet::denied(),
  376. Error::InvalidAddress => SbiRet::invalid_address(),
  377. Error::AlreadyAvailable => SbiRet::already_available(),
  378. Error::AlreadyStarted => SbiRet::already_started(),
  379. Error::AlreadyStopped => SbiRet::already_stopped(),
  380. Error::NoShmem => SbiRet::no_shmem(),
  381. Error::InvalidState => SbiRet::invalid_state(),
  382. Error::BadRange => SbiRet::bad_range(),
  383. Error::Timeout => SbiRet::timeout(),
  384. Error::Io => SbiRet::io(),
  385. Error::DeniedLocked => SbiRet::denied_locked(),
  386. Error::Custom(error) => SbiRet {
  387. error,
  388. value: T::ZERO,
  389. },
  390. }
  391. }
  392. }
  393. impl SbiRet {
  394. /// Converts to a [`Result`] of value and error.
  395. #[inline]
  396. pub const fn into_result(self) -> Result<usize, Error> {
  397. match self.error {
  398. RET_SUCCESS => Ok(self.value),
  399. RET_ERR_FAILED => Err(Error::Failed),
  400. RET_ERR_NOT_SUPPORTED => Err(Error::NotSupported),
  401. RET_ERR_INVALID_PARAM => Err(Error::InvalidParam),
  402. RET_ERR_DENIED => Err(Error::Denied),
  403. RET_ERR_INVALID_ADDRESS => Err(Error::InvalidAddress),
  404. RET_ERR_ALREADY_AVAILABLE => Err(Error::AlreadyAvailable),
  405. RET_ERR_ALREADY_STARTED => Err(Error::AlreadyStarted),
  406. RET_ERR_ALREADY_STOPPED => Err(Error::AlreadyStopped),
  407. RET_ERR_NO_SHMEM => Err(Error::NoShmem),
  408. RET_ERR_INVALID_STATE => Err(Error::InvalidState),
  409. RET_ERR_BAD_RANGE => Err(Error::BadRange),
  410. RET_ERR_TIMEOUT => Err(Error::Timeout),
  411. RET_ERR_IO => Err(Error::Io),
  412. RET_ERR_DENIED_LOCKED => Err(Error::DeniedLocked),
  413. unknown => Err(Error::Custom(unknown as _)),
  414. }
  415. }
  416. /// Returns `true` if current SBI return succeeded.
  417. ///
  418. /// # Examples
  419. ///
  420. /// Basic usage:
  421. ///
  422. /// ```
  423. /// # use sbi_spec::binary::SbiRet;
  424. /// let x = SbiRet::success(0);
  425. /// assert_eq!(x.is_ok(), true);
  426. ///
  427. /// let x = SbiRet::failed();
  428. /// assert_eq!(x.is_ok(), false);
  429. /// ```
  430. #[must_use = "if you intended to assert that this is ok, consider `.unwrap()` instead"]
  431. #[inline]
  432. pub const fn is_ok(&self) -> bool {
  433. matches!(self.error, RET_SUCCESS)
  434. }
  435. /// Returns `true` if the SBI call succeeded and the value inside of it matches a predicate.
  436. ///
  437. /// # Examples
  438. ///
  439. /// Basic usage:
  440. ///
  441. /// ```
  442. /// # use sbi_spec::binary::SbiRet;
  443. /// let x = SbiRet::success(2);
  444. /// assert_eq!(x.is_ok_and(|x| x > 1), true);
  445. ///
  446. /// let x = SbiRet::success(0);
  447. /// assert_eq!(x.is_ok_and(|x| x > 1), false);
  448. ///
  449. /// let x = SbiRet::no_shmem();
  450. /// assert_eq!(x.is_ok_and(|x| x > 1), false);
  451. /// ```
  452. #[must_use]
  453. #[inline]
  454. pub fn is_ok_and(self, f: impl FnOnce(usize) -> bool) -> bool {
  455. self.into_result().is_ok_and(f)
  456. }
  457. /// Returns `true` if current SBI return is an error.
  458. ///
  459. /// # Examples
  460. ///
  461. /// Basic usage:
  462. ///
  463. /// ```
  464. /// # use sbi_spec::binary::SbiRet;
  465. /// let x = SbiRet::success(0);
  466. /// assert_eq!(x.is_err(), false);
  467. ///
  468. /// let x = SbiRet::not_supported();
  469. /// assert_eq!(x.is_err(), true);
  470. /// ```
  471. #[must_use = "if you intended to assert that this is err, consider `.unwrap_err()` instead"]
  472. #[inline]
  473. pub const fn is_err(&self) -> bool {
  474. !self.is_ok()
  475. }
  476. /// Returns `true` if the result is an error and the value inside of it matches a predicate.
  477. ///
  478. /// # Examples
  479. ///
  480. /// ```
  481. /// # use sbi_spec::binary::{SbiRet, Error};
  482. /// let x = SbiRet::denied();
  483. /// assert_eq!(x.is_err_and(|x| x == Error::Denied), true);
  484. ///
  485. /// let x = SbiRet::invalid_address();
  486. /// assert_eq!(x.is_err_and(|x| x == Error::Denied), false);
  487. ///
  488. /// let x = SbiRet::success(0);
  489. /// assert_eq!(x.is_err_and(|x| x == Error::Denied), false);
  490. /// ```
  491. #[must_use]
  492. #[inline]
  493. pub fn is_err_and(self, f: impl FnOnce(Error) -> bool) -> bool {
  494. self.into_result().is_err_and(f)
  495. }
  496. /// Converts from `SbiRet` to [`Option<usize>`].
  497. ///
  498. /// Converts `self` into an [`Option<usize>`], consuming `self`,
  499. /// and discarding the error, if any.
  500. ///
  501. /// # Examples
  502. ///
  503. /// Basic usage:
  504. ///
  505. /// ```
  506. /// # use sbi_spec::binary::SbiRet;
  507. /// let x = SbiRet::success(2);
  508. /// assert_eq!(x.ok(), Some(2));
  509. ///
  510. /// let x = SbiRet::invalid_param();
  511. /// assert_eq!(x.ok(), None);
  512. /// ```
  513. // fixme: should be pub const fn once this function in Result is stablized in constant
  514. #[inline]
  515. pub fn ok(self) -> Option<usize> {
  516. self.into_result().ok()
  517. }
  518. /// Converts from `SbiRet` to [`Option<Error>`].
  519. ///
  520. /// Converts `self` into an [`Option<Error>`], consuming `self`,
  521. /// and discarding the success value, if any.
  522. ///
  523. /// # Examples
  524. ///
  525. /// Basic usage:
  526. ///
  527. /// ```
  528. /// # use sbi_spec::binary::{SbiRet, Error};
  529. /// let x = SbiRet::success(2);
  530. /// assert_eq!(x.err(), None);
  531. ///
  532. /// let x = SbiRet::denied();
  533. /// assert_eq!(x.err(), Some(Error::Denied));
  534. /// ```
  535. // fixme: should be pub const fn once this function in Result is stablized in constant
  536. #[inline]
  537. pub fn err(self) -> Option<Error> {
  538. self.into_result().err()
  539. }
  540. /// Maps a `SbiRet` to `Result<U, Error>` by applying a function to a
  541. /// contained success value, leaving an error value untouched.
  542. ///
  543. /// This function can be used to compose the results of two functions.
  544. ///
  545. /// # Examples
  546. ///
  547. /// Gets detail of a PMU counter and judge if it is a firmware counter.
  548. ///
  549. /// ```
  550. /// # use sbi_spec::binary::SbiRet;
  551. /// # use core::mem::size_of;
  552. /// # mod sbi_rt {
  553. /// # use sbi_spec::binary::SbiRet;
  554. /// # const TYPE_MASK: usize = 1 << (core::mem::size_of::<usize>() - 1);
  555. /// # pub fn pmu_counter_get_info(_: usize) -> SbiRet { SbiRet::success(TYPE_MASK) }
  556. /// # }
  557. /// // We assume that counter index 42 is a firmware counter.
  558. /// let counter_idx = 42;
  559. /// // Masks PMU counter type by setting highest bit in `usize`.
  560. /// const TYPE_MASK: usize = 1 << (size_of::<usize>() - 1);
  561. /// // Highest bit of returned `counter_info` represents whether it's
  562. /// // a firmware counter or a hardware counter.
  563. /// let is_firmware_counter = sbi_rt::pmu_counter_get_info(counter_idx)
  564. /// .map(|counter_info| counter_info & TYPE_MASK != 0);
  565. /// // If that bit is set, it is a firmware counter.
  566. /// assert_eq!(is_firmware_counter, Ok(true));
  567. /// ```
  568. #[inline]
  569. pub fn map<U, F: FnOnce(usize) -> U>(self, op: F) -> Result<U, Error> {
  570. self.into_result().map(op)
  571. }
  572. /// Returns the provided default (if error),
  573. /// or applies a function to the contained value (if success).
  574. ///
  575. /// Arguments passed to `map_or` are eagerly evaluated;
  576. /// if you are passing the result of a function call,
  577. /// it is recommended to use [`map_or_else`],
  578. /// which is lazily evaluated.
  579. ///
  580. /// [`map_or_else`]: SbiRet::map_or_else
  581. ///
  582. /// # Examples
  583. ///
  584. /// ```
  585. /// # use sbi_spec::binary::SbiRet;
  586. /// let x = SbiRet::success(3);
  587. /// assert_eq!(x.map_or(42, |v| v & 0b1), 1);
  588. ///
  589. /// let x = SbiRet::invalid_address();
  590. /// assert_eq!(x.map_or(42, |v| v & 0b1), 42);
  591. /// ```
  592. #[inline]
  593. pub fn map_or<U, F: FnOnce(usize) -> U>(self, default: U, f: F) -> U {
  594. self.into_result().map_or(default, f)
  595. }
  596. /// Maps a `SbiRet` to `usize` value by applying fallback function `default` to
  597. /// a contained error, or function `f` to a contained success value.
  598. ///
  599. /// This function can be used to unpack a successful result
  600. /// while handling an error.
  601. ///
  602. /// # Examples
  603. ///
  604. /// Basic usage:
  605. ///
  606. /// ```
  607. /// # use sbi_spec::binary::SbiRet;
  608. /// let k = 21;
  609. ///
  610. /// let x = SbiRet::success(3);
  611. /// assert_eq!(x.map_or_else(|e| k * 2, |v| v & 0b1), 1);
  612. ///
  613. /// let x = SbiRet::already_available();
  614. /// assert_eq!(x.map_or_else(|e| k * 2, |v| v & 0b1), 42);
  615. /// ```
  616. #[inline]
  617. pub fn map_or_else<U, D: FnOnce(Error) -> U, F: FnOnce(usize) -> U>(
  618. self,
  619. default: D,
  620. f: F,
  621. ) -> U {
  622. self.into_result().map_or_else(default, f)
  623. }
  624. /// Maps a `SbiRet` to `Result<T, F>` by applying a function to a
  625. /// contained error as [`Error`] struct, leaving success value untouched.
  626. ///
  627. /// This function can be used to pass through a successful result while handling
  628. /// an error.
  629. ///
  630. /// # Examples
  631. ///
  632. /// Basic usage:
  633. ///
  634. /// ```
  635. /// # use sbi_spec::binary::{SbiRet, Error};
  636. /// fn stringify(x: Error) -> String {
  637. /// if x == Error::AlreadyStarted {
  638. /// "error: already started!".to_string()
  639. /// } else {
  640. /// "error: other error!".to_string()
  641. /// }
  642. /// }
  643. ///
  644. /// let x = SbiRet::success(2);
  645. /// assert_eq!(x.map_err(stringify), Ok(2));
  646. ///
  647. /// let x = SbiRet::already_started();
  648. /// assert_eq!(x.map_err(stringify), Err("error: already started!".to_string()));
  649. /// ```
  650. #[inline]
  651. pub fn map_err<F, O: FnOnce(Error) -> F>(self, op: O) -> Result<usize, F> {
  652. self.into_result().map_err(op)
  653. }
  654. /// Calls a function with a reference to the contained value if current SBI call succeeded.
  655. ///
  656. /// Returns the original result.
  657. ///
  658. /// # Examples
  659. ///
  660. /// ```
  661. /// # use sbi_spec::binary::SbiRet;
  662. /// // Assume that SBI debug console have read 512 bytes into a buffer.
  663. /// let ret = SbiRet::success(512);
  664. /// // Inspect the SBI DBCN call result.
  665. /// let idx = ret
  666. /// .inspect(|x| println!("bytes written: {x}"))
  667. /// .map(|x| x - 1)
  668. /// .expect("SBI DBCN call failed");
  669. /// assert_eq!(idx, 511);
  670. /// ```
  671. #[inline]
  672. pub fn inspect<F: FnOnce(&usize)>(self, f: F) -> Self {
  673. if let Ok(ref t) = self.into_result() {
  674. f(t);
  675. }
  676. self
  677. }
  678. /// Calls a function with a reference to the contained value if current SBI result is an error.
  679. ///
  680. /// Returns the original result.
  681. ///
  682. /// # Examples
  683. ///
  684. /// ```
  685. /// # use sbi_spec::binary::SbiRet;
  686. /// // Assume that SBI debug console write operation failed for invalid parameter.
  687. /// let ret = SbiRet::invalid_param();
  688. /// // Print the error if SBI DBCN call failed.
  689. /// let ret = ret.inspect_err(|e| eprintln!("failed to read from SBI console: {e:?}"));
  690. /// ```
  691. #[inline]
  692. pub fn inspect_err<F: FnOnce(&Error)>(self, f: F) -> Self {
  693. if let Err(ref e) = self.into_result() {
  694. f(e);
  695. }
  696. self
  697. }
  698. // TODO: pub fn iter(&self) -> Iter
  699. // TODO: pub fn iter_mut(&mut self) -> IterMut
  700. /// Returns the contained success value, consuming the `self` value.
  701. ///
  702. /// # Panics
  703. ///
  704. /// Panics if self is an SBI error with a panic message including the
  705. /// passed message, and the content of the SBI state.
  706. ///
  707. /// # Examples
  708. ///
  709. /// Basic usage:
  710. ///
  711. /// ```should_panic
  712. /// # use sbi_spec::binary::SbiRet;
  713. /// let x = SbiRet::already_stopped();
  714. /// x.expect("Testing expect"); // panics with `Testing expect`
  715. /// ```
  716. #[inline]
  717. pub fn expect(self, msg: &str) -> usize {
  718. self.into_result().expect(msg)
  719. }
  720. /// Returns the contained success value, consuming the `self` value.
  721. ///
  722. /// # Panics
  723. ///
  724. /// Panics if self is an SBI error, with a panic message provided by the
  725. /// SBI error converted into [`Error`] struct.
  726. ///
  727. /// # Examples
  728. ///
  729. /// Basic usage:
  730. ///
  731. /// ```
  732. /// # use sbi_spec::binary::SbiRet;
  733. /// let x = SbiRet::success(2);
  734. /// assert_eq!(x.unwrap(), 2);
  735. /// ```
  736. ///
  737. /// ```should_panic
  738. /// # use sbi_spec::binary::SbiRet;
  739. /// let x = SbiRet::failed();
  740. /// x.unwrap(); // panics
  741. /// ```
  742. #[inline]
  743. pub fn unwrap(self) -> usize {
  744. self.into_result().unwrap()
  745. }
  746. // Note: No unwrap_or_default as we cannot determine a meaningful default value for a successful SbiRet.
  747. /// Returns the contained error as [`Error`] struct, consuming the `self` value.
  748. ///
  749. /// # Panics
  750. ///
  751. /// Panics if the self is SBI success value, with a panic message
  752. /// including the passed message, and the content of the success value.
  753. ///
  754. /// # Examples
  755. ///
  756. /// Basic usage:
  757. ///
  758. /// ```should_panic
  759. /// # use sbi_spec::binary::SbiRet;
  760. /// let x = SbiRet::success(10);
  761. /// x.expect_err("Testing expect_err"); // panics with `Testing expect_err`
  762. /// ```
  763. #[inline]
  764. pub fn expect_err(self, msg: &str) -> Error {
  765. self.into_result().expect_err(msg)
  766. }
  767. /// Returns the contained error as [`Error`] struct, consuming the `self` value.
  768. ///
  769. /// # Panics
  770. ///
  771. /// Panics if the self is SBI success value, with a custom panic message provided
  772. /// by the success value.
  773. ///
  774. /// # Examples
  775. ///
  776. /// ```should_panic
  777. /// # use sbi_spec::binary::SbiRet;
  778. /// let x = SbiRet::success(2);
  779. /// x.unwrap_err(); // panics with `2`
  780. /// ```
  781. ///
  782. /// ```
  783. /// # use sbi_spec::binary::{SbiRet, Error};
  784. /// let x = SbiRet::not_supported();
  785. /// assert_eq!(x.unwrap_err(), Error::NotSupported);
  786. /// ```
  787. #[inline]
  788. pub fn unwrap_err(self) -> Error {
  789. self.into_result().unwrap_err()
  790. }
  791. // TODO: pub fn into_ok(self) -> usize and pub fn into_err(self) -> Error
  792. // once `unwrap_infallible` is stablized
  793. /// Returns `res` if self is success value, otherwise otherwise returns the contained error
  794. /// of `self` as [`Error`] struct.
  795. ///
  796. /// Arguments passed to `and` are eagerly evaluated; if you are passing the
  797. /// result of a function call, it is recommended to use [`and_then`], which is
  798. /// lazily evaluated.
  799. ///
  800. /// [`and_then`]: SbiRet::and_then
  801. ///
  802. /// # Examples
  803. ///
  804. /// Basic usage:
  805. ///
  806. /// ```
  807. /// # use sbi_spec::binary::{SbiRet, Error};
  808. /// let x = SbiRet::success(2);
  809. /// let y = SbiRet::invalid_param().into_result();
  810. /// assert_eq!(x.and(y), Err(Error::InvalidParam));
  811. ///
  812. /// let x = SbiRet::denied();
  813. /// let y = SbiRet::success(3).into_result();
  814. /// assert_eq!(x.and(y), Err(Error::Denied));
  815. ///
  816. /// let x = SbiRet::invalid_address();
  817. /// let y = SbiRet::already_available().into_result();
  818. /// assert_eq!(x.and(y), Err(Error::InvalidAddress));
  819. ///
  820. /// let x = SbiRet::success(4);
  821. /// let y = SbiRet::success(5).into_result();
  822. /// assert_eq!(x.and(y), Ok(5));
  823. /// ```
  824. // fixme: should be pub const fn once this function in Result is stablized in constant
  825. // fixme: should parameter be `res: SbiRet`?
  826. #[inline]
  827. pub fn and<U>(self, res: Result<U, Error>) -> Result<U, Error> {
  828. self.into_result().and(res)
  829. }
  830. /// Calls `op` if self is success value, otherwise returns the contained error
  831. /// as [`Error`] struct.
  832. ///
  833. /// This function can be used for control flow based on `SbiRet` values.
  834. ///
  835. /// # Examples
  836. ///
  837. /// ```
  838. /// # use sbi_spec::binary::{SbiRet, Error};
  839. /// fn sq_then_to_string(x: usize) -> Result<String, Error> {
  840. /// x.checked_mul(x).map(|sq| sq.to_string()).ok_or(Error::Failed)
  841. /// }
  842. ///
  843. /// assert_eq!(SbiRet::success(2).and_then(sq_then_to_string), Ok(4.to_string()));
  844. /// assert_eq!(SbiRet::success(1_000_000_000_000).and_then(sq_then_to_string), Err(Error::Failed));
  845. /// assert_eq!(SbiRet::invalid_param().and_then(sq_then_to_string), Err(Error::InvalidParam));
  846. /// ```
  847. #[inline]
  848. pub fn and_then<U, F: FnOnce(usize) -> Result<U, Error>>(self, op: F) -> Result<U, Error> {
  849. self.into_result().and_then(op)
  850. }
  851. /// Returns `res` if self is SBI error, otherwise returns the success value of `self`.
  852. ///
  853. /// Arguments passed to `or` are eagerly evaluated; if you are passing the
  854. /// result of a function call, it is recommended to use [`or_else`], which is
  855. /// lazily evaluated.
  856. ///
  857. /// [`or_else`]: Result::or_else
  858. ///
  859. /// # Examples
  860. ///
  861. /// Basic usage:
  862. ///
  863. /// ```
  864. /// # use sbi_spec::binary::{SbiRet, Error};
  865. /// let x = SbiRet::success(2);
  866. /// let y = SbiRet::invalid_param().into_result();
  867. /// assert_eq!(x.or(y), Ok(2));
  868. ///
  869. /// let x = SbiRet::denied();
  870. /// let y = SbiRet::success(3).into_result();
  871. /// assert_eq!(x.or(y), Ok(3));
  872. ///
  873. /// let x = SbiRet::invalid_address();
  874. /// let y = SbiRet::already_available().into_result();
  875. /// assert_eq!(x.or(y), Err(Error::AlreadyAvailable));
  876. ///
  877. /// let x = SbiRet::success(4);
  878. /// let y = SbiRet::success(100).into_result();
  879. /// assert_eq!(x.or(y), Ok(4));
  880. /// ```
  881. // fixme: should be pub const fn once this function in Result is stablized in constant
  882. // fixme: should parameter be `res: SbiRet`?
  883. #[inline]
  884. pub fn or<F>(self, res: Result<usize, F>) -> Result<usize, F> {
  885. self.into_result().or(res)
  886. }
  887. /// Calls `op` if self is SBI error, otherwise returns the success value of `self`.
  888. ///
  889. /// This function can be used for control flow based on result values.
  890. ///
  891. ///
  892. /// # Examples
  893. ///
  894. /// Basic usage:
  895. ///
  896. /// ```
  897. /// # use sbi_spec::binary::{SbiRet, Error};
  898. /// fn is_failed(x: Error) -> Result<usize, bool> { Err(x == Error::Failed) }
  899. ///
  900. /// assert_eq!(SbiRet::success(2).or_else(is_failed), Ok(2));
  901. /// assert_eq!(SbiRet::failed().or_else(is_failed), Err(true));
  902. /// ```
  903. #[inline]
  904. pub fn or_else<F, O: FnOnce(Error) -> Result<usize, F>>(self, op: O) -> Result<usize, F> {
  905. self.into_result().or_else(op)
  906. }
  907. /// Returns the contained success value or a provided default.
  908. ///
  909. /// Arguments passed to `unwrap_or` are eagerly evaluated; if you are passing
  910. /// the result of a function call, it is recommended to use [`unwrap_or_else`],
  911. /// which is lazily evaluated.
  912. ///
  913. /// [`unwrap_or_else`]: SbiRet::unwrap_or_else
  914. ///
  915. /// # Examples
  916. ///
  917. /// Basic usage:
  918. ///
  919. /// ```
  920. /// # use sbi_spec::binary::SbiRet;
  921. /// let default = 2;
  922. /// let x = SbiRet::success(9);
  923. /// assert_eq!(x.unwrap_or(default), 9);
  924. ///
  925. /// let x = SbiRet::invalid_param();
  926. /// assert_eq!(x.unwrap_or(default), default);
  927. /// ```
  928. // fixme: should be pub const fn once this function in Result is stablized in constant
  929. #[inline]
  930. pub fn unwrap_or(self, default: usize) -> usize {
  931. self.into_result().unwrap_or(default)
  932. }
  933. /// Returns the contained success value or computes it from a closure.
  934. ///
  935. /// # Examples
  936. ///
  937. /// Basic usage:
  938. ///
  939. /// ```
  940. /// # use sbi_spec::binary::{SbiRet, Error};
  941. /// fn invalid_use_zero(x: Error) -> usize { if x == Error::InvalidParam { 0 } else { 3 } }
  942. ///
  943. /// assert_eq!(SbiRet::success(2).unwrap_or_else(invalid_use_zero), 2);
  944. /// assert_eq!(SbiRet::invalid_param().unwrap_or_else(invalid_use_zero), 0);
  945. /// ```
  946. #[inline]
  947. pub fn unwrap_or_else<F: FnOnce(Error) -> usize>(self, op: F) -> usize {
  948. self.into_result().unwrap_or_else(op)
  949. }
  950. /// Returns the contained success value, consuming the `self` value,
  951. /// without checking that the `SbiRet` contains an error value.
  952. ///
  953. /// # Safety
  954. ///
  955. /// Calling this method on an `SbiRet` containing an error value results
  956. /// in *undefined behavior*.
  957. ///
  958. /// # Examples
  959. ///
  960. /// ```
  961. /// # use sbi_spec::binary::{SbiRet, Error};
  962. /// let x = SbiRet::success(3);
  963. /// assert_eq!(unsafe { x.unwrap_unchecked() }, 3);
  964. /// ```
  965. ///
  966. /// ```no_run
  967. /// # use sbi_spec::binary::SbiRet;
  968. /// let x = SbiRet::no_shmem();
  969. /// unsafe { x.unwrap_unchecked(); } // Undefined behavior!
  970. /// ```
  971. #[inline]
  972. pub unsafe fn unwrap_unchecked(self) -> usize {
  973. unsafe { self.into_result().unwrap_unchecked() }
  974. }
  975. /// Returns the contained `Error` value, consuming the `self` value,
  976. /// without checking that the `SbiRet` does not contain a success value.
  977. ///
  978. /// # Safety
  979. ///
  980. /// Calling this method on an `SbiRet` containing a success value results
  981. /// in *undefined behavior*.
  982. ///
  983. /// # Examples
  984. ///
  985. /// ```no_run
  986. /// # use sbi_spec::binary::SbiRet;
  987. /// let x = SbiRet::success(4);
  988. /// unsafe { x.unwrap_unchecked(); } // Undefined behavior!
  989. /// ```
  990. ///
  991. /// ```
  992. /// # use sbi_spec::binary::{SbiRet, Error};
  993. /// let x = SbiRet::failed();
  994. /// assert_eq!(unsafe { x.unwrap_err_unchecked() }, Error::Failed);
  995. /// ```
  996. #[inline]
  997. pub unsafe fn unwrap_err_unchecked(self) -> Error {
  998. unsafe { self.into_result().unwrap_err_unchecked() }
  999. }
  1000. }
  1001. impl IntoIterator for SbiRet {
  1002. type Item = usize;
  1003. type IntoIter = core::result::IntoIter<usize>;
  1004. /// Returns a consuming iterator over the possibly contained value.
  1005. ///
  1006. /// The iterator yields one value if the result contains a success value, otherwise none.
  1007. ///
  1008. /// # Examples
  1009. ///
  1010. /// ```
  1011. /// # use sbi_spec::binary::SbiRet;
  1012. /// let x = SbiRet::success(5);
  1013. /// let v: Vec<usize> = x.into_iter().collect();
  1014. /// assert_eq!(v, [5]);
  1015. ///
  1016. /// let x = SbiRet::not_supported();
  1017. /// let v: Vec<usize> = x.into_iter().collect();
  1018. /// assert_eq!(v, []);
  1019. /// ```
  1020. #[inline]
  1021. fn into_iter(self) -> Self::IntoIter {
  1022. self.into_result().into_iter()
  1023. }
  1024. }
  1025. // TODO: implement Try and FromResidual for SbiRet once those traits are stablized
  1026. /*
  1027. impl core::ops::Try for SbiRet {
  1028. type Output = usize;
  1029. type Residual = Result<core::convert::Infallible, Error>;
  1030. #[inline]
  1031. fn from_output(output: Self::Output) -> Self {
  1032. SbiRet::success(output)
  1033. }
  1034. #[inline]
  1035. fn branch(self) -> core::ops::ControlFlow<Self::Residual, Self::Output> {
  1036. self.into_result().branch()
  1037. }
  1038. }
  1039. impl core::ops::FromResidual<Result<core::convert::Infallible, Error>> for SbiRet {
  1040. #[inline]
  1041. #[track_caller]
  1042. fn from_residual(residual: Result<core::convert::Infallible, Error>) -> Self {
  1043. match residual {
  1044. Err(e) => e.into(),
  1045. }
  1046. }
  1047. }
  1048. /// ```
  1049. /// # use sbi_spec::binary::SbiRet;
  1050. /// fn test() -> SbiRet {
  1051. /// let value = SbiRet::failed()?;
  1052. /// SbiRet::success(0)
  1053. /// }
  1054. /// assert_eq!(test(), SbiRet::failed());
  1055. /// ```
  1056. mod test_try_trait_for_sbiret {}
  1057. */
  1058. #[cfg(test)]
  1059. mod tests {
  1060. use super::*;
  1061. #[test]
  1062. #[rustfmt::skip]
  1063. fn rustsbi_sbi_ret_constructors() {
  1064. assert_eq!(SbiRet::success(0), SbiRet { value: 0, error: 0 });
  1065. assert_eq!(SbiRet::success(1037), SbiRet { value: 1037, error: 0 });
  1066. assert_eq!(SbiRet::success(usize::MAX), SbiRet { value: usize::MAX, error: 0 });
  1067. assert_eq!(SbiRet::failed(), SbiRet { value: 0, error: usize::MAX - 1 + 1 });
  1068. assert_eq!(SbiRet::not_supported(), SbiRet { value: 0, error: usize::MAX - 2 + 1 });
  1069. assert_eq!(SbiRet::invalid_param(), SbiRet { value: 0, error: usize::MAX - 3 + 1 });
  1070. assert_eq!(SbiRet::denied(), SbiRet { value: 0, error: usize::MAX - 4 + 1 });
  1071. assert_eq!(SbiRet::invalid_address(), SbiRet { value: 0, error: usize::MAX - 5 + 1 });
  1072. assert_eq!(SbiRet::already_available(), SbiRet { value: 0, error: usize::MAX - 6 + 1 });
  1073. assert_eq!(SbiRet::already_started(), SbiRet { value: 0, error: usize::MAX - 7 + 1 });
  1074. assert_eq!(SbiRet::already_stopped(), SbiRet { value: 0, error: usize::MAX - 8 + 1 });
  1075. assert_eq!(SbiRet::no_shmem(), SbiRet { value: 0, error: usize::MAX - 9 + 1 });
  1076. assert_eq!(SbiRet::invalid_state(), SbiRet { value: 0, error: usize::MAX - 10 + 1 });
  1077. assert_eq!(SbiRet::bad_range(), SbiRet { value: 0, error: usize::MAX - 11 + 1 });
  1078. assert_eq!(SbiRet::timeout(), SbiRet { value: 0, error: usize::MAX - 12 + 1 });
  1079. assert_eq!(SbiRet::io(), SbiRet { value: 0, error: usize::MAX - 13 + 1 });
  1080. assert_eq!(SbiRet::denied_locked(), SbiRet { value: 0, error: usize::MAX - 14 + 1 });
  1081. }
  1082. }