📄 f500_oscillator_cmos.c
字号:
//-----------------------------------------------------------------------------
// F500_Oscillator_CMOS.c
//-----------------------------------------------------------------------------
// Copyright 2008 Silicon Laboratories, Inc.
// http://www.silabs.com
//
// Program Description:
//
// This example demonstrates how to initialize for and switch to an external
// CMOS Oscillator.
//
// P0.0 -> /SYSCLK
// P0.3 -> XTAL2 - External CMOS Oscillator input
// P1.3 -> LED
//
// How To Test:
//
// 1) Define the input CMOS clock frequency using <CMOS_clock>
// 2) Compile and download code to a 'F500 device.
// 3) Connect the CMOS input clock to XTAL2 / P0.3 (or J2 on the TB)
// 4) Run the code:
// - the test will blink an LED at a frequency based on the CMOS
// Oscillator
// - the 'F500 will also output the SYSCLK to a port pin (P0.0) for
// observation
//
//
// Target: C8051F500 (Side A of a C8051F500-TB)
// Tool chain: Keil C51 8.0 / Keil EVAL C51
// Command Line: None
//
// Release 1.0 / 04 MAR 2008 (GP)
// -Initial Revision
//-----------------------------------------------------------------------------
// Includes
//-----------------------------------------------------------------------------
#include <compiler_defs.h>
#include <C8051F500_defs.h> // SFR declarations
//-----------------------------------------------------------------------------
// Global CONSTANTS
//-----------------------------------------------------------------------------
#define CMOS_clock 3062500 // CMOS oscillator frequency
// Timer2 using SYSCLK as its time base
// Timer2 counts 65536 SYSCLKs per Timer2 interrupt
// LED target flash rate = 10 Hz (may be slower, depending on CMOS_clock)
#define LED_interrupt_count CMOS_clock / 65536 / 10
SBIT (LED, SFR_P1, 3); // LED == 1 means ON
//-----------------------------------------------------------------------------
// Function PROTOTYPES
//-----------------------------------------------------------------------------
void OSCILLATOR_Init (void);
void PORT_Init (void);
void TIMER2_Init (void);
INTERRUPT_PROTO (TIMER2_ISR, INTERRUPT_TIMER2);
//-----------------------------------------------------------------------------
// MAIN Routine
//-----------------------------------------------------------------------------
//
// Main routine performs all configuration tasks, switches to the external CMOS
// oscillator, and loops forever, blinking the LED.
//
//-----------------------------------------------------------------------------
void main (void)
{
SFRPAGE = ACTIVE_PAGE; // Set SFR Page for PCA0MD
PCA0MD &= ~0x40; // Disable the watchdog timer
OSCILLATOR_Init (); // Initialize External CMOS Oscillator
PORT_Init (); // Initialize crossbar and GPIO
TIMER2_Init (); // Init Timer2 to generate
// interrupts for the LED.
EA = 1; // Enable global interrupts
while (1); // Spin forever
}
//-----------------------------------------------------------------------------
// Initialization Routines
//-----------------------------------------------------------------------------
//-----------------------------------------------------------------------------
// OSCILLATOR_Init ()
//-----------------------------------------------------------------------------
//
// Return Value : None
// Parameters : None
//
// This routine initializes for and switches to the External CMOS Oscillator
// and also enables the missing clock detector reset.
//
// Note: In RC, capacitor, or CMOS clock configuration, the clock source
// should be wired to the XTAL2 pin as shown in Option 2, 3, or 4 in the
// Oscillators section of the datasheet.
//
//-----------------------------------------------------------------------------
void OSCILLATOR_Init (void)
{
U8 SFRPAGE_save = SFRPAGE;
SFRPAGE = ACTIVE_PAGE;
RSTSRC = 0x04; // Enable missing clock detector
SFRPAGE = CONFIG_PAGE;
P0MDOUT &= ~0x08; // Configure P0.3 as open-drain
P0MDIN |= 0x08; // In CMOS clock mode, the associated
// pin should be configured as a
// digital input.
OSCXCN = 0x20; // External Oscillator is a CMOS clock
// (no divide by 2 stage)
// XFCN bit settings do not apply to a
// CMOS clock
CLKSEL = 0x01; // Switch to the external CMOS clock
SFRPAGE = SFRPAGE_save;
}
//-----------------------------------------------------------------------------
// Port_Init ()
//-----------------------------------------------------------------------------
//
// Return Value : None
// Parameters : None
//
// Configure the Crossbar and GPIO ports.
//
// P0.0 digital push-pull /SYSCLK
// P0.3 digital open-drain XTAL2 - External CMOS Oscillator input
// P1.3 digital push-pull LED
//
//-----------------------------------------------------------------------------
void PORT_Init (void)
{
U8 SFRPAGE_save = SFRPAGE;
SFRPAGE = CONFIG_PAGE;
P0MDOUT |= 0x01; // Enable /SYSCLK as a push-pull output
P1MDOUT |= 0x08; // Enable LED as a push-pull output
P0SKIP |= 0x08; // Skip the XTAL2 pin
P1SKIP |= 0x08; // Skip the LED pin
XBR1 = 0x02; // Route /SYSCLK to a port pin
XBR1 = 0x40; // Enable crossbar and weak pull-ups
SFRPAGE = SFRPAGE_save;
}
//-----------------------------------------------------------------------------
// TIMER2_Init ()
//-----------------------------------------------------------------------------
//
// Return Value : None
// Parameters : None
//
// Configure Timer2 to 16-bit auto-reload and generate an interrupt after the
// maximum possible time (TMR2RL = 0x0000).
//
//-----------------------------------------------------------------------------
void TIMER2_Init (void)
{
U8 SFRPAGE_save = SFRPAGE;
SFRPAGE = ACTIVE_PAGE;
CKCON |= 0x30; // Timer2 clock is system clock
TMR2RL = 0x0000; // Init reload value
TMR2 = 0xFFFF; // Set to reload immediately
ET2 = 1; // Enable Timer2 interrupts
TR2 = 1; // Start Timer2
SFRPAGE = SFRPAGE_save;
}
//-----------------------------------------------------------------------------
// Interrupt Service Routines
//-----------------------------------------------------------------------------
//-----------------------------------------------------------------------------
// Timer2_ISR
//-----------------------------------------------------------------------------
//
// This routine changes the state of the LED whenever Timer2 overflows a
// certain number of times
//
//-----------------------------------------------------------------------------
INTERRUPT (TIMER2_ISR, INTERRUPT_TIMER2)
{
static U16 count = 0;
TF2H = 0; // Clear Timer2 interrupt flag
if (count == LED_interrupt_count) // If counter reaches target overflows
{
LED = ~LED; // Change state of LED
count = 0; // Reset counter
}
else
{
count++; // Count overflow
}
}
//-----------------------------------------------------------------------------
// End Of File
//-----------------------------------------------------------------------------
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -