| /* | 
 |  * Copyright (c) 2023 Grinn | 
 |  * SPDX-License-Identifier: Apache-2.0 | 
 |  */ | 
 |  | 
 | #define DT_DRV_COMPAT adi_ad559x_dac | 
 |  | 
 | #include <zephyr/drivers/dac.h> | 
 | #include <zephyr/kernel.h> | 
 | #include <zephyr/sys/byteorder.h> | 
 |  | 
 | #include <zephyr/drivers/mfd/ad559x.h> | 
 |  | 
 | #include <zephyr/logging/log.h> | 
 | 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) |