/* * Copyright (c) 2020 Libre Solar Technologies GmbH * * SPDX-License-Identifier: Apache-2.0 */ #define DT_DRV_COMPAT st_stm32_dac #include #include #include #include #include #include #define LOG_LEVEL CONFIG_DAC_LOG_LEVEL #include LOG_MODULE_REGISTER(dac_stm32); #include /* some low-end MCUs have DAC with only one channel */ #ifdef LL_DAC_CHANNEL_2 #define STM32_CHANNEL_COUNT 2 #else #define STM32_CHANNEL_COUNT 1 #endif /* first channel always named 1 */ #define STM32_FIRST_CHANNEL 1 #define CHAN(n) LL_DAC_CHANNEL_##n static const uint32_t table_channels[] = { CHAN(1), #ifdef LL_DAC_CHANNEL_2 CHAN(2), #endif }; /* Read-only driver configuration */ struct dac_stm32_cfg { DAC_TypeDef *base; struct stm32_pclken pclken; }; /* Runtime driver data */ struct dac_stm32_data { uint8_t channel_count; uint8_t resolution; }; static int dac_stm32_write_value(struct device *dev, uint8_t channel, uint32_t value) { struct dac_stm32_data *data = dev->driver_data; const struct dac_stm32_cfg *cfg = dev->config_info; if (channel - STM32_FIRST_CHANNEL >= data->channel_count || channel < STM32_FIRST_CHANNEL) { LOG_ERR("Channel %d is not valid", channel); return -EINVAL; } if (data->resolution == 8) { LL_DAC_ConvertData8RightAligned(cfg->base, table_channels[channel - STM32_FIRST_CHANNEL], value); } else if (data->resolution == 12) { LL_DAC_ConvertData12RightAligned(cfg->base, table_channels[channel - STM32_FIRST_CHANNEL], value); } return 0; } static int dac_stm32_channel_setup(struct device *dev, const struct dac_channel_cfg *channel_cfg) { struct dac_stm32_data *data = dev->driver_data; const struct dac_stm32_cfg *cfg = dev->config_info; if ((channel_cfg->channel_id - STM32_FIRST_CHANNEL >= data->channel_count) || (channel_cfg->channel_id < STM32_FIRST_CHANNEL)) { LOG_ERR("Channel %d is not valid", channel_cfg->channel_id); return -EINVAL; } if ((channel_cfg->resolution == 8) || (channel_cfg->resolution == 12)) { data->resolution = channel_cfg->resolution; } else { LOG_ERR("Resolution not supported"); return -ENOTSUP; } /* enable output buffer by default */ LL_DAC_SetOutputBuffer(cfg->base, table_channels[channel_cfg->channel_id - STM32_FIRST_CHANNEL], LL_DAC_OUTPUT_BUFFER_ENABLE); LL_DAC_Enable(cfg->base, table_channels[channel_cfg->channel_id - STM32_FIRST_CHANNEL]); LOG_DBG("Channel setup succeeded!"); return 0; } static int dac_stm32_init(struct device *dev) { const struct dac_stm32_cfg *cfg = dev->config_info; /* enable clock for subsystem */ struct device *clk = device_get_binding(STM32_CLOCK_CONTROL_NAME); if (clock_control_on(clk, (clock_control_subsys_t *) &cfg->pclken) != 0) { return -EIO; } return 0; } static const struct dac_driver_api api_stm32_driver_api = { .channel_setup = dac_stm32_channel_setup, .write_value = dac_stm32_write_value }; #define STM32_DAC_INIT(index) \ \ static const struct dac_stm32_cfg dac_stm32_cfg_##index = { \ .base = (DAC_TypeDef *)DT_INST_REG_ADDR(index), \ .pclken = { \ .enr = DT_INST_CLOCKS_CELL(index, bits), \ .bus = DT_INST_CLOCKS_CELL(index, bus), \ }, \ }; \ static struct dac_stm32_data dac_stm32_data_##index = { \ .channel_count = STM32_CHANNEL_COUNT \ }; \ \ DEVICE_AND_API_INIT(dac_##index, DT_INST_LABEL(index), \ &dac_stm32_init, &dac_stm32_data_##index, \ &dac_stm32_cfg_##index, POST_KERNEL, \ CONFIG_KERNEL_INIT_PRIORITY_DEFAULT, \ &api_stm32_driver_api); DT_INST_FOREACH_STATUS_OKAY(STM32_DAC_INIT)