作者在某STM32的應用中外接了一塊DS1302,借鑒原來寫過的PIC代碼,很快移植成功。在這里與大家分享一下。(1)硬件 CLK PA4 DAT PA5 RST PA6
(2)硬件初始化 CLK與RST均為輸出,而DAT是開漏型的輸出。因為在這種方式下,IO口的讀仍然存在,因此是真正的雙向IO模式。
/*DSCK -PA4 DAT PA5 RST DAT*/
#define ds1302Clk GPIO_Pin_4 //與時鐘線相連的芯片的管腳
#define ds1302Dat GPIO_Pin_5 //與數據線相連的芯片的管腳
#define ds1302Rst GPIO_Pin_6 //與復位端相連的芯片的管腳
/* PA 4,6,為輸出*/
GPIO_InitStructure.GPIO_Pin = ds1302Clk | ds1302Rst ;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStructure);
//PA5配置為開漏模式,此模式下可以實現真下的雙向IO
GPIO_InitStructure.GPIO_Pin = ds1302Dat;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD;
GPIO_Init(GPIOA, &GPIO_InitStructure);
(3)配置完畢,然后實現的代碼就很簡單了,從現成的代碼移植過來。
#define WrEnDisCmd 0x8e //寫允許/禁止指令代碼
#define WrEnDat 0x00 //寫允許數據
#define WrDisDat 0x80 //寫禁止數據
#define OscEnDisCmd 0x80 //振蕩器允許/禁止指令代碼
#define OscEnDat 0x00 //振蕩器允許數據
#define OscDisDat 0x80 //振蕩器禁止數據
#define WrMulti 0xbe //寫入多個字節的指令代碼
#define WrSingle 0x84 //寫入單個字節的指令代碼
#define RdMulti 0xbf //讀出多個字節的指令代碼
////以上這些#define均放在一個ds1302.h文件中。
void SendDat_1302(u8 Dat)
{ u8 i;
u8 cTmp;
for(i=0;i<8;i++)
{
cTmp=Dat&LSB; //數據端等于tmp數據的末位值
if(cTmp) //1
GPIO_SetBits(GPIOA,ds1302Dat);
else
GPIO_ResetBits(GPIOA,ds1302Dat);
Dat>>=1;
GPIO_SetBits(GPIOA,ds1302Clk);
uDelay(1);
GPIO_ResetBits(GPIOA,ds1302Clk);
}
}
/*寫入1個或者多個字節,第1個參數是相關命令
#define WrMulti 0xbe //寫入多個字節的指令代碼
#define WrSingle 0x84 //寫入單個字節的指令代碼
第2個參數是待寫入的值
第3個參數是待寫入數組的指針
*/
void WriteByte_1302(u8 CmdDat,u8 Num,u8 *pSend)
{
u8 i=0;
GPIO_ResetBits(GPIOA,ds1302Rst);
uDelay(1);
GPIO_SetBits(GPIOA,ds1302Rst);
SendDat_1302(CmdDat);
for(i=0;i<Num;i++)
{ SendDat_1302(*(pSend+i));
}
GPIO_ResetBits(GPIOA,ds1302Rst);
}
/*讀出字節,第一個參數是命令
#define RdMulti 0xbf //讀出多個字節的指令代碼
第2個參數是讀出的字節數,第3個是指收數據數組指針
*/
void RecByte_1302(u8 CmdDat,u8 Num,u8 *pRec)
{
u8 i,j,tmp,cTmp;
GPIO_ResetBits(GPIOA,ds1302Rst);//復位引腳為低電平
uDelay(1);
GPIO_ResetBits(GPIOA,ds1302Clk);
uDelay(1);
GPIO_SetBits(GPIOA,ds1302Rst);
SendDat_1302(CmdDat); //發送命令
for(i=0;i<Num;i++)
{ for(j=0;j<8;j++)
{ tmp>>=1;
cTmp=GPIO_ReadInputDataBit(GPIOA,ds1302Dat);
if(cTmp)
tmp|=0x80;
GPIO_SetBits(GPIOA,ds1302Clk);
uDelay(1);
GPIO_ResetBits(GPIOA,ds1302Clk);
}
*(pRec+i)=tmp;
}
uDelay(1);
GPIO_ResetBits(GPIOA,ds1302Rst);//復位引腳為低電平
}
/*
當寫保護寄存器的最高位為0時,允許數據寫入寄存器。
寫保護寄存器可以通過命令字節8E、8F來規定禁止寫入/讀出。寫保護位不能在多字節傳送模式下寫入。
當寫保護寄存器的最高位為1時,禁止數據寫入寄存器。
時鐘停止位操作:當把秒寄存器的第7位時鐘停止位設置為0時起動時鐘開始
當把秒寄存器的第7位時鐘停止位設置為1時,時鐘振蕩器停止。
根據傳入的參數決定相關命令,
第一個參數:命令字,第2個參數:寫入的數據
寫允許命令;8EH,00H
寫禁止命令;8EH,80H
振蕩器允許命令;80H,00H
振蕩器禁止命令;80H,80H
*/
void WrCmd(u8 CmdDat,u8 CmdWord)
{ u8 CmdBuf[2];
CmdBuf[0]=CmdWord;
WriteByte_1302(CmdDat,1,CmdBuf);
}
main函數中調用如下:
u8 Ds1302SendBuf[8]={0x30,0x32,0x01,0x10,0x01,0x01,0x08,33}; //發送數據緩沖區
u8 Ds1302RecBuf[8]; //接收數據緩沖區
WrCmd(WrEnDisCmd,WrEnDat); //寫允許
WrCmd(OscEnDisCmd,OscEnDat); //振蕩器允許
WriteByte(WrMulti,8,Ds1302SendBuf);//將時間值送到DS1302中
RecByte(RdMulti,8,Ds1302RecBuf); // 讀出來看看,
......
|