历史上的今天
返回首页

历史上的今天

今天是:2024年10月18日(星期五)

正在发生

2019年10月18日 | 浅谈msp430f5529入门(2)——时钟配置、例程分析

2019-10-18 来源:eefocus

前天写了关于DCO,FLL的一些问题,而其他的如XT1,XT2和1xx系列几乎没有区别,而且相比比较简单,就不另外讨论了。现在总结一下55系列的时钟配置,我以TI官网提供的例程为例子进行分析,只要肯折腾,收获是不会少的。而且我发现了例程里的注释有错误,可能会给大家带来学习时的困扰,我在下文会指出。


有人可能会问TI的例程怎么找,我在这里也说一下。在TI官网搜索需要的芯片,在结果的右边会有该芯片的资料,点进去后找到“工具和软件”,点它之后再软件的培训内容里可以找到例程下载,就是名字里有examples的zip。


呵呵,可能已经有人这么尝试过了,但F5529的这里并没有例程。不过这并不表明TI没有提供5529的例程。另辟蹊径吧骚年,我估摸着一个系列的例程都是通用的,于是就再搜了一遍5526,结果就在那里找到了55xx的例程,也就是我下面要分析的例程。


例程下载链接


-----------------------------------可以开宰了------------------------------------


包里有关时钟配置的例程一共有9个,其实弄好前面几个,后面只是大同小异而已。


(1)首先,第一个,最简单的一个,因为它根本就没有配置时钟......不过,它还是很有用,因为它给我们呈现了单片机上电后的时钟状况。


//   ACLK = REFO = 32.768kHz, MCLK = SMCLK = Default 1MHz

#include

 

int main(void)

{

  volatile unsigned int i;

 

  WDTCTL = WDTPW+WDTHOLD;                   // Stop WDT

  P1DIR |= BIT1;                            // P1.1 output

 

  P1DIR |= BIT0;                            // ACLK set out to pins

  P1SEL |= BIT0;                            

  P2DIR |= BIT2;                            // SMCLK set out to pins

  P2SEL |= BIT2;                            

  P7DIR |= BIT7;                            // MCLK set out to pins

  P7SEL |= BIT7;                              

 

  while(1)

  {

    P1OUT ^= BIT1;

    __delay_cycles(60000);                  // Delay

  }

}

        博主我刚开始学时就很纠结这PUC后的默认时钟状况:


After a PUC, the UCS module default configuration is:

• XT1 in LF mode is selected as the oscillator source for XT1CLK. XT1CLK is selected for ACLK.

• DCOCLKDIV is selected for MCLK.

• DCOCLKDIV is selected for SMCLK.

• FLL operation is enabled and XT1CLK is selected as the FLL reference clock, FLLREFCLK.

• On devices that have XIN and XOUT shared with general-purpose I/O, XIN and XOUT pins are set to

general-purpose I/Os and XT1 remains disabled until the I/O ports are configured for XT1 operation. If

XIN and XOUT are not shared with general-purpose I/O, XT1 is enabled.

• When available, XT2IN and XT2OUT pins are set to general-purpose I/Os and XT2 is disabled.


刚上电后,说是ACLK和FLL默认以XT1为源,但你又不打开XT1,那问题纠结了我很久,用示波器测试测得ACLK是大约327668Hz,但使用手册后面有说这两个时钟在XT1LF失效后会转向REFOCLK,它是内部的32768Hz的时钟(那时我还没看- -||)。


真相就是这么一回事:

这是我用CCS仿真调试时暂停截的图,图示程序已经执行到while(1)里了,打开寄存器窗口看到XT1LFOFFG和DCOFFG置1了。前者被置位就是因为ACLK和FLL处于默认状态一直在向XT1取源,而XT1被禁止失效所以XT1LFOFFG被置位了,而程序还能跑就是因为XT1LF失效时它们两个时钟暂时向REFOCLK取源。后者为什么也置1了呢?它没有向XT1取源啊。呵呵,上一章说过,默认状态的DCO由FLL稳定,而FLL的时钟源XT1出问题所以DCOFFG就置1了。


因此,即使大家不需要调整单片机的时钟,在程序的开头也请先配置一下时钟,养成这个习惯,至少把XT1打开,因为XT1比REFOCLK稳定多了。


(2)第二个例程,很值得去折腾,它设置了FLL来稳定DCO到8MHz。噢不!是16MHz.


//   ACLK = REFO = 32kHz, MCLK = SMCLK = 8MHz

