2019-04-04 00:20:39 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2019 Gunar Schorcht
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser General
|
|
|
|
* Public License v2.1. See the file LICENSE in the top level directory for more
|
|
|
|
* details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup cpu_esp32
|
|
|
|
* @ingroup drivers_periph_adc
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Low-level DAC driver implementation
|
|
|
|
*
|
|
|
|
* @author Gunar Schorcht <gunar@schorcht.net>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "board.h"
|
|
|
|
#include "periph/dac.h"
|
|
|
|
|
|
|
|
#include "adc_arch.h"
|
|
|
|
#include "adc_ctrl.h"
|
|
|
|
#include "esp_common.h"
|
|
|
|
#include "gpio_arch.h"
|
|
|
|
#include "rom/ets_sys.h"
|
|
|
|
#include "soc/rtc_io_struct.h"
|
|
|
|
#include "soc/rtc_cntl_struct.h"
|
|
|
|
#include "soc/sens_reg.h"
|
|
|
|
#include "soc/sens_struct.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
/* declaration of external functions */
|
|
|
|
extern void _adc2_ctrl_init(void);
|
|
|
|
|
|
|
|
/* forward declarations of internal functions */
|
|
|
|
static bool _dac_conf_check(void);
|
|
|
|
static bool _dac_module_initialized = false;
|
|
|
|
|
|
|
|
/* external variable declarations */
|
|
|
|
extern const gpio_t _gpio_rtcio_map[];
|
|
|
|
|
|
|
|
int8_t dac_init (dac_t line)
|
|
|
|
{
|
2019-05-02 16:47:12 +02:00
|
|
|
CHECK_PARAM_RET (line < DAC_NUMOF, DAC_NOLINE)
|
2019-04-04 00:20:39 +02:00
|
|
|
|
|
|
|
if (!_dac_module_initialized) {
|
|
|
|
/* do some configuration checks */
|
|
|
|
if (!_dac_conf_check()) {
|
|
|
|
return DAC_NOLINE;
|
|
|
|
}
|
|
|
|
_dac_module_initialized = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
_adc2_ctrl_init();
|
|
|
|
|
2019-05-02 16:47:12 +02:00
|
|
|
uint8_t rtcio = _gpio_rtcio_map[dac_channels[line]];
|
2019-04-04 00:20:39 +02:00
|
|
|
uint8_t idx;
|
|
|
|
|
|
|
|
/* try to initialize the pin as DAC ouput */
|
|
|
|
if (gpio_get_pin_usage(_adc_hw[rtcio].gpio) != _GPIO) {
|
|
|
|
LOG_TAG_ERROR("dac", "GPIO%d is used for %s and cannot be used as "
|
|
|
|
"DAC output\n", _adc_hw[rtcio].gpio,
|
|
|
|
gpio_get_pin_usage_str(_adc_hw[rtcio].gpio));
|
|
|
|
return DAC_NOLINE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* disable the output of the pad */
|
|
|
|
RTCIO.enable_w1tc.val = BIT(_adc_hw[rtcio].rtc_gpio);
|
|
|
|
|
|
|
|
switch (rtcio) {
|
|
|
|
case RTCIO_DAC1: /* GPIO25, RTC6 */
|
|
|
|
case RTCIO_DAC2: /* GPIO26, RTC7 */
|
|
|
|
idx = rtcio - RTCIO_DAC1;
|
|
|
|
RTCIO.pad_dac[idx].mux_sel = 1; /* route to RTC */
|
|
|
|
RTCIO.pad_dac[idx].fun_sel = 0; /* function ADC2_CH8, ADC2_CH9 */
|
|
|
|
RTCIO.pad_dac[idx].fun_ie = 0; /* input disabled */
|
|
|
|
RTCIO.pad_dac[idx].rue = 0; /* pull-up disabled */
|
|
|
|
RTCIO.pad_dac[idx].rde = 0; /* pull-down disabled */
|
|
|
|
|
|
|
|
RTCIO.pad_dac[idx].dac_xpd_force = 1; /* use RTC pad not the FSM*/
|
|
|
|
RTCIO.pad_dac[idx].xpd_dac = 1; /* DAC powered on */
|
|
|
|
break;
|
|
|
|
|
|
|
|
default: return DAC_NOLINE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* set pin usage type */
|
|
|
|
gpio_set_pin_usage(_adc_hw[rtcio].gpio, _DAC);
|
|
|
|
|
|
|
|
/* don't use DMA */
|
|
|
|
SENS.sar_dac_ctrl1.dac_dig_force = 0;
|
|
|
|
|
|
|
|
/* disable CW generators and invert DAC signal */
|
|
|
|
SENS.sar_dac_ctrl1.sw_tone_en = 0;
|
|
|
|
SENS.sar_dac_ctrl2.dac_cw_en1 = 0;
|
|
|
|
SENS.sar_dac_ctrl2.dac_cw_en2 = 0;
|
|
|
|
|
|
|
|
return DAC_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
void dac_set (dac_t line, uint16_t value)
|
|
|
|
{
|
2019-05-02 16:47:12 +02:00
|
|
|
CHECK_PARAM (line < DAC_NUMOF);
|
|
|
|
RTCIO.pad_dac[_gpio_rtcio_map[dac_channels[line]] - RTCIO_DAC1].dac = value >> 8;
|
2019-04-04 00:20:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void dac_poweroff (dac_t line)
|
|
|
|
{
|
2019-05-02 16:47:12 +02:00
|
|
|
CHECK_PARAM (line < DAC_NUMOF);
|
2019-04-04 00:20:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void dac_poweron (dac_t line)
|
|
|
|
{
|
2019-05-02 16:47:12 +02:00
|
|
|
CHECK_PARAM (line < DAC_NUMOF);
|
2019-04-04 00:20:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool _dac_conf_check(void)
|
|
|
|
{
|
2019-05-02 16:47:12 +02:00
|
|
|
for (unsigned i = 0; i < DAC_NUMOF; i++) {
|
|
|
|
if (_gpio_rtcio_map[dac_channels[i]] != RTCIO_DAC1 &&
|
|
|
|
_gpio_rtcio_map[dac_channels[i]] != RTCIO_DAC2) {
|
2019-04-04 00:20:39 +02:00
|
|
|
LOG_TAG_ERROR("dac", "GPIO%d cannot be used as DAC line\n",
|
2019-05-02 16:47:12 +02:00
|
|
|
dac_channels[i]);
|
2019-04-04 00:20:39 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void dac_print_config(void)
|
|
|
|
{
|
|
|
|
ets_printf("\tDAC\t\tpins=[ ");
|
|
|
|
#if defined(DAC_GPIOS)
|
2019-05-02 16:47:12 +02:00
|
|
|
for (unsigned i = 0; i < DAC_NUMOF; i++) {
|
|
|
|
ets_printf("%d ", dac_channels[i]);
|
2019-04-04 00:20:39 +02:00
|
|
|
}
|
|
|
|
#endif /* defined(DAC_GPIOS) */
|
|
|
|
ets_printf("]\n");
|
|
|
|
}
|