datasheet

PIC振荡器配置与时钟切换

2018-07-30来源: eefocus关键字:PIC  振荡器配置  时钟切换

单片机编程就是C语言+寄存器设置。 
以前对PIC振荡器的配置都是拿来主义,把别人的代码拿过来用就行了。这两天特意研究下振荡器的配置与时钟切换。在mplab IDE和C30编译器下,针对PIC24FJxx系列单片机完成的测试。 
配置振荡器最主要的目的就是为了设置机械时钟Fosc,此时钟给CPU和外设提供时钟源。但为了降低功耗又不中断外设正常通信,此系列PIC保证CPU与外设的时钟同步情况下,增加了打盹模式,用于降低CPU运行时钟速度,以达到节能效果。 
个人是这么理解的,CPU时钟就是代码运行时钟,决定代码运行速度;外设时钟就是中断、定时器、输入捕捉、输出比较、UART、SPI等外设的时钟源。 
时钟框图(From:datasheet) 
在时钟框图中可看出,由四个振荡器提供时钟源,包括两个外部振荡器(主振荡器、辅助振荡器)和两个内部振荡器(FRC振荡器——快速RC、LPRC振荡器——低功耗RC)。 
时钟模式共11种: 
1. XT——1M到4M的石英晶体振荡器(主振荡器) 
2. HS——超过4M的石英晶体振荡器(主振荡器) 
3. EC——低于1M的陶瓷振荡器(主振荡器) 
4. XTPLL——带PLL模块的主振荡器(主振荡器) 
5. HSPLL——带PLL模块的主振荡器(主振荡器) 
6. ECPLL——带PLL模块的主振荡器(主振荡器) 
7. FRCPLL——带后分频器和PLL模块的快速RC振荡器(内部振荡器) 
8. FRCDIV——带后分频器的快速RC振荡器(内部振荡器) 
9. FRC——快速RC振荡器(内部振荡器) 
10. LPRC——低功耗RC振荡器(内部振荡器) 
11. SOSC——辅助振荡器(辅助振荡器) 
注:①PLL模块是一个锁相环(Phase Lock Loop)倍频器,可提高4倍频率,②这里共有11时钟模式,在配置时,有一个保留模式,但无HSPLL模式

一. 振荡器配置

一般为了减少外围电路,采用内部振荡器,其时钟频率最高可达32M。如果需要严格的时钟频率,而内部振荡器又无法匹配上,那才考虑外部振荡器。所以一般按以下步骤来配置(后面对应寄存器名): 
1. 是否使用主振荡器——配置位CW2的POSCMD1:POSCMD0 
2. 选择初始振荡器,即时钟模式——配置位CW2的FNOSC2:FNOSC0 
3. 配置OSCO引脚功能,在EC和非主振荡器时钟模式下,不占用此引脚,可配置成Fosc/2时钟输出CLKO,或普通I/O口RA3——配置位CW2的OSCIOFCN 
4. 【不使用SOSC模式可忽略此步】配置SOSC辅助振荡器使能位——OSCCON的SOSCEN 
5. 【不使用FRCPLL、FRCDIV模式可忽略此步】配置FRC后分频比——CLKDIV的RCIDV2:RCDIV0 
6. 【不使用打盹模式可忽略此步】配置CPU与外设的时钟比——CLKDIV的DOZE2:DOZE0 
7. 【不使用打盹模式可忽略此步】配置中断是否影响打盹使能位(DOZEN)——CLKDIV的RIO 
8. 【不使用打盹模式可忽略此步】使能打盹模式——CLKDIV的DOZEN 
9. 【不调节FRC振荡器频率,应忽略此步】校准FRC振荡器频率——OSCTUN的TUN5:TUN0 
(四个寄存器位组成参考文章末尾)

