s3c2440硬件篇之六:系統(tǒng)時鐘和定時器
S3C2440有三種時鐘:FCLK(用于CPU核),HCLK(用于主機模塊),PCLK(用于外設).兩種PLL(鎖相環(huán)):MPLL(用于設置FCLK,HCLK,PCLK),UPLL(用于設置USB設備),.
S3C2440的CPU核工作電壓為1.2V時,主頻FCLK可以達到300M,CPU核工作電壓為1.3V時,主頻FCLK可以達到400M.為了降低電磁干擾,降低板間的布線要求,s3c2410/s3c2440外接的晶振通常很小,一般為12M,那么如何達到主頻FCLK的400M的呢?------PLL倍頻。
一.設置主頻FCLK主要是通過MPLL來軟件實現(xiàn)倍頻。MPLL主要由3個值MDIV,PDIV,SDIV來決定。而這3個值是由MPLLCON寄存器決定的,MPLLCON的第12位到第19位的值為MDIV,如下所示。
*對于MPLLCON寄存器,[19:12]為MDIV,[9:4]為PDIV,[1:0]為SDIV
*有如下計算公式:
*S3C2410:MPLL(FCLK)=(m*Fin)/(p*2^s)
*S3C2410:MPLL(FCLK)=(2*m*Fin)/(p*2^s)
*其中:m=MDIV+8,p=PDIV+2,s=SDIV
*對于本開發(fā)板,F(xiàn)in=12MHz
二.設置好了MPLLCON寄存器也就基本上算是設置好了FCLK,可以在此基礎上設置HCLK,PCLK,主要是設置分頻比,主要通過設置CLKDIV寄存器設置。
三.代碼詳解:(參考韋東山大哥代碼)
1.設置/啟動MPLL
#defineS3C2410_MPLL_200MHZ((0x5c<<12)|(0x04<<4)|(0x00))
#defineS3C2440_MPLL_200MHZ((0x5c<<12)|(0x01<<4)|(0x02))
/*
* 對于MPLLCON寄存器,[19:12]為MDIV,[9:4]為PDIV,[1:0]為SDIV
* 有如下計算公式:
* S3C2410: MPLL(FCLK) = (m * Fin)/(p * 2^s)
* S3C2410: MPLL(FCLK) = (2 * m * Fin)/(p * 2^s)
* 其中: m = MDIV + 8, p = PDIV + 2, s = SDIV
* 對于本開發(fā)板,F(xiàn)in = 12MHz
* 設置CLKDIVN,令分頻比為:FCLK:HCLK:PCLK=1:2:4,
* FCLK=200MHz,HCLK=100MHz,PCLK=50MHz
*/
voidclock_init(void)
{
// LOCKTIME = 0x00ffffff; // 使用默認值即可
CLKDIVN=0x03;// FCLK:HCLK:PCLK=1:2:4, HDIVN=1,PDIVN=1
/* 如果HDIVN非0,CPU的總線模式應該從“fast bus mode”變?yōu)椤癮synchronous bus mode” */
__asm__(
"mrc p15, 0, r1, c1, c0, 0n"/* 讀出控制寄存器 */
"orr r1, r1, #0xc0000000n"/* 設置為“asynchronous bus mode” */
"mcr p15, 0, r1, c1, c0, 0n"/* 寫入控制寄存器 */
);
/* 判斷是S3C2410還是S3C2440 */
if((GSTATUS1==0x32410000)||(GSTATUS1==0x32410002))
{
MPLLCON=S3C2410_MPLL_200MHZ;/* 現(xiàn)在,F(xiàn)CLK=200MHz,HCLK=100MHz,PCLK=50MHz */
}
else
{
MPLLCON=S3C2440_MPLL_200MHZ;/* 現(xiàn)在,F(xiàn)CLK=200MHz,HCLK=100MHz,PCLK=50MHz */
}
}
2.初始化定時器0,并設置中斷。
/*
* Timer input clock Frequency = PCLK / {prescaler value+1} / {divider value}
* {prescaler value} = 0~255
* {divider value} = 2, 4, 8, 16
* 本實驗的Timer0的時鐘頻率=100MHz/(99+1)/(16)=62500Hz
* 設置Timer0 0.5秒鐘觸發(fā)一次中斷:
*/
voidtimer0_init(void)
{
TCFG0=99;// 預分頻器0 = 99
TCFG1=0x03;// 選擇16分頻
TCNTB0=31250;// 0.5秒鐘觸發(fā)一次中斷
TCON|=(1<<1);// 手動更新
TCON=0x09;// 自動加載,清“手動更新”位,啟動定時器0
}
/*
* 定時器0中斷使能
*/
voidinit_irq(void)
{
// 定時器0中斷使能
INTMSK&=(~(1<<10));
}
3.執(zhí)行make生成timer.bin。燒入Nandflash中后運行,即可看到4個LED每1S閃爍一次。