From d8914f064fe735890156a1e39f80ece9d9ee39f1 Mon Sep 17 00:00:00 2001 From: gdisirio Date: Wed, 27 Feb 2013 15:57:45 +0000 Subject: git-svn-id: svn://svn.code.sf.net/p/chibios/svn/trunk@5337 35acf78f-673a-0410-8e92-d51de3d6d3f4 --- demos/PPC-SPC563M-GCC/halconf.h | 2 +- demos/PPC-SPC563M-GCC/mcuconf.h | 5 + os/hal/platforms/SPC563Mxx/platform.mk | 6 +- os/hal/platforms/SPC563Mxx/spc563m_registry.h | 5 + os/hal/platforms/SPC5xx/EQADC_v1/adc_lld.c | 105 +++++++++++++---- os/hal/platforms/SPC5xx/EQADC_v1/adc_lld.h | 163 ++++++++++++++++++++------ 6 files changed, 224 insertions(+), 62 deletions(-) diff --git a/demos/PPC-SPC563M-GCC/halconf.h b/demos/PPC-SPC563M-GCC/halconf.h index 9bff08736..a9d890562 100644 --- a/demos/PPC-SPC563M-GCC/halconf.h +++ b/demos/PPC-SPC563M-GCC/halconf.h @@ -52,7 +52,7 @@ * @brief Enables the ADC subsystem. */ #if !defined(HAL_USE_ADC) || defined(__DOXYGEN__) -#define HAL_USE_ADC FALSE +#define HAL_USE_ADC TRUE #endif /** diff --git a/demos/PPC-SPC563M-GCC/mcuconf.h b/demos/PPC-SPC563M-GCC/mcuconf.h index 77f9ab2e8..da2b259e9 100644 --- a/demos/PPC-SPC563M-GCC/mcuconf.h +++ b/demos/PPC-SPC563M-GCC/mcuconf.h @@ -42,6 +42,11 @@ BIUCR_PFLIM_ON_MISS | \ BIUCR_BFEN) +/* + * ADC driver settings. + */ +#define SPC5_ADC_USE_EQADC_Q0 TRUE + /* * SERIAL driver system settings. */ diff --git a/os/hal/platforms/SPC563Mxx/platform.mk b/os/hal/platforms/SPC563Mxx/platform.mk index 04e9a9739..989cfb09b 100644 --- a/os/hal/platforms/SPC563Mxx/platform.mk +++ b/os/hal/platforms/SPC563Mxx/platform.mk @@ -1,9 +1,11 @@ # List of all the SPC56x platform files. PLATFORMSRC = ${CHIBIOS}/os/hal/platforms/SPC563Mxx/hal_lld.c \ - ${CHIBIOS}/os/hal/platforms/SPC5xx/ESCI_v1/serial_lld.c \ - ${CHIBIOS}/os/hal/platforms/SPC5xx/SIU_v1/pal_lld.c + ${CHIBIOS}/os/hal/platforms/SPC5xx/EQADC_v1/adc_lld.c \ + ${CHIBIOS}/os/hal/platforms/SPC5xx/SIU_v1/pal_lld.c \ + ${CHIBIOS}/os/hal/platforms/SPC5xx/ESCI_v1/serial_lld.c # Required include directories PLATFORMINC = ${CHIBIOS}/os/hal/platforms/SPC563Mxx \ + ${CHIBIOS}/os/hal/platforms/SPC5xx/EQADC_v1 \ ${CHIBIOS}/os/hal/platforms/SPC5xx/ESCI_v1 \ ${CHIBIOS}/os/hal/platforms/SPC5xx/SIU_v1 diff --git a/os/hal/platforms/SPC563Mxx/spc563m_registry.h b/os/hal/platforms/SPC563Mxx/spc563m_registry.h index 72de59156..bf2529c7a 100644 --- a/os/hal/platforms/SPC563Mxx/spc563m_registry.h +++ b/os/hal/platforms/SPC563Mxx/spc563m_registry.h @@ -31,6 +31,11 @@ * @name SPC563Mxx capabilities * @{ */ +/* eQADC attributes.*/ +#define SPC5_HAS_EQADC TRUE +#define SPC5_EQADC1_HANDLER vector146 +#define SPC5_EQADC1_NUMBER 146 + /* eSCI attributes.*/ #define SPC5_HAS_ESCIA TRUE #define SPC5_ESCIA_HANDLER vector146 diff --git a/os/hal/platforms/SPC5xx/EQADC_v1/adc_lld.c b/os/hal/platforms/SPC5xx/EQADC_v1/adc_lld.c index 083d84500..5bd1113f7 100644 --- a/os/hal/platforms/SPC5xx/EQADC_v1/adc_lld.c +++ b/os/hal/platforms/SPC5xx/EQADC_v1/adc_lld.c @@ -36,15 +36,8 @@ /** * @brief ADCD1 driver identifier. */ -#if SPC5_ADC_USE_EQADC1_Q0 || defined(__DOXYGEN__) -ADCDriver ADCD10; -#endif - -/** - * @brief ADCD20 driver identifier. - */ -#if SPC5_ADC_USE_EQADC2_Q0 || defined(__DOXYGEN__) -ADCDriver ADCD20; +#if SPC5_ADC_USE_EQADC_Q0 || defined(__DOXYGEN__) +ADCDriver ADCD1; #endif /*===========================================================================*/ @@ -55,6 +48,79 @@ ADCDriver ADCD20; /* Driver local functions. */ /*===========================================================================*/ +/** + * @brief Enables a CFIFO. + * + * @param[in] fifo the FIFO identifier + * @param[in] cfcr CFCR register value + * @param[in] idcr IDCR register value + * + * @notapi + */ +static void cfifo_enable(adcfifo_t fifo, uint16_t cfcr, uint16_t idcr) { + + EQADC.CFCR[fifo].R = cfcr; + EQADC.IDCR[fifo].R = idcr; +} + + +/** + * @brief Disables a CFIFO and the associated resources. + * + * @param[in] fifo the FIFO identifier + * + * @notapi + */ +static void cfifo_disable(adcfifo_t fifo) { + + /* Disables the CFIFO.*/ + EQADC.CFCR[fifo].R = EQADC_CFCR_MODE_DISABLED; + + /* Disables Interrupts and DMAs of the CFIFO.*/ + EQADC.IDCR[fifo].R = 0; + + /* Waits for the CFIFO to become idle.*/ + while ((EQADC.CFSR.R & (0xC0000000 >> (fifo * 2))) != 0) + ; + + /* Invalidates the CFIFO.*/ + EQADC.CFCR[fifo].R = EQADC_CFCR_CFINV | EQADC_CFCR_MODE_DISABLED; + + /* Clears all Interrupts and eDMA flags for the CFIFO.*/ + EQADC.FISR[fifo].R = EQADC_FISR_CLEAR_MASK; + + /* Clears the Tx Count Registers for the CFIFO.*/ + EQADC.CFTCR[fifo].R = 0; +} + +/** + * @brief Pushes a command into the CFIFO0. + * + * @param[in] cmd the command + * + * @notapi + */ +static void cfifo0_push_command(adccommand_t cmd) { + + while (EQADC.FISR[0].B.CFCTR >= 4) + ; + EQADC.CFPR[0].R = cmd; +} + +/** + * @brief Waits until the RFIFO0 contains the specified number of entries. + * + * @param[in] n number of entries + * + * @notapi + */ +static void cfifo0_wait_rfifo(uint32_t n) { + + while (EQADC.FISR[0].B.RFCTR < n) + ; + EQADC.FISR[0].R = EQADC_FISR_CLEAR_MASK; +} + /*===========================================================================*/ /* Driver interrupt handlers. */ /*===========================================================================*/ @@ -70,15 +136,10 @@ ADCDriver ADCD20; */ void adc_lld_init(void) { -#if SPC5_ADC_USE_EQADC1_Q0 +#if SPC5_ADC_USE_EQADC_Q0 /* Driver initialization.*/ - adcObjectInit(&ADCD10); -#endif /* SPC5_ADC_USE_EQADC1_Q0 */ - -#if SPC5_ADC_USE_EQADC2_Q0 - /* Driver initialization.*/ - adcObjectInit(&ADCD20); -#endif /* SPC5_ADC_USE_EQADC2_Q0 */ + adcObjectInit(&ADCD1); +#endif /* SPC5_ADC_USE_EQADC_Q0 */ } /** @@ -92,11 +153,11 @@ void adc_lld_start(ADCDriver *adcp) { if (adcp->state == ADC_STOP) { /* Enables the peripheral.*/ -#if SPC5_ADC_USE_EQADC1_Q0 - if (&ADCD10 == adcp) { +#if SPC5_ADC_USE_EQADC_Q0 + if (&ADCD1 == adcp) { } -#endif /* SPC5_ADC_USE_EQADC1_Q0 */ +#endif /* SPC5_ADC_USE_EQADC_Q0 */ } /* Configures the peripheral.*/ @@ -115,11 +176,11 @@ void adc_lld_stop(ADCDriver *adcp) { /* Resets the peripheral.*/ /* Disables the peripheral.*/ -#if SPC5_ADC_USE_EQADC1_Q0 +#if SPC5_ADC_USE_EQADC_Q0 if (&ADCD1 == adcp) { } -#endif /* SPC5_ADC_USE_EQADC1_Q0 */ +#endif /* SPC5_ADC_USE_EQADC_Q0 */ } } diff --git a/os/hal/platforms/SPC5xx/EQADC_v1/adc_lld.h b/os/hal/platforms/SPC5xx/EQADC_v1/adc_lld.h index e04da1b69..5e63489f2 100644 --- a/os/hal/platforms/SPC5xx/EQADC_v1/adc_lld.h +++ b/os/hal/platforms/SPC5xx/EQADC_v1/adc_lld.h @@ -30,28 +30,110 @@ /*===========================================================================*/ /** - * @name CFIFO commands + * @name Internal registers indexes * @{ */ -#define ADC_CFIFO_CHANNEL_MASK (0xFFU << 8)/**< @brief Channel number mask.*/ -#define ADC_CFIFO_CHANNEL(n) ((n) << 8) /**< @brief Channel number. */ -#define ADC_CFIFO_FMT_RJU (0U << 16) /**< @brief Unsigned samples. */ -#define ADC_CFIFO_FMT_RJS (1U << 16) /**< @brief Signed samples. */ -#define ADC_CFIFO_TSR (1U << 17) /**< @brief Time stamp request. */ -#define ADC_CFIFO_LST_MASK (3U << 18) /**< @brief Sample time. */ -#define ADC_CFIFO_LST_2 (0U << 18) /**< @brief 2 clock cycles. */ -#define ADC_CFIFO_LST_8 (1U << 18) /**< @brief 8 clock cycles. */ -#define ADC_CFIFO_LST_64 (2U << 18) /**< @brief 64 clock cycles. */ -#define ADC_CFIFO_LST_128 (3U << 18) /**< @brief 128 clock cycles. */ -#define ADC_CFIFO_MSG_MASK (15U << 20) /**< @brief Message mask. */ -#define ADC_CFIFO_MSG_RFIFO(n) ((n) << 20) /**< @brief Result in RFIFO0..5.*/ -#define ADC_CFIFO_MSG_NULL (6U << 20) /**< @brief Null message. */ -#define ADC_CFIFO_CAL (1U << 24) /**< @brief Calibrated result. */ -#define ADC_CFIFO_BN_MASK (1U << 25) /**< @brief Buffer number mask. */ -#define ADC_CFIFO_BN(n) ((n) << 25) /**< @brief Buffer number. */ -#define ADC_CFIFO_REP (1U << 29) /**< @brief Repeat loop flag. */ -#define ADC_CFIFO_PAUSE (1U << 30) /**< @brief Pause flag. */ -#define ADC_CFIFO_EOQ (1U << 31) /**< @brief End of queue flag. */ +#define ADC_REG_CR 0x1 +#define ADC_REG_TSCR 0x2 +#define ADC_REG_TBCR 0x3 +#define ADC_REG_GCCR 0x4 +#define ADC_REG_OCCR 0x5 +#define ADC_REG_AC1GCCR 0x31 +#define ADC_REG_AC1OCCR 0x32 +#define ADC_REG_AC2GCCR 0x35 +#define ADC_REG_AC2OCCR 0x36 +#define ADC_REG_AC1CR 0x30 +#define ADC_REG_AC2CR 0x34 +#define ADC_REG_AC3CR 0x38 +#define ADC_REG_AC4CR 0x3C +#define ADC_REG_AC5CR 0x40 +#define ADC_REG_AC6CR 0x44 +#define ADC_REG_AC7CR 0x48 +#define ADC_REG_AC8CR 0x4C +#define ADC_REG_PUDCR(n) (0x70 + (n)) +#define ADC_REG_PUDCR0 0x70UL +#define ADC_REG_PUDCR1 0x71UL +#define ADC_REG_PUDCR2 0x72UL +#define ADC_REG_PUDCR3 0x73UL +#define ADC_REG_PUDCR4 0x74UL +#define ADC_REG_PUDCR5 0x75UL +#define ADC_REG_PUDCR6 0x76UL +#define ADC_REG_PUDCR7 0x77UL +/** @} */ + +/** + * @name EQADC CFCR registers definitions + * @{ + */ +#define EQADC_CFCR_CFEEE0 (1U << 12) +#define EQADC_CFCR_STRME0 (1U << 11) +#define EQADC_CFCR_SSE (1U << 10) +#define EQADC_CFCR_CFINV (1U << 9) +#define EQADC_CFCR_MODE_MASK (15U << 4) +#define EQADC_CFCR_MODE(n) ((n) << 4) +#define EQADC_CFCR_MODE_DISABLED EQADC_CFCR_MODE(0) +#define EQADC_CFCR_MODE_SWSS EQADC_CFCR_MODE(1) +#define EQADC_CFCR_MODE_HWSS_LL EQADC_CFCR_MODE(2) +#define EQADC_CFCR_MODE_HWSS_HL EQADC_CFCR_MODE(3) +#define EQADC_CFCR_MODE_HWSS_FE EQADC_CFCR_MODE(4) +#define EQADC_CFCR_MODE_HWSS_RE EQADC_CFCR_MODE(5) +#define EQADC_CFCR_MODE_HWSS_BE EQADC_CFCR_MODE(6) +#define EQADC_CFCR_MODE_MODE_SWCS EQADC_CFCR_MODE(9) +#define EQADC_CFCR_MODE_HWCS_LL EQADC_CFCR_MODE(10) +#define EQADC_CFCR_MODE_HWCS_HL EQADC_CFCR_MODE(11) +#define EQADC_CFCR_MODE_HWCS_FE EQADC_CFCR_MODE(12) +#define EQADC_CFCR_MODE_HWCS_RE EQADC_CFCR_MODE(13) +#define EQADC_CFCR_MODE_HWCS_BE EQADC_CFCR_MODE(14) +#define EQADC_CFCR_AMODE0_MASK (15U << 0) +#define EQADC_CFCR_AMODE0(n) ((n) << 0) +/** @} */ + +/** + * @name EQADC FISR registers definitions + * @{ + */ +#define EQADC_FISR_POPNXTPTR_MASK (15U << 0) +#define EQADC_FISR_RFCTR_MASK (15U << 4) +#define EQADC_FISR_TNXTPTR_MASK (15U << 8) +#define EQADC_FISR_CFCTR_MASK (15U << 12) +#define EQADC_FISR_RFDF (1U << 17) +#define EQADC_FISR_RFOF (1U << 19) +#define EQADC_FISR_CFFF (1U << 25) +#define EQADC_FISR_SSS (1U << 26) +#define EQADC_FISR_CFUF (1U << 27) +#define EQADC_FISR_EOQF (1U << 28) +#define EQADC_FISR_PF (1U << 29) +#define EQADC_FISR_TORF (1U << 30) +#define EQADC_FISR_NCF (1U << 31) +#define EQADC_FISR_CLEAR_MASK (EQADC_FISR_NCF | EQADC_FISR_TORF | \ + EQADC_FISR_PF | EQADC_FISR_EOQF | \ + EQADC_FISR_CFUF | EQADC_FISR_RFOF | \ + EQADC_FISR_RFDF) +/** @} */ + +/** + * @name EQADC Conversion commands + * @{ + */ +#define EQADC_CONV_CHANNEL_MASK (0xFFU << 8)/**< @brief Channel number mask.*/ +#define EQADC_CONV_CHANNEL(n) ((n) << 8) /**< @brief Channel number. */ +#define EQADC_CONV_FMT_RJU (0U << 16) /**< @brief Unsigned samples. */ +#define EQADC_CONV_FMT_RJS (1U << 16) /**< @brief Signed samples. */ +#define EQADC_CONV_TSR (1U << 17) /**< @brief Time stamp request. */ +#define EQADC_CONV_LST_MASK (3U << 18) /**< @brief Sample time. */ +#define EQADC_CONV_LST_2 (0U << 18) /**< @brief 2 clock cycles. */ +#define EQADC_CONV_LST_8 (1U << 18) /**< @brief 8 clock cycles. */ +#define EQADC_CONV_LST_64 (2U << 18) /**< @brief 64 clock cycles. */ +#define EQADC_CONV_LST_128 (3U << 18) /**< @brief 128 clock cycles. */ +#define EQADC_CONV_MSG_MASK (15U << 20) /**< @brief Message mask. */ +#define EQADC_CONV_MSG_RFIFO(n) ((n) << 20) /**< @brief Result in RFIFO0..5.*/ +#define EQADC_CONV_MSG_NULL (6U << 20) /**< @brief Null message. */ +#define EQADC_CONV_CAL (1U << 24) /**< @brief Calibrated result. */ +#define EQADC_CONV_BN_MASK (1U << 25) /**< @brief Buffer number mask. */ +#define EQADC_CONV_BN(n) ((n) << 25) /**< @brief Buffer number. */ +#define EQADC_CONV_REP (1U << 29) /**< @brief Repeat loop flag. */ +#define EQADC_CONV_PAUSE (1U << 30) /**< @brief Pause flag. */ +#define EQADC_CONV_EOQ (1U << 31) /**< @brief End of queue flag. */ /** @} */ /*===========================================================================*/ @@ -67,17 +149,8 @@ * @details If set to @p TRUE the support for EQADC1 queue 0 is included. * @note The default is @p FALSE. */ -#if !defined(SPC5_ADC_USE_EQADC1_Q0) || defined(__DOXYGEN__) -#define SPC5_ADC_USE_EQADC1_Q0 FALSE -#endif - -/** - * @brief ADCD20 driver enable switch. - * @details If set to @p TRUE the support for EQADC2 queue 0 is included. - * @note The default is @p FALSE. - */ -#if !defined(SPC5_ADC_USE_EQADC2_Q0) || defined(__DOXYGEN__) -#define SPC5_ADC_USE_EQADC2_Q0 FALSE +#if !defined(SPC5_ADC_USE_EQADC_Q0) || defined(__DOXYGEN__) +#define SPC5_ADC_USE_EQADC_Q0 FALSE #endif /** @} */ @@ -85,10 +158,30 @@ /* Derived constants and error checks. */ /*===========================================================================*/ +#if SPC5_ADC_USE_EQADC_Q0 && !SPC5_HAS_EQADC +#error "EQADC1 not present in the selected device" +#endif + +#if !SPC5_ADC_USE_EQADC_Q0 +#error "ADC driver activated but no EQADC peripheral assigned" +#endif + /*===========================================================================*/ /* Driver data structures and types. */ /*===========================================================================*/ +/*! + * @brief FIFO unit numeric IDs. + */ +typedef enum { + ADC_FIFO_0 = 0, + ADC_FIFO_1 = 1, + ADC_FIFO_2 = 2, + ADC_FIFO_3 = 3, + ADC_FIFO_4 = 4, + ADC_FIFO_5 = 5 +} adcfifo_t; + /** * @brief ADC command data type. */ @@ -241,12 +334,8 @@ struct ADCDriver { /* External declarations. */ /*===========================================================================*/ -#if SPC5_ADC_USE_EQADC1_Q0 && !defined(__DOXYGEN__) -extern ADCDriver ADCD10; -#endif - -#if SPC5_ADC_USE_EQADC2_Q0 && !defined(__DOXYGEN__) -extern ADCDriver ADCD20; +#if SPC5_ADC_USE_EQADC_Q0 && !defined(__DOXYGEN__) +extern ADCDriver ADCD1; #endif #ifdef __cplusplus -- cgit v1.2.3