commit
						25b870d811
					
				| @ -111,6 +111,8 @@ impl<'d, T: Instance> Spi<'d, T> { | |||||||
|             for &b in data { |             for &b in data { | ||||||
|                 while !p.sr().read().tnf() {} |                 while !p.sr().read().tnf() {} | ||||||
|                 p.dr().write(|w| w.set_data(b as _)); |                 p.dr().write(|w| w.set_data(b as _)); | ||||||
|  |                 while !p.sr().read().rne() {} | ||||||
|  |                 let _ = p.dr().read(); | ||||||
|             } |             } | ||||||
|             self.flush(); |             self.flush(); | ||||||
|         } |         } | ||||||
| @ -140,10 +142,17 @@ impl<'d, T: Instance> Spi<'d, T> { | |||||||
|         let (presc, postdiv) = calc_prescs(freq); |         let (presc, postdiv) = calc_prescs(freq); | ||||||
|         let p = self.inner.regs(); |         let p = self.inner.regs(); | ||||||
|         unsafe { |         unsafe { | ||||||
|  |             // disable
 | ||||||
|  |             p.cr1().write(|w| w.set_sse(false)); | ||||||
|  | 
 | ||||||
|  |             // change stuff
 | ||||||
|             p.cpsr().write(|w| w.set_cpsdvsr(presc)); |             p.cpsr().write(|w| w.set_cpsdvsr(presc)); | ||||||
|             p.cr0().modify(|w| { |             p.cr0().modify(|w| { | ||||||
|                 w.set_scr(postdiv); |                 w.set_scr(postdiv); | ||||||
|             }); |             }); | ||||||
|  | 
 | ||||||
|  |             // enable
 | ||||||
|  |             p.cr1().write(|w| w.set_sse(true)); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -30,3 +30,6 @@ cortex-m-rt = "0.6.13" | |||||||
| embedded-hal    = { version = "0.2.4" } | embedded-hal    = { version = "0.2.4" } | ||||||
| panic-probe = { version = "0.2.0", features = ["print-defmt"] } | panic-probe = { version = "0.2.0", features = ["print-defmt"] } | ||||||
| futures = { version = "0.3.8", default-features = false, features = ["async-await", "cfg-target-has-atomic", "unstable"] } | futures = { version = "0.3.8", default-features = false, features = ["async-await", "cfg-target-has-atomic", "unstable"] } | ||||||
|  | display-interface-spi = "0.4.1" | ||||||
|  | embedded-graphics = "0.7.1" | ||||||
|  | st7789 = "0.6.1" | ||||||
|  | |||||||
							
								
								
									
										
											BIN
										
									
								
								examples/rp/assets/ferris.raw
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								examples/rp/assets/ferris.raw
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										215
									
								
								examples/rp/src/bin/spi_display.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										215
									
								
								examples/rp/src/bin/spi_display.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,215 @@ | |||||||
|  | #![no_std] | ||||||
|  | #![no_main] | ||||||
|  | #![feature(asm)] | ||||||
|  | #![feature(min_type_alias_impl_trait)] | ||||||
|  | #![feature(impl_trait_in_bindings)] | ||||||
|  | #![feature(type_alias_impl_trait)] | ||||||
|  | #![allow(incomplete_features)] | ||||||
|  | 
 | ||||||
|  | #[path = "../example_common.rs"] | ||||||
|  | mod example_common; | ||||||
|  | 
 | ||||||
|  | use core::cell::RefCell; | ||||||
|  | use core::fmt::Debug; | ||||||
|  | 
 | ||||||
|  | use defmt::*; | ||||||
|  | use display_interface_spi::SPIInterfaceNoCS; | ||||||
|  | use embassy::executor::Spawner; | ||||||
|  | use embassy::time::Delay; | ||||||
|  | use embassy_rp::gpio::NoPin; | ||||||
|  | use embassy_rp::peripherals; | ||||||
|  | use embassy_rp::spi; | ||||||
|  | use embassy_rp::spi::Spi; | ||||||
|  | use embassy_rp::{gpio, Peripherals}; | ||||||
|  | use embedded_graphics::image::{Image, ImageRawLE}; | ||||||
|  | use embedded_graphics::mono_font::ascii::FONT_10X20; | ||||||
|  | use embedded_graphics::mono_font::MonoTextStyle; | ||||||
|  | use embedded_graphics::pixelcolor::Rgb565; | ||||||
|  | use embedded_graphics::prelude::*; | ||||||
|  | use embedded_graphics::primitives::{PrimitiveStyleBuilder, Rectangle}; | ||||||
|  | use embedded_graphics::text::Text; | ||||||
|  | use gpio::{Level, Output}; | ||||||
|  | use st7789::{Orientation, ST7789}; | ||||||
|  | 
 | ||||||
|  | #[embassy::main] | ||||||
|  | async fn main(_spawner: Spawner, p: Peripherals) { | ||||||
|  |     info!("Hello World!"); | ||||||
|  | 
 | ||||||
|  |     let bl = p.PIN_13; | ||||||
|  |     let rst = p.PIN_15; | ||||||
|  |     let display_cs = p.PIN_9; | ||||||
|  |     let dcx = p.PIN_8; | ||||||
|  |     let miso = p.PIN_12; | ||||||
|  |     let mosi = p.PIN_11; | ||||||
|  |     let clk = p.PIN_10; | ||||||
|  |     let touch_cs = p.PIN_16; | ||||||
|  |     //let touch_irq = p.PIN_17;
 | ||||||
|  | 
 | ||||||
|  |     // create SPI
 | ||||||
|  |     let mut config = spi::Config::default(); | ||||||
|  |     config.frequency = DISPLAY_FREQ; | ||||||
|  |     config.phase = spi::Phase::CaptureOnSecondTransition; | ||||||
|  |     config.polarity = spi::Polarity::IdleHigh; | ||||||
|  | 
 | ||||||
|  |     let spi = RefCell::new(SpiState { | ||||||
|  |         last_mode: SpiMode::Display, | ||||||
|  |         spi: Spi::new(p.SPI1, clk, mosi, miso, NoPin, config), | ||||||
|  |         display_cs: Output::new(display_cs, Level::Low), | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     let mut touch = Touch::new(TouchSpi(&spi), Output::new(touch_cs, Level::High)); | ||||||
|  | 
 | ||||||
|  |     let dcx = Output::new(dcx, Level::Low); | ||||||
|  |     let rst = Output::new(rst, Level::Low); | ||||||
|  |     // dcx: 0 = command, 1 = data
 | ||||||
|  | 
 | ||||||
|  |     // Enable LCD backlight
 | ||||||
|  |     let _bl = Output::new(bl, Level::High); | ||||||
|  | 
 | ||||||
|  |     // display interface abstraction from SPI and DC
 | ||||||
|  |     let di = SPIInterfaceNoCS::new(DisplaySpi(&spi), dcx); | ||||||
|  | 
 | ||||||
|  |     // create driver
 | ||||||
|  |     let mut display = ST7789::new(di, rst, 240, 320); | ||||||
|  | 
 | ||||||
|  |     // initialize
 | ||||||
|  |     display.init(&mut Delay).unwrap(); | ||||||
|  | 
 | ||||||
|  |     // set default orientation
 | ||||||
|  |     display.set_orientation(Orientation::Landscape).unwrap(); | ||||||
|  | 
 | ||||||
|  |     display.clear(Rgb565::BLACK).unwrap(); | ||||||
|  | 
 | ||||||
|  |     let raw_image_data = ImageRawLE::new(include_bytes!("../../assets/ferris.raw"), 86); | ||||||
|  |     let ferris = Image::new(&raw_image_data, Point::new(34, 68)); | ||||||
|  | 
 | ||||||
|  |     // Display the image
 | ||||||
|  |     ferris.draw(&mut display).unwrap(); | ||||||
|  | 
 | ||||||
|  |     let style = MonoTextStyle::new(&FONT_10X20, Rgb565::GREEN); | ||||||
|  |     Text::new( | ||||||
|  |         "Hello embedded_graphics \n + embassy + RP2040!", | ||||||
|  |         Point::new(20, 200), | ||||||
|  |         style, | ||||||
|  |     ) | ||||||
|  |     .draw(&mut display) | ||||||
|  |     .unwrap(); | ||||||
|  | 
 | ||||||
|  |     loop { | ||||||
|  |         if let Some((x, y)) = touch.read() { | ||||||
|  |             let style = PrimitiveStyleBuilder::new() | ||||||
|  |                 .fill_color(Rgb565::BLUE) | ||||||
|  |                 .build(); | ||||||
|  | 
 | ||||||
|  |             Rectangle::new(Point::new(x - 1, y - 1), Size::new(3, 3)) | ||||||
|  |                 .into_styled(style) | ||||||
|  |                 .draw(&mut display) | ||||||
|  |                 .unwrap(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[derive(Debug, Clone, Copy, PartialEq, Eq)] | ||||||
|  | enum SpiMode { | ||||||
|  |     Display, | ||||||
|  |     Touch, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct SpiState { | ||||||
|  |     spi: Spi<'static, peripherals::SPI1>, | ||||||
|  |     display_cs: Output<'static, peripherals::PIN_9>, | ||||||
|  | 
 | ||||||
|  |     last_mode: SpiMode, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const DISPLAY_FREQ: u32 = 64_000_000; | ||||||
|  | const TOUCH_FREQ: u32 = 200_000; | ||||||
|  | 
 | ||||||
|  | struct DisplaySpi<'a>(&'a RefCell<SpiState>); | ||||||
|  | impl<'a> embedded_hal::blocking::spi::Write<u8> for DisplaySpi<'a> { | ||||||
|  |     type Error = core::convert::Infallible; | ||||||
|  | 
 | ||||||
|  |     fn write(&mut self, words: &[u8]) -> Result<(), Self::Error> { | ||||||
|  |         let this = &mut *self.0.borrow_mut(); | ||||||
|  |         if this.last_mode != SpiMode::Display { | ||||||
|  |             this.spi.set_frequency(DISPLAY_FREQ); | ||||||
|  |             this.display_cs.set_low(); | ||||||
|  |             this.last_mode = SpiMode::Display; | ||||||
|  |         } | ||||||
|  |         this.spi.write(words); | ||||||
|  |         Ok(()) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct TouchSpi<'a>(&'a RefCell<SpiState>); | ||||||
|  | impl<'a> embedded_hal::blocking::spi::Transfer<u8> for TouchSpi<'a> { | ||||||
|  |     type Error = core::convert::Infallible; | ||||||
|  | 
 | ||||||
|  |     fn transfer<'w>(&mut self, words: &'w mut [u8]) -> Result<&'w [u8], Self::Error> { | ||||||
|  |         let this = &mut *self.0.borrow_mut(); | ||||||
|  |         if this.last_mode != SpiMode::Touch { | ||||||
|  |             this.spi.set_frequency(TOUCH_FREQ); | ||||||
|  |             this.display_cs.set_high(); | ||||||
|  |             this.last_mode = SpiMode::Touch; | ||||||
|  |         } | ||||||
|  |         this.spi.transfer(words); | ||||||
|  |         Ok(words) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct Calibration { | ||||||
|  |     x1: i32, | ||||||
|  |     x2: i32, | ||||||
|  |     y1: i32, | ||||||
|  |     y2: i32, | ||||||
|  |     sx: i32, | ||||||
|  |     sy: i32, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const CALIBRATION: Calibration = Calibration { | ||||||
|  |     x1: 3880, | ||||||
|  |     x2: 340, | ||||||
|  |     y1: 262, | ||||||
|  |     y2: 3850, | ||||||
|  |     sx: 320, | ||||||
|  |     sy: 240, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct Touch< | ||||||
|  |     SPI: embedded_hal::blocking::spi::Transfer<u8>, | ||||||
|  |     CS: embedded_hal::digital::v2::OutputPin, | ||||||
|  | > { | ||||||
|  |     spi: SPI, | ||||||
|  |     cs: CS, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<SPI: embedded_hal::blocking::spi::Transfer<u8>, CS: embedded_hal::digital::v2::OutputPin> | ||||||
|  |     Touch<SPI, CS> | ||||||
|  | where | ||||||
|  |     SPI::Error: Debug, | ||||||
|  |     CS::Error: Debug, | ||||||
|  | { | ||||||
|  |     pub fn new(spi: SPI, cs: CS) -> Self { | ||||||
|  |         Self { spi, cs } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     pub fn read(&mut self) -> Option<(i32, i32)> { | ||||||
|  |         self.cs.set_low().unwrap(); | ||||||
|  |         let mut buf = [0x90, 0x00, 0x00, 0xd0, 0x00, 0x00]; | ||||||
|  |         self.spi.transfer(&mut buf).unwrap(); | ||||||
|  |         self.cs.set_high().unwrap(); | ||||||
|  | 
 | ||||||
|  |         let x = ((buf[1] as u32) << 5 | (buf[2] as u32) >> 3) as i32; | ||||||
|  |         let y = ((buf[4] as u32) << 5 | (buf[5] as u32) >> 3) as i32; | ||||||
|  | 
 | ||||||
|  |         let cal = &CALIBRATION; | ||||||
|  | 
 | ||||||
|  |         let x = ((x - cal.x1) * cal.sx / (cal.x2 - cal.x1)).clamp(0, cal.sx); | ||||||
|  |         let y = ((y - cal.y1) * cal.sy / (cal.y2 - cal.y1)).clamp(0, cal.sy); | ||||||
|  |         if x == 0 && y == 0 { | ||||||
|  |             None | ||||||
|  |         } else { | ||||||
|  |             Some((x, y)) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user