草庐IT

合宙AIR105(四): SPI, MAX7219 8x8LED驱动

Milton 2023-03-28 原文

目录

Air105 的 SPI

Air105 包含五组普通SPI, 可以以半/全双工, 同步, 串行的方式通信. 可以被配置成主模式并为从设备提供时钟(SCK), 还能以多主配置方式工作. 这里不介绍QSPI(高速SPI).

SPI 整体参数

  • SPI时钟由 PCLK 提供, SPI_CLK = PCLK (AIR105绝大部分外设的时钟都是PCLK)
  • 支持协议Motorola Serial Peripheral Interface (SPI), Texas Instruments Serial Protocol (SSP), National Semiconductor Microwire
  • 包含硬件收发FIFO, 深度为16
  • 独立硬件收发FIFO, 可配收发FIFO中断阈值
  • SPI0 支持主或者从(主/从地址不同), 从模式支持CS拉低持续接收
  • 4到16位数据帧格式选择
  • 支持全双工, 半双工模式
  • 收发, 错误中断检测
  • 支持DMA

硬件收发FIFO

包含2个独立的深度为16的收发FIFO

  • 对寄存器DR写 -> 写入发送FIFO
  • 对寄存器DR读 <- 取自接收FIFO
  • 收发FIFO有独立的中断阈值设定, 当数据符合设定阈值时产生中断
  • 收发FIFO有独立的DMA阈值设定, 当数据符合设定阈值时产生DMA请求

主从模式

  • SPI0包括2组寄存器组SPIM0 和 SPIS0, 分别用于实现主模式和从模式, 2组寄存器组结构相同, 地址不同
    • 主模式下SPI相应初始化及数据收发操作由SPIMx完成
    • 从模式下,SPI相应初始化及数据接收操作由SPISx完成
  • SPI0外设工作模式使用 SYSCTRL 寄存器中 PHER_CTRL 相应位切换
  • SPI1 - SPI4只支持主模式, (SPIM1 - SPIM4)

SPI相关代码

SPI初始化

以驱动 MAX7219 为例, 初始化 SPI0 的代码, 使用 GPIO_PinRemapConfig 复用后, 不需要再单独进行GPIO初始化

#define MAX7219_SPIx        SPIM0
#define MAX7219_CS_PORT     GPIOC
#define MAX7219_CS_PIN      GPIO_Pin_13

#define MAX7219_SPIx_PORT   GPIOC
#define MAX7219_SPIx_PINS   GPIO_Pin_12|GPIO_Pin_14|GPIO_Pin_15
#define MAX7219_SPIx_REMAP  GPIO_Remap_2

void MAX7219_SPI_Init(void)
{
    SPI_InitTypeDef SPI_InitStructure;

    SYSCTRL_APBPeriphClockCmd(SYSCTRL_APBPeriph_GPIO | SYSCTRL_APBPeriph_SPI0, ENABLE);
    SYSCTRL_APBPeriphResetCmd(SYSCTRL_APBPeriph_SPI0, ENABLE);

    //SPI0
    GPIO_PinRemapConfig(MAX7219_SPIx_PORT, MAX7219_SPIx_PINS, MAX7219_SPIx_REMAP);

    SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;
    SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;
    SPI_InitStructure.SPI_CPOL = SPI_CPOL_Low;
    SPI_InitStructure.SPI_CPHA = SPI_CPHA_1Edge;
    SPI_InitStructure.SPI_NSS = SPI_NSS_0;
    SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_256;
    // 这两项不能省略, 否则 SPI_Init 初始化会阻塞
    SPI_InitStructure.SPI_RXFIFOFullThreshold = SPI_RXFIFOFullThreshold_1;
    SPI_InitStructure.SPI_TXFIFOEmptyThreshold = SPI_TXFIFOEmptyThreshold_10;

    SPI_Init(MAX7219_SPIx, &SPI_InitStructure);
    SPI_Cmd(MAX7219_SPIx, ENABLE);
}

CS使用主动控制, 因此PC13单独初始化GPIO

void MAX7219_GPIO_Init(void)
{
    GPIO_InitTypeDef GPIO_InitStruct;

    SYSCTRL_APBPeriphClockCmd(SYSCTRL_APBPeriph_GPIO, ENABLE);

    GPIO_InitStruct.GPIO_Pin = MAX7219_CS_PIN;
    GPIO_InitStruct.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_InitStruct.GPIO_Remap = GPIO_Remap_1;
    GPIO_Init(MAX7219_CS_PORT, &GPIO_InitStruct);
}

SPI主动发送和接收

通过调用SPI_SendDataSPI_ReceiveData完成

