kernel/hil/crc.rs
1// Licensed under the Apache License, Version 2.0 or the MIT License.
2// SPDX-License-Identifier: Apache-2.0 OR MIT
3// Copyright Tock Contributors 2022.
4
5//! Interface for CRC computation.
6
7use crate::utilities::leasable_buffer::SubSliceMut;
8use crate::ErrorCode;
9
10/// Client for CRC algorithm implementations
11///
12/// Implement this trait and use [`Crc::set_client`] in order to
13/// receive callbacks from the CRC implementation.
14pub trait Client {
15 /// Called when the current data chunk has been processed by the
16 /// CRC engine. Further data may be supplied when this callback is
17 /// received.
18 fn input_done(&self, result: Result<(), ErrorCode>, buffer: SubSliceMut<'static, u8>);
19
20 /// Called when the CRC computation is finished.
21 fn crc_done(&self, result: Result<CrcOutput, ErrorCode>);
22}
23
24/// CRC algorithms
25///
26/// In all cases, input bytes are bit-reversed (i.e., consumed from LSB to MSB.)
27///
28/// Algorithms prefixed with `Sam4L` are native to that chip and thus require
29/// no software post-processing on platforms using it.
30///
31#[derive(Copy, Clone)]
32pub enum CrcAlgorithm {
33 /// Polynomial 0x04C11DB7, output reversed then inverted
34 /// ("CRC-32")
35 Crc32,
36 /// Polynomial 0x1EDC6F41, output reversed then inverted
37 /// ("CRC-32C" / "Castagnoli")
38 Crc32C,
39 /// Polynomial 0x1021, no output post-processing ("CRC-16-CCITT")
40 Crc16CCITT,
41}
42
43/// CRC output type
44///
45/// Individual CRC algorithms can have different output lengths. This
46/// type represents the different [`CrcAlgorithm`] outputs
47/// respectively.
48#[derive(Copy, Clone)]
49pub enum CrcOutput {
50 /// Output of [`CrcAlgorithm::Crc32`]
51 Crc32(u32),
52 /// Output of [`CrcAlgorithm::Crc32C`]
53 Crc32C(u32),
54 /// Output of [`CrcAlgorithm::Crc16CCITT`]
55 Crc16CCITT(u16),
56}
57
58impl CrcOutput {
59 pub fn algorithm(&self) -> CrcAlgorithm {
60 match self {
61 CrcOutput::Crc32(_) => CrcAlgorithm::Crc32,
62 CrcOutput::Crc32C(_) => CrcAlgorithm::Crc32C,
63 CrcOutput::Crc16CCITT(_) => CrcAlgorithm::Crc16CCITT,
64 }
65 }
66}
67
68pub trait Crc<'a> {
69 /// Set the client to be used for callbacks of the CRC
70 /// implementation.
71 fn set_client(&self, client: &'a dyn Client);
72 /// Check whether a given CRC algorithm is supported by a CRC
73 /// implementation.
74 ///
75 /// Returns true if the algorithm specified is supported.
76 fn algorithm_supported(&self, algorithm: CrcAlgorithm) -> bool;
77
78 /// Set the CRC algorithm to use.
79 ///
80 /// Calling this method may enable the CRC engine in case of a
81 /// physical unit.
82 ///
83 /// If the device is currently processing a chunk of data or
84 /// calculating a CRC, this operation will be refused, returning
85 /// [`ErrorCode::BUSY`]. If a CRC calculation currently has
86 /// pending data, it will be cancelled and the CRC engine's state
87 /// reset.
88 ///
89 /// [`ErrorCode::NOSUPPORT`] will be returned if the algorithm
90 /// requested is not supported. To non-invasively check whether a
91 /// given algorithm is supported by a CRC implementation, use
92 /// [`Crc::algorithm_supported`].
93 fn set_algorithm(&self, algorithm: CrcAlgorithm) -> Result<(), ErrorCode>;
94
95 /// Input chunked data into the CRC implementation.
96 ///
97 /// Calling this method may enable the CRC engine in case of a
98 /// physical unit.
99 ///
100 /// If [`Crc::set_algorithm`] has not been invoked before, this
101 /// method must return [`ErrorCode::RESERVE`].
102 ///
103 /// If the device is currently already processing a chunk of data
104 /// or calculating a CRC, [`ErrorCode::BUSY`] must be returned.
105 ///
106 /// After the chunk of data has been processed,
107 /// [`Client::input_done`] is called.
108 ///
109 /// The implementation may only read a part of the passed
110 /// [`SubSliceMut`]. It will return the bytes read and will
111 /// resize the returned [`SubSliceMut`] appropriately prior to
112 /// passing it back through [`Client::input_done`].
113 fn input(
114 &self,
115 data: SubSliceMut<'static, u8>,
116 ) -> Result<(), (ErrorCode, SubSliceMut<'static, u8>)>;
117
118 /// Request calculation of the CRC.
119 ///
120 /// Calling this method may enable the CRC engine in case of a
121 /// physical unit.
122 ///
123 /// If [`Crc::set_algorithm`] has not been invoked before, this
124 /// method must return [`ErrorCode::RESERVE`].
125 ///
126 /// If the device is currently processing a chunk of data or
127 /// calculating a CRC, [`ErrorCode::BUSY`] must be returned.
128 ///
129 /// After the CRC has been calculated, [`Client::crc_done`] is
130 /// called.
131 fn compute(&self) -> Result<(), ErrorCode>;
132
133 /// Disable the CRC unit until susequent calls to methods which
134 /// will enable the CRC unit again.
135 fn disable(&self);
136}