#include

 

int main(void)

{

  volatile unsigned int i;

 

  WDTCTL = WDTPW+WDTHOLD;                   // Stop WDT

  P1DIR |= BIT1;                            // P1.1 output

 

  P1DIR |= BIT0;                            // ACLK set out to pins

  P1SEL |= BIT0;                            

  P2DIR |= BIT2;                            // SMCLK set out to pins

  P2SEL |= BIT2;                            

  P7DIR |= BIT7;                            // MCLK set out to pins

  P7SEL |= BIT7;           

 

  UCSCTL3 = SELREF_2;                       // Set DCO FLL reference = REFO

  UCSCTL4 |= SELA_2;                        // Set ACLK = REFO

  UCSCTL0 = 0x0000;                         // Set lowest possible DCOx, MODx

 

  // Loop until XT1,XT2 & DCO stabilizes - In this case only DCO has to stabilize

  do

  {

    UCSCTL7 &= ~(XT2OFFG + XT1LFOFFG + DCOFFG);

                                            // Clear XT2,XT1,DCO fault flags

    SFRIFG1 &= ~OFIFG;                      // Clear fault flags

  }while (SFRIFG1&OFIFG);                   // Test oscillator fault flag

  __bis_SR_register(SCG0);                  // Disable the FLL control loop

  UCSCTL1 = DCORSEL_5;                      // Select DCO range 16MHz operation

  UCSCTL2 |= 249;                           // Set DCO Multiplier for 8MHz

                                            // (N + 1) * FLLRef = Fdco

                                            // (249 + 1) * 32768 = 8MHz

  __bic_SR_register(SCG0);                  // Enable the FLL control loop

 

  // Worst-case settling time for the DCO when the DCO range bits have been

  // changed is n x 32 x 32 x f_MCLK / f_FLL_reference. See UCS chapter in 5xx

  // UG for optimization.

  // 32 x 32 x 8 MHz / 32,768 Hz = 250000 = MCLK cycles for DCO to settle

  __delay_cycles(250000);

  while(1)

  {

    P1OUT ^= BIT0;                          // Toggle P1.0

    __delay_cycles(600000);                 // Delay

  }

}


18和19行,将这两个时钟的时钟源取为了REFO,这样,就没有任何时钟在向XT1请求时钟源,DCO也没有问题,故而下面的do{}while语句能通过。这个语句就是检测时钟有没有问题,如果晶体正在起振,还没有达到工作状态,会在这个循环里等待晶体起振完毕。XT2OFFG,XT1LFOFFG,DCOFFG,OFIFG都必须由软件清零。另外,POR后,OFIFG是被置位了的。


比如我把UCSCTL4 |= SELA_2;注释掉,这样,ACLK也会向REFO申请时钟源,但程序运行下来就并不一样了。


它会卡在这个循环里出不去,和上面(1)的情况一样,ACLK的寄存器里的默认值还是在向XT1申请时钟源,XT1不干活,XT1LFOFFG就置位了。


20行那里其实可以不配置,因为默认就已经是0x0000,而FLL会自动调整UCSCTL0。


这个程序的精髓就在于30-35这4行(啊咧?4行?),这是配置FLL的标准流程,在调整有关FLL的寄存器前,无比要先禁止FLL,不然可能会产生不可预料的结果。


上一章说到,用FLL稳定DCO,只要设置好FLL的时钟,再就是配置DCORSEL和FLLN就好了。利用公式:


      ·fDCOCLK= D × (N + 1) × (fFLLREFCLK÷ n)

        ·fDCOCLKDIV= (N + 1) × (fFLLREFCLK÷ n)

就可以计算出所需的值,纳尼?这和注释里的不一样?呵呵,是的,真的是注释错了。程序里(249 + 1) * 32768 = 8MHz计算的是fDCOCLKDIV!而不是注释里写的fDCOCLK!而注释里的那条看起来像公式的根本就不是个东西!


那为什么我测出来的真的是它说的大约8MHz呢?呵呵,因为它说错了,SMCLK并没有使用DCOCLK,它还是默认地使用了DCOCLKDIV,所以那条式子计算出来的的的确确是SMCLK的频率,但因为它错了两次,1.(N + 1) * FLLRef = Fdco.2.SMCLK使用DCOCLK。所以就被它绕回来了。没想到堂堂TI的例程,也会出错。被我这菜鸟抓到了。


为了揭穿TI的真面目,我把19行改为了


