⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 f02x_oscillator_capacitor.c

📁 C8051f020 的源码包括所有常用部分的
💻 C
字号:
//-----------------------------------------------------------------------------
// F02x_Oscillator_Capacitor.c
//-----------------------------------------------------------------------------
// Copyright 2006 Silicon Laboratories, Inc.
// http://www.silabs.com
//
// Program Description:
//
// This example demonstrates how to initialize for and switch to a Capacitor
// configuration for the external oscillator.
// - Pinout:
//    P0.0 -> /SYSCLK
//
//    P1.6 -> LED
//
//    all other port pins unused
//
//    XTAL1 and XTAL2 -> External Capacitor Configuration
//
//
// How To Test:
//
// 1) Define the VDD voltage, capacitor value, and desired frequency using
//      <AVplus_Value>, <Cap_Value>, and <Desired_Freq>.
// 2) Compile and download code to a 'F02x device.
// 3) Run the code:
//      - the test will blink an LED
//      - the 'F02x will also output the SYSCLK to a port pin (P0.0) for
//         observation
//
//
// FID:            02X000013
// Target:         C8051F02x
// Tool chain:     Keil C51 7.50 / Keil EVAL C51
// Command Line:   None
//
// Release 1.0
//    -Initial Revision (TP)
//    -9 JUNE 2006
//

//-----------------------------------------------------------------------------
// Includes
//-----------------------------------------------------------------------------

#include <C8051F020.h>                 // SFR declarations

//-----------------------------------------------------------------------------
// Global CONSTANTS
//-----------------------------------------------------------------------------

// Note: The AVplus_Value, Cap_Value, Desired_Freq, and KFactor numbers are
// formatted specifically to keep all numbers whole (not decimals) and within
// a representable range (32 bits).

#define AVplus_Value 3300              // Value of AV+ for the 'F0xx in mV
                                       // VDD should be no higher than 3600 mV

#define Cap_Value 56                   // Value of the external capacitor, in
                                       // pF
                                       // The capacitor should be no greater
                                       // than 100 pF.

#define Desired_Freq 100000            // roughly the desired frequency, in Hz
                                       // Note: Debugging cannot occur when the
                                       // SYSCLK is less than 32 kHz.  The
                                       // capacitor external oscillator
                                       // includes a divide by 2 stage.

sfr16 RCAP2 = 0xCA;                    // Timer2 reload value
sfr16 TMR2 = 0xCC;                     // Timer2 counter

sbit LED = P1^6;                       // LED='1' means ON

//-----------------------------------------------------------------------------
// Function PROTOTYPES
//-----------------------------------------------------------------------------

void SYSCLK_Init (void);
void ExtCapOsc_Init (void);
void Port_Init (void);
void Timer2_Init (void);

void Timer2_ISR (void);

