155 lines
3.6 KiB
C

/**************************************************************************//**
* @copyright (C) 2020 Nuvoton Technology Corp. All rights reserved.
*
* SPDX-License-Identifier: Apache-2.0
*
* Change Logs:
* Date Author Notes
* 2020-9-16 Philo First version
*
******************************************************************************/
#include <rtconfig.h>
#include <rtdevice.h>
#include "NuMicro.h"
#ifdef BSP_USING_ADC
/* Private define ---------------------------------------------------------------*/
/* Private Typedef --------------------------------------------------------------*/
struct nu_adc
{
struct rt_adc_device dev;
char *name;
ADC_T *adc_base;
int adc_reg_tab;
int adc_max_ch_num;
};
typedef struct nu_adc *nu_adc_t;
/* Private functions ------------------------------------------------------------*/
static rt_err_t nu_adc_enabled(struct rt_adc_device *device, rt_uint32_t channel, rt_bool_t enabled);
static rt_err_t nu_get_adc_value(struct rt_adc_device *device, rt_uint32_t channel, rt_uint32_t *value);
/* Public functions ------------------------------------------------------------*/
int rt_hw_adc_init(void);
/* Private variables ------------------------------------------------------------*/
static struct nu_adc nu_adc_arr [] =
{
#if defined(BSP_USING_ADC0)
{
.name = "adc0",
.adc_base = ADC,
.adc_max_ch_num = 15,
},
#endif
{0}
};
static const struct rt_adc_ops nu_adc_ops =
{
nu_adc_enabled,
nu_get_adc_value,
};
typedef struct rt_adc_ops *rt_adc_ops_t;
/* nu_adc_enabled - Enable ADC clock and wait for ready */
static rt_err_t nu_adc_enabled(struct rt_adc_device *device, rt_uint32_t channel, rt_bool_t enabled)
{
ADC_T *adc_base = ((nu_adc_t)device)->adc_base;
int *padc_reg_tab = &((nu_adc_t)device)->adc_reg_tab;
RT_ASSERT(device != RT_NULL);
if (channel >= ((nu_adc_t)device)->adc_max_ch_num)
return -(RT_EINVAL);
if (enabled)
{
ADC_POWER_ON(adc_base);
if (*padc_reg_tab == 0)
{
ADC_Open(adc_base, ADC_ADCR_DIFFEN_SINGLE_END, ADC_ADCR_ADMD_SINGLE, (0x1 << channel));
}
*padc_reg_tab |= (0x1 << channel);
}
else
{
*padc_reg_tab &= ~(0x1 << channel);
if (*padc_reg_tab == 0)
{
ADC_Close(adc_base);
}
ADC_POWER_DOWN(adc_base);
}
return RT_EOK;
}
static rt_err_t nu_get_adc_value(struct rt_adc_device *device, rt_uint32_t channel, rt_uint32_t *value)
{
RT_ASSERT(device != RT_NULL);
RT_ASSERT(value != RT_NULL);
ADC_T *adc_base = ((nu_adc_t)device)->adc_base;
int *padc_reg_tab = &((nu_adc_t)device)->adc_reg_tab;
if (channel >= ((nu_adc_t)device)->adc_max_ch_num)
{
*value = 0xFFFFFFFF;
return -(RT_EINVAL);
}
if ((*padc_reg_tab & (1 << channel)) == 0)
{
*value = 0xFFFFFFFF;
return -(RT_EBUSY);
}
ADC_SET_INPUT_CHANNEL(adc_base, (0x1<<channel));
ADC_CLR_INT_FLAG(adc_base, ADC_ADF_INT);
ADC_ENABLE_INT(adc_base, ADC_ADF_INT);
ADC_START_CONV(adc_base);
while (ADC_GET_INT_FLAG(adc_base, ADC_ADF_INT) == 0);
*value = ADC_GET_CONVERSION_DATA(adc_base, channel);
return RT_EOK;
}
int rt_hw_adc_init(void)
{
rt_err_t result = -RT_ERROR;
int nu_sel = 0;
while (nu_adc_arr[nu_sel].name != 0)
{
nu_adc_arr[nu_sel].adc_reg_tab = 0;
result = rt_hw_adc_register(&nu_adc_arr[nu_sel].dev, nu_adc_arr[nu_sel].name, &nu_adc_ops, NULL);
RT_ASSERT(result == RT_EOK);
nu_sel++;
}
return (int)result;
}
INIT_BOARD_EXPORT(rt_hw_adc_init);
#endif //#if defined(BSP_USING_ADC)