登录
首页 嵌入式系统 微软嵌入式
回帖 发帖
正文

主题:使用GPIO控制SPI接口的AD芯片

点击:898 回复:0

在实际应用中,英创的嵌入式工控主板经常需要与客户外部扩展的AD芯片相连。一般来讲AD单元的扩展有两种方法,一种是通过英创工控主板的精简ISA总线扩展,另一种则是通过同步串口的方法,如SPI、I2C接口,与AD连接。前一种方法所涉及的AD芯片一般具有并行接口,如MAX197等;而后一种方法的AD芯片则带有SPI或I2C接口。采用SPI或I2C接口的AD芯片,可使芯片的管脚数大幅减少,进一步使芯片本身的尺寸也大幅减小,从而大大扩展了这些AD芯片的应用范围。为了方便广大客户在英创的嵌入式工控主板上快速应用这类AD芯片,本文将介绍如何通过EM9160工控主板的GPIO信号来控制TI公司的带有SPI接口的TLC2543 AD芯片。
    TI公司的TLC2543是一款支持11路模拟输入,量化分辨率12-bit的低成本AD芯片。EM9160是英创公司的一款预装Windows CE实时操作系统的高性价比ARM9工控主板产品。EM9160最多可支持16位方向可独立设置的GPIO,这些GPIO均可被用来作为同步串口接口SPI的信号。在本文以下部分,SPI信号方向都是以工控主板EM9160为参考的。4线制的SPI接口其接口信号包括:
    1. SPI_CS:SPI片选信号,低电平有效;从EM9160输出,接到TLC2543。
    2. SPI_CK:SPI接口的同步时钟信号;从EM9160输出,接到TLC2543。
    3. SPI_DO:SPI接口数据输出,从EM9160输出的转换命令,输入到TLC2543。
    4. SPI_DI: SPI接口数据输入,从AD芯片输出的转换数据,输入到EM9160。
    用EM9160的GPIO仿真SPI接口的第一步是根据具体的设计情况,选择合适的GPIO信号来作为SPI的各个信号,用C代码可表述如下:
    #include 'em9160_dio_ex.h'
    #include 'em9160_isa_dio.h'
    #define GPIO0_PIN   0x0001
    #define GPIO1_PIN   0x0002
    #define GPIO2_PIN   0x0004
    #define GPIO3_PIN   0x0008
    #define GPIO4_PIN   0x0010
    #define GPIO5_PIN   0x0020
    #define GPIO6_PIN   0x0040
    #define GPIO7_PIN   0x0080
    #define GPIO8_PIN   0x0100
    #define GPIO9_PIN   0x0200
    #define GPIO10_PIN   0x0400
    #define GPIO11_PIN   0x0800
    #define GPIO12_PIN   0x1000
    #define GPIO13_PIN   0x2000
    #define GPIO14_PIN   0x4000
    #define GPIO15_PIN   0x8000
    //
    // 输入输出方向是以主板为参考来定义的。
    //
    #define SPI_CS_PIN   GPIO0_PIN   //可根据实际情况更改
    #define SPI_CK_PIN   GPIO1_PIN   //可根据实际情况更改
    #define SPI_DI_PIN   GPIO2_PIN   //可根据实际情况更改
    #define SPI_DO_PIN   GPIO3_PIN   //可根据实际情况更改
    第二步是实现SPI各个控制信号的操作函数,即各个控制信号的置位和清零以及输入状态的读入。通过调用EM9160_ISA_DIO.LIB中的相关GPIO函数,函数原型定义在头文件“em9160_dio_ex.h”中,可很容易实现下列函数:
    /////////////////////////////////////////////////////////////////////////////
    // SPI接口各管脚控制函数
    /////////////////////////////////////////////////////////////////////////////
    void  Set_SPI_CS()  //SPI片选置高,注意SPI_CS片选一般是低有效信号
    {
          PIO_OutSetEx( SPI_CS_PIN );
    }
    void  Clear_SPI_CS()  //SPI片选清零,注意SPI_CS片选一般是低有效信号
    {
          PIO_OutClearEx( SPI_CS_PIN );
    }
    void  Set_SPI_CK()  //SPI时钟置高,注意SPI_CK初始状态为低
    {
          PIO_OutSetEx( SPI_CK_PIN );
    }
    void  Clear_SPI_CK()  //SPI时钟置低,注意SPI_CK初始状态为低
    {
         PIO_OutClearEx( SPI_CK_PIN );
    }
    void  Set_SPI_DO()  //SPI数据输出高电平
    {
          PIO_OutSetEx( SPI_DO_PIN );
    }
    void  Clear_SPI_DO()  //SPI数据输出低电平
    {
          PIO_OutClearEx( SPI_DO_PIN );
    }
    int   Get_SPI_DI()   //读取SPI数据输入电平,'0'表示低电平,'1'表示高电平
    {
          UINT16 uState;
          PIO_StateEx( &uState );
          if(uState & SPI_DI_PIN)
          {
                return 1;
          }
          return 0;
    }
    void  Init_SPI()   // 设置SPI接口各控制信号,只初始化阶段运行一次。
    {
          Set_SPI_CS();
          Clear_SPI_CK();
          Clear_SPI_DO();
          //设置SPI_CS、SPI_CK、SPI_DO为数据输出
          PIO_OutEnableEx( SPI_CS_PIN | SPI_CK_PIN | SPI_DO_PIN );
          //设置SPI_DI为数据输入
          PIO_OutDisableEx( SPI_DI_PIN );
    }
    第三步就是根据SPI的时序,构造相应的读写函数。TLC2543是4线制SPI接口,因此它的读写操作是同时进行的,即所谓全双工串行数据传输。在构造函数时,需要仔细研究AD芯片数据手册上提供的SPI接口时序关系,如下图所示:
    这里需要注意的有以下几点:
    1. 在SPI_CS片选有效后,TLC2543将把上次AD转换的数据,按MSB在先的顺序,呈现在SPI_DI信号线上,并在SPI_CK的下降沿更新数据;
    2. SPI_CK的上升沿将把对AD芯片的操作指令锁存到AD芯片,输出的数据也是按MSB在先的顺序。
    3. 输入AD的操作指令只有8个bit,而从AD读出的转换数据有12个bit,在读入低4bit时,输入指令用“0”填充。
    4. 芯片数据手册中串行输入输出数据与我们的定义SPI_DO和SPI_DI是正好相反的。
    根据上述时序构造的启动AD转换并读取上次转换结果的函数如下:
   ///////////////////////////////////////////////////////////////////////////////////////
   // 输入参数uCmdCode:发送给AD芯片的转换命令,具体内容参考AD数据手册
   // 输出参数pADData:从AD读取的数据,低12-bit有效
    //////////////////////////////////////////////////////////////////////////////////////
    BOOL  ReadAD( UCHAR uCmdCode, UINT16* pADData )
    {
          int             i1;
          volatile UINT16 ui1, uCmd16;
          // activiate AD chipselection
          Clear_SPI_CS();      
          // wait 1.4us before clocking 1st bit (AD TLC2543 required)
          EM9160_DelayInUs( 2 );    
          uCmd16 = (UINT16)uCmdCode << 4; // convert cmd to 12-bit format
          ui1 = 0;        // save shift-out data from AD
          for ( i1 = 0; i1 < 12; i1++ )    // set coverting channel
          {
                ui1 = ui1 << 1;
                if(Get_SPI_DI())     // read AD_DOUT
                {
                      ui1 = ui1 | 0x0001;
                }
                if( uCmd16 & 0x0800 )   // issue Cmd onto AD_DIN, MSB first
                {
                      Set_SPI_DO();
                }
                else
                {
                      Clear_SPI_DO();
                }
                EM9160_DelayInUs( 1 );   // insert delay if required
                Set_SPI_CK();     // AD_CLK low-to-high
                EM9160_DelayInUs( 1 );   // insert delay if required
                Clear_SPI_CK();     // AD_CLK high-tolow
                EM9160_DelayInUs( 1 );   // insert delay if required
 
                uCmd16 = uCmd16 << 1;
          }
          // assign ui1 to ADdata
          *pADData = ui1;
          // de-activiate AD chipselection
          Set_SPI_CS();      

          // wait for next AD data ready if necessary
          Sleep(1);        
          return TRUE;
    }
    在程序中最后的Sleep(1),是为了保证在下次调用函数时,AD的数据已转换完毕。应用程序也可采用其他方法来保证AD有足够时间,在应用程序再次调用ReadAD(…)前已完成数据转换。特别需要注意的是,第一次调用ReadAD(…)读取的数据是无意义的,因为此时还没有设置转换命令。在SPI输入输出过程中,是否加入适当的延时,主要是由AD芯片SPI接口的响应速度来决定的,客户可查看所选AD芯片,如TLC2543,的数据手册,就可获得正确的选择。
    尽管本文是以EM9160为例来介绍如何构造SPI接口的,这个方法也完全适合英创公司的其他嵌入式工控主板产品,如EM9000、EM9161、EM9260、ETR232i等。对于不同的主板,主要的修改在第二步骤中对SPI接口信号操作函数的实现上。此外,英创公司还准备了3线制SPI接口以及I2C接口的范例参考代码,需要使用的英创用户可联系免费获得。
[查看全文]
[关于英创]
[更多文章]
[技术论坛]
本文PDF格式下载
10-04-22 10:46

工控新闻

更多新闻资讯