Rust OLED显示驱动库ssd1306的使用,支持I2C和SPI接口的SSD1306显示屏控制

Rust OLED显示驱动库ssd1306的使用,支持I2C和SPI接口的SSD1306显示屏控制

以下是使用ssd1306库控制SSD1306 OLED显示屏的完整示例代码,支持I2C接口:

#![no_std]
#![no_main]

use cortex_m_rt::{entry, exception, ExceptionFrame};
use embedded_graphics::{
    image::{Image, ImageRaw},
    pixelcolor::BinaryColor,
    prelude::*,
};
use panic_halt as _;
use ssd1306::{prelude::*, I2CDisplayInterface, Ssd1306};
use stm32f1xx_hal::{
    i2c::{BlockingI2c, DutyCycle, Mode},
    prelude::*,
    stm32,
};

#[entry]
fn main() -> ! {
    let dp = stm32::Peripherals::take().unwrap();

    let mut flash = dp.FLASH.constrain();
    let mut rcc = dp.RCC.constrain();

    let clocks = rcc.cfgr.freeze(&mut flash.acr);

    let mut afio = dp.AFIO.constrain(&mut rcc.apb2);

    let mut gpiob = dp.GPIOB.split(&mut rcc.apb2);

    let scl = gpiob.pb8.into_alternate_open_drain(&mut gpiob.crh);
    let sda = gpiob.pb9.into_alternate_open_drain(&mut gpiob.crh);

    let i2c = BlockingI2c::i2c1(
        dp.I2C1,
        (scl, sda),
        &mut afio.mapr,
        Mode::Fast {
            frequency: 400_000.hz(),
            duty_cycle: DutyCycle::Ratio2to1,
        },
        clocks,
        &mut rcc.apb1,
        1000,
        10,
        1000,
        1000,
    );

    let interface = I2CDisplayInterface::new(i2c);
    let mut display = Ssd1306::new(interface, DisplaySize128x64, DisplayRotation::Rotate0)
        .into_buffered_graphics_mode();
    display.init().unwrap();

    let raw: ImageRaw<BinaryColor> = ImageRaw::new(include_bytes!("./rust.raw"), 64);

    let im = Image::new(&raw, Point::new(32, 0));

    im.draw(&mut display).unwrap();

    display.flush().unwrap();

    loop {}
}

#[exception]
fn HardFault(ef: &ExceptionFrame) -> ! {
    panic!("{:#?}", ef);
}

使用说明

  1. 首先需要添加ssd1306库到你的Cargo.toml中:

    ssd1306 = "0.10.0"
    
  2. 示例代码展示了如何通过I2C接口控制SSD1306 OLED显示屏:

    • 初始化I2C接口
    • 创建显示接口
    • 初始化显示屏
    • 显示图像
  3. 代码中使用了一个名为"rust.raw"的图像文件,你需要准备自己的图像文件或修改代码显示其他内容。

SPI接口示例

以下是使用SPI接口控制SSD1306 OLED显示屏的示例代码:

#![no_std]
#![no_main]

use cortex_m_rt::{entry, exception, ExceptionFrame};
use embedded_graphics::{
    mono_font::{ascii::FONT_6X10, MonoText},
    pixelcolor::BinaryColor,
    prelude::*,
    text::Text,
};
use panic_halt as _;
use ssd1306::{prelude::*, Ssd1306};
use stm32f1xx_hal::{
    gpio::gpiob::{PB13, PB14, PB15},
    gpio::{Alternate, Output, PushPull},
    pac,
    prelude::*,
    spi::{Mode, Phase, Polarity, Spi},
};

#[entry]
fn main() -> ! {
    let dp = pac::Peripherals::take().unwrap();
    let cp = cortex_m::Peripherals::take().unwrap();

    let mut flash = dp.FLASH.constrain();
    let mut rcc = dp.RCC.constrain();

    let clocks = rcc.cfgr.freeze(&mut flash.acr);

    let mut gpiob = dp.GPIOB.split(&mut rcc.apb2);

    // SPI1 pins
    let sck = gpiob.pb13.into_alternate_push_pull(&mut gpiob.crh);
    let miso = gpiob.pb14;
    let mosi = gpiob.pb15.into_alternate_push_pull(&mut gpiob.crh);

    // Chip Select pin
    let cs = gpiob.pb12.into_push_pull_output(&mut gpiob.crh);
    
    // Data/Command pin
    let dc = gpiob.pb1.into_push_pull_output(&mut gpiob.crl);
    
    // Reset pin
    let res = gpiob.pb0.into_push_pull_output(&mut gpiob.crl);

    let spi_mode = Mode {
        polarity: Polarity::IdleLow,
        phase: Phase::CaptureOnFirstTransition,
    };

    let spi = Spi::spi1(
        dp.SPI1,
        (sck, miso, mosi),
        spi_mode,
        8.mhz(),
        clocks,
        &mut rcc.apb2,
    );

    let interface = display_interface_spi::SPIInterface::new(spi, dc, cs);
    let mut display = Ssd1306::new(interface, DisplaySize128x64, DisplayRotation::Rotate0)
        .into_buffered_graphics_mode();
    
    display.init().unwrap();
    display.clear();

    let text = "Hello, Rust!";
    Text::new(text, Point::new(0, 16), MonoTextStyle::new(&FONT_6X10, BinaryColor::On))
        .draw(&mut display)
        .unwrap();

    display.flush().unwrap();

    loop {}
}

#[exception]
fn HardFault(ef: &ExceptionFrame) -> ! {
    panic!("{:#?}", ef);
}

主要特性

  • 支持I2C和SPI(4线)接口
  • 提供图形绘制功能
  • 支持文本显示
  • 支持图像显示
  • 支持多种显示尺寸和旋转方向

