1
0
mirror of https://github.com/arduino/Arduino.git synced 2025-01-22 11:52:14 +01:00

400 lines
9.9 KiB
C
Raw Normal View History

2012-06-12 18:56:47 +02:00
/* ----------------------------------------------------------------------------
* SAM Software Package License
* ----------------------------------------------------------------------------
* Copyright (c) 2011-2012, Atmel Corporation
2011-10-25 15:52:09 +02:00
*
2012-06-12 18:56:47 +02:00
* All rights reserved.
2011-10-25 15:52:09 +02:00
*
* Redistribution and use in source and binary forms, with or without
2012-06-12 18:56:47 +02:00
* modification, are permitted provided that the following condition is met:
2011-10-25 15:52:09 +02:00
*
2012-06-12 18:56:47 +02:00
* - Redistributions of source code must retain the above copyright notice,
* this list of conditions and the disclaimer below.
2011-10-25 15:52:09 +02:00
*
2012-06-12 18:56:47 +02:00
* Atmel's name may not be used to endorse or promote products derived from
* this software without specific prior written permission.
*
2012-06-12 18:56:47 +02:00
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
2012-06-12 18:56:47 +02:00
* DISCLAIMED. IN NO EVENT SHALL ATMEL 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.
* ----------------------------------------------------------------------------
*/
2011-10-25 15:52:09 +02:00
#include "../chip.h"
2011-10-25 15:52:09 +02:00
/// @cond 0
/**INDENT-OFF**/
#ifdef __cplusplus
extern "C" {
#endif
2011-10-25 15:52:09 +02:00
/**INDENT-ON**/
/// @endcond
/**
* \defgroup sam_drivers_adc_group Analog-to-digital Converter (ADC)
*
2012-06-12 18:56:47 +02:00
* Driver for the Analog-to-digital Converter. This driver provides access to the main
* features of the ADC controller.
*
* @{
*/
2012-06-12 18:56:47 +02:00
#if SAM3U_SERIES
2011-10-25 15:52:09 +02:00
/**
* \brief Initialize the given ADC with the specified ADC clock and startup time.
2011-10-25 15:52:09 +02:00
*
* \param p_adc Pointer to an ADC instance.
* \param ul_mck Main clock of the device (in Hz).
* \param ul_adc_clock Analog-to-Digital conversion clock (in Hz).
2012-06-12 18:56:47 +02:00
* \param ul_startuptime ADC startup time value (value in us).
* Please refer to the product datasheet for more details.
2012-06-12 18:56:47 +02:00
* \param ul_offmode_startuptime ADC off mode startup time value (value in us).
* Please refer to the product datasheet for more details.
2011-10-25 15:52:09 +02:00
*
* \return 0 on success.
2011-10-25 15:52:09 +02:00
*/
uint32_t adc12b_init(Adc12b *p_adc, const uint32_t ul_mck, const uint32_t ul_adc_clock,
const uint32_t ul_startuptime, const uint32_t ul_offmode_startuptime)
2011-10-25 15:52:09 +02:00
{
uint32_t ul_prescal, ul_startup, ul_offmode;
p_adc->ADC12B_CR = ADC12B_CR_SWRST;
/* Reset Mode Register */
p_adc->ADC12B_MR = 0;
/* Reset PDC transfer */
p_adc->ADC12B_PTCR = (ADC12B_PTCR_RXTDIS | ADC12B_PTCR_TXTDIS);
p_adc->ADC12B_RCR = 0;
p_adc->ADC12B_RNCR = 0;
ul_prescal = ul_mck / (2 * ul_adc_clock) - 1;
ul_startup = ((ul_adc_clock / 1000000) * ul_startuptime / 8) - 1;
ul_offmode = ((ul_adc_clock / 1000000) * ul_offmode_startuptime / 8) -
1;
p_adc->ADC12B_MR |=
ADC12B_MR_PRESCAL(ul_prescal) | ((ul_startup <<
ADC12B_MR_STARTUP_Pos) &
ADC12B_MR_STARTUP_Msk);
p_adc->ADC12B_EMR |= (ul_offmode << 16) & (0xffu << 16);
return 0;
2011-10-25 15:52:09 +02:00
}
2011-10-25 15:52:09 +02:00
/**
* \brief Configure conversion resolution.
2011-10-25 15:52:09 +02:00
*
* \param p_adc Pointer to an ADC instance.
* \param resolution ADC resolution.
2011-10-25 15:52:09 +02:00
*/
void adc12b_set_resolution(Adc12b *p_adc, const enum adc_resolution_t resolution)
2011-10-25 15:52:09 +02:00
{
p_adc->ADC12B_MR |= (resolution << 4) & ADC12B_MR_LOWRES;
2011-10-25 15:52:09 +02:00
}
2011-10-25 15:52:09 +02:00
/**
* \brief Configure conversion trigger and free run mode.
2011-10-25 15:52:09 +02:00
*
* \param p_adc Pointer to an ADC instance.
* \param trigger Conversion trigger.
2011-10-25 15:52:09 +02:00
*/
void adc12b_configure_trigger(Adc12b *p_adc, const enum adc_trigger_t trigger)
2011-10-25 15:52:09 +02:00
{
p_adc->ADC12B_MR |= trigger;
2011-10-25 15:52:09 +02:00
}
2011-10-25 15:52:09 +02:00
/**
* \brief Configure ADC power saving mode.
*
* \param p_adc Pointer to an ADC instance.
2012-06-12 18:56:47 +02:00
* \param uc_sleep ADC_MR_SLEEP_NORMAL keeps the ADC Core and reference
* voltage circuitry ON between conversions.
* ADC_MR_SLEEP_SLEEP keeps the ADC Core and reference voltage circuitry
* OFF between conversions.
* \param uc_offmode 0 Standby Mode (if Sleep Bit = 1), 1 Off Mode.
2011-10-25 15:52:09 +02:00
*/
void adc12b_configure_power_save(Adc12b *p_adc, const uint8_t uc_sleep,
uint8_t uc_offmode)
2011-10-25 15:52:09 +02:00
{
p_adc->ADC12B_MR |= ((uc_sleep << 5) & ADC12B_MR_SLEEP);
p_adc->ADC12B_EMR |= uc_offmode;
2011-10-25 15:52:09 +02:00
}
2011-10-25 15:52:09 +02:00
/**
* \brief Configure ADC timing.
2011-10-25 15:52:09 +02:00
*
* \param p_adc Pointer to an ADC instance.
* \param ul_sh ADC sample and hold time = uc_sh / ADC clock.
2011-10-25 15:52:09 +02:00
*/
void adc12b_configure_timing(Adc12b *p_adc, const uint32_t ul_sh)
2011-10-25 15:52:09 +02:00
{
p_adc->ADC12B_MR |= ADC12B_MR_SHTIM(ul_sh);
2011-10-25 15:52:09 +02:00
}
2011-10-25 15:52:09 +02:00
/**
* \brief Start ADC conversion.
2011-10-25 15:52:09 +02:00
*
2012-06-12 18:56:47 +02:00
* \note If one of the hardware event is selected as ADC trigger,
* this function can NOT start ADC conversion.
2011-10-25 15:52:09 +02:00
*
* \param p_adc Pointer to an ADC instance.
2011-10-25 15:52:09 +02:00
*/
void adc12b_start(Adc12b *p_adc)
2011-10-25 15:52:09 +02:00
{
p_adc->ADC12B_CR = ADC12B_CR_START;
2011-10-25 15:52:09 +02:00
}
2011-10-25 15:52:09 +02:00
/**
* \brief Stop ADC conversion.
* \param p_adc Pointer to an ADC instance.
2011-10-25 15:52:09 +02:00
*/
void adc12b_stop(Adc12b *p_adc)
2011-10-25 15:52:09 +02:00
{
p_adc->ADC12B_CR = ADC12B_CR_SWRST;
2011-10-25 15:52:09 +02:00
}
2011-10-25 15:52:09 +02:00
/**
* \brief Enable the specified ADC channel.
2011-10-25 15:52:09 +02:00
*
* \param p_adc Pointer to an ADC instance.
* \param adc_ch ADC channel number.
2011-10-25 15:52:09 +02:00
*/
void adc12b_enable_channel(Adc12b *p_adc, const enum adc_channel_num_t adc_ch)
2011-10-25 15:52:09 +02:00
{
p_adc->ADC12B_CHER = 1 << adc_ch;
2011-10-25 15:52:09 +02:00
}
/**
* \brief Enable all ADC channels.
2011-10-25 15:52:09 +02:00
*
* \param p_adc Pointer to an ADC instance.
2011-10-25 15:52:09 +02:00
*/
void adc12b_enable_all_channel(Adc12b *p_adc)
2011-10-25 15:52:09 +02:00
{
p_adc->ADC12B_CHER = 0xFF;
2011-10-25 15:52:09 +02:00
}
/**
* \brief Disable the specified ADC channel.
2011-10-25 15:52:09 +02:00
*
* \param p_adc Pointer to an ADC instance.
* \param adc_ch ADC channel number.
*/
void adc12b_disable_channel(Adc12b *p_adc, const enum adc_channel_num_t adc_ch)
{
p_adc->ADC12B_CHDR = 1 << adc_ch;
}
/**
* \brief Disable all ADC channel.
2011-10-25 15:52:09 +02:00
*
* \param p_adc Pointer to an ADC instance.
2011-10-25 15:52:09 +02:00
*/
void adc12b_disable_all_channel(Adc12b *p_adc)
2011-10-25 15:52:09 +02:00
{
p_adc->ADC12B_CHDR = 0xFF;
2011-10-25 15:52:09 +02:00
}
2011-11-21 12:16:54 +01:00
/**
* \brief Read the ADC channel status.
2011-11-21 12:16:54 +01:00
*
* \param p_adc Pointer to an ADC instance.
* \param adc_ch ADC channel number.
2011-11-21 12:16:54 +01:00
*
* \retval 1 if channel is enabled.
* \retval 0 if channel is disabled.
2011-11-21 12:16:54 +01:00
*/
uint32_t adc12b_get_channel_status(const Adc12b *p_adc, const enum adc_channel_num_t adc_ch)
2011-11-21 12:16:54 +01:00
{
return p_adc->ADC12B_CHSR & (1 << adc_ch);
2011-11-21 12:16:54 +01:00
}
2011-10-25 15:52:09 +02:00
/**
* \brief Read the ADC result data of the specified channel.
2011-10-25 15:52:09 +02:00
*
* \param p_adc Pointer to an ADC instance.
* \param adc_ch ADC channel number.
2011-10-25 15:52:09 +02:00
*
* \return ADC value of the specified channel.
2011-10-25 15:52:09 +02:00
*/
uint32_t adc12b_get_channel_value(const Adc12b *p_adc,const enum adc_channel_num_t adc_ch)
2011-10-25 15:52:09 +02:00
{
uint32_t dwData = 0;
2011-10-25 15:52:09 +02:00
if (15 >= adc_ch) {
dwData = *(p_adc->ADC12B_CDR + adc_ch);
}
2011-10-25 15:52:09 +02:00
return dwData;
2011-10-25 15:52:09 +02:00
}
/**
* \brief Read the last ADC result data.
2011-10-25 15:52:09 +02:00
*
* \param p_adc Pointer to an ADC instance.
2011-10-25 15:52:09 +02:00
*
* \return ADC latest value.
2011-10-25 15:52:09 +02:00
*/
uint32_t adc12b_get_latest_value(const Adc12b *p_adc)
2011-10-25 15:52:09 +02:00
{
return p_adc->ADC12B_LCDR;
2011-10-25 15:52:09 +02:00
}
2011-10-25 15:52:09 +02:00
/**
* \brief Enable differential input for all channels.
2011-10-25 15:52:09 +02:00
*
* \param p_adc Pointer to an ADC instance.
2011-10-25 15:52:09 +02:00
*/
void adc12b_enable_differential_input(Adc12b *p_adc)
2011-10-25 15:52:09 +02:00
{
p_adc->ADC12B_ACR |= (0x01u << 16);
2011-10-25 15:52:09 +02:00
}
/**
* \brief Disable differential input for the specified channel.
2011-10-25 15:52:09 +02:00
*
* \param p_adc Pointer to an ADC instance.
2011-10-25 15:52:09 +02:00
*/
void adc12b_disable_differential_input(Adc12b *p_adc)
2011-10-25 15:52:09 +02:00
{
p_adc->ADC12B_ACR &= (0x01u << 16);
2011-10-25 15:52:09 +02:00
}
/**
* \brief Enable analog signal offset for the specified channel.
2011-10-25 15:52:09 +02:00
*
* \param p_adc Pointer to an ADC instance.
2011-10-25 15:52:09 +02:00
*/
void adc12b_enable_input_offset(Adc12b *p_adc)
2011-10-25 15:52:09 +02:00
{
p_adc->ADC12B_ACR |= (0x01u << 17);
2011-10-25 15:52:09 +02:00
}
/**
* \brief Disable analog signal offset for the specified channel.
2011-10-25 15:52:09 +02:00
*
* \param p_adc Pointer to an ADC instance.
2011-10-25 15:52:09 +02:00
*/
void adc12b_disable_input_offset(Adc12b *p_adc)
2011-10-25 15:52:09 +02:00
{
p_adc->ADC12B_ACR &= (0x01u << 17);
2011-10-25 15:52:09 +02:00
}
2011-10-25 15:52:09 +02:00
/**
* \brief Configure input gain for the specified channel.
2011-10-25 15:52:09 +02:00
*
* \param p_adc Pointer to an ADC instance.
* \param gain Gain value for the input.
2011-10-25 15:52:09 +02:00
*/
void adc12b_set_input_gain(Adc12b *p_adc, const enum adc_gainvalue_t gain)
2011-10-25 15:52:09 +02:00
{
p_adc->ADC12B_ACR |= (0x03u & gain);
2011-10-25 15:52:09 +02:00
}
/**
* \brief Return the actual ADC clock.
*
* \param p_adc Pointer to an ADC instance.
* \param ul_mck Main clock of the device (in Hz).
*
* \retval 0 The actual ADC clock (in Hz).
*/
uint32_t adc12b_get_actual_adc_clock(const Adc12b *p_adc, const uint32_t ul_mck)
2011-10-25 15:52:09 +02:00
{
uint32_t ul_adcfreq;
uint32_t ul_prescal;
/* ADCClock = MCK / ( (PRESCAL+1) * 2 ) */
ul_prescal = ((p_adc->ADC12B_MR & ADC12B_MR_PRESCAL_Msk) >>
ADC12B_MR_PRESCAL_Pos);
ul_adcfreq = ul_mck / ((ul_prescal + 1) * 2);
return ul_adcfreq;
2011-10-25 15:52:09 +02:00
}
2011-10-25 15:52:09 +02:00
/**
* \brief Enable ADC interrupts.
2011-10-25 15:52:09 +02:00
*
* \param p_adc Pointer to an ADC instance.
* \param ul_source Interrupts to be enabled.
2011-10-25 15:52:09 +02:00
*/
void adc12b_enable_interrupt(Adc12b *p_adc, const uint32_t ul_source)
2011-10-25 15:52:09 +02:00
{
p_adc->ADC12B_IER = ul_source;
2011-10-25 15:52:09 +02:00
}
/**
* \brief Disable ADC interrupts.
2011-10-25 15:52:09 +02:00
*
* \param p_adc Pointer to an ADC instance.
* \param ul_source Interrupts to be disabled.
2011-10-25 15:52:09 +02:00
*/
void adc12b_disable_interrupt(Adc12b *p_adc, const uint32_t ul_source)
2011-10-25 15:52:09 +02:00
{
p_adc->ADC12B_IDR = ul_source;
2011-10-25 15:52:09 +02:00
}
2011-10-25 15:52:09 +02:00
/** \brief Read ADC interrupt mask.
*
* \param p_adc Pointer to an ADC instance.
2011-10-25 15:52:09 +02:00
*
* \return The interrupt mask value.
2011-10-25 15:52:09 +02:00
*/
uint32_t adc12b_get_interrupt_mask(const Adc12b *p_adc)
2011-10-25 15:52:09 +02:00
{
return p_adc->ADC12B_IMR;
}
2011-10-25 15:52:09 +02:00
/**
* \brief Read ADC interrupt status.
2011-10-25 15:52:09 +02:00
*
* \param p_adc Pointer to an ADC instance.
2011-10-25 15:52:09 +02:00
*
* \retval ADC interrupt status.
*/
uint32_t adc12b_get_status(const Adc12b *p_adc)
2011-10-25 15:52:09 +02:00
{
return p_adc->ADC12B_SR;
2011-10-25 15:52:09 +02:00
}
2011-10-25 15:52:09 +02:00
/**
* \brief Adapt performance versus power consumption.
2011-10-25 15:52:09 +02:00
*
2012-06-12 18:56:47 +02:00
* \note Please refer to ADC Characteristics in the product datasheet
* for more details.
2011-10-25 15:52:09 +02:00
*
* \param p_adc Pointer to an ADC instance.
* \param uc_ibctl ADC Bias current control.
2011-10-25 15:52:09 +02:00
*/
void adc12b_set_bias_current(Adc12b *p_adc, const uint8_t uc_ibctl)
2011-10-25 15:52:09 +02:00
{
p_adc->ADC12B_ACR |= ADC12B_ACR_IBCTL(uc_ibctl);
2011-10-25 15:52:09 +02:00
}
2011-10-25 15:52:09 +02:00
/**
* \brief Get PDC registers base address.
2011-10-25 15:52:09 +02:00
*
* \param p_adc Pointer to an ADC instance.
2011-10-25 15:52:09 +02:00
*
* \return ADC PDC register base address.
2011-10-25 15:52:09 +02:00
*/
Pdc *adc12b_get_pdc_base(const Adc12b *p_adc)
2011-10-25 15:52:09 +02:00
{
return PDC_ADC12B;
2011-10-25 15:52:09 +02:00
}
#endif // SAM3U_SERIES
//@}
2011-10-25 15:52:09 +02:00
/// @cond 0
/**INDENT-OFF**/
#ifdef __cplusplus
}
#endif
2011-10-25 15:52:09 +02:00
/**INDENT-ON**/
/// @endcond