模擬信號輸出是經(jīng)常會遇到的應(yīng)用需求,解決的辦法應(yīng)多種,但我們使用最多的還是數(shù)模轉(zhuǎn)換。對于不同的數(shù)模轉(zhuǎn)換器我們需要為其編寫適用的驅(qū)動程序,在這一篇中我們就來考慮如何實現(xiàn)DAC8552高精度模數(shù)轉(zhuǎn)換器的驅(qū)動程序。
1、功能概述
??該DAC8552是一個16位,雙通道,電壓輸出數(shù)模轉(zhuǎn)換器(DAC)提供低功率操作和靈活的串行主機(jī)接口。每個芯片上的精確輸出放大器允許軌到軌輸出擺動,以實現(xiàn)在2.7V到5.5V的供應(yīng)范圍。該設(shè)備支持標(biāo)準(zhǔn)三線串行接口,能夠操作與輸入數(shù)據(jù)時鐘頻率高達(dá)30MHz的VDD = 5V。
1.1、功能結(jié)構(gòu)
??DAC8552這種設(shè)備在正常情況下的低功耗使得它非常適合便攜式、電池驅(qū)動設(shè)備和其他低功耗應(yīng)用。采用SOIC-8的封裝形式,引腳定義如下:
??DAC8552需要一個外部參考電壓來設(shè)置每個DAC通道的輸出范圍。DAC8552還包括一個電源上電復(fù)位電路,以確保DAC輸出功率能夠輸出到零,并保持在那里,直到獲取一個有效的寫入值。DAC8552擁有一個SPI串行接口,該接口提供了靈活的功能。
??從上述結(jié)構(gòu)圖可知,DAC8552每次僅能操作一路輸出,因為全部的操作都是通過同一個移位寄存器來實現(xiàn)的。
1.2、移位寄存器
??DAC8552有一個24位的輸入移位寄存器,前面8位用來作控制位,后面16位用作數(shù)據(jù)位。具體如下圖所示:
??在前面的8位控制位中,DB23和DB22是保留位必須為“0”,DB21(LDB)位和DB20(LDA)用于控制后面的16位數(shù)據(jù)適用于加載哪一個輸出通道還是Power_Down命令。DB19沒有定義,DAC8552不關(guān)心該位的具體數(shù)值。DB18為緩沖器選擇位,用于控制數(shù)據(jù)的目標(biāo)通道是DAC A還是DAC B。后續(xù)的DB17(PD1)和DB16(PD0)用于選擇Power_Down的模式。具體的命令如下表中描述:
??至于Power_Down的模式有幾種選擇,如下表所示:
2、驅(qū)動設(shè)計與實現(xiàn)
??我們已經(jīng)了解了DAC8552的基本結(jié)構(gòu)及寄存器命令,接下來我們將根據(jù)這些認(rèn)知設(shè)計DAC8552的驅(qū)動程序。
2.1、對象定義
??在設(shè)計DAC8552的驅(qū)動程序之前,我們先來考慮一下DAC8552的對象定義問題。我們作為一個對象一般會包括屬性和操作兩個方面的內(nèi)容。我們先來分析DAC8552對象應(yīng)該包含有哪些屬性。屬性用于標(biāo)識對象的某些特性,DAC8552通過SPI總線下發(fā)數(shù)據(jù)和命令,我們沒有發(fā)現(xiàn)什么需要特別標(biāo)記的特性,所以我們不需要為DAC8552對象設(shè)計屬性。
??我們再來看一看,DAC8552對象需要實現(xiàn)哪些操作。首先DAC8552使用SPI總線進(jìn)行通訊,而SPI總線采用片選信號來區(qū)分不同的節(jié)點,所以我們需要操作DAC8552的片選信號,而片選型號的操作顯然依賴于特定的操作平臺,所以我們將控制其片選信號作為DAC8552對象的一個操作。另外,DAC8552作為模擬量輸出對象,我們需要向其發(fā)送命令和數(shù)據(jù),而向其發(fā)送數(shù)據(jù)和命令也依賴于具體的操作平臺,所以應(yīng)將其作為對象的一個操作來實現(xiàn)。據(jù)此我們可以定義DAC8552的對象類型如下:
/* 定義DAC8552對象類型 */
typedef struct DAC8552Object {
void (*WriteDataToDAC)(uint8_t *tData,uint16_t tSize); //向DAC發(fā)送數(shù)據(jù)
void (*ChipSelcet)(DAC8552CSType cs); //片選信號
}DAC8552xObjectType;
??我們定義了DAC8552的對象類型,但當(dāng)我們使用其聲明一個對象時,并不能直接使用,我們需要對對象進(jìn)行初始化,這就需要我們設(shè)計一個對象初始化的函數(shù)。對象初始化函數(shù)處理對象相關(guān)的屬性和操作的配置,具體實現(xiàn)如下:
/*初始化DAC8552對象*/
void DAC8552Initialization(DAC8552xObjectType *dac, //DAC8552對象變量
DAC8552WriteType write, //寫數(shù)據(jù)函數(shù)指針
DAC8552ChipSelectType cs //片選操作函數(shù)指針
)
{
if((dac==NULL)||(write==NULL))
{
return;
}
if(cs!=NULL)
{
dac->ChipSelcet=cs;
}
else
{
dac->ChipSelcet=DefaultChipSelect;
}
}
2.2、對象操作
??我們已經(jīng)定義了DAC8552的對象類型并為DAC8552對象設(shè)計了初始化函數(shù),接下來我們看一看DAC8552所要實現(xiàn)的操作。對于DAC8552對象來說,我們對其操作無非就是寫其移位寄存器以實現(xiàn)命令和數(shù)據(jù)的下發(fā)。從其數(shù)據(jù)表中我們可以看到操作移位寄存器的時序如下所示:
??根據(jù)我們前面對DAC8552相關(guān)數(shù)據(jù)的了解以及上述時序圖,我們可以封裝對其移位寄存器的操作函數(shù)如下:
/*操作DAC8552輸出通道*/
void SetDAC8552ChannelValue(DAC8552xObjectType *dac,DAC8552LDType ld,DAC8552BSType bs,DAC8552PDType pd,uint16_t data)
{
uint32_t inputShiftData=0;
uint8_t sData[3];
inputShiftData=data;
inputShiftData=inputShiftData|(ld<<20);
inputShiftData=inputShiftData|(bs<<18);
inputShiftData=inputShiftData|(pd<<16);
sData[0]=(uint8_t)(inputShiftData>>16);
sData[1]=(uint8_t)(inputShiftData>>8);
sData[2]=(uint8_t)inputShiftData;
dac->ChipSelcet(DAC8552CS_Enable);
dac->WriteDataToDAC(sData,3);
dac->ChipSelcet(DAC8552CS_Disable);
}
3、驅(qū)動的使用
??我們設(shè)計了DAC8552的對象驅(qū)動,但這個驅(qū)動是否正確我們需要驗證一下。所以接下來我們設(shè)計一個簡單的例子來實現(xiàn)對驅(qū)動程序的驗證。
3.1、聲明并初始化對象
??我們使用設(shè)計的驅(qū)動程序操作DAC8552時,首先需要使用我們定義的對象類型聲明一個DAC8552對象。
DAC8552xObjectType dac8552;
??聲明了這個對象變量之后,我們還需要使用初始化函數(shù)對其進(jìn)行初始化方可使用。這一初始化函數(shù)擁有3個參數(shù):
DAC8552xObjectType *dac, //DAC8552對象變量
DAC8552WriteType write, //寫數(shù)據(jù)函數(shù)指針
DAC8552ChipSelectType cs //片選操作函數(shù)指針
??第一個參數(shù)正是我們要初始化的對象變量;第二個參數(shù)為向DAC8552寫命令和數(shù)據(jù)的函數(shù)指針;第三個參數(shù)是片選信號操作函數(shù)指針。這兩個函數(shù)指針需要我們實現(xiàn)。它們的原型如下:
/* 向DAC發(fā)送數(shù)據(jù)函數(shù)指針類型 */
typedef void (*DAC8552WriteType)(uint8_t *tData,uint16_t tSize);
/* 片選操作函數(shù)指針類型 */
typedef void (*DAC8552ChipSelectType)(DAC8552CSType cs);
??我們根據(jù)函數(shù)原型定義,在具體的實現(xiàn)平臺上實現(xiàn)它們,如我們在STM32平臺上實現(xiàn)如下:
/*定義片選信號函數(shù)*/
void DAC8552CS(DAC8552CSType en)
{
if(DAC8552CS_Enable==en)
{
HAL_GPIO_WritePin(GPIOF, GPIO_PIN_4, GPIO_PIN_RESET);
}
else
{
HAL_GPIO_WritePin(GPIOF, GPIO_PIN_4, GPIO_PIN_SET);
}
}
/*定義發(fā)送數(shù)據(jù)函數(shù)*/
void DAC8552TransmitData(uint8_t *wData,uint16_t wSize)
{
HAL_SPI_Transmit (&dac8552hspi, wData, wSize, 1000);
}
??我們將對象變量以及上面實現(xiàn)的2個函數(shù)的函數(shù)指針作為參數(shù)傳遞給DAC8552對象初始化函數(shù)來實現(xiàn)對象變量的初始化。具體如下:
DAC8552Initialization(&dac8552, //DAC8552對象變量
DAC8552TransmitData, //寫數(shù)據(jù)函數(shù)指針
DAC8552CS //片選操作函數(shù)指針
);
3.2、基于對象進(jìn)行操作
??初始化對象變量后,我們就可以基于該對象變量實現(xiàn)我們對DAC8552的操作了。我們已經(jīng)封裝了對其移位寄存器操作的函數(shù),直接調(diào)用該函數(shù)來說實現(xiàn)我們的操作。一個簡單的實現(xiàn)函數(shù)如下:
/* 修改DAC8552的通道輸出 */
void DAC8552Operation(void)
{
uint16_t wData=0;
wData=(uint16_t)(65535*tValueA/100);
SetDAC8552ChannelValue(&dac8552, //所操作的DAC對象
DAC8552_LoadA, //加載的通道
DAC8552BS_BufferA, //選擇的緩存
DAC8552PD_Normal, //Power-Down設(shè)置
wData //所寫的數(shù)據(jù)
);
wData=(uint16_t)(65535*tValueB/100);
SetDAC8552ChannelValue(&dac8552, //所操作的DAC對象
DAC8552_LoadB, //加載的通道
DAC8552BS_BufferB, //選擇的緩存
DAC8552PD_Normal, //Power-Down設(shè)置
wData //所寫的數(shù)據(jù)
);
}
??在這個例子中我們分別通過百分比設(shè)定值調(diào)整了A、B通道的輸出,實現(xiàn)在正常模式下操作A或者B通道,并更新指定的緩存。
4、應(yīng)用總結(jié)
??我們設(shè)計并實現(xiàn)了DAC8552模數(shù)轉(zhuǎn)換器的驅(qū)動程序,并且設(shè)計了一個簡單的應(yīng)用來驗證這一驅(qū)動程序的正確性。所得到的結(jié)果證明驅(qū)動的設(shè)計是沒有問題的,實際上我們已經(jīng)將其運用到實際的項目中,效果良好。
??在使用驅(qū)動程序時需要注意,片選信號并非必須實現(xiàn)。因為有些時候我們可能需要在硬件上直接將其選中,此時添加片選操作函數(shù)是沒有什么意義的,我們可以在初始化時傳入NULL來完成。
評論
查看更多