MSP430的SPI协议分析和简单实例

2019-07-23 16:01发布

MSP430的SPI协议分析和简单实例
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
5条回答
yiyigirl2014
1楼-- · 2019-07-23 20:54
SPI:SPI是Motorola首先在其MC68HCXX系列处理器上定义的,它是一种同步的高速串行通信协议,有关SPI协议的详细内容,参考:SPI_互动百科。

MSP430对SPI的支持:当msp430USART模块控制器UxCTL的位SYNC置位时,USART模块工作于同步模式,对于149即工作于SPI模式,若是169,USART0可以支持I2C,可以通过另一控制位I2C控制,I2C位0则工作于SPI。在SPI模式下,允许单片机以确定的速率发送和接收7位或8位数据。

同步通信与异步通信类似;同步通信和异步通信寄存器资源一致,具体寄存器的不同位之间的功能存在差异;具体寄存器内容参见TI提供的用户指南。

USART模块的SPI操作可以是3线和4线,其信号如下:
SIMO:从进主出,主机模式下,数据输出;从机模式下,数据输入。
SOMI:从出主进,主机模式下,数据输入;从机模式下,数据输出。
UCLK:USART SPI模式时钟,信号有主机输出,从机输入。
STE:从机模式发送接收允许控制脚,用于4线模式,控制多主从系统中多个从机,避免发生冲突。
yiyigirl2014
2楼-- · 2019-07-24 02:50

初始化函数:SpiMasterInit,实现主机模式的初始化工作,函数内容如下:
char SpiMasterInit(long baud,char dataBits,char mode,char clkMode)
{
    long int brclk;                 //波特率发生器时钟频率

    UxCTL |= SWRST;                 //初始

    //反馈选择位,为1,发送的数被自己接收,用于测试,正常使用时注释掉
    //UxCTL |= LISTEN;

    UxCTL |= SYNC + MM;             //SPI 主机模式

    //时钟源设置
    UxTCTL &=~ (SSEL0+SSEL1);       //清除之前的时钟设置
    if(baud<=16364)                 //
    {
      UxTCTL |= SSEL0;              //ACLK,降低功耗
      brclk = 32768;                //波特率发生器时钟频率=ACLK(32768)
    }
    else
    {
      UxTCTL |= SSEL1;              //SMCLK,保证速度
      brclk = 1000000;              //波特率发生器时钟频率=SMCLK(1MHz)
    }

    //------------------------设置波特率-------------------------   
    if(baud < 300||baud > 115200)   //波特率超出范围
    {
        return 0;
    }
    //设置波特率寄存器
    int fen = brclk / baud;         //分频系数
    if(fen<2)return (0);            //分频系数必须大于2
    else
    {
        UxBR0 = fen / 256;
        UxBR1 = fen % 256;
    }

    //------------------------设置数据位-------------------------   
    switch(dataBits)
    {
        case 7:case'7': UxCTL &=~ CHAR; break;      //7位数据
        case 8:case'8': UxCTL |= CHAR;  break;      //8位数据
        default :       return(0);                  //参数错误
    }
    //------------------------设置模式---------------------------   
    switch(mode)
    {
        case 3:case'3': UxTCTL |= STC;  USPI3ON;    break;  //三线模式
        case 4:case'4': UxTCTL &=~ STC; USPI4ON;    break;  //四线模式
        default :       return(0);                          //参数错误
    }

    //------------------------设置UCLK模式-----------------------  
    switch(clkMode)
    {
        case 0:case'0': UxTCTL &=~ CKPH; UxTCTL &=~ CKPL;   break;  //模式0
        case 1:case'1': UxTCTL &=~ CKPH; UxTCTL |= CKPL;    break;  //模式1
        case 2:case'2': UxTCTL |= CKPH;  UxTCTL &=~ CKPL;   break;  //模式2
        case 3:case'3': UxTCTL |= CKPH;  UxTCTL |= CKPL;    break;  //模式3
        default :       return(0);                                  //参数错误
    }

    UxME |= USPIEx;             //模块使能

    UCTL0 &= ~SWRST;            // Initialize USART state machine

    UxIE |= URXIEx + UTXIEx;    // Enable USART0 RX interrupt

    return(1);                  //设置成功
}