//-----------------------------------------------------------------------------
// MAIN Routine
//-----------------------------------------------------------------------------
//
// Main routine performs all configuration tasks, switches to the external
// crystal oscillator, and loops forever, blinking the LED.
//
//-----------------------------------------------------------------------------
void main (void) 
{
   WDTCN = 0xDE;                       // Disable watchdog timer
   WDTCN = 0xAD;

   SYSCLK_Init ();                     // Initialize system clock to 16MHz

   ExtCapOsc_Init ();                  // Initialize for and switch to the
                                       // external capacitor 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
//-----------------------------------------------------------------------------

//-----------------------------------------------------------------------------
// SYSCLK_Init ()
//-----------------------------------------------------------------------------
//
// Return Value : None
// Parameters   : None
//
// This routine initializes the system clock to use the internal 16 MHz
// oscillator as its clock source.  Also enables missing clock detector reset.
//
//-----------------------------------------------------------------------------

void SYSCLK_Init (void)
{
   OSCICN |= 0x03;                     // Configure internal oscillator for
                                       // its highest frequency (16 MHz)

   OSCICN |= 0x80;                     // Enable missing clock detector
}

//-----------------------------------------------------------------------------
// ExtCapOsc_Init ()
//-----------------------------------------------------------------------------
//
// Return Value : None
// Parameters   : None
//
// This routine initializes for and switches to an external capacitor for the
// External Oscillator.
//
//-----------------------------------------------------------------------------
void ExtCapOsc_Init (void)
{
   // Error checks on the input values
   // The capacitor should be no greater than 100 pF.
   #if (Cap_Value > 100)
      #error "Defined Capacitor Value outside allowable range!"
      #define XFCN 0
   // The AV+ value should not be above 3.6 V, according to the Table in
   // Chapter 3 of the datasheet.
   #elif (AVplus_Value > 3600)
      #error "Defined AV+ Value outside allowable range!"
      #define XFCN 0
   #endif

   // Set the appropriate XFCN bits for the oscillator frequency
   //
   //   XFCN     C (XOSCMD = 10x)
   //   000      K Factor = 0.44 => 440
   //   001      K Factor = 1.4 => 1400
   //   010      K Factor = 4.4 => 4400
   //   011      K Factor = 13 => 13000
   //   100      K Factor = 38 => 38000
   //   101      K Factor = 100 => 100000
   //   110      K Factor = 420 => 420000
   //   111      K Factor = 1400 => 1400000

   // from the datasheet: f (in MHz) = KF / (C x AV+ (in V))
   // f (in MHz) x C x AV+ (in V) = KF
   // f (in Hz) x C x AV+ (in V) = KF x 1000000
   // f (in Hz) x C x AV+ (in mV) = KF x 1000000 x 1000
   // f (in Hz) x C x AV+ (in mV) = KF x 1000000000
   //
   // Desired_Freq x Cap_Value x AVplus_Value = KFactor x 1000000000
   // (Desired_Freq x Cap_Value x AVplus_Value) / 1000000 = KFactor x 1000
   // (Desired_Freq)/1000 x (Cap_Value x AVplus_Value)/1000 = KFactor x 1000
   #define KFactor (Desired_Freq/1000)*((Cap_Value*AVplus_Value)/1000)

   // Try to determine the closest XFCN setting to the KFactor value
   //
   // For example, if the KFactor is 2000, the XFCN value should be 2.6,
   // but if the KFactor is 1600, the XFCN value should be 0.87
   #if (KFactor <= 920L) // (440 + 1400) / 2
      #define XFCN 0
   #elif (KFactor <= 2900L) // (1400 + 4400) / 2
      #define XFCN 1
   #elif (KFactor <= 8700L) // (4400 + 13000) / 2
      #define XFCN 2
   #elif (KFactor <= 25500L) // (13000 + 38000) / 2
      #define XFCN 3
   #elif (KFactor <= 69000L) // (38000 + 100000) / 2
      #define XFCN 4
   #elif (KFactor <= 260000L) // (100000 + 420000) / 2
      #define XFCN 5
   #elif (KFactor <= 910000L) // (420000 + 1400000) / 2
      #define XFCN 6
   #else // > (420000 + 1400000) / 2
      #define XFCN 7
   #endif

   OSCXCN = 0x40;                      // External Oscillator is an external
                                       // capacitor (divide by 2 stage)

   OSCXCN |= XFCN;

   // RC and C modes typically require no startup time.

   OSCICN |= 0x08;                     // Switch the system clock to the
                                       // external oscillator circuit.
}

//-----------------------------------------------------------------------------
// Port_Init ()
//-----------------------------------------------------------------------------
//
// Return Value : None
// Parameters   : None
//
// Configure the Crossbar and GPIO ports.
//
// P0.0   digital   push-pull     /SYSCLK
//
// P1.6   digital   push-pull     LED
//
//-----------------------------------------------------------------------------
void PORT_Init (void)
{
   P0MDOUT |= 0x01;                    // Enable /SYSCLK as a push-pull output

   P1MDOUT |= 0x40;                    // Enable LED as a push-pull output

   XBR1 = 0x80;                        // Route /SYSCLK to a port pin
   XBR2 = 0x40;                        // Enable crossbar and weak pull-ups
}

//-----------------------------------------------------------------------------
// Timer2_Init ()
//-----------------------------------------------------------------------------
//
// Return Value : None
// Parameters   : None
//
// Configure Timer2 to 16-bit auto-reload and generate an interrupt for an
// approximate 10 Hz blinking rate.
//
//-----------------------------------------------------------------------------
void Timer2_Init (void)
{
   T2CON = 0x00;                       // Stop Timer2; Clear TF2;
                                       // auto-reload mode

   CKCON &= ~0x20;                     // use SYSCLK/12 as timebase

   RCAP2 = -(Desired_Freq/2/12/10);    // Init reload value;
                                       // Capacitor external oscillator with
                                       // divide by 2 stage, Timer2 uses
                                       // SYSCLK/12 for timebase, aim for a
                                       // blink rate of roughly 10 Hz
   TMR2 = -(Desired_Freq/2/12/10);     // Init the Timer2 value.

   ET2 = 1;                            // Enable Timer2 interrupts

   TR2 = 1;                            // Start Timer2
}

//-----------------------------------------------------------------------------
// Interrupt Service Routines
//-----------------------------------------------------------------------------

//-----------------------------------------------------------------------------
// Timer2_ISR
//-----------------------------------------------------------------------------
//
// This routine changes the state of the LED whenever Timer2 overflows.
//
//-----------------------------------------------------------------------------
void Timer2_ISR (void) interrupt 5
{
   TF2 = 0;                            // Clear Timer2 interrupt flag

   LED = ~LED;                         // Change state of LED
}

//-----------------------------------------------------------------------------
// End Of File
//-----------------------------------------------------------------------------

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -