137 SPI+DMA

#include "bat32g137.h"
#include "sci.h"
#include "stdio.h"

// bref: spi dma发送
// para:
// note:
static void spi_send_dma(uint8_t *data,uint16_t len)
{
	#define SPI10_DMA_VECTOR   14
	#define DMA_SRAM_OFFSET    0
	
	DMAVEC->VEC[SPI10_DMA_VECTOR] = DMA_SRAM_OFFSET;
	DMAVEC->CTRL[DMA_SRAM_OFFSET].DMACR = (0 << CTRL_DMACR_SZ_Pos) | (0 << CTRL_DMACR_CHNE_Pos) |
									  (0 << CTRL_DMACR_DAMOD_Pos) | (1 << CTRL_DMACR_SAMOD_Pos) |
									  (0 << CTRL_DMACR_MODE_Pos);
	DMAVEC->CTRL[DMA_SRAM_OFFSET].DMBLS = 1;
	DMAVEC->CTRL[DMA_SRAM_OFFSET].DMACT = len - 1;
	DMAVEC->CTRL[DMA_SRAM_OFFSET].DMRLD = len - 1;
	DMAVEC->CTRL[DMA_SRAM_OFFSET].DMSAR = (uint32_t)data+1;
	DMAVEC->CTRL[DMA_SRAM_OFFSET].DMDAR = (uint32_t)&SCI0->SIO10;
	
	/* init DMA registers */
	CGC->PER1 |= CGC_PER1_DMAEN_Msk;
	DMA->DMABAR = DMAVEC_BASE;
	DMA->DMAEN1 |= (1 << 6);
	SCI0->SIO10 = *data;
	// 等待传输完成
//	INT->IF[SPI10_IRQn].IFL = 0;
//	while(INT->IF[SPI10_IRQn].IFL == 0);
	//INT->IF[SPI10_IRQn].IFL = 0;
	while(DMA->DMAEN1 & (1 << 6));
}



int main(void)
{
	spi_mode_t  mode = SPI_MODE_0;
	// 开启外设时钟
	CGC->PER0 |= CGC_PER0_SCI0EN_Msk;
	// 停止通道
    SCI0->ST0 |= _0004_SCI_CH2_STOP_TRG_ON;
	// 分频器
    SCI0->SPS0 &= ~SCI0_SPS0_PRS01_Msk;
    SCI0->SPS0 |= (0 << SCI0_SPS0_PRS01_Pos);
	// 设置为主模式
    SCI0->SIR02 = _0004_SCI_SIRMN_FECTMN | _0002_SCI_SIRMN_PECTMN | _0001_SCI_SIRMN_OVCTMN;
	// 设置为SPI模式 传输结束产生中断
    SCI0->SMR02 = _0020_SMRMN_DEFAULT_VALUE | _8000_SCI_CLOCK_SELECT_CK01 | _0000_SCI_CLOCK_MODE_CKS |
                  _0000_SCI_TRIGGER_SOFTWARE | _0000_SCI_MODE_SPI | _0001_SCI_BUFFER_EMPTY;
	// 设置SPI 极性
    SCI0->SCR02 = _0004_SCRMN_DEFAULT_VALUE | _8000_SCI_TRANSMISSION | mode | _0000_SCI_INTSRE_MASK |
                  _0000_SCI_PARITY_NONE | _0000_SCI_MSB | _0000_SCI_STOP_NONE | _0003_SCI_LENGTH_8;

	//(2+1)*2 = 6 
	// 48/6 = 8M
    SCI0->SDR02 = 2 << 9; 

    /* Set output enable */
    if ((mode == SPI_MODE_0) || (mode == SPI_MODE_1))
    {
        SCI0->SO0 &= ~_0400_SCI_CH2_CLOCK_OUTPUT_1;
    }
    if ((mode == SPI_MODE_2) || (mode == SPI_MODE_3))
    {
        SCI0->SO0 |= _0400_SCI_CH2_CLOCK_OUTPUT_1;
    }
    SCI0->SOE0 |= _0004_SCI_CH2_OUTPUT_ENABLE;
	SCI0->SO0 |= _0004_SCI_CH2_DATA_OUTPUT_1;
    SCI0->SOE0 |= _0004_SCI_CH2_OUTPUT_ENABLE;
    SCI0->SS0 |= _0004_SCI_CH2_START_TRG_ON;
//    /* Set P02/SDO10 pin */
    PORT->PMC0 &= ~(1<<2);
    PORT->P0 |= 1<<2;
    PORT->PM0 &= ~(1<<2);
    /* Set P03/SDI10 pin */
    PORT->PMC0 &= 0xF7U;
    PORT->PM0 |= 0x08U;
    /* Set P04/SCK10 pin */
    PORT->PMC0 &= 0xEFU;
    PORT->P0 |= 0x10;
    PORT->PM0 &= ~(1<<4);
	// 
	PORT->PMC0 &= ~(1<<5);
    PORT->PM0 &= ~(1<<5);
	
	PORT->P0 &=~(1<<5); 
	
	// 开启对应的中断
	 /* clear INTSPI10 interrupt flag */
    INTC_ClearPendingIRQ(SPI10_IRQn);
    NVIC_ClearPendingIRQ(SPI10_IRQn);
    /* enable INTSPI10 interrupt */
    INTC_EnableIRQ(SPI10_IRQn);
    NVIC_EnableIRQ(SPI10_IRQn);
	
	// 使能通道
	SCI0->SS0 |= _0004_SCI_CH2_START_TRG_ON;
	
	static uint8_t buff[] = {0xAA,0x55,0x12,0x34};
	while(1)
	{
		spi_send_dma(buff,sizeof(buff));
	}
}

