SPI初始化分析

KV56中SPI的初始化通过SPI_MasterInit(void)函数,在SPI.c文件下。
具体代码
代码清单1

void SPI_MasterInit(void)
{
   /* Configure the SPI corresponding Pins */
  PORTC_PCR4 = PORT_PCR_MUC(2);    //SPI0_PCS0
  PORTC_PCR5 = PORT_PCR_MUC(2);    //SPI0_SCK
  PORTC_PCR6 = PORT_PCR_MUC(2);    //SPI0_SOUT
  PORTC_PCR7 = PORT_PCR_MUC(2);    //SPI0_SIN

  /* SPI MCR Configure */
  SPI0_MCR &= ~SPI_MCR__MDIS_MASK;      //使能模式,注意此句必须是‘=’号,MDIS默认为1,此时无法设置DIS_TSF和DIS_RSF
  SPI0_MCR |= SPI_MCR__HALT_MASK;        //SPI Stop Transfer
  SPI0_MCR |= SPI_MCR__MSTR_MASK        //主机模式
                     |   SPI_MCR_PCSIS(0x01)          //PCSx 信号的空闲状态为高 Pcs Active Low(idles high)
                     |   SPI_MCR_CLR_TXF_MASK
                     |   SPI_MCR_CLR_RXF_MASK
                     |   SPI_MCR_DIS_TXF(FALSE) | SPI_MCR_DIS_RXF(FALSE);                          //TX RX FIFO is enableed.
  /* SPI CTAR0 Configure */
  SPI0_CTAR0 = SPI_CTAR_FMSZ(7);            //注意这句话必须是 = 号,因为fmsz默认为1111,SPI发送数据帧大小设定,实际数据位为该值+1,8bit

  /* SCK baud rate f(SCK) = [f(Busclk)/PBR]*[(1+DBR)/BR],f(SCK)最大值为f(Busclk)/2,f(Busclk) = 120MHz */
  SPI0_CTAR0  |= SPI_CTAR_BR(0);                          // BR = 0,2分频
              | SPI_CTAR_BR(2);                           // BR = 2,5分频
              | SPI_CTAR_PCSSCK(0);                       // PCS to SCK Prescaler value is 1
              | SPI_CTAR_CSSCK(2);                        //CSSCK为PCSx有效到SCK有效时间间隔value is 8
              | SPI_CTAR_PASC(1);                         //PASC为SCK无效到PCSx无效的时间间隔value is 3
              | SPI_CTAR_DBR(0);                          //DBR is 50/50 dury cycle

  /* SPI RSER Configure */
  SPI_RSER = 0;                                            //Disabled all interrupt requests 
  
  SPI0_SR |= SPI_SR_EOQF_MASK          //Clear End of Queue flag
          | SPI_SR_TFFF_MASK           //Clear Transmit FIFO Fill flag
          | SPI_SR_TCF_MASK            //Clear Transmit Complete flag

  SPI0_PUSHR |= SPI_PUSHR_PCS(1)       //Select PCS0 signals are to be asserted for the transfer
             | SPI_PUSHR_CONT_MASK;

  return;

}

分析代码清单2

代码清单2

   /* Configure the SPI corresponding Pins */
  PORTC_PCR4 = PORT_PCR_MUC(2);    //SPI0_PCS0
  PORTC_PCR5 = PORT_PCR_MUC(2);    //SPI0_SCK
  PORTC_PCR6 = PORT_PCR_MUC(2);    //SPI0_SOUT
  PORTC_PCR7 = PORT_PCR_MUC(2);    //SPI0_SIN

注释:
Configure the SPI corresponding Pins:配置SPI对应的管脚。
配置的是哪几个引脚?PTC4、PTC5、PTC6、PTC7。
分别对应:
SPI的PCS0、SCK、SOUT、SIN。
PCS0、SCK、SOUT、SIN是啥?因为以前没看过SPI这块,有点菜。
因为MKV56和k60都是飞思卡尔系列,所以先到K60上查一下。


k60-SPI信号线

再看KV56


KV56-SPI信号线1.png

KV56-SPI信号线2.png

原来把PTC4、PTC5、PTC6、PTC7配置了信号线。如何配置的呢?
代码清单2用的是PORT_PCR_MUC(2)。
看一下PORT_PCR_MUC(2)是如何实现的:
代码清单3-配置SPI信号线引脚

#define PORT_PCR_MUX(x)         (((uint32_t)(((uint32_t)(x)) << PORT_PCR_MUX_SHIFT)) & PORT_PCR_MUX_MASK)
其中
#define  PORT_PCR_MUX_SHIFT  8U
#define  PORT_PCR_MUX_MASK 0xF00U。

所以PORT_PCR_MUC(2)的结果是0x10’0000‘0000。
0x10’0000‘0000代表什么意思?
真没想出来。
后来想到,既然通过PORT_PCR_MUC(2)配置了引脚,那肯定和PORT有关。
去查PTC4


