/* * Copyright (c) 2006-2022, RT-Thread Development Team * * SPDX-License-Identifier: Apache-2.0 * * Change Logs: * Date Author Notes * 2011-02-22 onelife Initial creation for EFM32 * 2011-07-27 onelife Modify according to ADC driver changes */ /***************************************************************************//** * @addtogroup efm32 * @{ ******************************************************************************/ /* Includes ------------------------------------------------------------------*/ #include "board.h" #include "drv_adc.h" #if defined(RT_USING_MISC) /* Private typedef -----------------------------------------------------------*/ /* Private define ------------------------------------------------------------*/ /* Private macro -------------------------------------------------------------*/ #ifdef RT_MISC_DEBUG #define misc_debug(format,args...) rt_kprintf(format, ##args) #else #define misc_debug(format,args...) #endif /* Private constants ---------------------------------------------------------*/ static rt_device_t adc0; static struct efm32_adc_control_t control = \ {ADC_MODE_SINGLE, {}, {0, (rt_uint8_t)EFM32_NO_DMA}}; /* Private variables ---------------------------------------------------------*/ /* Private function prototypes -----------------------------------------------*/ rt_int32_t efm32_misc_getCelsius(rt_uint32_t adcSample); /* Private functions ---------------------------------------------------------*/ /***************************************************************************//** * @brief * Get current temperature value in degree celsius * * @details * * @note * * @return * Temperature value (signed integer) in degree celsius times 100 * ******************************************************************************/ rt_int32_t rt_hw_get_temp(void) { ADC_InitSingle_TypeDef singleInit = ADC_INITSINGLE_DEFAULT; struct efm32_adc_result_t result; rt_uint32_t temp; /* Set input to temperature sensor. Acquisition time must be 256 cycles. Reference must be 1.25V */ singleInit.acqTime = adcAcqTime32; singleInit.reference = adcRef1V25; singleInit.input = adcSingleInpTemp; control.single.init = &singleInit; adc0->control(adc0, RT_DEVICE_CTRL_ADC_MODE, &control); result.mode = control.mode; result.buffer = (void *)&temp; adc0->control(adc0, RT_DEVICE_CTRL_RESUME, &result); adc0->control(adc0, RT_DEVICE_CTRL_ADC_RESULT, &result); return efm32_misc_getCelsius(temp); } /***************************************************************************//** * @brief * Get current VDD value in volt * * @details * * @note * * @return * VDD value (unsigned integer) in volt times 100 * ******************************************************************************/ rt_uint32_t rt_hw_get_vdd(void) { ADC_InitSingle_TypeDef singleInit = ADC_INITSINGLE_DEFAULT; struct efm32_adc_result_t result; rt_uint32_t vdd; /* Set input to temperature sensor. Reference must be 1.25V */ singleInit.acqTime = adcAcqTime32; singleInit.reference = adcRef1V25; singleInit.input = adcSingleInpVDDDiv3; control.single.init = &singleInit; adc0->control(adc0, RT_DEVICE_CTRL_ADC_MODE, &control); result.mode = control.mode; result.buffer = (void *)&vdd; adc0->control(adc0, RT_DEVICE_CTRL_RESUME, &result); adc0->control(adc0, RT_DEVICE_CTRL_ADC_RESULT, &result); return (vdd * 125 * 3) / 4096; } /***************************************************************************//** * @brief * Initialize all the miscellaneous drivers * * @details * * @note * * @return * Error code ******************************************************************************/ rt_err_t rt_hw_misc_init(void) { do { /* Find ADC device */ adc0 = rt_device_find(RT_ADC0_NAME); if (adc0 == RT_NULL) { misc_debug("Misc err: Can't find device: %s!\n", RT_ADC0_NAME); break; } misc_debug("Misc: Find device %s\n", RT_ADC0_NAME); return RT_EOK; } while (0); misc_debug("Misc err: Init failed!\n"); return -RT_ERROR; } /***************************************************************************//** * @brief * Convert ADC result to degree celsius. * * @details * * @note * See section 2.3.4 in the reference manual for details on this calculatoin * * @param adcResult * Raw value from ADC to be converted to celsius * * @return * The temperature value (signed integer) in degrees celsius times 100 * ******************************************************************************/ rt_int32_t efm32_misc_getCelsius(rt_uint32_t adcResult) { /* Factory calibration temperature from device information page. */ rt_int32_t cal_temp = ((DEVINFO->CAL & _DEVINFO_CAL_TEMP_MASK) \ >> _DEVINFO_CAL_TEMP_SHIFT) * 100; /* Factory calibration value from device information page. */ rt_int32_t cal_value = ((DEVINFO->ADC0CAL2 & _DEVINFO_ADC0CAL2_TEMP1V25_MASK) \ >> _DEVINFO_ADC0CAL2_TEMP1V25_SHIFT) * 10000; /* Temperature gradient (from datasheet) in (ADC unit / degree celsius * 100) */ rt_int32_t t_grad = -385; return (cal_temp - (cal_value - (rt_int32_t)adcResult * 10000) / t_grad); } /******************************************************************************* * Export to FINSH ******************************************************************************/ #ifdef RT_USING_FINSH #include <finsh.h> void list_temp(void) { rt_int32_t temp = rt_hw_get_temp(); rt_kprintf("Temperature is %2d.%02d C\n", temp / 100, temp % 100); } FINSH_FUNCTION_EXPORT(list_temp, list current temperature value.) void list_vdd(void) { rt_uint32_t vdd = rt_hw_get_vdd(); rt_kprintf("VDD is %1d.%02d V\n", vdd / 100, vdd % 100); } FINSH_FUNCTION_EXPORT(list_vdd, list current VDD value.) #endif /* RT_USING_FINSH */ #endif /* defined(RT_USING_MISC) */ /***************************************************************************//** * @} ******************************************************************************/