void IRQ13_Handler(void) __attribute__((alias("spi10_interrupt")));
/***********************************************************************************************************************
* Function Name: spi10_interrupt
* Description  : None
* Arguments    : None
* Return Value : None
***********************************************************************************************************************/
void spi10_interrupt(void)
{
    volatile uint8_t err_type;
    volatile uint8_t sio_dummy;

    INTC_ClearPendingIRQ(SPI10_IRQn);
    err_type = (uint8_t)(SCI0->SSR02 & _0001_SCI_OVERRUN_ERROR);
    SCI0->SIR02 = (uint16_t)err_type;

    if (1U == err_type)
    {
        //spi10_callback_error(err_type);    /* overrun error occurs */
		
    }
    else
    {
		if(PORT->P0 & (1<<5))
		{
			PORT->P0 &=~(1<<5);
		}
		else
		{
			PORT->P0 |=(1<<5);
		}
        
    }
}

DMA需要找SPI的DMA向量位置,找一个内存设置为 DMA的信息块。

DMA机制是利用截获SPI的中断来启动DMA传输,所以可以先写一个数据启动DMA传输。

while(DMA->DMAEN1 & (1 << 6));这个意思是传输完成6 bit自动变为0.

也可以判断中断方式来确定DMA是否传输完成,由于本例子中开启了中断,所以不使用本方法判断DMA传输结束。

10-15
### SPIDMA结合使用 在STM32中,SPIDMA结合使用时,发送数据时,在每次TXE(发送缓冲区空标志)被设置为’1’时发出DMA请求,DMA控制器则写数据至SPI_DR寄存器,TXE标志因此而被清除;接收时,在每次RXNE(接收缓冲区非空标志)被设置为’1’时发出DMA请求,DMA控制器则从SPI_DR寄存器读出数据,RXNE标志因此被清除。使用时需要设置外设地址、存储器地址、传输数据量、通道的配置信息,然后使能DMA通道启动传输[^3]。 ### SPIDMA结合原理 SPI允许芯片与外部设备以半/全双工、同步、串行方式通信,其通信使用MISO、MOSI、SCLK、CS四根线。DMA用来提供在外设和存储器之间或者存储器和存储器之间的高速数据传输,无须CPU干预,数据可以通过DMA快速地移动,节省了CPU的资源来做其他操作。结合时,SPI进行数据的串行通信,而DMA负责数据在SPI与存储器之间的高速搬运,实现数据高效传输[^1]。 ### SPIDMA结合应用场景 - **数据采集系统**:在数据采集系统中,常常需要将传感器采集到的数据传输到处理器中进行处理。使用SPIDMA结合,可以将传感器的数据通过SPI接口快速传输到存储器中,而不需要CPU的干预,从而提高系统的实时性和处理效率。例如将ADC采集到的数据通过SPI传输,再用DMA搬运到缓存器,最后传输到串口等外设上[^1]。 - **显示控制**:在一些显示系统中,需要将图像数据从存储器传输到显示屏上。使用SPIDMA结合,可以将图像数据快速传输到显示屏的驱动芯片中,实现高速显示。例如使用STM32F103的SPI+DMA功能控制WS2812灯带[^2]。 以下是一个简单的伪代码示例来展示SPIDMA结合的基本配置思路: ```c // 初始化SPI void SPI_Init(void) { // 配置SPI相关寄存器,如时钟、模式、数据格式等 } // 初始化DMA void DMA_Init(void) { // 设置外设地址 // 设置存储器地址 // 设置传输数据量 // 配置DMA通道 // 使能DMA通道 } // 启动SPI+DMA传输 void SPI_DMA_Transfer(void) { SPI_Init(); DMA_Init(); // 启动SPI // 启动DMA传输 } ```
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值