KV56_PORT_PCR_MUC(2)———0x10’0000‘0000含义.png

对照上图一看,大致猜到应该是将PTC4、PTC5、PTC6、PTC7复用了SPI0。
去找GPIO->PORTx-PCRn看引脚是怎么实现复用的。
照例先看k60(中文),再看KV56(英文)。如下图,可以知道K60的PORTx-PCRn的第8-10位用来配置引脚复用。


引脚复用-k60.png

KV56的第8-11用来配置引脚复用。0x10’0000‘0000的 bit9=1,即选择ATL2


引脚复用-KV56.png

小结:

代码清单2这4句实现了2个功能
1.配置SPI信号线引脚;
2.对引脚实现功能复用(SPI)
代码清单2

   /* Configure the SPI corresponding Pins */
  PORTC_PCR4 = PORT_PCR_MUC(2);    //SPI0_PCS0
  PORTC_PCR5 = PORT_PCR_MUC(2);    //SPI0_SCK
  PORTC_PCR6 = PORT_PCR_MUC(2);    //SPI0_SOUT
  PORTC_PCR7 = PORT_PCR_MUC(2);    //SPI0_SIN

分析代码清单3

这里明显是在配置SPI的MCR寄存器。

/* SPI MCR Configure */
  SPI0_MCR &= ~SPI_MCR__MDIS_MASK;      //使能模式,注意此句必须是‘=’号,MDIS默认为1,此时无法设置DIS_TSF和DIS_RSF
  SPI0_MCR |= SPI_MCR__HALT_MASK;        //SPI Stop Transfer
  SPI0_MCR |= SPI_MCR__MSTR_MASK        //主机模式
                     |   SPI_MCR_PCSIS(0x01)          //PCSx 信号的空闲状态为高 Pcs Active Low(idles high)
                     |   SPI_MCR_CLR_TXF_MASK
                     |   SPI_MCR_CLR_RXF_MASK
                     |   SPI_MCR_DIS_TXF(FALSE) | SPI_MCR_DIS_RXF(FALSE);                          //TX RX FIFO is enableed.
  SPI0_MCR &= ~SPI_MCR__MDIS_MASK;      //使能模式,注意此句必须是‘=’号,MDIS默认为1,此时无法设置DIS_TSF和DIS_RSF
#include SPI_MCR__MDIS_MASK    0x4000U
so,  SPI0_MCR = SPI0_MCR  & 11 1111 1111 1111
而
#define   SPI0_MCR     SPI_MCR_REG(SPI0_BASE_PTR)     
#define   SPI_MCR_REG(base)   ((base)->MCR),
#define    SPI0_BASE_PTR    0x4002C000u
即 SPI0_MCR    ((0x4002C000u)->MCR)
SPI0_MCR     =  0x4002C000u & 11 1111 1111 1111 = 0;
SPI0_MCR |= SPI_MCR__HALT_MASK;    //SPI Stop Transfer
#define   SPI_MCR__HALT_MASK   0x1u,
so  SPI0_MCR = 0x1
配置MCR寄存器实现'停止传输'功能.png

继续分析代码清单3

  SPI0_MCR |= SPI_MCR__MSTR_MASK        //主机模式
           |   SPI_MCR_PCSIS(0x01)          //PCSx 信号的空闲状态为高 Pcs Active Low(idles high)
           |   SPI_MCR_CLR_TXF_MASK
           |   SPI_MCR_CLR_RXF_MASK
           |   SPI_MCR_DIS_TXF(FALSE) | SPI_MCR_DIS_RXF(FALSE);                          //TX RX FIFO is enableed.

 SPI_MCR__MSTR_MASK     0x8000'0000U    
 SPI_MCR_PCSIS(0x01)       1 0000 0000 0000 0000
 SPI_MCR_CLR_TXF_MASK      0x800U
 SPI_MCR_CLR_RXF_MASK      0x400U
 SPI_MCR_DIS_TXF(FALSE)  = 0
 SPI_MCR_DIS_RXF(FALSE) = 0
  so
  SPI0_MCR  = 1000 0000 8000 0C00
    =1 0000 0000 0000 0000 0000 0000 0000 1000 0000 0000 0000 0000 1100 0000 0000 

其中
(3.1) SPI_MCR__MSTR_MASK

 SPI_MCR__MSTR_MASK     0x8000'0000U   
0x8000'0000.png

(3.2) SPI_MCR_PCSIS(0x01)

 SPI_MCR_PCSIS(0x01)       1 0000 0000 0000 0000
SPI_MCR_PCSIS.png

(3.3)SPI_MCR_CLR_TXF_MASK

SPI_MCR_CLR_TXF_MASK      0x800U
CLR_TXF.png

(3.4)SPI_MCR_CLR_RXF_MASK

SPI_MCR_CLR_RXF_MASK      0x400U
CLR_RXF.png