在不使用时钟切换的模式下,以上9步就可解决振荡器的配置问题。上面涉及到4个寄存器CW2、OSCCON、CLKDIV和OSCTUN,要配置他们可不是直接使用“_SOSCEN = 1;”或“OSCCONbits.SOSCEN = 1;”那样简单。CLKDIV和OSCTUN两个寄存器可按前面的方式进行配置,而CW2与OSCCON需要通过其他方式进行配置: 
1)CW2——两种方法 
可通过IDE自带配置位界面(Configure->Configuration Bits…)直接选择 
或在#include包含文件位置后,使用代码_CONFIG2(value),value为配置数值,如: 
_CONFIG2(POSCMOD_NONE & FNOSC_FRCPLL & OSCIOFNC_ON & FCKSM_CSECME) 
2)OSCCON——也有两种方法 
首先说明下,OSCCON是个核心寄存器,不是可以随便编辑的,用了两把锁把它的高低字节分别锁起来了。所以要编辑它,必须先解锁。高字节OSCCON<15:8>写序列为:连续将78h和9Ah写入高字节进行解锁,立即写入需要的数值。低字节OSCCON<7:0>写序列为:连续将46h和57h写入低字节进行解锁,立即写入需要的数值。 
因此,要先区分所编辑位属于OSCCON高字节还是低字节,再按要求进行解锁和写入。 
第一种方法,使用内置函数,__builtin_write_OSCCONH(value)来配置OSCCON高字节,和__builtin_write_OSCCONL(value)来配置OSCCON低字节。使用内置函数,不需要考虑解锁,编译成汇编代码已经包含了解锁序列,见下图(参考C30编译器的帮助文件hlpMPLABC30.chm) 
写入OSCCON的内置函数 
第二种方法,直接使用汇编语言,嵌套在C语言中。发现C30不支持#asm和#endasm的多行汇编,就使用单行嵌入“asm(instruction);”,望知道的大侠告诉一声(谢谢^_^)。

    asm("mov #OSCCONH, w1");

    asm("mov #0x78, w2");

    asm("mov #0x9A, w3");

    asm("mov #0x00, w4"); //0x00 is the value will write to OSCCONH

    asm("mov.b w2, [w1]");

    asm("mov.b w3, [w1]");

    asm("mov.b w4, [w1]");


    asm("mov #OSCCONL, w1");

    asm("mov #0x46, w2");

    asm("mov #0x57, w3");

    asm("mov #0x01, w4"); //0x01 is the value will write to OSCCONL

    asm("mov.b w2, [w1]");

    asm("mov.b w3, [w1]");

    asm("mov.b w4, [w1]");


相信用C语言写程序的人都会选内置函数。 

以下是个完整的代码,使用FRCPLL时钟模式,Fosc=32M,用定时器T1开关LED灯,实现每3s切换一次状态:


#include


_CONFIG1(JTAGEN_OFF & GCP_OFF & FWDTEN_OFF)

_CONFIG2(POSCMOD_NONE & FNOSC_FRCPLL & OSCIOFNC_ON) // clock mode:FRCPLL, OSCO/RA3 functions as port I/O


typedef char BYTE;

typedef unsigned int WORD;

typedef unsigned long int DWORD;


#define LED_TRIS        _TRISA3

#define LED_ON()        _LATA3 = 1

#define LED_OFF()       _LATA3 = 0

#define LED_TRIGGER()   _LATA3 = ~_LATA3


#define TIME_TEN_MICROSECOND    300


WORD ledTriggerCount = 0;


void IC_Initialize(void)

{

    /*Oscillator configuration :32M FRCPLL*/

    __builtin_write_OSCCONL(0x00);

    CLKDIVbits.RCDIV = 0b000; //FRC postscaler divide by 1, is 8M


    /*enable DOZE mode*/

    //CLKDIVbits.DOZE = 0b001; 

    //CLKDIVbits.DOZEN = 1; //enable DOZE bit


    /*initialize T1*/

    T1CONbits.TCS = 0; // internal clock

    T1CONbits.TGATE = 0; //disable Gated time accumulation

    T1CONbits.TCKPS = 0b01; //prescale =1:8, T1 = 2*8/fosc = 0.5us

    T1CONbits.TON =0;

    TMR1 = 0;

    PR1 = 20000; // time on a cycle is 10us

    _T1MD = 0; //default value, enable clock source to T1

    _T1IP = 7; //highest priority

    _T1IF = 0;

    _T1IE = 1; //enable T1 interrupt

}


int main(void)

{

    IC_Initialize();

    LED_TRIS =0;

    LED_ON();

    T1CONbits.TON =1; //T1 start

    while(1);


    return 0;

}


void __attribute__ ((__interrupt__, no_auto_psv)) _T1Interrupt(void)

{

    if(++ledTriggerCount == TIME_TEN_MICROSECOND){

        ledTriggerCount = 0;

        LED_TRIGGER();

    }

    _T1IF = 0;

}


