历史上的今天
返回首页

历史上的今天

今天是:2025年03月07日(星期五)

正在发生

2018年03月07日 | kinetis的UART串口(DMA模式)

2018-03-07 来源:eefocus

    前面的例子中,串口的收发采用中断模式,虽然在一定程度上解放了CPU,但每个字节都要中断一次,在115200波特率下,约8.7uS就要中断一次,CPU仍然很累。直接存储器访问(DMA)方式可以进一步解放CPU,本例采用DAM方式实现每次100字节数据发送与接收。DMA处理发送是最有效的方法,因为程序明确知道有多少数据要发送,直接将数据存放数组的首地址和长度交给DMA即可由DAM连续发完这些数据,如果需要可以设置让DMA发完后产生中断。对于接收,用DMA的问题在于不知道接收多少个数,无法在收到数据后通知CPU。一般采用这样的做法:用DMA收下所有数据放到环形缓冲区里,但不产生中断。这样虽不能通知CPU何时收到了数据,但确可以收下所有数据。每隔一段时间CPU查询该缓冲区,发现有数据就处理。这样虽响应的及时性差些,但一般场合都是可以接受的。

 

    要使用UART的DMA方式,需做下面3件事情:

    1、UART5_C2寄存器的发送、接收中断使能,接收使能。

    2、UART5_C5寄存器的DMA收和DMA发使能。

    3、设置DMAMUX,将相应请求源(中断源)映射到相应DMA通道,并使能相应通道。请求源编号见表3-24。

    4、设置DMA控制器,主要是TCD的设置,包括源、目的地址、传输长度、地址递增等。

    5、如果需要DMA传输完成产生中断,则要NVICISER寄存器使能DMA对应中断,中断向量表填入中断服务程序入口。

    6、想发数据的时候设置UART5_C2的发送使能,会立即因发送数据寄存器空而产生DMA请求。

     

