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

 2009-6-18         

        在实际应用中,英创的嵌入式工控主板经常需要与客户外部扩展的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接口的范例参考代码,需要使用的英创用户可联系免费获得。