]> git.karo-electronics.de Git - oswald.git/blobdiff - metawatch/F5xx_F6xx_Core_Lib/HAL_UCS.c
Power saving changes, add new fonts, bitmaps and screens
[oswald.git] / metawatch / F5xx_F6xx_Core_Lib / HAL_UCS.c
diff --git a/metawatch/F5xx_F6xx_Core_Lib/HAL_UCS.c b/metawatch/F5xx_F6xx_Core_Lib/HAL_UCS.c
new file mode 100644 (file)
index 0000000..ae015f6
--- /dev/null
@@ -0,0 +1,340 @@
+/*******************************************************************************
+ *
+ * HAL_UCS.c
+ * Provides Functions to Initialize the UCS/FLL and clock sources
+ * 
+ *
+ * Copyright (C) 2010 Texas Instruments Incorporated - http://www.ti.com/ 
+ * 
+ * 
+ *  Redistribution and use in source and binary forms, with or without 
+ *  modification, are permitted provided that the following conditions 
+ *  are met:
+ *
+ *    Redistributions of source code must retain the above copyright 
+ *    notice, this list of conditions and the following disclaimer.
+ *
+ *    Redistributions in binary form must reproduce the above copyright
+ *    notice, this list of conditions and the following disclaimer in the 
+ *    documentation and/or other materials provided with the   
+ *    distribution.
+ *
+ *    Neither the name of Texas Instruments Incorporated nor the names of
+ *    its contributors may be used to endorse or promote products derived
+ *    from this software without specific prior written permission.
+ *
+ *  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 
+ *  "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 
+ *  LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ *  A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 
+ *  OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 
+ *  SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT 
+ *  LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ *  DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ *  THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 
+ *  (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 
+ *  OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ * 
+ * Created: Version 1.0 11/24/2009
+ * Updated: Version 2.0 12/15/2010
+ *          Added Functions: XT2_Stop() and XT1_Stop()
+ *          Modified all functions to preserve drive settings
+ *  
+ ******************************************************************************/
+
+#include "msp430.h"
+#include "HAL_UCS.h"
+// #include "hal_calibration.h"
+
+/*******************************************************************************
+ * Check and define required Defines
+ ******************************************************************************/
+#ifndef XT1LFOFFG               // Defines if not available in header file
+#define XT1LFOFFG   0
+#endif
+
+#ifndef XT1HFOFFG               // Defines if not available in header file
+#define XT1HFOFFG   0
+#endif
+
+#ifndef XT2OFFG                 // Defines if not available in header file
+#define XT2OFFG     0
+#endif
+
+#ifndef XTS                     // Defines if not available in header file
+#define XTS         0
+#endif
+
+#ifndef XT2DRIVE_3              // Defines if not available in header file
+#define XT2DRIVE_3  0
+#endif
+
+
+/*******************************************************************************
+ * \brief   Initializes FLL of the UCS
+ *
+ * \param fsystem  Required system frequency (MCLK) in kHz
+ * \param ratio    Ratio between fsystem and FLLREFCLK
+ ******************************************************************************/
+static void Init_FLL(unsigned int fsystem, unsigned int ratio);
+
+
+#if 0
+void LFXT_Start(unsigned int xtdrive)
+{
+  // If the drive setting is not already set to maximum
+  // Set it to max for LFXT startup
+  if ((UCSCTL6 & XT1DRIVE_3)!= XT1DRIVE_3) { 
+    UCSCTL6_L |= XT1DRIVE1_L + XT1DRIVE0_L; // Highest drive setting for XT1startup
+  }
+
+  while (SFRIFG1 & OFIFG) {   // Check OFIFG fault flag
+    UCSCTL7 &= ~(DCOFFG+XT1LFOFFG+XT1HFOFFG+XT2OFFG); // Clear OSC flaut Flags fault flags
+    SFRIFG1 &= ~OFIFG;        // Clear OFIFG fault flag
+  }
+  
+  UCSCTL6 = (UCSCTL6 & ~(XT1DRIVE_3)) | (xtdrive); // set requested Drive mode
+}
+#endif
+
+unsigned int LFXT_Start_Timeout(unsigned int xtdrive, unsigned int timeout)
+{
+  // add in capacitor setting
+  //SetOscillatorCapacitorValues();
+  
+  // If the drive setting is not already set to maximum
+  // Set it to max for LFXT startup
+  if ((UCSCTL6 & XT1DRIVE_3)!= XT1DRIVE_3) { 
+   UCSCTL6_L |= XT1DRIVE1_L+XT1DRIVE0_L; // Highest drive setting for XT1startup
+  }
+
+  while ((SFRIFG1 & OFIFG) && timeout--){   // Check OFIFG fault flag
+    UCSCTL7 &= ~(DCOFFG+XT1LFOFFG+XT1HFOFFG+XT2OFFG); // Clear OSC flaut Flags fault flags
+    SFRIFG1 &= ~OFIFG;        // Clear OFIFG fault flag
+  }
+  
+  UCSCTL6 = (UCSCTL6 & ~(XT1DRIVE_3)) |(xtdrive); // set Drive mode
+  
+  // add in capacitor setting
+  //SetOscillatorCapacitorValues();
+  
+  if (timeout)
+    return (UCS_STATUS_OK);
+  else
+    return (UCS_STATUS_ERROR);
+}
+
+#if 0
+void XT1_Start(unsigned int xtdrive)
+{
+  // Check if drive value is the expected one
+  if ((UCSCTL6 & XT1DRIVE_3) != xtdrive) {
+    UCSCTL6 &= ~XT1DRIVE_3;                 // Clear XT1drive field
+    UCSCTL6 |= xtdrive;                     // Set requested value
+  }
+  
+  UCSCTL6 &= ~XT1OFF;                       // Enable XT1
+  UCSCTL6 |= XTS;                           // Enable HF mode
+
+  while (SFRIFG1 & OFIFG) {   // Check OFIFG fault flag
+    UCSCTL7 &= ~(DCOFFG+XT1LFOFFG+XT1HFOFFG+XT2OFFG); // Clear OSC flaut Flags
+    SFRIFG1 &= ~OFIFG;        // Clear OFIFG fault flag
+  }
+}
+
+unsigned int XT1_Start_Timeout(unsigned int xtdrive, unsigned int timeout)
+{
+  // Check if drive value is the expected one
+  if ((UCSCTL6 & XT1DRIVE_3) != xtdrive) {
+    UCSCTL6 &= ~XT1DRIVE_3;                 // Clear XT1drive field
+    UCSCTL6 |= xtdrive;                     // Set requested value
+  }
+  
+  UCSCTL6 &= ~XT1OFF;                       // Enable XT1
+  UCSCTL6 |= XTS;                           // Enable HF mode
+
+  while ((SFRIFG1 & OFIFG) && timeout--) {  // Check OFIFG fault flag
+    UCSCTL7 &= ~(DCOFFG+XT1LFOFFG+XT1HFOFFG+XT2OFFG); // Clear OSC flaut Flags
+    SFRIFG1 &= ~OFIFG;                      // Clear OFIFG fault flag
+  }
+  
+  if (timeout) {
+    return UCS_STATUS_OK;
+  }
+  else {
+    return UCS_STATUS_ERROR;
+  }
+}
+
+void XT1_Bypass(void)
+{
+  UCSCTL6 |= XT1BYPASS;
+
+  while (SFRIFG1 & OFIFG) {   // Check OFIFG fault flag
+    UCSCTL7 &= ~(DCOFFG+XT1LFOFFG+XT1HFOFFG+XT2OFFG); // Clear OSC flaut Flags
+    SFRIFG1 &= ~OFIFG;        // Clear OFIFG fault flag
+  }
+}
+#endif
+
+void XT1_Stop(void)
+{
+  UCSCTL6 |= XT1OFF;                         // Switch off XT1 oscillator
+}
+
+#if 0
+void XT2_Start(unsigned int xtdrive)
+{
+  // Check if drive value is the expected one
+  if ((UCSCTL6 & XT2DRIVE_3) != xtdrive) {
+    UCSCTL6 &= ~XT2DRIVE_3;                 // Clear XT2drive field
+    UCSCTL6 |= xtdrive;                     // Set requested value
+  }
+  
+  UCSCTL6 &= ~XT2OFF; 
+  
+  while (SFRIFG1 & OFIFG) {                 // Check OFIFG fault flag
+    UCSCTL7 &= ~(DCOFFG+XT1LFOFFG+XT1HFOFFG+XT2OFFG); // Clear OSC flaut Flags
+    SFRIFG1 &= ~OFIFG;                      // Clear OFIFG fault flag
+  }
+}
+
+unsigned int XT2_Start_Timeout(unsigned int xtdrive, unsigned int timeout)
+{
+  // Check if drive value is the expected one
+  if ((UCSCTL6 & XT2DRIVE_3) != xtdrive)  {
+    UCSCTL6 &= ~XT2DRIVE_3;                 // Clear XT2drive field
+    UCSCTL6 |= xtdrive;                     // Set requested value
+  }
+  
+  UCSCTL6 &= ~XT2OFF; 
+
+  while ((SFRIFG1 & OFIFG) && timeout--) {  // Check OFIFG fault flag
+    UCSCTL7 &= ~(DCOFFG+XT1LFOFFG+XT1HFOFFG+XT2OFFG); // Clear OSC flaut Flags
+    SFRIFG1 &= ~OFIFG;        // Clear OFIFG fault flag
+  }
+  
+  if (timeout) {
+    return UCS_STATUS_OK;
+  }
+  else {
+    return UCS_STATUS_ERROR;
+  }
+}
+
+void XT2_Bypass(void)
+{
+#ifdef XT2BYPASS              // On devices without XT2 this function will be empty
+  UCSCTL6 |= XT2BYPASS;
+
+  while (SFRIFG1 & OFIFG) {   // Check OFIFG fault flag
+    UCSCTL7 &= ~(DCOFFG+XT1LFOFFG+XT1HFOFFG+XT2OFFG); // Clear OSC flaut Flags
+    SFRIFG1 &= ~OFIFG;        // Clear OFIFG fault flag
+  }
+#endif
+}
+
+void XT2_Stop(void)
+{
+  UCSCTL6 |= XT2OFF;                         // Switch off XT2 oscillator
+}
+#endif
+
+void Init_FLL_Settle(unsigned int fsystem, unsigned int ratio)
+{
+  volatile unsigned int x;
+
+  // x = ratio * 32;
+  x = ratio << 5;
+  Init_FLL(fsystem, ratio);
+  
+  /* from changes in Init_FLL we know that fll is now enabled */
+  while (x--) {
+   __delay_cycles(30); 
+  }
+}
+
+static void Init_FLL(unsigned int fsystem, unsigned int ratio)
+{
+  unsigned int d, dco_div_bits;
+  unsigned int mode;
+  
+#if 0
+  unsigned int srRegisterState;
+#endif
+  
+  mode = 0;
+
+  /* we only run this at startup and we want the fll enabled on exit */
+#if 0
+  // Save actual state of FLL loop control, then disable it. This is needed to
+  // prevent the FLL from acting as we are making fundamental modifications to
+  // the clock setup.
+  srRegisterState = __get_SR_register() & SCG0;
+#endif
+  
+  d = ratio;
+  dco_div_bits = FLLD__2;        // Have at least a divider of 2
+  
+  if (fsystem > 16000) {
+    d >>= 1 ;
+    mode = 1;
+  }
+  else {
+    fsystem <<= 1;               // fsystem = fsystem * 2
+  }
+
+  while (d > 512) {
+    dco_div_bits = dco_div_bits + FLLD0;  // Set next higher div level
+    d >>= 1;
+  }
+
+  // Disable FLL
+  __bis_SR_register(SCG0);  
+  
+  UCSCTL0 = 0x0000;              // Set DCO to lowest Tap
+
+  UCSCTL2 &= ~(0x03FF);          // Reset FN bits
+  UCSCTL2 = dco_div_bits | (d - 1);
+
+  if (fsystem <= 630)            //           fsystem < 0.63MHz
+       UCSCTL1 = DCORSEL_0;
+  else if (fsystem <  1250)      // 0.63MHz < fsystem < 1.25MHz
+       UCSCTL1 = DCORSEL_1;
+  else if (fsystem <  2500)      // 1.25MHz < fsystem <  2.5MHz
+       UCSCTL1 = DCORSEL_2;
+  else if (fsystem <  5000)      // 2.5MHz  < fsystem <    5MHz
+       UCSCTL1 = DCORSEL_3;
+  else if (fsystem <  10000)     // 5MHz    < fsystem <   10MHz
+       UCSCTL1 = DCORSEL_4;
+  else if (fsystem <  20000)     // 10MHz   < fsystem <   20MHz
+       UCSCTL1 = DCORSEL_5;
+  else if (fsystem <  40000)     // 20MHz   < fsystem <   40MHz
+       UCSCTL1 = DCORSEL_6;
+  else
+       UCSCTL1 = DCORSEL_7;
+
+  // Re-enable FLL
+  __bic_SR_register(SCG0);
+  
+  while (SFRIFG1 & OFIFG) {                               // Check OFIFG fault flag
+    UCSCTL7 &= ~(DCOFFG+XT1LFOFFG+XT1HFOFFG+XT2OFFG);     // Clear OSC flaut Flags
+    SFRIFG1 &= ~OFIFG;                                    // Clear OFIFG fault flag
+  }
+
+  /* Init fll is only run at startup - We want the fll enabled when
+   * this function is complete (so don't save a restore setting)
+   */
+#if 0
+  // Restore previous SCG0
+  __bis_SR_register(srRegisterState);                    
+#endif
+  
+  if (mode == 1) {                                               // fsystem > 16000
+    SELECT_MCLK_SMCLK(SELM__DCOCLK + SELS__DCOCLK);       // Select DCOCLK
+  }
+  else {
+    SELECT_MCLK_SMCLK(SELM__DCOCLKDIV + SELS__DCOCLKDIV); // Select DCODIVCLK
+  }
+  
+}