stm32_cec/lib.rs
1//! HDMI CEC driver for STM32 microcontrollers
2#![no_std]
3#![cfg_attr(docsrs, feature(doc_cfg), feature(doc_auto_cfg))]
4
5mod reg;
6
7pub use reg::{Cfgr, Cr, Regs};
8
9/// Interrupt flags.
10///
11/// All interrupts are cleared by writing `1` to the bit field.
12pub mod irq {
13 /// TX-missing acknowledge error interrupt
14 ///
15 /// In transmission mode, TXACKE is set by hardware to inform application
16 /// that no acknowledge was received.
17 /// In case of broadcast transmission, TXACKE informs application that a
18 /// negative acknowledge was received.
19 /// TXACKE aborts message transmission and clears TXSOM and TXEOM controls.
20 pub const TXACK: u32 = 1 << 12;
21 /// TX-error
22 ///
23 /// In transmission mode, TXERR is set by hardware if the CEC initiator
24 /// detects low impedance on the CEC line while it is released.
25 /// TXERR aborts message transmission and clears TXSOM and TXEOM controls.
26 pub const TXERR: u32 = 1 << 11;
27 /// TX-buffer underrun
28 ///
29 /// In transmission mode, TXUDR is set by hardware if application was not in
30 /// time to load TXDR before of next byte transmission.
31 /// TXUDR aborts message transmission and clears TXSOM and TXEOM control
32 /// bits.
33 pub const TXUDR: u32 = 1 << 10;
34 /// End of transmission
35 ///
36 /// TXEND is set by hardware to inform application that the last byte of the
37 /// CEC message has been successfully transmitted.
38 /// TXEND clears the TXSOM and TXEOM control bits.
39 pub const TXEND: u32 = 1 << 9;
40 /// TX-byte request
41 ///
42 /// TXBR is set by hardware to inform application that the next transmission
43 /// data has to be written to TXDR.
44 /// TXBR is set when the 4th bit of currently transmitted byte is sent.
45 /// Application must write the next byte to TXDR within six nominal data-bit
46 /// periods before transmission underrun error occurs (TXUDR).
47 pub const TXBR: u32 = 1 << 8;
48 /// Arbitration lost
49 ///
50 /// ARBLST is set by hardware to inform application that CEC device is
51 /// switching to reception due to arbitration lost event following the TXSOM
52 /// command.
53 /// ARBLST can be due either to a contending CEC device starting earlier or
54 /// starting at the same time but with higher HEADER priority.
55 /// After ARBLST assertion TXSOM bit keeps pending for next transmission
56 /// attempt.
57 pub const ARBLST: u32 = 1 << 7;
58 /// RX-missing acknowledge.
59 ///
60 /// In receive mode, RXACKE is set by hardware to inform application that no
61 /// acknowledge was seen on the CEC line.
62 /// RXACKE applies only for broadcast messages and in listen mode also for
63 /// not directly addressed messages (destination address not enabled in OAR).
64 /// RXACKE aborts message reception.
65 pub const RXACK: u32 = 1 << 6;
66 /// RX-long bit period error
67 ///
68 /// LBPE is set by hardware in case a data-bit waveform is detected with
69 /// long bit period error.
70 /// LBPE is set at the end of the maximum bit-extension tolerance allowed by
71 /// RXTOL, in case falling edge is still longing.
72 /// LBPE always stops reception of the CEC message.
73 /// LBPE generates an error-bit on the CEC line if LBPEGEN = 1.
74 /// In case of broadcast, error-bit is generated even in case of LBPEGEN = 0.
75 pub const LBPE: u32 = 1 << 5;
76 /// RX-short bit period error
77 ///
78 /// SBPE is set by hardware in case a data-bit waveform is detected with
79 /// short bit period error.
80 /// SBPE is set at the time the anticipated falling edge occurs.
81 /// SBPE generates an error-bit on the CEC line.
82 pub const SBPE: u32 = 1 << 4;
83 /// RX-bit rising error
84 ///
85 /// BRE is set by hardware in case a data-bit waveform is detected with bit
86 /// rising error.
87 /// BRE is set either at the time the misplaced rising edge occurs, or at
88 /// the end of the maximum BRE tolerance allowed by RXTOL, in case rising
89 /// edge is still longing.
90 /// BRE stops message reception if BRESTP = 1.
91 /// BRE generates an error-bit on the CEC line if BREGEN = 1.
92 pub const BRE: u32 = 1 << 3;
93 /// RX-overrun
94 ///
95 /// RXOVR is set by hardware if RXBR is not yet cleared at the time a new
96 /// byte is received on the CEC line and stored into RXD.
97 /// RXOVR assertion stops message reception so that no acknowledge is sent.
98 /// In case of broadcast, a negative acknowledge is sent.
99 pub const RXOVR: u32 = 1 << 2;
100 /// End of reception
101 ///
102 /// RXEND is set by hardware to inform application that the last byte of a
103 /// CEC message is received from the CEC line and stored into the RXD buffer.
104 /// RXEND is set at the same time of RXBR.
105 pub const RXEND: u32 = 1 << 1;
106 /// RX-byte received
107 ///
108 /// The RXBR bit is set by hardware to inform application that a new byte
109 /// has been received from the CEC line and stored into the RXD buffer.
110 pub const RXBR: u32 = 1;
111
112 /// Bitmask of all interrupts.
113 pub const ALL: u32 = TXACK
114 | TXERR
115 | TXUDR
116 | TXEND
117 | TXBR
118 | ARBLST
119 | RXACK
120 | LBPE
121 | SBPE
122 | BRE
123 | RXOVR
124 | RXEND
125 | RXBR;
126}
127
128/// Physical Address
129#[derive(Debug, Copy, Clone, Eq, PartialEq, Ord, PartialOrd, Hash)]
130#[allow(dead_code)]
131pub struct PhysAddr {
132 addr: u32,
133}
134
135#[cfg(feature = "defmt")]
136impl defmt::Format for PhysAddr {
137 fn format(&self, fmt: defmt::Formatter) {
138 defmt::write!(
139 fmt,
140 "{:X}.{:X}.{:X}.{:X}",
141 (self.addr >> 24) as u8,
142 (self.addr >> 16) as u8,
143 (self.addr >> 8) as u8,
144 self.addr as u8,
145 )
146 }
147}
148
149impl core::fmt::Display for PhysAddr {
150 fn fmt(&self, f: &mut core::fmt::Formatter) -> core::fmt::Result {
151 write!(
152 f,
153 "{:X}.{:X}.{:X}.{:X}",
154 (self.addr >> 24) as u8,
155 (self.addr >> 16) as u8,
156 (self.addr >> 8) as u8,
157 self.addr as u8,
158 )
159 }
160}
161
162/// Logical Address
163#[derive(Debug, Copy, Clone, Eq, PartialEq, Ord, PartialOrd, Hash)]
164#[cfg_attr(feature = "defmt", derive(defmt::Format))]
165#[repr(u8)]
166pub enum LogiAddr {
167 /// Television
168 Tv = 0x0,
169 /// Recording device 1
170 RecDev1 = 0x1,
171 /// Recording device 2
172 RecDev2 = 0x2,
173 /// Tuner 1
174 Tuner1 = 0x3,
175 /// Playback device 1
176 PlaybackDev = 0x4,
177 // Audio system
178 AudioSys = 0x5,
179 /// Tuner 2
180 Tuner2 = 0x6,
181 /// Tuner 3
182 Tuner3 = 0x7,
183 /// Playback Device 2
184 PlaybackDev2 = 0x8,
185 /// Recording Device 3
186 RecDev3 = 0x9,
187 /// Tuner 4
188 Tuner4 = 0xA,
189 /// Playback Device 3
190 PlaybackDev3 = 0xB,
191 /// Reserved 1
192 Rsvd1 = 0xC,
193 /// Reserved 2
194 Rsvd2 = 0xD,
195 /// Free use
196 FreeUse = 0xE,
197 /// Unregistered (as source address) or Broadcast (as destination address).
198 Broadcast = 0xF,
199}
200
201impl From<LogiAddr> for u8 {
202 #[inline]
203 fn from(addr: LogiAddr) -> Self {
204 addr as u8
205 }
206}
207
208impl TryFrom<u8> for LogiAddr {
209 type Error = u8;
210
211 fn try_from(addr: u8) -> Result<Self, Self::Error> {
212 match addr {
213 x if x == Self::Tv as u8 => Ok(Self::Tv),
214 x if x == Self::RecDev1 as u8 => Ok(Self::RecDev1),
215 x if x == Self::RecDev2 as u8 => Ok(Self::RecDev2),
216 x if x == Self::Tuner1 as u8 => Ok(Self::Tuner1),
217 x if x == Self::PlaybackDev as u8 => Ok(Self::PlaybackDev),
218 x if x == Self::AudioSys as u8 => Ok(Self::AudioSys),
219 x if x == Self::Tuner2 as u8 => Ok(Self::Tuner2),
220 x if x == Self::Tuner3 as u8 => Ok(Self::Tuner3),
221 x if x == Self::PlaybackDev2 as u8 => Ok(Self::PlaybackDev2),
222 x if x == Self::RecDev3 as u8 => Ok(Self::RecDev3),
223 x if x == Self::Tuner4 as u8 => Ok(Self::Tuner4),
224 x if x == Self::PlaybackDev3 as u8 => Ok(Self::PlaybackDev3),
225 x if x == Self::Rsvd1 as u8 => Ok(Self::Rsvd1),
226 x if x == Self::Rsvd2 as u8 => Ok(Self::Rsvd2),
227 x if x == Self::FreeUse as u8 => Ok(Self::FreeUse),
228 x if x == Self::Broadcast as u8 => Ok(Self::Broadcast),
229 _ => Err(addr),
230 }
231 }
232}
233
234/// HDMI-CEC driver.
235#[derive(Debug)]
236#[cfg_attr(feature = "defmt", derive(defmt::Format))]
237pub struct Cec<const BASE: usize> {
238 regs: Regs<BASE>,
239}
240
241impl<const BASE: usize> Cec<BASE> {
242 /// Create a new HDMI CEC driver.
243 ///
244 /// # Safety
245 ///
246 /// 1. The HDMI CEC source clock must be enabled.
247 /// 2. The HDMI CEC pin must be configured.
248 /// 3. The HDMI CEC peripheral should be reset before.
249 /// 4. The HDMI CEC registers provided by the PAC should be dropped.
250 /// 5. The generic BASE parameter must be correct or bad things will happen.
251 ///
252 /// # Panics
253 ///
254 /// Panics if reading CFGR does not return the value written.
255 /// This occurs when the HDMI CEC peripheral clocks are not configured
256 /// correctly.
257 ///
258 /// # Example
259 ///
260 /// ```no_run
261 /// use stm32_cec::Cec;
262 ///
263 /// // device specific setup occurs here
264 /// // ...
265 ///
266 /// // valid address for the STM32H7
267 /// let mut cec: Cec<0x40006C00> = unsafe { Cec::<0x40006C00>::new() };
268 /// ```
269 #[inline]
270 pub unsafe fn new() -> Cec<BASE> {
271 let mut regs: Regs<BASE> = Regs { _priv: () };
272 regs.set_cr(Cr::DEFAULT);
273 const MY_CFGR: Cfgr = Cfgr::DEFAULT
274 .set_lstn(true)
275 .set_oar(0x8)
276 .set_lbpegen(true)
277 .set_bregen(true)
278 .set_brestp(true);
279 regs.set_cfgr(MY_CFGR);
280 assert_eq!(Regs::<BASE>::cfgr().oar(), 0x8);
281 regs.set_ier(irq::ALL);
282 regs.set_cr(Cr::EN);
283
284 Cec { regs }
285 }
286
287 fn poll_isr() -> u32 {
288 // TODO: timeout
289 loop {
290 let isr: u32 = Regs::<BASE>::isr();
291 if isr != 0 {
292 Regs::<BASE>::set_isr(isr);
293 return isr;
294 }
295 }
296 }
297
298 fn send_byte(&mut self, src: LogiAddr, dst: LogiAddr, data: u8) -> Result<(), u32> {
299 self.regs.set_txdr((u8::from(src) << 4) | u8::from(dst));
300 self.regs.set_cr(Cr::SOM);
301
302 let isr: u32 = Self::poll_isr();
303 if isr & irq::TXBR == irq::TXBR {
304 self.regs.set_txdr(data);
305 self.regs.set_cr(Cr::EOM);
306 let isr: u32 = Self::poll_isr();
307 if isr & irq::TXEND == irq::TXEND {
308 Ok(())
309 } else {
310 Err(isr)
311 }
312 } else {
313 Err(isr)
314 }
315 }
316
317 /// Power off devices.
318 ///
319 /// # Example
320 ///
321 /// Turn everything off.
322 ///
323 /// ```no_run
324 /// use stm32_cec::{Cec, LogiAddr};
325 ///
326 /// let mut cec = unsafe { stm32_cec::Cec::<0x40006C00>::new() };
327 /// cec.set_standby(LogiAddr::Broadcast, LogiAddr::Broadcast)?;
328 /// # Ok::<(), u32>(())
329 /// ```
330 pub fn set_standby(&mut self, src: LogiAddr, dst: LogiAddr) -> Result<(), u32> {
331 self.send_byte(src, dst, 0x36)
332 }
333
334 /// Power on the TV.
335 ///
336 /// # Example
337 ///
338 /// Turn everything on.
339 ///
340 /// ```no_run
341 /// use stm32_cec::{Cec, LogiAddr};
342 ///
343 /// let mut cec = unsafe { stm32_cec::Cec::<0x40006C00>::new() };
344 /// cec.set_image_view_on(LogiAddr::Broadcast, LogiAddr::Broadcast)?;
345 /// # Ok::<(), u32>(())
346 /// ```
347 pub fn set_image_view_on(&mut self, src: LogiAddr, dst: LogiAddr) -> Result<(), u32> {
348 self.send_byte(src, dst, 0x04)
349 }
350}