spi速率计算公式为:PCLK / (CPSDVSR *[SCR+1])
公式出处:lpc1114用户手册SSP章节CR0寄存器的bit15:bit8定义的解释里面
PCLK是当前SSP的时钟,CPSDVSR是寄存器CPSR值,SCR是CR0寄存器bit15:bit8的值。
所以spi的速率受到了3个寄存器值的影响,这3个寄存器分别是:
1. LPC_SYSCON->SSP1CLKDIV或者 LPC_SYSCON->SSP0CLKDIV(这是SSP的分频寄存器)
2. LPC_SSP1->CR0(这是SSP控制寄存器0,其中bit15:bit8决定速率)
3. LPC_SSP1->CPSR(这是SSP时钟预分频寄存器)
得到这3个寄存器的值,就可以计算出当前的spi速率值,例如下面的SPI1初始化函数:
void SPI1_Init(void) { uint8_t i,Clear=Clear;//Clear=Clear:用这种语句形式解决编译产生的Waring:never used! LPC_SYSCON->PRESETCTRL |= (0x1<<2); //禁止LPC_SSP1复位 LPC_SYSCON->SYSAHBCLKCTRL |= (0x1<<18);//允许LPC_SSP1时钟 bit18 LPC_SYSCON->SSP1CLKDIV = 10; //10分频:50/10=5Mhz LPC_SYSCON->SYSAHBCLKCTRL |= (1<<16); // 使能IOCON时钟(bit16) LPC_IOCON->PIO2_1 &= ~0x07; LPC_IOCON->PIO2_1 |= 0x02; //把PIO2_1选择为LPC_SSP CLK LPC_IOCON->PIO2_2 &= ~0x07; LPC_IOCON->PIO2_2 |= 0x02; //把PIO2_2选择为LPC_SSP MISO LPC_IOCON->PIO2_3 &= ~0x07; LPC_IOCON->PIO2_3 |= 0x02; //把PIO2_3选择为LPC_SSP MOSI LPC_SYSCON->SYSAHBCLKCTRL &= ~(1<<16); // 禁能IOCON时钟(bit16) // 8位数据传输,SPI模式, CPOL = 1, CPHA = 1,空闲时CLK为1,SCR = 4 LPC_SSP1->CR0 = 0x04C7; // 预分频值(注意:这里必须为偶数 2~254) LPC_SSP1->CPSR = 10; LPC_SSP1->CR1 &= ~(1<<0);//LBM=0:正常模式 LPC_SSP1->CR1 &= ~(1<<2);//MS=0:主机模式 LPC_SSP1->CR1 |= (1<<1);//SSE=1:使能SPI1 //清空RxFIFO,LPC1114收发均有8帧FIFO,每帧可放置4~16位数据 for ( i = 0; i < 8; i++ ) { Clear = LPC_SSP1->DR;//读数据寄存器DR将清空RxFIFO } }
上面例子中,主频50MHz, LPC_SYSCON->SSP1CLKDIV值为10,即PCLK=5MHz; LPC_SSP1->CR0=0x04c7,即bit15:bit8为4,即SCR=4;LPC_SSP1->CPSR=10;带入公式计算得出,现在的SPI速率应该是100KHz.
上图中,示波器,横向每格表示5微秒,图中一个周期就是10微秒,即100KHz,实测与理论完全一致。