许可证

该库采用以下许可证之一:

  • Apache License, Version 2.0
  • MIT license

你可以选择其中任意一种许可证使用该库。


1 回复

Rust OLED显示驱动库ssd1306的使用指南

简介

ssd1306是Rust生态中一个用于控制SSD1306 OLED显示屏的驱动库,支持I2C和SPI两种通信接口。该库提供了简单易用的API来操作这种常见的单色OLED显示屏。

特性

  • 支持I2C和SPI接口
  • 提供基本绘图功能(点、线、矩形、圆形等)
  • 支持文本显示
  • 内置多种显示控制功能(对比度、滚动、反转等)
  • embedded-graphics库兼容

完整示例代码

I2C接口完整示例

use ssd1306::{prelude::*, I2CDisplayInterface, Ssd1306};
use linux_embedded_hal::I2cdev;
use embedded_graphics::{
    mono_font::{ascii::FONT_6X10, MonoTextStyle},
    pixelcolor::BinaryColor,
    prelude::*,
    text::Text,
    primitives::{Line, Rectangle, Circle},
    style::PrimitiveStyle,
};

fn main() -> Result<(), Box<dyn std::error::Error>> {
    // 初始化I2C设备
    let i2c = I2cdev::new("/dev/i2c-1")?;
    
    // 创建接口
    let interface = I2CDisplayInterface::new(i2c);
    
    // 创建128x64分辨率的显示驱动,旋转角度为0度
    let mut display = Ssd1306::new(
        interface,
        DisplaySize128x64,
        DisplayRotation::Rotate0,
    ).into_buffered_graphics_mode();
    
    // 初始化显示屏
    display.init()?;
    
    // 设置对比度为中等
    display.set_contrast(128)?;
    
    // 创建文本样式
    let text_style = MonoTextStyle::new(&FONT_6X10, BinaryColor::On);
    
    // 绘制文本
    Text::new("Rust SSD1306", Point::new(10, 10), text_style)
        .draw(&mut display)?;
    
    // 绘制线条
    Line::new(Point::new(0, 20), Point::new(127, 20))
        .into_styled(PrimitiveStyle::with_stroke(BinaryColor::On, 1))
        .draw(&mut display)?;
    
    // 绘制矩形
    Rectangle::new(Point::new(10, 30), Size::new(50, 20))
        .into_styled(PrimitiveStyle::with_stroke(BinaryColor::On, 1))
        .draw(&mut display)?;
    
    // 绘制圆形
    Circle::new(Point::new(80, 40), 15)
        .into_styled(PrimitiveStyle::with_fill(BinaryColor::On))
        .draw(&mut display)?;
    
    // 刷新显示
    display.flush()?;
    
    Ok(())
}

SPI接口完整示例

use ssd1306::{prelude::*, SpiInterface, Ssd1306};
use embedded_hal::spi::MODE_0;
use linux_embedded_hal::{
    Spidev, SpidevOptions,
    sysfs_gpio::Direction,
    Delay, Pin
};
use embedded_graphics::{
    pixelcolor::BinaryColor,
    prelude::*,
    primitives::{Line, Rectangle, Circle, Triangle},
    style::PrimitiveStyle,
    mono_font::{ascii::FONT_6X10, MonoTextStyle},
    text::Text,
};

fn main() -> Result<(), Box<dyn std::error::Error>> {
    // 配置SPI
    let mut spi = Spidev::open("/dev/spidev0.0")?;
    let options = SpidevOptions::new()
        .max_speed_hz(500_000)
        .mode(MODE_0)
        .build();
    spi.configure(&options)?;
    
    // 设置数据/命令(DC)和复位(RES)引脚
    let dc = Pin::new(24);
    dc.export()?;
    dc.set_direction(Direction::Out)?;
    
    let mut res = Pin::new(25);
    res.export()?;
    res.set_direction(Direction::Out)?;
    
    // 创建接口
    let interface = SpiInterface::new(spi, dc, Delay);
    
    // 创建显示驱动
    let mut display = Ssd1306::new(
        interface,
        DisplaySize128x64,
        DisplayRotation::Rotate0,
    ).into_buffered_graphics_mode();
    
    // 复位显示屏
    res.set_high()?;
    res.set_low()?;
    res.set_high()?;
    
    // 初始化显示屏
    display.init()?;
    
    // 设置显示反转
    display.set_invert(false)?;
    
    // 绘制三角形
    Triangle::new(
        Point::new(64, 10), 
        Point::new(54, 30), 
        Point::new(74, 30)
    )
    .into_styled(PrimitiveStyle::with_stroke(BinaryColor::On, 1))
    .draw(&mut display)?;
    
    // 绘制文本
    let style = MonoTextStyle::new(&FONT_6X10, BinaryColor::On);
    Text::new("SPI Demo", Point::new(30, 40), style)
        .draw(&mut display)?;
    
    // 绘制填充矩形
    Rectangle::new(Point::new(20, 50), Size::new(30, 10))
        .into_styled(PrimitiveStyle::with_fill(BinaryColor::On))
        .draw(&mut display)?;
    
    // 刷新显示
    display.flush()?;
    
    Ok(())
}

注意事项

  1. 确保正确配置了I2C或SPI接口权限
  2. 根据实际硬件调整引脚和接口路径
  3. 不同尺寸的显示屏需要选择正确的DisplaySize(如DisplaySize128x32DisplaySize128x64
  4. 在嵌入式环境中使用时,需要选择合适的HAL实现

这个库非常适合在嵌入式项目中使用Rust来控制SSD1306 OLED显示屏,无论是显示传感器数据、系统状态还是简单的用户界面。

回到顶部