(3.5) SPI_MCR_DIS_TXF(FALSE) | SPI_MCR_DIS_RXF(FALSE) = 0;开始传输


TXF -RXF.png

分析代码清单4

  SPI0_CTAR0 = SPI_CTAR_FMSZ(7);            //注意这句话必须是 = 号,因为fmsz默认为1111,SPI发送数据帧的大小设定,实际数据位为该值+1,8bit

  /* SCK baud rate f(SCK) = [f(Busclk)/PBR]*[(1+DBR)/BR],f(SCK)最大值为f(Busclk)/2,f(Busclk) = 120MHz */
  SPI0_CTAR0  |= SPI_CTAR_BR(0);                          // BR = 0,2分频
              | SPI_CTAR_BR(2);                           // BR = 2,5分频
              | SPI_CTAR_PCSSCK(0);                       // PCS to SCK Prescaler value is 1
              | SPI_CTAR_CSSCK(2);                        //CSSCK为PCSx有效到SCK有效时间间隔value is 8
              | SPI_CTAR_PASC(1);                         //PASC为SCK无效到PCSx无效的时间间隔value is 3
              | SPI_CTAR_DBR(0);                          //DBR is 50/50 dury cycle

  /* SPI RSER Configure */
  SPI_RSER = 0;                                            //Disabled all interrupt requests 

(4.1)

//注意这句话必须是 = 号,因为fmsz默认为1111,SPI发送数据帧大小设定,实际数据位为该值+1,8bit
  SPI0_CTAR0 = SPI_CTAR_FMSZ(7);   
           =111 1000 0000 0000 0000 0000 0000 0000

bit30\29\28\27为1


FMSZ.png

(4.2)

  /* SCK baud rate f(SCK) = [f(Busclk)/PBR]*[(1+DBR)/BR],f(SCK)最大值为f(Busclk)/2,f(Busclk) = 120MHz */
  SPI0_CTAR0  |= SPI_CTAR_BR(0);                          // BR = 0,2分频
              | SPI_CTAR_BR(2);                           // BR = 2,5分频
              | SPI_CTAR_PCSSCK(0);                       // PCS to SCK Prescaler value is 1
              | SPI_CTAR_CSSCK(2);                        //CSSCK为PCSx有效到SCK有效时间间隔value is 8
              | SPI_CTAR_PASC(1);                         //PASC为SCK无效到PCSx无效的时间间隔value is 3
              | SPI_CTAR_DBR(0);                          //DBR is 50/50 dury cycle

(4.2.1)

SPI_CTAR_BR(0)  ==0<<0U & FU =  0
BR.png

(4.2.2)
SPI_CTAR_PBR(2); // BR = 2,5分频

SPI_CTAR_PBR(2)  == 2<<16 & 0x3'0000 = 10 0000 0000 0000 0000

bit16 = 0,bit17=1;


PBR.png

(4.2.3)
SPI_CTAR_PCSSCK(0) // PCS to SCK Prescaler value is 1

SPI_CTAR_PCSSCK(0) = 0<<22 & 0xC0'0000
PCSSCK.png

(4.2.4)
SPI_CTAR_CSSCK(2); //CSSCK为PCSx有效到SCK有效时间间隔value is 8

SPI_CTAR_CSSCK(2) ==2<<12 & 0xF000 =10 0000 0000 0000
CSSCK.png

(4.2.5)
SPI_CTAR_PASC(1); //PASC为SCK无效到PCSx无效的时间间隔value is 3
其中SPI_CTAR_PASC(1) =1<<20 &0x30'0000 = 1 0000 0000 0000 0000 0000,bit20 ==1


PASC.png

(4.2.6)
SPI_CTAR_DBR(0); //DBR is 50/50 dury cycle
SPI_CTAR_DBR(0) =0<<31 & 0x8000'0000= 0000 0000 0000 0000 0000 0000 0000 0000,bit31等于0


DBR.png

代码清单5


  /* SPI RSER Configure */
  SPI_RSER = 0;                                            //Disabled all interrupt requests 
  
  SPI0_SR |= SPI_SR_EOQF_MASK          //Clear End of Queue flag
          | SPI_SR_TFFF_MASK           //Clear Transmit FIFO Fill flag
          | SPI_SR_TCF_MASK            //Clear Transmit Complete flag

  SPI0_PUSHR |= SPI_PUSHR_PCS(1)       //Select PCS0 signals are to be asserted for the transfer
             | SPI_PUSHR_CONT_MASK;

代码清单5这里就不在分析,和上面的分析是一样的。

总结

SPI的初始化,其实就是对各个寄存器的配置(各个寄存器位的值设置)。当然不是乱配置的,而是,基于我们要实现的SPI功能,进行相应的配置。
我这里只是,结合配置\初始化好的SPI程序,去学习操作手册。
没错,这篇文章,其实就是教你如何去看操作手册,以及,了解程序中是如何通过代码来设置寄存器实现SPI功能的

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352