程序注释已经比较详细,这里不再细说;如果要改为从机模式,把时钟设置和波特率设置去掉应该就可以了。

发送函数和接收函数:
void SpiWriteDat(char c)
{
    while (TxFlag==0) SpiLpm();  // 等待上一字节发完,并休眠
    TxFlag=0;                     //
    UxTXBUF=c;
}
char SpiReadDat()
{
    while (RxFlag==0) SpiLpm(); // 收到一字节?
    RxFlag=0;
    return(UxRXBUF);
}

发送和接收函数和异步通信里面的几乎一样,如果标志位为0,则等待改变为1,然后写入或读出;标志位在中断函数里被更改;中断函数如下:
#pragma vector=USARTxRX_VECTOR
__interrupt void UartRx()
{
    RxFlag=1;
    __low_power_mode_off_on_exit();
}
#pragma vector=USARTxTX_VECTOR
__interrupt void UartTx ()
{
    TxFlag=1;
    __low_power_mode_off_on_exit();
}

中断里面仅仅置标志位后,就退出低功耗;退出后即写入或者读取数据。

读取或写入函数调用的SpiLpm函数:
void SpiLpm()
{
    if(UxTCTL&SSEL0) LPM3;  //若以ACLK 作时钟,进入LPM3休眠(仅打开ACLK)
    else             LPM0;  //若以SMCLK作时钟,进入LPM0休眠(不关闭SMCLK)
}

根据不同情况进入低功耗,如果单片机其他地方不允许进入低功耗,可以更改这个函数。

程序部分就这么多了。需要的函数在头文件里面声明,方便使用。

3.使用示例:
程序使用方式和之前的程序库相同,加入c文件,包含h文件,调用初始化函数后即可掉用程序库中的函数。
#include "msp430x16x.h"   //430寄存器头文件
#include "Spi.h"         //串口通讯程序库头文件

void main()
{
    // Stop watchdog timer to prevent time out reset
    WDTCTL = WDTPW + WDTHOLD;

    ClkInit();
    // 主机模式,波特率25000,8位数据位,三线模式,时钟模式0(具体见spi.c)
    SpiMasterInit(25000,8,3,0);
    _EINT();


    while(1)                    //串口测试
    {
        SpiWriteDat(0X20);
        char a = SpiReadDat();
    }
}

这里只是一个简单的使用示例,详细的使用,将会在下一篇给出,下一篇:MSP430程序库<六>通过SPI操作AD7708;将会使用今天的程序库,完成SPI的通信部分。


4.注意事项:

SPI是全双工通信,每次写入(发送)8位/7位数据的同时,430的SPI主模块都会在发送后半个时钟周期读取采样的0/1信号,存入接收缓冲寄存器,所以,每次的写入,均有数据读取,但不一定是从设备发送回来的,这个地方在使用430主机模式的时候必须注意,很容易出错(我也是在调试AD7708的时候才注意到这个地方的);SPI的函数已经添加SpiWriteData函数,这个函数会在发送的同时返回发送完成半个时钟周期后的接收到的数据,方便使用;不建议使用前面的发送和读取函数,很容易出错;建议使用刚添加的这个函数,程序库已经更新,可以重新下载。函数SpiWriteData:
char SpiWriteData(char c)
{
    SpiWriteDat(c);
    return SpiReadDat();
}

发送后读取即可,程序比较简单。

新的示例程序:
void main()
{
    // Stop watchdog timer to prevent time out reset
    WDTCTL = WDTPW + WDTHOLD;

    ClkInit();
    // 主机模式,波特率25000,8位数据位,三线模式,时钟模式0(具体见spi.c)
    SpiMasterInit(25000,8,3,0);
    _EINT();


    while(1)                    //串口测试
    {
        SpiWriteData(0X20);     //只写入
        char a = SpiWriteData(0xff);    //只读取
    }
}