uint8_t MAX7219_SendByte(uint8_t byte)
{
    uint8_t data;
    SPI_SendData(MAX7219_SPIx, byte);
    while (RESET == SPI_GetFlagStatus(MAX7219_SPIx, SPI_FLAG_RXNE));
    data = SPI_ReceiveData(MAX7219_SPIx);
    return data;
}

Air105 SPI 驱动 MAX7219

又到了喜闻乐见的点灯环节. MAX7219 8x8的LED点阵是最简单的SPI设备了

接线

PIN脚的具体位置, 查看 https://wiki.luatos.com/_static/bom/Air105.html

* 3.3V            -> VCC
* GND             -> GND
* PC12 SPI0_CLK   -> CLK
* PC13 CS         -> CS
* PC14 SPI0_MOSI  -> DIN

代码

初始化需要的命令

#define DECODE_MODE  0x09
#define INTENSITY    0x0A
#define SCAN_LIMIT   0x0B
#define SHUT_DOWN    0x0C
#define DISPLAY_TEST 0x0F

片选指令

void MAX7219_ChipSelect(int state)
{
    if (state)
    {
        GPIO_SetBits(MAX7219_CS_PORT, MAX7219_CS_PIN);
    }
    else
    {
        GPIO_ResetBits(MAX7219_CS_PORT, MAX7219_CS_PIN);
    }
}

MAX7219通信的基础方法

void MAX7219_Write(uint8_t addr, uint8_t dat)
{
    MAX7219_ChipSelect(0);
    MAX7219_SendByte(addr);
    MAX7219_SendByte(dat);
    MAX7219_ChipSelect(1);
}

初始化代码

void MAX7219_Init(void)
{
    MAX7219_Write(SHUT_DOWN,    0x01);      // 0x00:shutdown, 0x01:normal
    MAX7219_Write(DECODE_MODE,  0x00);      // Bypass code B decoder, no-decode operation
    MAX7219_Write(SCAN_LIMIT,   0x07);      // Scan-limit, 0:1-digit, 1:2-digits, ... 7:8-digits
    MAX7219_Write(INTENSITY,    0x01);      // 0x00:min, 0xFF:max
    MAX7219_Write(DISPLAY_TEST, 0x00);      // 0x00:normal, 0x01:test mode
}

完整代码从这里下载

https://gitee.com/iosetting/air105_project/tree/master/Demos/SPI/SPI_MAX7219_8x8LED

