/* * Copyright (c) 2023 Grinn * SPDX-License-Identifier: Apache-2.0 */ #define DT_DRV_COMPAT adi_ad559x_dac #include #include #include #include #include LOG_MODULE_REGISTER(dac_ad559x, CONFIG_DAC_LOG_LEVEL); #define AD559X_DAC_RESOLUTION 12 #define AD559X_DAC_WR_POINTER 0x10 #define AD559X_DAC_WR_MSB_BIT BIT(15) #define AD559X_DAC_CHANNEL_SHIFT_VAL 12 struct dac_ad559x_config { const struct device *mfd_dev; bool double_output_range; }; struct dac_ad559x_data { uint8_t dac_conf; }; static int dac_ad559x_channel_setup(const struct device *dev, const struct dac_channel_cfg *channel_cfg) { const struct dac_ad559x_config *config = dev->config; struct dac_ad559x_data *data = dev->data; if (channel_cfg->channel_id >= AD559X_PIN_MAX) { LOG_ERR("Invalid channel number %d", channel_cfg->channel_id); return -EINVAL; } if (channel_cfg->resolution != AD559X_DAC_RESOLUTION) { LOG_ERR("Invalid resolution %d", channel_cfg->resolution); return -EINVAL; } if (channel_cfg->internal) { LOG_ERR("Internal channels not supported"); return -ENOTSUP; } data->dac_conf |= BIT(channel_cfg->channel_id); return mfd_ad559x_write_reg(config->mfd_dev, AD559X_REG_LDAC_EN, data->dac_conf); } static int dac_ad559x_write_value(const struct device *dev, uint8_t channel, uint32_t value) { const struct dac_ad559x_config *config = dev->config; uint16_t msg; if (channel >= AD559X_PIN_MAX) { LOG_ERR("Invalid channel number %d", channel); return -EINVAL; } if (value >= (1 << AD559X_DAC_RESOLUTION)) { LOG_ERR("Value %d out of range", value); return -EINVAL; } if (mfd_ad559x_has_pointer_byte_map(config->mfd_dev)) { return mfd_ad559x_write_reg(config->mfd_dev, AD559X_DAC_WR_POINTER | channel, value); } else { msg = sys_cpu_to_be16(AD559X_DAC_WR_MSB_BIT | channel << AD559X_DAC_CHANNEL_SHIFT_VAL | value); return mfd_ad559x_write_raw(config->mfd_dev, (uint8_t *)&msg, sizeof(msg)); } } static DEVICE_API(dac, dac_ad559x_api) = { .channel_setup = dac_ad559x_channel_setup, .write_value = dac_ad559x_write_value, }; static int dac_ad559x_init(const struct device *dev) { const struct dac_ad559x_config *config = dev->config; int ret; uint16_t reg_val; if (!device_is_ready(config->mfd_dev)) { return -ENODEV; } ret = mfd_ad559x_read_reg(config->mfd_dev, AD559X_REG_GEN_CTRL, 0, ®_val); if (ret < 0) { return ret; } if (config->double_output_range) { reg_val |= AD559X_DAC_RANGE; } else { reg_val &= ~AD559X_DAC_RANGE; } ret = mfd_ad559x_write_reg(config->mfd_dev, AD559X_REG_GEN_CTRL, reg_val); if (ret < 0) { return ret; } ret = mfd_ad559x_write_reg(config->mfd_dev, AD559X_REG_PD_REF_CTRL, AD559X_EN_REF); if (ret < 0) { return ret; } return 0; } #define DAC_AD559X_DEFINE(inst) \ static const struct dac_ad559x_config dac_ad559x_config##inst = { \ .mfd_dev = DEVICE_DT_GET(DT_INST_PARENT(inst)), \ .double_output_range = DT_INST_PROP(inst, double_output_range), \ }; \ \ struct dac_ad559x_data dac_ad559x_data##inst; \ \ DEVICE_DT_INST_DEFINE(inst, dac_ad559x_init, NULL, &dac_ad559x_data##inst, \ &dac_ad559x_config##inst, POST_KERNEL, CONFIG_MFD_INIT_PRIORITY, \ &dac_ad559x_api); DT_INST_FOREACH_STATUS_OKAY(DAC_AD559X_DEFINE)