UCSCTL4 &= ~SELS_4;   //取消默认的SELS_4

UCSCTL4 |= SELA_2 + SELS_3;               //设置SMCLK = DCO


测了一下SMCLK,16.828MHz,是接近计算值,结论正确!所以大家在计算使用DCO时要注意DCODIV的存在,在我感觉,DCODIV更像是老大。


而后我把目光盯向了DCORSEL,因为看的资料一直在说只要把公式里面4个量设置好就可以得到想要的DCO或DCODIV频率。那还设置DCORSEL干嘛?


于是,我把UCSCTL1 = DCORSEL_5;注释掉了,RCORSEL默认值是RCORSEL_2,再调试了一遍。


发现DCOFFG被置位了,而SMCLK得频率只有2.456MHz,不是计算值呀!


DCORSEL的作用是设定DCO的频率范围,详见datasheet里的下面这表:


可以看到,DCORSEL = 2时,DCO最大最大也就7.38MHz,没有达到我们要的16点多MHz。


也就是说,就算是使用FLL来稳定DCO,DCO大概的范围即DCORSEL还是需要设置的!


于是,我有把DCORSEL = 3,调了一次


DCOOFFG还是被置位了,SMCLK频率只有4.654MHz,看表,DCORSLE = 3时DCO最大是14.0,还没有达到计算值。


在把DCORSEL设置为4之后,一切都正常了,5,6,7都试过没问题,计算值16.384MHz包含在这些范围里面。所以结论是正确的!还是那句话:就算是使用FLL来稳定DCO,DCO大概的范围即DCORSEL还是需要设置的!这个范围只需要包含目标值就行。


(3)例程2终于被我折腾完了,后面的例程不过是换了下时钟源换了下频率,已经没多少研究的价值,不过这些是我都看了一遍之后的结论。


在例程3和4里,还是设置FLL,不过是3里面换了下数值,do{}while放下面而已,还是使用DCODIV,注释还是错的......


4里面启用了XT1,用了这3句代码:


  P5SEL |= BIT4+BIT5;                       // Port select XT1

  UCSCTL6 &= ~(XT1OFF);                     // XT1 On

  UCSCTL6 |= XCAP_3;                        // Internal load cap

        打开XIN和XOUT


置0XT1OFF


老实说电容我不知道怎么选,不过这里选了12pf,那就12pf吧。其实上电后XCAP里的值默认就是XCAP_3,所以这里可以不配置


启用了XT1别忘了要等它起振:


  // Loop until XT1 fault flag is cleared

  do

  {

    UCSCTL7 &= ~XT1LFOFFG;                  // Clear XT1 fault flags

  }while (UCSCTL7&XT1LFOFFG);               // Test XT1 fault flag


其他的没什么了,说一下XT1的牛逼,这里ACLK和SMCLK的计算值分别是


ACLK = 32768Hz, SMCLK = 2457600Hz。


我测出来的是


ACLK = 32767.5Hz, SMCLK = 2457560Hz。


非常高的精密度,而且非常稳定,数值就没有变化过。相比起使用REFO时的频率总是会有百分之零点几到一点几的误差而且会有波动,当然是XT1更胜一筹。


(4)好了,写了这么多,能分享的就是这些了,关于时钟这一块,就到此结束吧。

推荐阅读

史海拾趣

ELMOS公司的发展小趣事

ELMOS公司自创立以来,就专注于汽车电子领域的发展。多年来,公司不断积累在模拟混合信号集成电路设计方面的专业知识,形成了深厚的技术底蕴。这种长期的技术积累使得ELMOS在汽车和工业物理接口领域拥有广泛的产品线,特别是在供电和DC/DC方面积累了丰富的经验。这种技术积累不仅为ELMOS赢得了市场的认可,也为公司的持续发展奠定了坚实的基础。

Computer Conversions Corp公司的发展小趣事

在计算机行业快速发展的时代,Computer Conversions Corp公司凭借其敏锐的市场洞察力和技术实力,成功地把握住了行业发展的脉搏。公司曾面临一个重大挑战:如何使老旧的计算机硬件与新兴的软件兼容。为了解决这一问题,Computer Conversions Corp投入大量资源进行技术研发,最终开发出了一种独特的硬件转换器。这种转换器不仅解决了兼容性问题,还大幅提升了数据处理速度,赢得了市场的广泛认可。

Caddock公司的发展小趣事