二. 时钟切换


时钟切换按正常逻辑来理解,应该是告诉我一个新时钟模式,然后我切换过去就好了。对,就是这么简单。具体地寄存器操作步骤,看下面: 

1. 开启时钟切换功能,FCKSM1位必须清零——CW2的FCKSM1:FCKSM0 

2. 配置新时钟模式——OSCCON的NOSC2:NOSC0 

3. 开始切换——OSCCON的OSWEN 

三步完成时钟切换,但有四点要注意: 

1)主振荡器下的三个子模式(XT、HS和EC)是由配置位决定,他们之间无法切换的。这好理解,你用一台发电机给工厂发电,你要切换发电机,在不断电的情况下不好办吧,得先断电后再切换。这里要切换就要重新烧录程序并设置配置位 

2)使能PLL的主振荡器与FRCPLL之间也不能直接切换,但可通过先中转到FRC下再切换 

3)涉及到引脚或分频类的,要注意设置好,参考datasheet,这里不再赘述 

4)OSCCON的COSC2:COSC0可读出当前时钟模式,在切换前可先判断当前时钟模式 

下面实例代码,在FRCPLL(Fosc=32M)和FRC(Fosc=8M)模式之间循环切换,通过LED呈现状态结果。在FRCPLL模式下,LED亮2s,灭2s,然后切换到FRC模式下,亮8s,灭8s,再切换到FRCPLL模式下,如此循环:


#include


_CONFIG1(JTAGEN_OFF & GCP_OFF & FWDTEN_OFF)

_CONFIG2(POSCMOD_NONE & FNOSC_FRCPLL & OSCIOFNC_ON & FCKSM_CSECME) // OSCO/RA3 functions as port I/O, enable clock switch


typedef char BYTE;

typedef unsigned int WORD;

typedef unsigned long int DWORD;


#define LED_TRIS        _TRISA3

#define LED_ON()        _LATA3 = 1

#define LED_OFF()       _LATA3 = 0

#define LED_TRIGGER()   _LATA3 = ~_LATA3


#define TIME_TEN_MICROSECOND    200


WORD ledTriggerCount = 0;

BYTE  oscillatorSwitchCount = 0;


void IC_ClockSwitch(BYTE newOSC)

{

    _T1IE = 0; //disable all interrupts before switching    

    __builtin_write_OSCCONH(newOSC); //set new oscillator mode

    __builtin_write_OSCCONL(0x01);  //enable oscillator switch


    while(OSCCONbits.OSWEN); //waiting for a successful clock transition

    _T1IE = 1; //enable interrupt after switched

}


void IC_Initialize(void)

{

    //Oscillator configuration :32M FRCPLL

    CLKDIVbits.RCDIV = 0b000; //FRC postscaler divide by 1, input 4*PLL is 8M

    CLKDIVbits.DOZE = 0b001; //1:2, CPU clock:16M

    CLKDIVbits.DOZEN = 1; //enable DOZE bit


    //initialize T1

    T1CONbits.TCS = 0; // internal clock

    T1CONbits.TGATE = 0; //disable Gated time accumulation

    T1CONbits.TCKPS = 0b01; //prescale =1:8, T1 = 2*8/fosc = 0.5us

    T1CONbits.TON =0;

    TMR1 = 0;

    PR1 = 20000; // time on a cycle is 10us

    _T1MD = 0; //default value, enable clock source to T1

    _T1IP = 7; //highest priority

    _T1IF = 0;

    _T1IE = 1; //enable T1 interrupt


}


int main(void)

{

    IC_Initialize();


    LED_TRIS =0;

    LED_ON();

    T1CONbits.TON =1; //T1 start

    while(1);


    return 0;

}


void __attribute__ ((__interrupt__, no_auto_psv)) _T1Interrupt(void)

{

    if(++ledTriggerCount == TIME_TEN_MICROSECOND){

        ledTriggerCount = 0;

        LED_TRIGGER();

        if(++oscillatorSwitchCount == 2){

            oscillatorSwitchCount = 0;

            IC_ClockSwitch(~OSCCONbits.COSC&0x01);//switch clock mode between FRCPLL and FRC

        }

    }

    _T1IF = 0;

}


里面的代码“IC_ClockSwitch(~OSCCONbits.COSC&0x01);”与下面代码功能一样:


if(OSCCONbits.COSC == 0b001)

    IC_ClockSwitch(0x00); //FRCPLL -> FRC

else

    IC_ClockSwitch(0x01); //FRC -> FRCPLL

CW2

OSCCON

CLKDIV

OSCTUN


关键字:PIC  振荡器配置  时钟切换

编辑:什么鱼 引用地址:http://news.eeworld.com.cn/mcu/2018/ic-news073040589.html
本网站转载的所有的文章、图片、音频视频文件等资料的版权归版权所有人所有,本站采用的非本站原创文章及图片等内容无法一一联系确认版权者。如果本网所选内容的文章作者及编辑认为其作品不宜公开自由传播,或不应无偿使用,请及时通过电子邮件或电话通知我们,以迅速采取适当措施,避免给双方造成不必要的经济损失。

上一篇:Pic18F25K80 16位模式下的定时器0配置
下一篇:PIC单片机延时问题

关注eeworld公众号 快捷获取更多信息
关注eeworld公众号
快捷获取更多信息
关注eeworld服务号 享受更多官方福利
关注eeworld服务号
享受更多官方福利

推荐阅读

只有想不到没有做不到!DLP® Pico™ 芯片组还能这么玩

相信大家对于DLP Pico芯片组及其应用并不陌生。但大多数人只了解在自己相关的领域,DLP Pico芯片组可以做什么,比如微投或是抬头显示。其实它可以做的事情还有很多很多。DLP Pico芯片组支持从nHD到4K UHD的分辨率,可实现多种创新的高性能超便携紧凑型显示应用。产品开发人员可以选择符合应用要求的芯片组和光学模块,快速实现产品上市。DLP Pico芯片组也可用于非显示型应用,例如3D打印和3D机器视觉。下面我们就来列举DLP Pico芯片组的5大应用领域。Pico投影仪Pico投影仪可用作便携式大屏幕显示器,适用于任何具有视频输出的设备,例如笔记本电脑、智能手机、平板电脑和游戏机。此类投影仪可为用户提供简便轻型的方式
发表于 2019-06-06
只有想不到没有做不到!DLP® Pico™ 芯片组还能这么玩

TI DLP Pico微投技术让智能音箱如虎添翼

随着 IoT 解决方案在全球消费市场的普及,智能音箱正在逐步成为家庭消费电子的重要一部分。目前,智能音箱普遍具备按需虚拟助手功能和高品质的音频性能。因此,将显示功能融合到这些“一成不变的”电子产品中,便成为顺理成章的事。然而,在小型音箱上显示视频内容充满挑战。将从平板电脑到小尺寸电视的显示改装成紧凑、美观的外形设计非常不易,不过借助 TI DLP Pico 技术,可通过小巧的设计来实现大画面的投影显示。智能音箱中投影显示的作用向智能音箱添加显示屏以扩展其功能是一个自然而然的事情。汽车环境中的中控台显示屏应用正在迅速兴起,与之类似,家庭信息化/娱乐设备的类似视觉体验也将让消费者受益匪浅。现在,用户通过智能音箱请求内容的方式不同于
发表于 2019-05-16
TI DLP Pico微投技术让智能音箱如虎添翼

PIC单片机实现二进制码与压缩BCD码的相互转换

;                         ; 兼容ICD调试工具,必须加nop    goto        Main             ; 跳转至Main函数;*************************Main 函数的代码******************************Main         
发表于 2019-05-11

PIC单片机花式点亮LED

; nop                             ; 与ICD调试配合的nop    goto        main                ; 跳转到Main;**********************************************************************   
发表于 2019-05-11

PIC单片机实现冒泡排序算法

;      ; 复位入口地址    nop                          ; 兼容ICD调试工具,必须加nop    goto        Main             ; 跳转至Main函数;*************************Main 函数的代码
发表于 2019-05-11

PIC单片机-七段数码管的使用

;  //延时1ms    }} 附完整程序:/******************************************************************************** 标    题: 静态数码管显示* 跳线接法:断开P14上的短接帽* 功能描述: 数码管显示“0123”*******************************************************************************///#include <pic16f877a.h>#include <htc.h> 
发表于 2019-05-11

小广播

何立民专栏

单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号 电信业务审批[2006]字第258号函 京公海网安备110108001534 Copyright © 2005-2019 EEWORLD.com.cn, Inc. All rights reserved