hwmon: qpnp-adc: Fix register read/writes

Fix reading/writing to the correct ADC peripheral offset
address during SPMI read/writes. While adding multiple
channels an incorrect channel index initialization causes
the ADC reads to occur only on one channel. Fix it by
initializing the channel index only once at init.

The configure api has few register read/writes done
without using the bit mask field. Fix it by reading
the register and updating only the concerned bits.
Also add more comments to the configure api to list
out the steps for each register configuration and
fix using the correct print statements on error.

Change-Id: I0ce67163c5f27021f5fb0905fe601f4b1e0ee8b8
Signed-off-by: Siddartha Mohanadoss <smohanad@codeaurora.org>
This commit is contained in:
Siddartha Mohanadoss 2012-08-08 16:39:02 -07:00 committed by Stephen Boyd
parent 0e6afaa1c2
commit 6fe8f3c97a
3 changed files with 85 additions and 53 deletions

View file

@ -111,7 +111,7 @@ int32_t qpnp_adc_get_devicetree_data(struct spmi_device *spmi,
struct qpnp_vadc_amux *adc_channel_list;
struct qpnp_adc_properties *adc_prop;
struct qpnp_adc_amux_properties *amux_prop;
int count_adc_channel_list = 0, decimation, rc = 0;
int count_adc_channel_list = 0, decimation, rc = 0, i = 0;
if (!node)
return -EINVAL;
@ -133,7 +133,7 @@ int32_t qpnp_adc_get_devicetree_data(struct spmi_device *spmi,
return -ENOMEM;
}
adc_channel_list = devm_kzalloc(&spmi->dev,
(sizeof(struct qpnp_vadc_amux) * count_adc_channel_list),
sizeof(struct qpnp_vadc_amux) * count_adc_channel_list,
GFP_KERNEL);
if (!adc_channel_list) {
dev_err(&spmi->dev, "Unable to allocate memory\n");
@ -148,9 +148,12 @@ int32_t qpnp_adc_get_devicetree_data(struct spmi_device *spmi,
return -ENOMEM;
}
adc_qpnp->adc_channels = adc_channel_list;
adc_qpnp->amux_prop = amux_prop;
for_each_child_of_node(node, child) {
int channel_num, scaling, post_scaling, hw_settle_time;
int fast_avg_setup, calib_type, i = 0, rc;
int fast_avg_setup, calib_type, rc;
const char *calibration_param, *channel_name;
channel_name = of_get_property(child,
@ -216,8 +219,6 @@ int32_t qpnp_adc_get_devicetree_data(struct spmi_device *spmi,
adc_channel_list[i].fast_avg_setup = fast_avg_setup;
i++;
}
adc_qpnp->adc_channels = adc_channel_list;
adc_qpnp->amux_prop = amux_prop;
/* Get the ADC VDD reference voltage and ADC bit resolution */
rc = of_property_read_u32(node, "qcom,adc-vdd-reference",

View file

@ -142,7 +142,7 @@ static int32_t qpnp_iadc_read_reg(uint32_t reg, u8 *data)
int rc;
rc = spmi_ext_register_readl(iadc->adc->spmi->ctrl, iadc->adc->slave,
reg, data, 1);
(iadc->adc->offset + reg), data, 1);
if (rc < 0) {
pr_err("qpnp iadc read reg %d failed with %d\n", reg, rc);
return rc;
@ -159,7 +159,7 @@ static int32_t qpnp_iadc_write_reg(uint32_t reg, u8 data)
buf = &data;
rc = spmi_ext_register_writel(iadc->adc->spmi->ctrl, iadc->adc->slave,
reg, buf, 1);
(iadc->adc->offset + reg), buf, 1);
if (rc < 0) {
pr_err("qpnp iadc write reg %d failed with %d\n", reg, rc);
return rc;

View file

@ -37,6 +37,7 @@
#define QPNP_VADC_STATUS1_MEAS_INTERVAL_EN_STS BIT(2)
#define QPNP_VADC_STATUS1_REQ_STS BIT(1)
#define QPNP_VADC_STATUS1_EOC BIT(0)
#define QPNP_VADC_STATUS1_REQ_STS_EOC_MASK 0x3
#define QPNP_VADC_STATUS2 0x9
#define QPNP_VADC_STATUS2_CONV_SEQ_STATE 6
#define QPNP_VADC_STATUS2_FIFO_NOT_EMPTY_FLAG BIT(1)
@ -112,7 +113,7 @@ static int32_t qpnp_vadc_read_reg(int16_t reg, u8 *data)
int rc;
rc = spmi_ext_register_readl(vadc->adc->spmi->ctrl, vadc->adc->slave,
reg, data, 1);
(vadc->adc->offset + reg), data, 1);
if (rc < 0) {
pr_err("qpnp adc read reg %d failed with %d\n", reg, rc);
return rc;
@ -130,7 +131,7 @@ static int32_t qpnp_vadc_write_reg(int16_t reg, u8 data)
buf = &data;
rc = spmi_ext_register_writel(vadc->adc->spmi->ctrl, vadc->adc->slave,
reg, buf, 1);
(vadc->adc->offset + reg), buf, 1);
if (rc < 0) {
pr_err("qpnp adc write reg %d failed with %d\n", reg, rc);
return rc;
@ -200,61 +201,87 @@ static int32_t qpnp_vadc_enable(bool state)
int32_t qpnp_vadc_configure(
struct qpnp_adc_amux_properties *chan_prop)
{
struct qpnp_vadc_drv *vadc = qpnp_vadc;
u8 decimation = 0, conv_sequence = 0, conv_sequence_trig = 0;
u8 mode_ctrl = 0;
int rc = 0;
if (vadc->vadc_init_calib) {
/* Configure interrupt if calibration is complete */
rc = qpnp_vadc_write_reg(QPNP_VADC_INT_EN_SET,
QPNP_VADC_INT_EOC_BIT);
if (rc < 0) {
pr_err("qpnp adc configure error for interrupt setup\n");
pr_err("Configure error for interrupt setup\n");
return rc;
}
}
rc = qpnp_vadc_write_reg(QPNP_VADC_MODE_CTL, chan_prop->mode_sel);
/* Mode selection */
rc = qpnp_vadc_read_reg(QPNP_VADC_MODE_CTL, &mode_ctrl);
if (rc < 0) {
pr_err("qpnp adc configure error for mode selection\n");
pr_err("Mode configure read error\n");
return rc;
}
mode_ctrl |= chan_prop->mode_sel << QPNP_VADC_OP_MODE_SHIFT;
rc = qpnp_vadc_write_reg(QPNP_VADC_MODE_CTL, mode_ctrl);
if (rc < 0) {
pr_err("Mode configure write error\n");
return rc;
}
rc = qpnp_vadc_enable(true);
if (rc)
return rc;
/* Channel selection */
rc = qpnp_vadc_write_reg(QPNP_VADC_ADC_CH_SEL_CTL,
chan_prop->amux_channel);
if (rc < 0) {
pr_err("qpnp adc configure error for channel selection\n");
pr_err("Channel configure error\n");
return rc;
}
/* Digital parameter setup */
rc = qpnp_vadc_read_reg(QPNP_VADC_ADC_DIG_PARAM, &decimation);
if (rc < 0) {
pr_err("Digital parameter configure read error\n");
return rc;
}
decimation |= chan_prop->decimation <<
QPNP_VADC_ADC_DIG_DEC_RATIO_SEL_SHIFT;
rc = qpnp_vadc_write_reg(QPNP_VADC_ADC_DIG_PARAM, decimation);
if (rc < 0) {
pr_err("qpnp adc configure error for digital parameter setup\n");
pr_err("Digital parameter configure write error\n");
return rc;
}
/* HW settling time delay */
rc = qpnp_vadc_write_reg(QPNP_VADC_HW_SETTLE_DELAY,
chan_prop->hw_settle_time);
if (rc < 0) {
pr_err("qpnp adc configure error for hw settling time setup\n");
pr_err("HW settling time setup error\n");
return rc;
}
if (chan_prop->mode_sel == (ADC_OP_NORMAL_MODE <<
QPNP_VADC_OP_MODE_SHIFT)) {
/* Normal measurement mode */
rc = qpnp_vadc_write_reg(QPNP_VADC_FAST_AVG_CTL,
chan_prop->fast_avg_setup);
if (rc < 0) {
pr_err("qpnp adc fast averaging configure error\n");
pr_err("Fast averaging configure error\n");
return rc;
}
} else if (chan_prop->mode_sel == (ADC_OP_CONVERSION_SEQUENCER <<
QPNP_VADC_OP_MODE_SHIFT)) {
/* Conversion sequence mode */
conv_sequence = ((ADC_SEQ_HOLD_100US <<
QPNP_VADC_CONV_SEQ_HOLDOFF_SHIFT) |
ADC_CONV_SEQ_TIMEOUT_5MS);
rc = qpnp_vadc_write_reg(QPNP_VADC_CONV_SEQ_CTL,
conv_sequence);
if (rc < 0) {
pr_err("qpnp adc conversion sequence error\n");
pr_err("Conversion sequence error\n");
return rc;
}
@ -264,14 +291,15 @@ int32_t qpnp_vadc_configure(
rc = qpnp_vadc_write_reg(QPNP_VADC_CONV_SEQ_TRIG_CTL,
conv_sequence_trig);
if (rc < 0) {
pr_err("qpnp adc conversion trigger error\n");
pr_err("Conversion trigger error\n");
return rc;
}
}
/* Request conversion */
rc = qpnp_vadc_write_reg(QPNP_VADC_CONV_REQ, QPNP_VADC_CONV_REQ_SET);
if (rc < 0) {
pr_err("qpnp adc request conversion failed\n");
pr_err("Request conversion failed\n");
return rc;
}
@ -282,29 +310,34 @@ EXPORT_SYMBOL(qpnp_vadc_configure);
static int32_t qpnp_vadc_read_conversion_result(int32_t *data)
{
uint8_t rslt_lsb, rslt_msb;
int rc = 0;
int rc = 0, status = 0;
rc = qpnp_vadc_read_reg(QPNP_VADC_DATA0, &rslt_lsb);
if (rc < 0) {
pr_err("qpnp adc result read failed for data0 with %d\n", rc);
return rc;
status = qpnp_vadc_read_reg(QPNP_VADC_DATA0, &rslt_lsb);
if (status < 0) {
pr_err("qpnp adc result read failed for data0\n");
goto fail;
}
rc = qpnp_vadc_read_reg(QPNP_VADC_DATA1, &rslt_msb);
if (rc < 0) {
pr_err("qpnp adc result read failed for data1 with %d\n", rc);
return rc;
status = qpnp_vadc_read_reg(QPNP_VADC_DATA1, &rslt_msb);
if (status < 0) {
pr_err("qpnp adc result read failed for data1\n");
goto fail;
}
*data = (rslt_msb << 8) | rslt_lsb;
rc = qpnp_vadc_check_result(data);
if (rc < 0) {
status = qpnp_vadc_check_result(data);
if (status < 0) {
pr_err("VADC data check failed\n");
return rc;
goto fail;
}
return 0;
fail:
rc = qpnp_vadc_enable(false);
if (rc)
return rc;
return status;
}
static int32_t qpnp_vadc_read_status(int mode_sel)
@ -393,6 +426,7 @@ static uint32_t qpnp_vadc_calib_device(void)
rc = qpnp_vadc_read_reg(QPNP_VADC_STATUS1, &status1);
if (rc < 0)
return rc;
status1 &= QPNP_VADC_STATUS1_REQ_STS_EOC_MASK;
usleep_range(QPNP_VADC_CONV_TIME_MIN,
QPNP_VADC_CONV_TIME_MAX);
}
@ -414,11 +448,13 @@ static uint32_t qpnp_vadc_calib_device(void)
goto calib_fail;
}
status1 = 0;
while (status1 != (~QPNP_VADC_STATUS1_REQ_STS |
QPNP_VADC_STATUS1_EOC)) {
rc = qpnp_vadc_read_reg(QPNP_VADC_STATUS1, &status1);
if (rc < 0)
return rc;
status1 &= QPNP_VADC_STATUS1_REQ_STS_EOC_MASK;
usleep_range(QPNP_VADC_CONV_TIME_MIN,
QPNP_VADC_CONV_TIME_MAX);
}
@ -449,11 +485,13 @@ static uint32_t qpnp_vadc_calib_device(void)
goto calib_fail;
}
status1 = 0;
while (status1 != (~QPNP_VADC_STATUS1_REQ_STS |
QPNP_VADC_STATUS1_EOC)) {
rc = qpnp_vadc_read_reg(QPNP_VADC_STATUS1, &status1);
if (rc < 0)
return rc;
status1 &= QPNP_VADC_STATUS1_REQ_STS_EOC_MASK;
usleep_range(QPNP_VADC_CONV_TIME_MIN,
QPNP_VADC_CONV_TIME_MAX);
}
@ -475,11 +513,13 @@ static uint32_t qpnp_vadc_calib_device(void)
goto calib_fail;
}
status1 = 0;
while (status1 != (~QPNP_VADC_STATUS1_REQ_STS |
QPNP_VADC_STATUS1_EOC)) {
rc = qpnp_vadc_read_reg(QPNP_VADC_STATUS1, &status1);
if (rc < 0)
return rc;
status1 &= QPNP_VADC_STATUS1_REQ_STS_EOC_MASK;
usleep_range(QPNP_VADC_CONV_TIME_MIN,
QPNP_VADC_CONV_TIME_MAX);
}
@ -508,7 +548,7 @@ int32_t qpnp_vadc_conv_seq_request(enum qpnp_vadc_trigger trigger_channel,
struct qpnp_vadc_result *result)
{
struct qpnp_vadc_drv *vadc = qpnp_vadc;
int rc, scale_type, amux_prescaling;
int rc = 0, scale_type, amux_prescaling;
if (!vadc->vadc_init_calib) {
rc = qpnp_vadc_calib_device();
@ -521,10 +561,6 @@ int32_t qpnp_vadc_conv_seq_request(enum qpnp_vadc_trigger trigger_channel,
mutex_lock(&vadc->adc->adc_lock);
rc = qpnp_vadc_enable(true);
if (rc)
goto fail_unlock;
vadc->adc->amux_prop->amux_channel = channel;
vadc->adc->amux_prop->decimation =
vadc->adc->adc_channels[channel].adc_decimation;
@ -541,15 +577,15 @@ int32_t qpnp_vadc_conv_seq_request(enum qpnp_vadc_trigger trigger_channel,
<< QPNP_VADC_OP_MODE_SHIFT);
else {
pr_err("Invalid trigger channel:%d\n", trigger_channel);
goto fail;
goto fail_unlock;
}
vadc->adc->amux_prop->trigger_channel = trigger_channel;
rc = qpnp_vadc_configure(vadc->adc->amux_prop);
if (rc) {
pr_info("qpnp vadc configure failed with %d\n", rc);
goto fail;
pr_err("qpnp vadc configure failed with %d\n", rc);
goto fail_unlock;
}
wait_for_completion(&vadc->adc->adc_rslt_completion);
@ -557,13 +593,13 @@ int32_t qpnp_vadc_conv_seq_request(enum qpnp_vadc_trigger trigger_channel,
if (trigger_channel < ADC_SEQ_NONE) {
rc = qpnp_vadc_read_status(vadc->adc->amux_prop->mode_sel);
if (rc)
pr_info("Conversion sequence timed out - %d\n", rc);
pr_debug("Conversion sequence timed out - %d\n", rc);
}
rc = qpnp_vadc_read_conversion_result(&result->adc_code);
if (rc) {
pr_info("qpnp vadc read adc code failed with %d\n", rc);
goto fail;
pr_err("qpnp vadc read adc code failed with %d\n", rc);
goto fail_unlock;
}
amux_prescaling = vadc->adc->adc_channels[channel].chan_path_prescaling;
@ -576,17 +612,12 @@ int32_t qpnp_vadc_conv_seq_request(enum qpnp_vadc_trigger trigger_channel,
scale_type = vadc->adc->adc_channels[channel].adc_scale_fn;
if (scale_type >= SCALE_NONE) {
rc = -EBADF;
goto fail;
goto fail_unlock;
}
vadc_scale_fn[scale_type].chan(result->adc_code,
vadc->adc->adc_prop, vadc->adc->amux_prop->chan_prop, result);
fail:
rc = qpnp_vadc_enable(false);
if (rc)
pr_err("Disable ADC failed during configuration\n");
fail_unlock:
mutex_unlock(&vadc->adc->adc_lock);
@ -649,7 +680,7 @@ static int32_t qpnp_vadc_init_hwmon(struct spmi_device *spmi)
return 0;
hwmon_err_sens:
pr_info("Init HWMON failed for qpnp_adc with %d\n", rc);
pr_err("Init HWMON failed for qpnp_adc with %d\n", rc);
return rc;
}