* Remove ChannelCreator types and burst mode * Fix up I2sParallel * Always enable burst transfering descriptors * Configure burst transfer with a non-bool for future chip support * Reuse buffer preparation code * Update LoopBuf as well * Update lcd_cam tests * Rename config, fix changelog
822 lines
28 KiB
Rust
822 lines
28 KiB
Rust
//! # LCD - RGB/Digital Parallel Interface Mode
|
|
//!
|
|
//! ## Overview
|
|
//!
|
|
//! The LCD_CAM peripheral Dpi driver provides support for the DPI (commonly
|
|
//! know as RGB) format/timing. The driver mandates DMA (Direct Memory Access)
|
|
//! for efficient data transfer.
|
|
//!
|
|
//! ## Examples
|
|
//!
|
|
//! ### A display
|
|
//!
|
|
//! The following example shows how to setup and send a solid frame to a DPI
|
|
//! display.
|
|
//!
|
|
//! ```rust, no_run
|
|
#![doc = crate::before_snippet!()]
|
|
//! # use esp_hal::gpio::Level;
|
|
//! # use esp_hal::lcd_cam::{
|
|
//! # LcdCam,
|
|
//! # lcd::{
|
|
//! # ClockMode, Polarity, Phase,
|
|
//! # dpi::{Config, Dpi, Format, FrameTiming, self}
|
|
//! # }
|
|
//! # };
|
|
//! # use esp_hal::dma_loop_buffer;
|
|
//! # use esp_hal::dma::{Dma, DmaPriority};
|
|
//!
|
|
//! # let dma = Dma::new(peripherals.DMA);
|
|
//! # let channel = dma.channel0;
|
|
//!
|
|
//! # let mut dma_buf = dma_loop_buffer!(32);
|
|
//!
|
|
//! let lcd_cam = LcdCam::new(peripherals.LCD_CAM);
|
|
//!
|
|
//! let mut config = dpi::Config::default();
|
|
//! config.clock_mode = ClockMode {
|
|
//! polarity: Polarity::IdleLow,
|
|
//! phase: Phase::ShiftLow,
|
|
//! };
|
|
//! config.format = Format {
|
|
//! enable_2byte_mode: true,
|
|
//! ..Default::default()
|
|
//! };
|
|
//! config.timing = FrameTiming {
|
|
//! horizontal_active_width: 480,
|
|
//! horizontal_total_width: 520,
|
|
//! horizontal_blank_front_porch: 10,
|
|
//!
|
|
//! vertical_active_height: 480,
|
|
//! vertical_total_height: 510,
|
|
//! vertical_blank_front_porch: 10,
|
|
//!
|
|
//! hsync_width: 10,
|
|
//! vsync_width: 10,
|
|
//!
|
|
//! hsync_position: 0,
|
|
//! };
|
|
//! config.vsync_idle_level = Level::High;
|
|
//! config.hsync_idle_level = Level::High;
|
|
//! config.de_idle_level = Level::Low;
|
|
//! config.disable_black_region = false;
|
|
//!
|
|
//! let mut dpi = Dpi::new(lcd_cam.lcd, channel.tx, 1.MHz(), config)
|
|
//! .with_vsync(peripherals.GPIO3)
|
|
//! .with_hsync(peripherals.GPIO46)
|
|
//! .with_de(peripherals.GPIO17)
|
|
//! .with_pclk(peripherals.GPIO9)
|
|
//! // Blue
|
|
//! .with_data0(peripherals.GPIO10)
|
|
//! .with_data1(peripherals.GPIO11)
|
|
//! .with_data2(peripherals.GPIO12)
|
|
//! .with_data3(peripherals.GPIO13)
|
|
//! .with_data4(peripherals.GPIO14)
|
|
//! // Green
|
|
//! .with_data5(peripherals.GPIO21)
|
|
//! .with_data6(peripherals.GPIO8)
|
|
//! .with_data7(peripherals.GPIO18)
|
|
//! .with_data8(peripherals.GPIO45)
|
|
//! .with_data9(peripherals.GPIO38)
|
|
//! .with_data10(peripherals.GPIO39)
|
|
//! // Red
|
|
//! .with_data11(peripherals.GPIO40)
|
|
//! .with_data12(peripherals.GPIO41)
|
|
//! .with_data13(peripherals.GPIO42)
|
|
//! .with_data14(peripherals.GPIO2)
|
|
//! .with_data15(peripherals.GPIO1);
|
|
//!
|
|
//! let color: u16 = 0b11111_000000_00000; // RED
|
|
//! for chunk in dma_buf.chunks_mut(2) {
|
|
//! chunk.copy_from_slice(&color.to_le_bytes());
|
|
//! }
|
|
//!
|
|
//! let transfer = dpi.send(false, dma_buf).map_err(|e| e.0).unwrap();
|
|
//! transfer.wait();
|
|
//! # }
|
|
//! ```
|
|
|
|
use core::{
|
|
mem::ManuallyDrop,
|
|
ops::{Deref, DerefMut},
|
|
};
|
|
|
|
use fugit::HertzU32;
|
|
|
|
use crate::{
|
|
clock::Clocks,
|
|
dma::{ChannelTx, DmaChannelConvert, DmaEligible, DmaError, DmaPeripheral, DmaTxBuffer, Tx},
|
|
gpio::{interconnect::PeripheralOutput, Level, OutputSignal},
|
|
lcd_cam::{
|
|
calculate_clkm,
|
|
lcd::{ClockMode, DelayMode, Lcd, Phase, Polarity},
|
|
BitOrder,
|
|
ByteOrder,
|
|
},
|
|
peripheral::{Peripheral, PeripheralRef},
|
|
peripherals::LCD_CAM,
|
|
Blocking,
|
|
Mode,
|
|
};
|
|
|
|
/// Represents the RGB LCD interface.
|
|
pub struct Dpi<'d> {
|
|
lcd_cam: PeripheralRef<'d, LCD_CAM>,
|
|
tx_channel: ChannelTx<'d, Blocking, <LCD_CAM as DmaEligible>::Dma>,
|
|
}
|
|
|
|
impl<'d> Dpi<'d> {
|
|
/// Create a new instance of the RGB/DPI driver.
|
|
pub fn new<DM: Mode, CH>(
|
|
lcd: Lcd<'d, DM>,
|
|
channel: ChannelTx<'d, Blocking, CH>,
|
|
frequency: HertzU32,
|
|
config: Config,
|
|
) -> Self
|
|
where
|
|
CH: DmaChannelConvert<<LCD_CAM as DmaEligible>::Dma>,
|
|
{
|
|
let lcd_cam = lcd.lcd_cam;
|
|
|
|
let clocks = Clocks::get();
|
|
// Due to https://www.espressif.com/sites/default/files/documentation/esp32-s3_errata_en.pdf
|
|
// the LCD_PCLK divider must be at least 2. To make up for this the user
|
|
// provided frequency is doubled to match.
|
|
let (i, divider) = calculate_clkm(
|
|
(frequency.to_Hz() * 2) as _,
|
|
&[
|
|
clocks.xtal_clock.to_Hz() as _,
|
|
clocks.cpu_clock.to_Hz() as _,
|
|
clocks.crypto_pwm_clock.to_Hz() as _,
|
|
],
|
|
);
|
|
|
|
lcd_cam.lcd_clock().write(|w| unsafe {
|
|
// Force enable the clock for all configuration registers.
|
|
w.clk_en().set_bit();
|
|
w.lcd_clk_sel().bits((i + 1) as _);
|
|
w.lcd_clkm_div_num().bits(divider.div_num as _);
|
|
w.lcd_clkm_div_b().bits(divider.div_b as _);
|
|
w.lcd_clkm_div_a().bits(divider.div_a as _); // LCD_PCLK = LCD_CLK / 2
|
|
w.lcd_clk_equ_sysclk().clear_bit();
|
|
w.lcd_clkcnt_n().bits(2 - 1); // Must not be 0.
|
|
w.lcd_ck_idle_edge()
|
|
.bit(config.clock_mode.polarity == Polarity::IdleHigh);
|
|
w.lcd_ck_out_edge()
|
|
.bit(config.clock_mode.phase == Phase::ShiftHigh)
|
|
});
|
|
lcd_cam.lcd_user().modify(|_, w| w.lcd_reset().set_bit());
|
|
|
|
lcd_cam
|
|
.lcd_rgb_yuv()
|
|
.write(|w| w.lcd_conv_bypass().clear_bit());
|
|
|
|
lcd_cam.lcd_user().modify(|_, w| {
|
|
if config.format.enable_2byte_mode {
|
|
w.lcd_8bits_order().bit(false);
|
|
w.lcd_byte_order()
|
|
.bit(config.format.byte_order == ByteOrder::Inverted);
|
|
} else {
|
|
w.lcd_8bits_order()
|
|
.bit(config.format.byte_order == ByteOrder::Inverted);
|
|
w.lcd_byte_order().bit(false);
|
|
}
|
|
w.lcd_bit_order()
|
|
.bit(config.format.bit_order == BitOrder::Inverted);
|
|
w.lcd_2byte_en().bit(config.format.enable_2byte_mode);
|
|
|
|
// Only valid in Intel8080 mode.
|
|
w.lcd_cmd().clear_bit();
|
|
w.lcd_dummy().clear_bit();
|
|
|
|
// This needs to be explicitly set for RGB mode.
|
|
w.lcd_dout().set_bit()
|
|
});
|
|
|
|
let timing = &config.timing;
|
|
lcd_cam.lcd_ctrl().modify(|_, w| unsafe {
|
|
// Enable RGB mode, and input VSYNC, HSYNC, and DE signals.
|
|
w.lcd_rgb_mode_en().set_bit();
|
|
|
|
w.lcd_hb_front()
|
|
.bits((timing.horizontal_blank_front_porch as u16).saturating_sub(1));
|
|
w.lcd_va_height()
|
|
.bits((timing.vertical_active_height as u16).saturating_sub(1));
|
|
w.lcd_vt_height()
|
|
.bits((timing.vertical_total_height as u16).saturating_sub(1))
|
|
});
|
|
lcd_cam.lcd_ctrl1().modify(|_, w| unsafe {
|
|
w.lcd_vb_front()
|
|
.bits((timing.vertical_blank_front_porch as u8).saturating_sub(1));
|
|
w.lcd_ha_width()
|
|
.bits((timing.horizontal_active_width as u16).saturating_sub(1));
|
|
w.lcd_ht_width()
|
|
.bits((timing.horizontal_total_width as u16).saturating_sub(1))
|
|
});
|
|
lcd_cam.lcd_ctrl2().modify(|_, w| unsafe {
|
|
w.lcd_vsync_width()
|
|
.bits((timing.vsync_width as u8).saturating_sub(1));
|
|
w.lcd_vsync_idle_pol().bit(config.vsync_idle_level.into());
|
|
w.lcd_de_idle_pol().bit(config.de_idle_level.into());
|
|
w.lcd_hs_blank_en().bit(config.hs_blank_en);
|
|
w.lcd_hsync_width()
|
|
.bits((timing.hsync_width as u8).saturating_sub(1));
|
|
w.lcd_hsync_idle_pol().bit(config.hsync_idle_level.into());
|
|
w.lcd_hsync_position().bits(timing.hsync_position as u8)
|
|
});
|
|
|
|
lcd_cam.lcd_misc().modify(|_, w| unsafe {
|
|
// TODO: Find out what this field actually does.
|
|
// Set the threshold for Async Tx FIFO full event. (5 bits)
|
|
w.lcd_afifo_threshold_num().bits((1 << 5) - 1);
|
|
|
|
// Doesn't matter for RGB mode.
|
|
w.lcd_vfk_cyclelen().bits(0);
|
|
w.lcd_vbk_cyclelen().bits(0);
|
|
|
|
// 1: Send the next frame data when the current frame is sent out.
|
|
// 0: LCD stops when the current frame is sent out.
|
|
w.lcd_next_frame_en().clear_bit();
|
|
|
|
// Enable blank region when LCD sends data out.
|
|
w.lcd_bk_en().bit(!config.disable_black_region)
|
|
});
|
|
lcd_cam.lcd_dly_mode().modify(|_, w| unsafe {
|
|
w.lcd_de_mode().bits(config.de_mode as u8);
|
|
w.lcd_hsync_mode().bits(config.hsync_mode as u8);
|
|
w.lcd_vsync_mode().bits(config.vsync_mode as u8);
|
|
w
|
|
});
|
|
lcd_cam.lcd_data_dout_mode().modify(|_, w| unsafe {
|
|
w.dout0_mode().bits(config.output_bit_mode as u8);
|
|
w.dout1_mode().bits(config.output_bit_mode as u8);
|
|
w.dout2_mode().bits(config.output_bit_mode as u8);
|
|
w.dout3_mode().bits(config.output_bit_mode as u8);
|
|
w.dout4_mode().bits(config.output_bit_mode as u8);
|
|
w.dout5_mode().bits(config.output_bit_mode as u8);
|
|
w.dout6_mode().bits(config.output_bit_mode as u8);
|
|
w.dout7_mode().bits(config.output_bit_mode as u8);
|
|
w.dout8_mode().bits(config.output_bit_mode as u8);
|
|
w.dout9_mode().bits(config.output_bit_mode as u8);
|
|
w.dout10_mode().bits(config.output_bit_mode as u8);
|
|
w.dout11_mode().bits(config.output_bit_mode as u8);
|
|
w.dout12_mode().bits(config.output_bit_mode as u8);
|
|
w.dout13_mode().bits(config.output_bit_mode as u8);
|
|
w.dout14_mode().bits(config.output_bit_mode as u8);
|
|
w.dout15_mode().bits(config.output_bit_mode as u8)
|
|
});
|
|
|
|
lcd_cam.lcd_user().modify(|_, w| w.lcd_update().set_bit());
|
|
|
|
Self {
|
|
lcd_cam,
|
|
tx_channel: channel.degrade(),
|
|
}
|
|
}
|
|
|
|
/// Assign the VSYNC pin for the LCD_CAM.
|
|
///
|
|
/// Sets the specified pin to push-pull output and connects it to the VSYNC
|
|
/// signal.
|
|
pub fn with_vsync<S: PeripheralOutput>(self, pin: impl Peripheral<P = S> + 'd) -> Self {
|
|
crate::into_mapped_ref!(pin);
|
|
pin.set_to_push_pull_output(crate::private::Internal);
|
|
OutputSignal::LCD_V_SYNC.connect_to(pin);
|
|
|
|
self
|
|
}
|
|
|
|
/// Assign the HSYNC pin for the LCD_CAM.
|
|
///
|
|
/// Sets the specified pin to push-pull output and connects it to the HSYNC
|
|
/// signal.
|
|
pub fn with_hsync<S: PeripheralOutput>(self, pin: impl Peripheral<P = S> + 'd) -> Self {
|
|
crate::into_mapped_ref!(pin);
|
|
pin.set_to_push_pull_output(crate::private::Internal);
|
|
OutputSignal::LCD_H_SYNC.connect_to(pin);
|
|
|
|
self
|
|
}
|
|
|
|
/// Assign the DE pin for the LCD_CAM.
|
|
///
|
|
/// Sets the specified pin to push-pull output and connects it to the DE
|
|
/// signal.
|
|
pub fn with_de<S: PeripheralOutput>(self, pin: impl Peripheral<P = S> + 'd) -> Self {
|
|
crate::into_mapped_ref!(pin);
|
|
pin.set_to_push_pull_output(crate::private::Internal);
|
|
OutputSignal::LCD_H_ENABLE.connect_to(pin);
|
|
|
|
self
|
|
}
|
|
|
|
/// Assign the PCLK pin for the LCD_CAM.
|
|
///
|
|
/// Sets the specified pin to push-pull output and connects it to the PCLK
|
|
/// signal.
|
|
pub fn with_pclk<S: PeripheralOutput>(self, pin: impl Peripheral<P = S> + 'd) -> Self {
|
|
crate::into_mapped_ref!(pin);
|
|
pin.set_to_push_pull_output(crate::private::Internal);
|
|
OutputSignal::LCD_PCLK.connect_to(pin);
|
|
|
|
self
|
|
}
|
|
|
|
/// Assign the DATA_0 pin for the LCD_CAM.
|
|
///
|
|
/// Sets the specified pin to push-pull output and connects it to the DATA_0
|
|
/// signal.
|
|
pub fn with_data0<S: PeripheralOutput>(self, pin: impl Peripheral<P = S> + 'd) -> Self {
|
|
crate::into_mapped_ref!(pin);
|
|
pin.set_to_push_pull_output(crate::private::Internal);
|
|
OutputSignal::LCD_DATA_0.connect_to(pin);
|
|
|
|
self
|
|
}
|
|
|
|
/// Assign the DATA_1 pin for the LCD_CAM.
|
|
///
|
|
/// Sets the specified pin to push-pull output and connects it to the DATA_1
|
|
/// signal.
|
|
pub fn with_data1<S: PeripheralOutput>(self, pin: impl Peripheral<P = S> + 'd) -> Self {
|
|
crate::into_mapped_ref!(pin);
|
|
pin.set_to_push_pull_output(crate::private::Internal);
|
|
OutputSignal::LCD_DATA_1.connect_to(pin);
|
|
|
|
self
|
|
}
|
|
|
|
/// Assign the DATA_2 pin for the LCD_CAM.
|
|
///
|
|
/// Sets the specified pin to push-pull output and connects it to the DATA_2
|
|
/// signal.
|
|
pub fn with_data2<S: PeripheralOutput>(self, pin: impl Peripheral<P = S> + 'd) -> Self {
|
|
crate::into_mapped_ref!(pin);
|
|
pin.set_to_push_pull_output(crate::private::Internal);
|
|
OutputSignal::LCD_DATA_2.connect_to(pin);
|
|
|
|
self
|
|
}
|
|
|
|
/// Assign the DATA_3 pin for the LCD_CAM.
|
|
///
|
|
/// Sets the specified pin to push-pull output and connects it to the DATA_3
|
|
/// signal.
|
|
pub fn with_data3<S: PeripheralOutput>(self, pin: impl Peripheral<P = S> + 'd) -> Self {
|
|
crate::into_mapped_ref!(pin);
|
|
pin.set_to_push_pull_output(crate::private::Internal);
|
|
OutputSignal::LCD_DATA_3.connect_to(pin);
|
|
|
|
self
|
|
}
|
|
|
|
/// Assign the DATA_4 pin for the LCD_CAM.
|
|
///
|
|
/// Sets the specified pin to push-pull output and connects it to the DATA_4
|
|
/// signal.
|
|
pub fn with_data4<S: PeripheralOutput>(self, pin: impl Peripheral<P = S> + 'd) -> Self {
|
|
crate::into_mapped_ref!(pin);
|
|
pin.set_to_push_pull_output(crate::private::Internal);
|
|
OutputSignal::LCD_DATA_4.connect_to(pin);
|
|
|
|
self
|
|
}
|
|
|
|
/// Assign the DATA_5 pin for the LCD_CAM.
|
|
///
|
|
/// Sets the specified pin to push-pull output and connects it to the DATA_5
|
|
/// signal.
|
|
pub fn with_data5<S: PeripheralOutput>(self, pin: impl Peripheral<P = S> + 'd) -> Self {
|
|
crate::into_mapped_ref!(pin);
|
|
pin.set_to_push_pull_output(crate::private::Internal);
|
|
OutputSignal::LCD_DATA_5.connect_to(pin);
|
|
|
|
self
|
|
}
|
|
|
|
/// Assign the DATA_6 pin for the LCD_CAM.
|
|
///
|
|
/// Sets the specified pin to push-pull output and connects it to the DATA_6
|
|
/// signal.
|
|
pub fn with_data6<S: PeripheralOutput>(self, pin: impl Peripheral<P = S> + 'd) -> Self {
|
|
crate::into_mapped_ref!(pin);
|
|
pin.set_to_push_pull_output(crate::private::Internal);
|
|
OutputSignal::LCD_DATA_6.connect_to(pin);
|
|
|
|
self
|
|
}
|
|
|
|
/// Assign the DATA_7 pin for the LCD_CAM.
|
|
///
|
|
/// Sets the specified pin to push-pull output and connects it to the DATA_7
|
|
/// signal.
|
|
pub fn with_data7<S: PeripheralOutput>(self, pin: impl Peripheral<P = S> + 'd) -> Self {
|
|
crate::into_mapped_ref!(pin);
|
|
pin.set_to_push_pull_output(crate::private::Internal);
|
|
OutputSignal::LCD_DATA_7.connect_to(pin);
|
|
|
|
self
|
|
}
|
|
|
|
/// Assign the DATA_8 pin for the LCD_CAM.
|
|
///
|
|
/// Sets the specified pin to push-pull output and connects it to the DATA_8
|
|
/// signal.
|
|
pub fn with_data8<S: PeripheralOutput>(self, pin: impl Peripheral<P = S> + 'd) -> Self {
|
|
crate::into_mapped_ref!(pin);
|
|
pin.set_to_push_pull_output(crate::private::Internal);
|
|
OutputSignal::LCD_DATA_8.connect_to(pin);
|
|
|
|
self
|
|
}
|
|
|
|
/// Assign the DATA_9 pin for the LCD_CAM.
|
|
///
|
|
/// Sets the specified pin to push-pull output and connects it to the DATA_9
|
|
/// signal.
|
|
pub fn with_data9<S: PeripheralOutput>(self, pin: impl Peripheral<P = S> + 'd) -> Self {
|
|
crate::into_mapped_ref!(pin);
|
|
pin.set_to_push_pull_output(crate::private::Internal);
|
|
OutputSignal::LCD_DATA_9.connect_to(pin);
|
|
|
|
self
|
|
}
|
|
|
|
/// Assign the DATA_10 pin for the LCD_CAM.
|
|
///
|
|
/// Sets the specified pin to push-pull output and connects it to the
|
|
/// DATA_10 signal.
|
|
pub fn with_data10<S: PeripheralOutput>(self, pin: impl Peripheral<P = S> + 'd) -> Self {
|
|
crate::into_mapped_ref!(pin);
|
|
pin.set_to_push_pull_output(crate::private::Internal);
|
|
OutputSignal::LCD_DATA_10.connect_to(pin);
|
|
|
|
self
|
|
}
|
|
|
|
/// Assign the DATA_11 pin for the LCD_CAM.
|
|
///
|
|
/// Sets the specified pin to push-pull output and connects it to the
|
|
/// DATA_11 signal.
|
|
pub fn with_data11<S: PeripheralOutput>(self, pin: impl Peripheral<P = S> + 'd) -> Self {
|
|
crate::into_mapped_ref!(pin);
|
|
pin.set_to_push_pull_output(crate::private::Internal);
|
|
OutputSignal::LCD_DATA_11.connect_to(pin);
|
|
|
|
self
|
|
}
|
|
|
|
/// Assign the DATA_12 pin for the LCD_CAM.
|
|
///
|
|
/// Sets the specified pin to push-pull output and connects it to the
|
|
/// DATA_12 signal.
|
|
pub fn with_data12<S: PeripheralOutput>(self, pin: impl Peripheral<P = S> + 'd) -> Self {
|
|
crate::into_mapped_ref!(pin);
|
|
pin.set_to_push_pull_output(crate::private::Internal);
|
|
OutputSignal::LCD_DATA_12.connect_to(pin);
|
|
|
|
self
|
|
}
|
|
|
|
/// Assign the DATA_13 pin for the LCD_CAM.
|
|
///
|
|
/// Sets the specified pin to push-pull output and connects it to the
|
|
/// DATA_13 signal.
|
|
pub fn with_data13<S: PeripheralOutput>(self, pin: impl Peripheral<P = S> + 'd) -> Self {
|
|
crate::into_mapped_ref!(pin);
|
|
pin.set_to_push_pull_output(crate::private::Internal);
|
|
OutputSignal::LCD_DATA_13.connect_to(pin);
|
|
|
|
self
|
|
}
|
|
|
|
/// Assign the DATA_14 pin for the LCD_CAM.
|
|
///
|
|
/// Sets the specified pin to push-pull output and connects it to the
|
|
/// DATA_14 signal.
|
|
pub fn with_data14<S: PeripheralOutput>(self, pin: impl Peripheral<P = S> + 'd) -> Self {
|
|
crate::into_mapped_ref!(pin);
|
|
pin.set_to_push_pull_output(crate::private::Internal);
|
|
OutputSignal::LCD_DATA_14.connect_to(pin);
|
|
|
|
self
|
|
}
|
|
|
|
/// Assign the DATA_15 pin for the LCD_CAM.
|
|
///
|
|
/// Sets the specified pin to push-pull output and connects it to the
|
|
/// DATA_15 signal.
|
|
pub fn with_data15<S: PeripheralOutput>(self, pin: impl Peripheral<P = S> + 'd) -> Self {
|
|
crate::into_mapped_ref!(pin);
|
|
pin.set_to_push_pull_output(crate::private::Internal);
|
|
OutputSignal::LCD_DATA_15.connect_to(pin);
|
|
|
|
self
|
|
}
|
|
|
|
/// Sending out the [DmaTxBuffer] to the RGB/DPI interface.
|
|
///
|
|
/// - `next_frame_en`: Automatically send the next frame data when the
|
|
/// current frame is sent out.
|
|
pub fn send<TX: DmaTxBuffer>(
|
|
mut self,
|
|
next_frame_en: bool,
|
|
mut buf: TX,
|
|
) -> Result<DpiTransfer<'d, TX>, (DmaError, Self, TX)> {
|
|
let result = unsafe {
|
|
self.tx_channel
|
|
.prepare_transfer(DmaPeripheral::LcdCam, &mut buf)
|
|
}
|
|
.and_then(|_| self.tx_channel.start_transfer());
|
|
if let Err(err) = result {
|
|
return Err((err, self, buf));
|
|
}
|
|
|
|
// Reset LCD control unit and Async Tx FIFO
|
|
self.lcd_cam
|
|
.lcd_user()
|
|
.modify(|_, w| w.lcd_reset().set_bit());
|
|
self.lcd_cam
|
|
.lcd_misc()
|
|
.modify(|_, w| w.lcd_afifo_reset().set_bit());
|
|
|
|
self.lcd_cam.lcd_misc().modify(|_, w| {
|
|
// 1: Send the next frame data when the current frame is sent out.
|
|
// 0: LCD stops when the current frame is sent out.
|
|
w.lcd_next_frame_en().bit(next_frame_en)
|
|
});
|
|
|
|
// Start the transfer.
|
|
self.lcd_cam.lcd_user().modify(|_, w| {
|
|
w.lcd_update().set_bit();
|
|
w.lcd_start().set_bit()
|
|
});
|
|
|
|
Ok(DpiTransfer {
|
|
dpi: ManuallyDrop::new(self),
|
|
buffer_view: ManuallyDrop::new(buf.into_view()),
|
|
})
|
|
}
|
|
}
|
|
|
|
/// Represents an ongoing (or potentially finished) transfer using the RGB LCD
|
|
/// interface
|
|
pub struct DpiTransfer<'d, BUF: DmaTxBuffer> {
|
|
dpi: ManuallyDrop<Dpi<'d>>,
|
|
buffer_view: ManuallyDrop<BUF::View>,
|
|
}
|
|
|
|
impl<'d, BUF: DmaTxBuffer> DpiTransfer<'d, BUF> {
|
|
/// Returns true when [Self::wait] will not block.
|
|
pub fn is_done(&self) -> bool {
|
|
self.dpi
|
|
.lcd_cam
|
|
.lcd_user()
|
|
.read()
|
|
.lcd_start()
|
|
.bit_is_clear()
|
|
}
|
|
|
|
/// Stops this transfer on the spot and returns the peripheral and buffer.
|
|
pub fn stop(mut self) -> (Dpi<'d>, BUF) {
|
|
self.stop_peripherals();
|
|
let (dpi, view) = self.release();
|
|
(dpi, BUF::from_view(view))
|
|
}
|
|
|
|
/// Waits for the transfer to finish and returns the peripheral and buffer.
|
|
///
|
|
/// Note: If you specified `next_frame_en` as true in [Dpi::send], you're
|
|
/// just waiting for a DMA error when you call this.
|
|
pub fn wait(mut self) -> (Result<(), DmaError>, Dpi<'d>, BUF) {
|
|
while !self.is_done() {
|
|
core::hint::spin_loop();
|
|
}
|
|
|
|
// Stop the DMA.
|
|
//
|
|
// If the user sends more data to the DMA than the LCD_CAM needs for a single
|
|
// frame, the DMA will still be running after the LCD_CAM stops.
|
|
self.dpi.tx_channel.stop_transfer();
|
|
|
|
// Note: There is no "done" interrupt to clear.
|
|
|
|
let (dpi, view) = self.release();
|
|
let result = if dpi.tx_channel.has_error() {
|
|
Err(DmaError::DescriptorError)
|
|
} else {
|
|
Ok(())
|
|
};
|
|
|
|
(result, dpi, BUF::from_view(view))
|
|
}
|
|
|
|
fn release(mut self) -> (Dpi<'d>, BUF::View) {
|
|
// SAFETY: Since forget is called on self, we know that self.dpi and
|
|
// self.buffer_view won't be touched again.
|
|
let result = unsafe {
|
|
let dpi = ManuallyDrop::take(&mut self.dpi);
|
|
let view = ManuallyDrop::take(&mut self.buffer_view);
|
|
(dpi, view)
|
|
};
|
|
core::mem::forget(self);
|
|
result
|
|
}
|
|
|
|
fn stop_peripherals(&mut self) {
|
|
// Stop the LCD_CAM peripheral.
|
|
self.dpi
|
|
.lcd_cam
|
|
.lcd_user()
|
|
.modify(|_, w| w.lcd_start().clear_bit());
|
|
|
|
// Stop the DMA
|
|
self.dpi.tx_channel.stop_transfer();
|
|
}
|
|
}
|
|
|
|
impl<'d, BUF: DmaTxBuffer> Deref for DpiTransfer<'d, BUF> {
|
|
type Target = BUF::View;
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
&self.buffer_view
|
|
}
|
|
}
|
|
|
|
impl<'d, BUF: DmaTxBuffer> DerefMut for DpiTransfer<'d, BUF> {
|
|
fn deref_mut(&mut self) -> &mut Self::Target {
|
|
&mut self.buffer_view
|
|
}
|
|
}
|
|
|
|
impl<'d, BUF: DmaTxBuffer> Drop for DpiTransfer<'d, BUF> {
|
|
fn drop(&mut self) {
|
|
self.stop_peripherals();
|
|
|
|
// SAFETY: This is Drop, we know that self.dpi and self.buf_view
|
|
// won't be touched again.
|
|
let view = unsafe {
|
|
ManuallyDrop::drop(&mut self.dpi);
|
|
ManuallyDrop::take(&mut self.buffer_view)
|
|
};
|
|
let _ = BUF::from_view(view);
|
|
}
|
|
}
|
|
|
|
/// Configuration settings for the RGB/DPI interface.
|
|
#[non_exhaustive]
|
|
#[derive(Debug, Clone, Copy, PartialEq)]
|
|
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
|
|
pub struct Config {
|
|
/// Specifies the clock mode, including polarity and phase settings.
|
|
pub clock_mode: ClockMode,
|
|
|
|
/// Format of the byte data sent out.
|
|
pub format: Format,
|
|
|
|
/// Timing settings for the peripheral.
|
|
pub timing: FrameTiming,
|
|
|
|
/// The vsync signal level in IDLE state.
|
|
pub vsync_idle_level: Level,
|
|
|
|
/// The hsync signal level in IDLE state.
|
|
pub hsync_idle_level: Level,
|
|
|
|
/// The de signal level in IDLE state.
|
|
pub de_idle_level: Level,
|
|
|
|
/// If enabled, the hsync pulse will be sent out in vertical blanking lines.
|
|
/// i.e. When no valid data is actually sent out. Otherwise, hysnc
|
|
/// pulses will only be sent out in active region lines.
|
|
pub hs_blank_en: bool,
|
|
|
|
/// Disables blank region when LCD sends data out.
|
|
pub disable_black_region: bool,
|
|
|
|
/// The output LCD_DE is delayed by module clock LCD_CLK.
|
|
pub de_mode: DelayMode,
|
|
/// The output LCD_HSYNC is delayed by module clock LCD_CLK.
|
|
pub hsync_mode: DelayMode,
|
|
/// The output LCD_VSYNC is delayed by module clock LCD_CLK.
|
|
pub vsync_mode: DelayMode,
|
|
/// The output data bits are delayed by module clock LCD_CLK.
|
|
pub output_bit_mode: DelayMode,
|
|
}
|
|
|
|
impl Default for Config {
|
|
fn default() -> Self {
|
|
Config {
|
|
clock_mode: Default::default(),
|
|
format: Default::default(),
|
|
timing: Default::default(),
|
|
vsync_idle_level: Level::Low,
|
|
hsync_idle_level: Level::Low,
|
|
de_idle_level: Level::Low,
|
|
hs_blank_en: true,
|
|
disable_black_region: false,
|
|
de_mode: Default::default(),
|
|
hsync_mode: Default::default(),
|
|
vsync_mode: Default::default(),
|
|
output_bit_mode: Default::default(),
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Controls how the peripheral should treat data received from the DMA.
|
|
#[derive(Debug, Clone, Copy, PartialEq, Default)]
|
|
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
|
|
pub struct Format {
|
|
/// Configures the bit order for data transmission.
|
|
pub bit_order: BitOrder,
|
|
|
|
/// Configures the byte order for data transmission.
|
|
///
|
|
/// - In 8-bit mode, [ByteOrder::Inverted] means every two bytes are
|
|
/// swapped.
|
|
/// - In 16-bit mode, this controls the byte order (endianness).
|
|
pub byte_order: ByteOrder,
|
|
|
|
/// If true, the width of the output is 16 bits.
|
|
/// Otherwise, the width of the output is 8 bits.
|
|
pub enable_2byte_mode: bool,
|
|
}
|
|
|
|
/// The timing numbers for the driver to follow.
|
|
///
|
|
/// Note: The names of the fields in this struct don't match what you
|
|
/// would typically find in an LCD's datasheet. Carefully read the doc on each
|
|
/// field to understand what to set it to.
|
|
#[derive(Debug, Clone, Copy, PartialEq, Default)]
|
|
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
|
|
pub struct FrameTiming {
|
|
/// The horizontal total width of a frame (in units of PCLK).
|
|
///
|
|
/// This should be greater than `horizontal_blank_front_porch` +
|
|
/// `horizontal_active_width`.
|
|
///
|
|
/// Max is 4096 (12 bits).
|
|
pub horizontal_total_width: usize,
|
|
|
|
/// The horizontal blank front porch of a frame (in units of PCLK).
|
|
///
|
|
/// This is the number of PCLKs between the start of the line and the start
|
|
/// of active data in the line.
|
|
///
|
|
/// Note: This includes `hsync_width`.
|
|
///
|
|
/// Max is 2048 (11 bits).
|
|
pub horizontal_blank_front_porch: usize,
|
|
|
|
/// The horizontal active width of a frame. i.e. The number of pixels in a
|
|
/// line. This is typically the horizontal resolution of the screen.
|
|
///
|
|
/// Max is 4096 (12 bits).
|
|
pub horizontal_active_width: usize,
|
|
|
|
/// The vertical total height of a frame (in units of lines).
|
|
///
|
|
/// This should be greater than `vertical_blank_front_porch` +
|
|
/// `vertical_active_height`.
|
|
///
|
|
/// Max is 1024 (10 bits).
|
|
pub vertical_total_height: usize,
|
|
|
|
/// The vertical blank front porch height of a frame (in units of lines).
|
|
///
|
|
/// This is the number of (blank/invalid) lines before the start of the
|
|
/// frame.
|
|
///
|
|
/// Note: This includes `vsync_width`.
|
|
///
|
|
/// Max is 256 (8 bits).
|
|
pub vertical_blank_front_porch: usize,
|
|
|
|
/// The vertical active height of a frame. i.e. The number of lines in a
|
|
/// frame. This is typically the vertical resolution of the screen.
|
|
///
|
|
/// Max is 1024 (10 bits).
|
|
pub vertical_active_height: usize,
|
|
|
|
/// It is the width of LCD_VSYNC active pulse in a line (in units of lines).
|
|
///
|
|
/// Max is 128 (7 bits).
|
|
pub vsync_width: usize,
|
|
|
|
/// The width of LCD_HSYNC active pulse in a line (in units of PCLK).
|
|
///
|
|
/// This should be less than vertical_blank_front_porch, otherwise the hsync
|
|
/// pulse will overlap with valid pixel data.
|
|
///
|
|
/// Max is 128 (7 bits).
|
|
pub hsync_width: usize,
|
|
|
|
/// It is the position of LCD_HSYNC active pulse in a line (in units of
|
|
/// PCLK).
|
|
///
|
|
/// This should be less than horizontal_total_width.
|
|
///
|
|
/// Max is 128 (7 bits).
|
|
pub hsync_position: usize,
|
|
}
|