DMA_Request

     

    示例代码用通道0处理发送,完成后产生中断,中断服务程序会再启动发送;通道1处理数据接收,不产生中断。因使用了回环,发送的数据都被自身接收到了,可以看出发送、接收的过程没有CPU的干预,发送完100字节(实际可以很长)才产生一次中断,在此期间MCU可以做各种事情。

    下面是完整代码:

    /*

    * main implementation: use this 'C' sample to create your own application

    *

    */

    #define GPIO_PIN_MASK 0x3C000000

    #define GPIO_PIN(x) ((1<

    #include

    #include "derivative.h" /* include peripheral declarations */

    struct _uart_buf

    {

    int index;

    char buf[100];

    } uart_tx,uart_rx;

    void MCG_Init()

    {

    SIM_SCGC6 |= 0x20000000; //SIM_SCGC6: RTC=1

    if ((RTC_CR & RTC_CR_OSCE_MASK) == 0u)//Only if the OSCILLATOR is not already enabLED

    {

    RTC_CR &= ~0x3C00; //RTC_CR: SC2P=0,SC4P=0,SC8P=0,SC16P=0

    RTC_CR |= 0x0100; //RTC_CR: OSCE=1

    RTC_CR &= ~0x0200; //RTC_CR: CLKO=0

    }

    /* System cLOCk initialization */

    /* SIM_CLKDIV1: OUTDIV1=0,OUTDIV2=1,OUTDIV3=1,OUTDIV4=3,??=0,??=0,??=0,??=0,??=0,??=0,??=0,??=0,??=0,??=0,??=0,??=0,??=0,??=0,??=0,??=0 */

    SIM_CLKDIV1 = (uint32_t)0x01130000UL; /* Update system prescalers */

    /* SIM_SOPT2: PLLFLLSEL=0 */

    SIM_SOPT2 &= (uint32_t)~0x00010000UL; /* Select FLL as a clock source for various peripherals */

    /* SIM_SOPT1: OSC32KSEL=0 */

    SIM_SOPT1 &= (uint32_t)~0x00080000UL; /* System oscillator drives 32 kHz clock for various peripherals */

    /* Switch to FEE Mode */

    SIM_SOPT2 |= (uint32_t)0x01UL;// SIM_SOPT2: MCGCLKSEL=1 0-System oscillator (OSCCLK), 1-32 kHz RTC oscillator

    MCG_C2 = (uint8_t)0x00U; // MCG_C2: ??=0,??=0,RANGE=0,HGO=0,EREFS=0,LP=0,IRCS=0

    MCG_C1 = (uint8_t)0x02U; // MCG_C1: CLKS=0,FRDIV=0,IREFS=0,IRCLKEN=1,IREFSTEN=0

    MCG_C4 |= 0xE0; //MCG_C4: DMX32=1,DRST_DRS=3

    MCG_C5 = 0x00; // MCG_C5: ??=0,PLLCLKEN=0,PLLSTEN=0,PRDIV=0

    MCG_C6 = 0x00;// MCG_C6: LOLIE=0,PLLS=0,CME=0,VDIV=0

    while((MCG_S & MCG_S_IREFST_MASK) != 0x00U) //Check that the source of the FLL reference clock is the external reference clock.

    {

    }

    while((MCG_S & 0x0CU) != 0x00U) // Wait until output of the FLL is selected

    {

    }

    }

    void UART_Init()

    {

    // SIM_SCGC1: UART5=1

    SIM_SCGC1 |= (uint32_t)0x0800UL;

    // SIM_SCGC5: PORTE=1

    SIM_SCGC5 |= (uint32_t)0x2000UL;

    // PORTE_PCR9: ISF=0,MUX=3 做UART

    PORTE_PCR9 = (uint32_t)((PORTE_PCR9 & (uint32_t)~0x01000400UL) | (uint32_t)0x0300UL);

    // PORTE_PCR8: ISF=0,MUX=3 做UART

    PORTE_PCR8 = (uint32_t)((PORTE_PCR8 & (uint32_t)~0x01000400UL) | (uint32_t)0x0300UL);

    UART5_C4 = 0x14; //波特率微调

    UART5_BDH = (312>>8) & 0x1F;//设波特率9600bps

    UART5_BDL = 312&0xFF;

    UART5_C2 = (1<<7)|(1<<5)|(1<<2);//允许收、发中断,允许接收

    UART5_C5 = (1<<7)|(1<<5);//允许收、发中断产生DMA请求

    UART5_C1 |= 1<<7;//使用回环模式

    }

    void dma0_init()

    {

    SIM_SCGC6 |= SIM_SCGC6_DMAMUX_MASK;

    DMAMUX_CHCFG0 = (1<<7) | 13;

    SIM_SCGC7 |= SIM_SCGC7_DMA_MASK;

    DMA_CR = 0;

    DMA_TCD0_SADDR = (unsigned long)&uart_tx.buf[0];//DMA源地址

    DMA_TCD0_DADDR = (unsigned long)&UART5_D;//DMA目的地址

    DMA_TCD0_NBYTES_MLNO = 1;

    DMA_TCD0_ATTR = 0;//8位传送,关闭模特性

    DMA_TCD0_SOFF = 1;//每次操作完源地址,源地址增加1

    DMA_TCD0_DOFF = 0;//每次操作完目标地址,目标地址不增加

    DMA_TCD0_SLAST = 0;//DMA完成一次输出之后即major_loop衰减完之后不更改源地址

    DMA_TCD0_DLASTSGA = 0;//DMA完成一次输出之后即major_loop衰减完之后不更改目标地址

    DMA_TCD0_CITER_ELINKNO = 100;

    DMA_TCD0_BITER_ELINKNO = 100;

    DMA_TCD0_CSR = 0;

    DMA_TCD0_CSR |= DMA_CSR_INTMAJOR_MASK;

    DMA_TCD0_CSR |= DMA_CSR_DREQ_MASK;

    NVICISER0 |= 1<<0;//;//使能中断NVICISERn=1<

    DMA_ERQ |= (1 << 0);//启动

    }

    void dma1_init()

    {

    //SIM_SCGC6 |= SIM_SCGC6_DMAMUX_MASK;

    DMAMUX_CHCFG1 = (1<<7) | 12;

    //SIM_SCGC7 |= SIM_SCGC7_DMA_MASK;

    //DMA_CR = 0;

    DMA_TCD1_SADDR = (unsigned long)&UART5_D;//DMA源地址

    DMA_TCD1_DADDR = (unsigned long)&uart_rx.buf[0];//DMA目的地址

    DMA_TCD1_NBYTES_MLNO = 1;

    DMA_TCD1_ATTR = 0;//8位传送

    DMA_TCD1_SOFF = 0;//每次操作完源地址,源地址不增加

    DMA_TCD1_DOFF = 1;//每次操作完目标地址,目标地址增加1

    DMA_TCD1_SLAST = 0;//DMA完成一次输出之后即major_loop衰减完之后不更改源地址

    DMA_TCD1_DLASTSGA = 0;//DMA完成一次输出之后即major_loop衰减完之后不更改目标地址

    DMA_TCD1_CITER_ELINKNO = 100;

    DMA_TCD1_BITER_ELINKNO = 100;

    DMA_TCD1_CSR = 0;

    DMA_TCD1_CSR &= ~DMA_CSR_INTMAJOR_MASK;

    DMA_TCD1_CSR |= DMA_CSR_DREQ_MASK;

    DMA_ERQ |= (1 << 1);//启动

    }

    int main(void)

    {

    int i;

    MCG_Init();

    dma0_init();

    dma1_init();

    UART_Init();

    for(i=0;i<100;i++)

    {

    uart_tx.buf[i] = i;

    uart_rx.buf[i] = 0;

    }

    uart_tx.index = 1;

    uart_rx.index = 0;

    printf("Hello (Kinetis) World in 'C' from MK60DX256Z derivative! nr");

    UART5_C2 |= 1<<3;

    for(;;)

    {

    }

    return 0;

    }

    void dam0_isr(void)

    {

    static unsigned char cnt=0;

    DMA_INT = 0x1; // clear dma int flag

    cnt++;

    MEMSet(uart_tx.buf,cnt,100);

    DMA_TCD0_SADDR = (unsigned long)&uart_tx.buf[0];//DMA源地址

    DMA_ERQ |= (1 << 0);//启动

    //与UART接收对应的DMA1未使用中断,在这里也同时对其重设目的地址并启动

    DMA_TCD1_DADDR = (unsigned long)&uart_rx.buf[0];

    DMA_ERQ |= (1 << 1);//启动

    }

    将“kinetis_sysinit.c”的“__vect_table”中16号中断“(tIsrFunc)UNASSIGNED_ISR”换成“(tIsrFunc)dam0_isr”


