2024-12-15 20:49:22 +00:00
|
|
|
#![no_std]
|
|
|
|
|
#![no_main]
|
|
|
|
|
|
2024-12-20 16:47:22 +00:00
|
|
|
use core::cell::RefCell;
|
2024-12-15 21:40:00 +00:00
|
|
|
use display_interface_spi::SPIInterface;
|
2024-12-20 16:47:22 +00:00
|
|
|
use embassy_embedded_hal::shared_bus::blocking::spi::SpiDevice;
|
2024-12-15 20:49:22 +00:00
|
|
|
use embassy_executor::Spawner;
|
2024-12-20 16:47:22 +00:00
|
|
|
use embassy_nrf::{
|
|
|
|
|
bind_interrupts,
|
2024-12-21 09:30:14 +00:00
|
|
|
gpio::{AnyPin, Level, Output, OutputDrive, Pin},
|
2024-12-21 10:10:20 +00:00
|
|
|
peripherals, spim,
|
2024-12-20 16:47:22 +00:00
|
|
|
};
|
|
|
|
|
use embassy_sync::blocking_mutex::Mutex;
|
2024-12-21 10:10:20 +00:00
|
|
|
use embassy_time::{Duration, Timer};
|
2024-12-15 21:40:00 +00:00
|
|
|
use ili9341::Ili9341;
|
2024-12-15 20:49:22 +00:00
|
|
|
use {defmt_rtt as _, panic_probe as _};
|
|
|
|
|
|
2024-12-15 21:40:00 +00:00
|
|
|
use embedded_graphics::{
|
2024-12-21 09:40:44 +00:00
|
|
|
mono_font::{ascii::FONT_10X20, MonoTextStyle},
|
2024-12-21 10:10:20 +00:00
|
|
|
pixelcolor::Rgb565,
|
2024-12-20 16:47:22 +00:00
|
|
|
prelude::*,
|
|
|
|
|
primitives::{
|
2024-12-15 21:40:00 +00:00
|
|
|
Circle, PrimitiveStyle, PrimitiveStyleBuilder, Rectangle, StrokeAlignment, Triangle,
|
2024-12-20 16:47:22 +00:00
|
|
|
},
|
|
|
|
|
text::{Alignment, Text},
|
2024-12-15 21:40:00 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Setup interrupts
|
|
|
|
|
bind_interrupts!(struct Irqs {
|
|
|
|
|
// Setup interrupts for SPI (SERIAL3)
|
|
|
|
|
UARTE3_SPIM3_SPIS3_TWIM3_TWIS3 => embassy_nrf::spim::InterruptHandler<peripherals::SERIAL3>;
|
|
|
|
|
});
|
|
|
|
|
|
2024-12-21 10:16:31 +00:00
|
|
|
/// Embassy task to blink a LED
|
|
|
|
|
///
|
|
|
|
|
/// # Arguments
|
|
|
|
|
/// pin - Any valid pin is accepted
|
|
|
|
|
/// blink_delay - a `Duration` with the amount of time
|
|
|
|
|
/// to wait between LED toggles.
|
2024-12-21 09:30:14 +00:00
|
|
|
#[embassy_executor::task]
|
|
|
|
|
async fn blink(pin: AnyPin, blink_delay: Duration) -> ! {
|
|
|
|
|
let mut led = Output::new(pin, Level::Low, OutputDrive::Standard);
|
|
|
|
|
|
|
|
|
|
loop {
|
|
|
|
|
led.set_high();
|
|
|
|
|
Timer::after(blink_delay).await;
|
|
|
|
|
led.set_low();
|
|
|
|
|
Timer::after(blink_delay).await;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-12-21 10:16:31 +00:00
|
|
|
/// Create a function to run the demo
|
|
|
|
|
///
|
|
|
|
|
/// # Arguments
|
|
|
|
|
/// This function takes in a variable of a generic type `D`.
|
|
|
|
|
/// With the restriction that the generic type `D` implements
|
|
|
|
|
/// `DrawTarget`, aka that it is a display.
|
|
|
|
|
///
|
|
|
|
|
/// I've also said it needs to use a certain colormode, because
|
|
|
|
|
/// that is the color mode I use in the demo.
|
|
|
|
|
///
|
|
|
|
|
/// While this may seem a little clunky this function now accepts _any_
|
|
|
|
|
/// display as long as it uses the standard 565 colormode.
|
|
|
|
|
///
|
|
|
|
|
/// # Returns
|
|
|
|
|
/// The function returns a `Result` containing either an empty tuple
|
|
|
|
|
/// or the error type associated with the generic type `D`.
|
2024-12-21 10:09:21 +00:00
|
|
|
fn display_demo<D>(display: &mut D) -> Result<(), D::Error>
|
|
|
|
|
where
|
|
|
|
|
D: DrawTarget<Color = Rgb565>,
|
|
|
|
|
{
|
|
|
|
|
display.clear(Rgb565::CSS_DARK_OLIVE_GREEN)?;
|
2024-12-15 21:40:00 +00:00
|
|
|
|
2024-12-21 09:32:57 +00:00
|
|
|
// Create styles used by the drawing operations.
|
|
|
|
|
let thin_stroke = PrimitiveStyle::with_stroke(Rgb565::WHITE, 1);
|
|
|
|
|
let thick_stroke = PrimitiveStyle::with_stroke(Rgb565::WHITE, 3);
|
|
|
|
|
let border_stroke = PrimitiveStyleBuilder::new()
|
|
|
|
|
.stroke_color(Rgb565::WHITE)
|
|
|
|
|
.stroke_width(3)
|
|
|
|
|
.stroke_alignment(StrokeAlignment::Inside)
|
|
|
|
|
.build();
|
|
|
|
|
let fill = PrimitiveStyle::with_fill(Rgb565::WHITE);
|
2024-12-21 09:40:44 +00:00
|
|
|
let character_style = MonoTextStyle::new(&FONT_10X20, Rgb565::WHITE);
|
2024-12-21 09:32:57 +00:00
|
|
|
|
|
|
|
|
let yoffset = 10;
|
|
|
|
|
|
|
|
|
|
// Draw a 3px wide outline around the display.
|
|
|
|
|
display
|
|
|
|
|
.bounding_box()
|
|
|
|
|
.into_styled(border_stroke)
|
2024-12-21 10:09:21 +00:00
|
|
|
.draw(display)?;
|
2024-12-21 09:32:57 +00:00
|
|
|
|
|
|
|
|
// Draw a triangle.
|
|
|
|
|
Triangle::new(
|
|
|
|
|
Point::new(16, 16 + yoffset),
|
|
|
|
|
Point::new(16 + 16, 16 + yoffset),
|
|
|
|
|
Point::new(16 + 8, yoffset),
|
|
|
|
|
)
|
|
|
|
|
.into_styled(thin_stroke)
|
2024-12-21 10:09:21 +00:00
|
|
|
.draw(display)?;
|
2024-12-21 09:32:57 +00:00
|
|
|
|
|
|
|
|
// Draw a filled square
|
|
|
|
|
Rectangle::new(Point::new(52, yoffset), Size::new(16, 16))
|
|
|
|
|
.into_styled(fill)
|
2024-12-21 10:09:21 +00:00
|
|
|
.draw(display)?;
|
2024-12-21 09:32:57 +00:00
|
|
|
|
|
|
|
|
// Draw a circle with a 3px wide stroke.
|
|
|
|
|
Circle::new(Point::new(88, yoffset), 17)
|
|
|
|
|
.into_styled(thick_stroke)
|
2024-12-21 10:09:21 +00:00
|
|
|
.draw(display)?;
|
2024-12-21 09:32:57 +00:00
|
|
|
|
|
|
|
|
// Draw centered text.
|
|
|
|
|
let text = "embedded-graphics";
|
|
|
|
|
Text::with_alignment(
|
|
|
|
|
text,
|
|
|
|
|
display.bounding_box().center() + Point::new(0, 15),
|
|
|
|
|
character_style,
|
|
|
|
|
Alignment::Center,
|
|
|
|
|
)
|
2024-12-21 10:09:21 +00:00
|
|
|
.draw(display)?;
|
|
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
2024-12-15 21:40:00 +00:00
|
|
|
|
2024-12-21 10:09:21 +00:00
|
|
|
#[embassy_executor::main]
|
|
|
|
|
async fn main(spawner: Spawner) {
|
|
|
|
|
// Initialize the nRF
|
|
|
|
|
let p = embassy_nrf::init(Default::default());
|
|
|
|
|
|
|
|
|
|
// Spawn the blink task
|
|
|
|
|
spawner
|
2024-12-21 10:16:31 +00:00
|
|
|
.spawn(
|
|
|
|
|
// Here we give the task to spawn, `.degrade()` turns the specific
|
|
|
|
|
// pin on the nRF into the `anyPin` type used above.
|
|
|
|
|
// It is basically a conversion between hardware pin and generic pin.
|
|
|
|
|
blink(p.P0_02.degrade(), Duration::from_millis(250)),
|
|
|
|
|
)
|
2024-12-21 10:09:21 +00:00
|
|
|
.expect("Failed to start blink task");
|
|
|
|
|
|
|
|
|
|
////////////////
|
|
|
|
|
// Setup pins //
|
|
|
|
|
////////////////
|
|
|
|
|
|
|
|
|
|
let mosi = p.P0_13;
|
|
|
|
|
let miso = p.P0_12;
|
|
|
|
|
let sck = p.P0_30;
|
|
|
|
|
let cs_pin = p.P0_31;
|
|
|
|
|
let dc_pin = p.P0_20; // Data / Command pin for the display
|
|
|
|
|
let disp_rst_pin = p.P0_11;
|
|
|
|
|
|
|
|
|
|
// Initialize the CS, DC, and display reset pins as outputs
|
|
|
|
|
let chip_select = Output::new(cs_pin, Level::High, OutputDrive::Standard);
|
|
|
|
|
let direction_control = Output::new(dc_pin, Level::High, OutputDrive::Standard);
|
|
|
|
|
let display_reset = Output::new(disp_rst_pin, Level::High, OutputDrive::Standard);
|
|
|
|
|
|
|
|
|
|
///////////////
|
|
|
|
|
// Setup SPI //
|
|
|
|
|
///////////////
|
|
|
|
|
|
|
|
|
|
// Create an SPI config with the SPI frequency at 8MHz
|
|
|
|
|
let mut spim_config = spim::Config::default();
|
|
|
|
|
spim_config.frequency = spim::Frequency::M8;
|
|
|
|
|
|
|
|
|
|
// Grab the SPI Master peripheral form the nRF60
|
|
|
|
|
let spim = embassy_nrf::spim::Spim::new(p.SERIAL3, Irqs, sck, miso, mosi, spim_config);
|
|
|
|
|
// Use the SPI peripheral to create a shared SPI bus
|
|
|
|
|
let spi_bus: Mutex<embassy_sync::blocking_mutex::raw::NoopRawMutex, _> =
|
|
|
|
|
Mutex::new(RefCell::new(spim));
|
|
|
|
|
|
|
|
|
|
// Use the shared bus to create a SPI Device to use for the display
|
|
|
|
|
let spi_display = SpiDevice::new(&spi_bus, chip_select);
|
|
|
|
|
// Create a new SPI display interface
|
|
|
|
|
let iface = SPIInterface::new(spi_display, direction_control);
|
|
|
|
|
|
|
|
|
|
////////////////////////
|
|
|
|
|
// Create the display //
|
|
|
|
|
////////////////////////
|
|
|
|
|
|
|
|
|
|
// Some options for the display
|
|
|
|
|
let display_orientation = ili9341::Orientation::Landscape;
|
|
|
|
|
let display_size = ili9341::DisplaySize240x320;
|
2024-12-15 20:49:22 +00:00
|
|
|
|
2024-12-21 10:09:21 +00:00
|
|
|
// Create the display
|
|
|
|
|
let mut display = Ili9341::new(
|
|
|
|
|
iface,
|
|
|
|
|
display_reset,
|
|
|
|
|
&mut embassy_time::Delay,
|
|
|
|
|
display_orientation,
|
|
|
|
|
display_size,
|
|
|
|
|
)
|
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
|
|
// Now loop, re-running the demo every second
|
2024-12-15 20:49:22 +00:00
|
|
|
loop {
|
2024-12-21 10:09:21 +00:00
|
|
|
display_demo(&mut display).expect("Failed to run display demo");
|
|
|
|
|
Timer::after_secs(1).await;
|
2024-12-15 20:49:22 +00:00
|
|
|
}
|
2024-12-20 16:47:22 +00:00
|
|
|
}
|