yiyigirl2014
3楼-- · 2019-07-24 03:30
  • 硬件主要是MSP430的SPI接口和AD7708芯片的说用说明。
    msp430的SPI接口:支持主机模式和从机模式,且始终的极性和相位可调,在于AD转换芯片通信的时候,需要极性一致。
    AD7718 的外部引脚有28 个。按性质主要分为模拟、数字两个部分。模拟部分引脚有模拟输入、参考电压输入和模拟电源三类。模拟输入引脚可以配置为8通道或10通道的伪差分输入,他们共同参考AINCOM端。
    数字部分引脚有 SPI 接口、数据就绪、通用I/O 口和数字电源四类。SPI 接口的4 根标准信号线分别是片选信号CS 、串行时钟输入SCLK、串行数据输入DIN 和串行数据输出DOUT。当AD7718接在SPI 总线上时是从器件,从引脚CS 输入低电平信号使能AD7718。数据就绪RDY 是一个低电平有效的输出引脚。当所选通道数据寄存器中有有效数据时,输出低电平信号;数据被读出后,输出高电平。AD7718 的通用I/O 口是2 个一位口P1 和P2。它们既可配置成输入也可配置成输出,单片机通过SPI 口读写AD7718 片内相关寄存器实现对P1 和P2 的操作。它们扩展了单片机的I/O 接口能力。
    AD7718 的模拟电源和数字电源是分别供电的,都既可以采用+3V 供电,也可以采用+5V 供电。但必须一致,要么都用+3V,要么都用+5V。
    AD7708和AD7718是通过一组片内寄存器控制和配置的。这些寄存器的第一个是通信寄存器,它是用来控制转换器的所有操作。这些部件的所有通信必须先写通信寄存器指定要执行的下一个操作。上电或复位后,设备默认等待写通信寄存器。 STATUS寄存器包含转换器的操作条件的有关信息。 STATUS寄存器是只读寄存器。模式寄存器用于配置转换模式,校准,斩波(chop)启用/禁用,参考电压选择,通道配置和伪差分AINCOM模拟输入操作时的缓冲或无缓冲。模式寄存器是一个读/写寄存器。 ADC控制寄存器是一个读/写寄存器,用来选择活动的通道和编码输入范围和双极性/单极性操作。I/O控制寄存器是一个读/写寄存器,用于配置了2个I/O端口的操作。滤波寄存器是一个读/写寄存器,用于编码转换器的数据更新率。 ADC数据寄存器是一个只读寄存器,它包含在所选通道上的一个数据转换的结果。 ADC的失调寄存器读/写寄存器包含偏移校准数据。有五个偏移寄存器,每个全差分输入通道之一。当配置为伪差分输入模式下的通道共用偏移寄存器。 ADC增益寄存器是读/写寄存器,包含增益校准数据。有5个ADC增益寄存器,每个全差分输入通道之一。当配置为伪差分输入模式通道共享增益寄存器。该ADC包含工厂使用的测试寄存器,用户应不改变这些寄存器的操作条件。 ID寄存器是一个只读寄存器,用于硅识别目的。
    我用的硬件连线方式:430的P3.0接AD7708的CS端,P3.1-P3.2接对应的AD芯片的SPI口;RDY信号没有接;所以,程序使用的是查询方式,等待STATUS寄存器的RDY位指示转换完成。
    有关AD7708的详细信息可以参考它的datasheet;另外我对数据手册的寄存器部分和程序流程的部分进行了翻译,如果需要,可以在本博客底部的附件中下载。

yiyigirl2014
4楼-- · 2019-07-24 07:59
 精彩回答 2  元偷偷看……
yiyigirl2014
5楼-- · 2019-07-24 10:16
mode是程序记录的上一次输入的MODE寄存器的内容,因为串口读取需要时间,为了获取更快的速度,程序记录了这两个变量,以供使用

一周热门 更多>