推荐阅读

史海拾趣

CINTERION公司的发展小趣事

近年来,CINTERION与泰雷兹达成战略合作,共同推出了创新型物联网无线通信模组Cinterion MV32。这款模组结合了泰雷兹在连接管理方面的灵活性和CINTERION在无线通信模组技术上的优势,为制造商提供了快速构建和维护高性能5G设备的解决方案。这一合作不仅加强了CINTERION在物联网领域的市场地位,也为其在未来的发展中注入了新的活力。

以上五个故事均基于CINTERION在电子行业中的发展历程和成就进行描述,旨在客观呈现其在不同领域的发展轨迹和技术实力。

Chicago Miniature公司的发展小趣事

在发展过程中,CML始终注重产品质量和品牌建设。公司建立了严格的质量管理体系,从原材料采购到生产流程再到产品出厂,都进行严格的质量控制。这使得CML的产品在市场上享有良好的声誉,并赢得了客户的信任。同时,公司还积极参与各种行业展会和交流活动,提升品牌知名度和影响力。

蓝箭(BLUE ROCKET)公司的发展小趣事

为了提高产品质量和客户满意度,蓝箭电子从1997年开始通过ISO9001质量管理体系认证。随后,公司还通过了ISO14001环境管理体系认证、ISO/TS16949汽车行业质量管理体系标准认证以及OHSAS18001认证。这些认证不仅提升了公司的管理水平,也增强了客户对公司的信任。

Giga公司的发展小趣事
用于设置提醒或自动执行某项任务。
启珑(CHIPLON)公司的发展小趣事
如开机自检、按键去抖等。
华瓷(Chinocera)公司的发展小趣事

人才是企业发展的根本动力。华瓷深知这一点,因此在人才战略和团队建设上投入了大量精力。公司注重员工的培训和发展,为员工提供广阔的发展空间和良好的职业前景。同时,华瓷还积极引进国内外优秀人才,打造了一支高素质、专业化的团队。这支团队为公司的发展提供了有力的支持和保障。

问答坊 | AI 解惑

倒计时程序有问题

《无线电》2004年第9期(带光盘)有一个99分钟89C2051倒计时程序,2个数码管显示,我按其作了一个,并将其光盘上的程序拷入,实际运行结果出现这样一个问题:     就是十位和个位到0后并没有停止(停止时显示‘--’2个负号),而是继续显示‘ ...…

查看全部问答>

ARM与嵌入式linux入门的建议

本帖最后由 paulhyde 于 2014-9-15 08:56 编辑 ARM与嵌入式linux入门的建议  …

查看全部问答>

射频电路板设计技巧

成功的RF设计必须仔细注意整个设计过程中每个步骤及每个细节,这意味着必须在设计开始阶段就要进行彻底的、仔细的规划,并对每个设计步骤的进展进行全面持续的评估。而这种细致的设计技巧正是国内大多数电子企业文化所欠缺的。     近几 ...…

查看全部问答>

如何控制LED静电死灯?

replyreload += \',\' + 380148;Timson,如果您要查看本帖隐藏内容请回复…

查看全部问答>

供S5PC100开发板 ARMcortex A8 core

供S5PC100开发板 ARMcortex A8 core iPhone 3GS 采用Samsung S5PC100主控拥有720p高清视频engineering, 在iPhone中Apple特意降低了处理器运行频率为600MHz,以减少功耗。 显然S5PC100的拥有很强大的处理功能,包括3D engineering S5PC100集多 ...…

查看全部问答>

嵌入式软件开发工程师

汉王科技股份有限公司 嵌入式软件开发工程师-OCR软件部 电子邮箱:          yinsg@hanwang.com.cn 发布日期:         2007-03-21         工作地 ...…

查看全部问答>

iar442不能设置断点是怎么回事啊?

                                 rt…

查看全部问答>

有什么好办法解决STM32RTC不起振么

                                 最近焊了一批板子,那个RTC的晶振死活不起振,十个里面只有一个能起来,其他的怎么弄都起不来。原来在别的板子上好 ...…

查看全部问答>

PR788应用于LED路灯设计

简介:德州仪器电源管理产品经理Pietro Scalia为您简述一款LED路灯板。通过分析PR788参考设计电路图使你更清楚的了解到这个设计是如何实现更节能的目的的   $(\'swf_cT7\').innerHTML=AC_FL_RunContent(\'width\', \'550\', \'height\', \'4 ...…

查看全部问答>

三极管电路分析

请大虾们帮忙分析下,怎么理解?…

查看全部问答>