有关合宙AIR105(四): SPI, MAX7219 8x8LED驱动的更多相关文章

  1. ruby-on-rails - 如何在 Ruby on Rails 中实现由 JSF 2.0 (Primefaces) 驱动的 UI 魔法 - 2

    按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visitthehelpcenter指导。关闭10年前。问题1)我想知道ruby​​onrails是否有功能类似于primefaces的gem。我问的原因是如果您使用primefaces(http://www.primefaces.org/showcase-labs/ui/home.jsf),开发人员无需担心javascript或jquery的东西。据我所知,JSF是一个规范,基于规范的各种可用实现,prim

  2. FOHEART H1数据手套驱动Optitrack光学动捕双手运动(Unity3D) - 2

    本教程将在Unity3D中混合Optitrack与数据手套的数据流,在人体运动的基础上,添加双手手指部分的运动。双手手背的角度仍由Optitrack提供,数据手套提供双手手指的角度。 01  客户端软件分别安装MotiveBody与MotionVenus并校准人体与数据手套。MotiveBodyMotionVenus数据手套使用、校准流程参照:https://gitee.com/foheart_1/foheart-h1-data-summary.git02  数据转发打开MotiveBody软件的Streaming,开始向Unity3D广播数据;MotionVenus中设置->选项选择Unit

  3. SPI接收数据异常问题总结 - 2

    SPI接收数据左移一位问题目录SPI接收数据左移一位问题一、问题描述二、问题分析三、探究原理四、经验总结最近在工作在学习调试SPI的过程中遇到一个问题——接收数据整体向左移了一位(1bit)。SPI数据收发是数据交换,因此接收数据时从第二个字节开始才是有效数据,也就是数据整体向右移一个字节(1byte)。请教前辈之后也没有得到解决,通过在网上查阅前人经验终于解决问题,所以写一个避坑经验总结。实际背景:MCU与一款芯片使用spi通信,MCU作为主机,芯片作为从机。这款芯片采用的是它规定的六线SPI,多了两根线:RDY和INT,这样从机就可以主动请求主机给主机发送数据了。一、问题描述根据从机芯片手

  4. ruby - 运行测试时静音 Chrome 驱动程序控制台输出 - 2

    我使用的是最新版本的Chrome(32.0.1700.107)和Chrome驱动程序(V2.8)。但是当我在Ruby中使用以下代码运行示例测试时:require'selenium-webdriver'WAIT=Selenium::WebDriver::Wait.new(timeout:100)$driver=Selenium::WebDriver.for:chrome$driver.manage.window.maximize$driver.navigate.to'https://www.google.co.in'defapps_hoverele_hover=$driver.find_

  5. node.js - 从未编写过任何自动化测试,我应该如何开始行为驱动开发? - 2

    按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visitthehelpcenter指导。关闭9年前。多年来,我一直在使用多种语言进行编程,并且认为自己总体上相当擅长。但是,我从未编写过任何自动化测试:没有单元测试,没有TDD,没有BDD,什么都没有。我已经尝试开始为我的项目编写适当的测试套件。我可以看到在进行任何更改后能够自动测试项目中所有代码的理论值(value)。我可以看到像RSpec和Mocha这样的测试框架应该如何使设置和运行所述测试变得相当容易

  6. ruby-on-rails - 在 Rails/Capybara/Poltergeist 规范中使用 url_for 将驱动程序发送到 example.com 而不是应用程序 - 2

    如果我在功能规范中调用url_for,它会返回一个以http://www.example.com/开头的绝对URL.Capybara会很乐意尝试加载该站点上的页面,但这与我的应用程序无关。以下是重现该问题的最少步骤:从这个Gemfile开始:source'https://rubygems.org'gem"sqlite3"gem"jquery-rails"gem"draper"gem"rails",'4.1.0'gem"therubyracer"gem"uglifier"gem"rspec-rails"gem"capybara"gem"poltergeist"gem"launchy"运行

  7. 驱动开发:内核无痕隐藏自身分析 - 2

    在笔者前面有一篇文章《驱动开发:断链隐藏驱动程序自身》通过摘除驱动的链表实现了断链隐藏自身的目的,但此方法恢复时会触发PG会蓝屏,偶然间在网上找到了一个作者介绍的一种方法,觉得有必要详细分析一下他是如何实现的进程隐藏的,总体来说作者的思路是最终寻找到MiProcessLoaderEntry的入口地址,该函数的作用是将驱动信息加入链表和移除链表,运用这个函数即可动态处理驱动的添加和移除问题。MiProcessLoaderEntry(pDriverObject->DriverSection,1)添加MiProcessLoaderEntry(pDriverObject->DriverSection,

  8. ruby - 事件/观察者驱动的 Ruby on Rails - 2

    我有一个适用于事件/监听器模型的应用程序。发布了几种不同类型的数据(事件),然后许多不同的事情可能需要也可能不需要对该数据(监听器)采取行动。监听器的发生没有特定的顺序,每个监听器将决定是否需要对事件采取行动。Rails应用程序有哪些工具可以完成此任务?我希望自己不必这样做(尽管我可以。这没什么大不了的。)编辑:观察者模式可能是更好的选择 最佳答案 查看EventMachine.它是一个非常流行的Ruby事件处理库。它看起来相当不错,而且很多其他库似乎都在利用它(Cramp)。这是一个很好的介绍:http://rubylearnin

  9. MicroBlaze在纯FPGA下 Xilinx SDK固化程序到外部SPI FLASH - 2

    外部SPIFLASH:MicronN25Q128A13ESE40G(128Mbit(16MByte))FPGA:XC7A100T CPU:Microblaze第一种情况:Microblaze在简单的应用,比如运行LED,IIC,SPI,UART之类的低俗接口驱动,或做一些简单的辅助型工作时,一般生成的applicationelf文件都不大,在10几KB或者几十,百几KB,此时使用FPGA内部的BRAM资源已经足够。XC7A100T本身就有600几KB的BRAM资源。这种情况下直接将硬件流文件和elf文件合并为download.bit文件,在直接烧录到外部SPIFLAH即可。1.Xilinx--

  10. 51单片机——74HC595的应用(SPI实践) - 2

    目录SPI总线SPI总线概述 SPI总线分类SPI优点及缺点SPI接口硬件原理SPI四种工作模式 74HC595应用74HC595芯片概述74HC595封装及管脚功能74HC595工作原理 ​编辑 74HC595串行转并行点亮LED灯 程序实现  Proteus运行结构示意图SPI总线SPI总线概述 SPI(SerialPeripheralinterface):串行外围设备接口 用途:用来在微控制器与外围设备芯片之间实现数据交换 特点:高速、全双工、同步 SPI总线分类四线制全双工SPI(同时收发)MISO    主机输入/从机输出MOSI    主机输出/从机输入SCLK   串行时钟CS或

随机推荐