随着电子行业的不断发展,精度和电阻稳定性的要求越来越高。Caddock公司积极投入研发,成功开发出Tetrinox电阻系统。这一技术具有低温度系数和超高稳定性,使得电阻器和电阻器网络在精密和超精密模拟系统中得到了广泛应用。

德立电子(DDY)公司的发展小趣事

作为一家有远见的企业,德立电子深知绿色发展的重要性。公司积极响应国家环保政策,推行绿色生产和绿色供应链管理。同时,德立电子还积极参与社会公益事业,回馈社会,履行企业社会责任。

请注意,以上故事框架仅供参考,具体内容和细节需根据实际情况进行填充和完善。

成都振芯/国腾(CORPRO)公司的发展小趣事

随着公司的发展,CORPRO在2007年成功并购了成都国星通信有限公司。这次并购为CORPRO带来了卫星导航定位终端产品业务,进一步拓宽了公司的业务范围。成都国星通信有限公司作为国腾电子集团联合中国电子科技集团第十研究所、第三十研究所和电子科技大学共同组建的企业,其卫星导航定位技术为CORPRO的发展注入了新的活力。

ELMEC Technology Of America Inc公司的发展小趣事

在电子行业,技术创新是企业持续发展的关键。为了保持竞争力,ELMEC不断投入研发资金,引进高端人才,加强技术创新。他们与多所知名大学和研究机构建立合作关系,共同开展前沿技术研究。这些努力使得ELMEC在人工智能、物联网等新兴领域取得了多项重要突破,为公司的发展注入了新的活力。

问答坊 | AI 解惑

正弦波信号发生器VHDL源代码

本帖最后由 paulhyde 于 2014-9-15 09:44 编辑 拿出来恭喜  …

查看全部问答>

50Mhz分频的问题

怎么将50M的时钟分频成153600hz? 用普通的方法好像不能解决,因为除不尽。 我设想用一个pll,倍频后,再分 高手指教…

查看全部问答>

有操作系统的嵌入式设备和没有操作系统的嵌入式设备之间的区别

有操作系统的嵌入式设备和没有操作系统的嵌入式设备之间的区别,以及有操作系统的嵌入式设备的优点是什么??…

查看全部问答>

初学者问几个问题,关于wince与arm,望高人指教

首先,我用的是微芯力科的ws-430评估板,cpu为at91rm9200,要求用wince开发。 评估板附带的资料只有linux以及ucos系统的,没有wince的资料,打电话到微芯力科公司,被告知该板不支持wince,因为wince要求某尺寸的真彩屏幕,而我的板子上是一个128 ...…

查看全部问答>

如何验证下位机的串行发送程序

我现在没有芯片,只是在keilc中写的串行通信程序,我想验证一下它能不能实现发送字节,只有一台的情况下该如何做?其中发送字节程序是? void sendbyte(unsigned char word) { SBUF=word; while(TI==0);   TI=0; } …

查看全部问答>

原以为写个usb驱动很简单

DOS下的usb storage驱动一个比一个不好用,我一急就决定自己写一个,结果搞了20天了,还一头浆糊,怎么办。想来想去还是原始社会好啊…

查看全部问答>

AD快捷键大全

挺全的,ad6的快捷键大全,很实用的…

查看全部问答>

如何使用MSP430外部计数信号TACLK

急求高手指点,我的做法是引脚TACLK连接外部频率信号,对外部信号直接计数,不知道为什么总是得不到计数值。 MSP430F2274 P1SEL |= BIT0; P1DIR &= ~BIT0; TACTL = TASSEL_0 + ID_0 + MC_2;…

查看全部问答>

空中鼠标-MSP-EXP430FR5739实验板实现

玩跑跑卡丁车突发奇想,可不可以用团购来的MSP-EXP430FR5739实验板来控制赛车!基本思路是这样的,利用EXP430FR5739实验板上的三轴加速度传感器来检测前后左右上下位置信息,利用串口发送到上位机程序,上位机软件是在VS2010下写的MFC程序,利用该 ...…

查看全部问答>

Kinetis KE06 GPIO测试

Kinetis KE06 GPIO测试 IDE-Version: μVision V5.11.1.0 例程包:KEXX_DRIVERS_V1.2.1_DEVD.RAR 该包用鼠标右键的“解压文件” ,解压后的文件名还是这个,不然DOWN时会出错! 1、打开例程 相关文件:gpio.c,gpio.h主要GPIO操作都在 ...…

查看全部问答>