blob: c37d7fa128178d4d7b216744506876a254d6f71a [file] [log] [blame]
/*
* Copyright (c) 2012-2020, The Linux Foundation. All rights reserved.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 and
* only version 2 as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
*/
#include <linux/module.h>
#include <linux/regmap.h>
#include <linux/of.h>
#include <linux/err.h>
#include <linux/spmi.h>
#include <linux/platform_device.h>
#include <linux/of_irq.h>
#include <linux/interrupt.h>
#include <linux/spinlock.h>
#include <linux/iio/consumer.h>
#include "adc-tm.h"
#include "../thermal_core.h"
#define ADC_TM_STATUS2 0x09
#define ADC_TM_STATUS_LOW 0x0a
#define ADC_TM_STATUS_HIGH 0x0b
#define ADC_TM_NUM_BTM 0x0f
#define ADC_TM_ADC_DIG_PARAM 0x42
#define ADC_TM_FAST_AVG_CTL 0x43
#define ADC_TM_FAST_AVG_EN BIT(7)
#define ADC_TM_MEAS_INTERVAL_CTL 0x44
#define ADC_TM_MEAS_INTERVAL_CTL2 0x45
#define ADC_TM_MEAS_INTERVAL_CTL_660 0x50
#define ADC_TM_MEAS_INTERVAL_CTL2_660 0x51
#define ADC_TM_MEAS_INTERVAL_CTL2_SHIFT 0x4
#define ADC_TM_MEAS_INTERVAL_CTL2_MASK 0xf0
#define ADC_TM_MEAS_INTERVAL_CTL3_MASK 0xf
#define ADC_TM_EN_CTL1 0x46
#define ADC_TM_EN BIT(7)
#define ADC_TM_CONV_REQ 0x47
#define ADC_TM_CONV_REQ_EN BIT(7)
#define ADC_TM_Mn_ADC_CH_SEL_CTL(n) ((n * 8) + 0x60)
#define ADC_TM_Mn_LOW_THR0(n) ((n * 8) + 0x61)
#define ADC_TM_Mn_LOW_THR1(n) ((n * 8) + 0x62)
#define ADC_TM_Mn_HIGH_THR0(n) ((n * 8) + 0x63)
#define ADC_TM_Mn_HIGH_THR1(n) ((n * 8) + 0x64)
#define ADC_TM_Mn_MEAS_INTERVAL_CTL(n) ((n * 8) + 0x65)
#define ADC_TM_Mn_CTL(n) ((n * 8) + 0x66)
#define ADC_TM_CTL_HW_SETTLE_DELAY_MASK 0xf
#define ADC_TM_CTL_CAL_SEL 0x30
#define ADC_TM_CTL_CAL_SEL_MASK_SHIFT 4
#define ADC_TM_CTL_CAL_VAL 0x40
#define ADC_TM_Mn_EN(n) ((n * 8) + 0x67)
#define ADC_TM_Mn_MEAS_EN BIT(7)
#define ADC_TM_Mn_HIGH_THR_INT_EN BIT(1)
#define ADC_TM_Mn_LOW_THR_INT_EN BIT(0)
#define ADC_TM_LOWER_MASK(n) ((n) & 0x000000ff)
#define ADC_TM_UPPER_MASK(n) (((n) & 0xffffff00) >> 8)
#define ADC_TM_Mn_DATA0(n) ((n * 2) + 0xa0)
#define ADC_TM_Mn_DATA1(n) ((n * 2) + 0xa1)
#define ADC_TM_DATA_SHIFT 8
static struct adc_tm_trip_reg_type adc_tm_ch_data[] = {
[ADC_TM_CHAN0] = {ADC_TM_M0_ADC_CH_SEL_CTL},
[ADC_TM_CHAN1] = {ADC_TM_M1_ADC_CH_SEL_CTL},
[ADC_TM_CHAN2] = {ADC_TM_M2_ADC_CH_SEL_CTL},
[ADC_TM_CHAN3] = {ADC_TM_M3_ADC_CH_SEL_CTL},
[ADC_TM_CHAN4] = {ADC_TM_M4_ADC_CH_SEL_CTL},
[ADC_TM_CHAN5] = {ADC_TM_M5_ADC_CH_SEL_CTL},
[ADC_TM_CHAN6] = {ADC_TM_M6_ADC_CH_SEL_CTL},
[ADC_TM_CHAN7] = {ADC_TM_M7_ADC_CH_SEL_CTL},
};
static struct adc_tm_reverse_scale_fn adc_tm_rscale_fn[] = {
[SCALE_R_ABSOLUTE] = {adc_tm_absolute_rthr},
};
static int adc_tm5_get_temp(struct adc_tm_sensor *sensor, int *temp)
{
int ret, milli_celsius;
int emul_temp;
if (!sensor || !sensor->adc)
return -EINVAL;
emul_temp = sensor->emul_temperature;
if (emul_temp) {
*temp = emul_temp;
return 0;
}
ret = iio_read_channel_processed(sensor->adc, &milli_celsius);
if (ret < 0)
return ret;
*temp = milli_celsius;
return 0;
}
static int adc_tm5_set_emul_temp(struct adc_tm_sensor *sensor, int emul_temp)
{
sensor->emul_temperature = emul_temp;
return 0;
}
static int32_t adc_tm5_read_reg(struct adc_tm_chip *chip,
int16_t reg, u8 *data, int len)
{
int ret;
ret = regmap_bulk_read(chip->regmap, (chip->base + reg), data, len);
if (ret < 0)
pr_err("adc-tm read reg %d failed with %d\n", reg, ret);
return ret;
}
static int32_t adc_tm5_write_reg(struct adc_tm_chip *chip,
int16_t reg, u8 *data, int len)
{
int ret;
ret = regmap_bulk_write(chip->regmap, (chip->base + reg), data, len);
if (ret < 0)
pr_err("adc-tm write reg %d failed with %d\n", reg, ret);
return ret;
}
static int32_t adc_tm5_reg_update(struct adc_tm_chip *chip,
uint16_t addr, u8 mask, bool state)
{
u8 reg_value = 0;
int ret;
ret = adc_tm5_read_reg(chip, addr, &reg_value, 1);
if (ret < 0) {
pr_err("read failed for addr:0x%x\n", addr);
return ret;
}
reg_value = reg_value & ~mask;
if (state)
reg_value |= mask;
pr_debug("state:%d, reg:0x%x with bits:0x%x and mask:0x%x\n",
state, addr, reg_value, ~mask);
ret = adc_tm5_write_reg(chip, addr, &reg_value, 1);
if (ret < 0) {
pr_err("write failed for addr:%x\n", addr);
return ret;
}
return ret;
}
static int32_t adc_tm5_get_btm_idx(struct adc_tm_chip *chip,
uint32_t btm_chan, uint32_t *btm_chan_idx)
{
int i;
for (i = 0; i < ADC_TM_CHAN_NONE; i++) {
if (adc_tm_ch_data[i].btm_amux_ch == btm_chan) {
*btm_chan_idx = i;
return 0;
}
}
return -EINVAL;
}
static int32_t adc_tm5_enable(struct adc_tm_chip *chip)
{
int rc = 0;
u8 data = 0;
data = ADC_TM_EN;
rc = adc_tm5_write_reg(chip, ADC_TM_EN_CTL1, &data, 1);
if (rc < 0) {
pr_err("adc-tm enable failed\n");
return rc;
}
data = ADC_TM_CONV_REQ_EN;
rc = adc_tm5_write_reg(chip, ADC_TM_CONV_REQ, &data, 1);
if (rc < 0) {
pr_err("adc-tm request conversion failed\n");
return rc;
}
return rc;
}
static int adc_tm5_configure(struct adc_tm_sensor *sensor,
uint32_t btm_chan_idx)
{
struct adc_tm_chip *chip = sensor->chip;
u8 buf[8], cal_sel;
int ret = 0;
ret = adc_tm5_read_reg(chip,
ADC_TM_Mn_ADC_CH_SEL_CTL(btm_chan_idx), buf, 8);
if (ret < 0) {
pr_err("adc-tm block read failed with %d\n", ret);
return ret;
}
/* Update ADC channel select */
buf[0] = sensor->adc_ch;
/* Update timer select */
buf[5] = sensor->timer_select;
/* Set calibration select, hw_settle delay */
cal_sel = (u8) (sensor->cal_sel << ADC_TM_CTL_CAL_SEL_MASK_SHIFT);
buf[6] &= (u8) ~ADC_TM_CTL_HW_SETTLE_DELAY_MASK;
buf[6] |= (u8) sensor->hw_settle_time;
buf[6] &= (u8) ~ADC_TM_CTL_CAL_SEL;
buf[6] |= (u8) cal_sel;
buf[7] |= ADC_TM_Mn_MEAS_EN;
ret = adc_tm5_write_reg(chip,
ADC_TM_Mn_ADC_CH_SEL_CTL(btm_chan_idx), buf, 1);
if (ret < 0) {
pr_err("adc-tm channel select failed\n");
return ret;
}
ret = adc_tm5_write_reg(chip,
ADC_TM_Mn_MEAS_INTERVAL_CTL(btm_chan_idx), &buf[5], 1);
if (ret < 0) {
pr_err("adc-tm timer select failed\n");
return ret;
}
ret = adc_tm5_write_reg(chip,
ADC_TM_Mn_CTL(btm_chan_idx), &buf[6], 1);
if (ret < 0) {
pr_err("adc-tm parameter select failed\n");
return ret;
}
ret = adc_tm5_write_reg(chip,
ADC_TM_Mn_EN(btm_chan_idx), &buf[7], 1);
if (ret < 0) {
pr_err("adc-tm monitoring enable failed\n");
return ret;
}
return 0;
}
static int32_t adc_tm_add_to_list(struct adc_tm_chip *chip,
uint32_t dt_index,
struct adc_tm_param *param)
{
struct adc_tm_client_info *client_info = NULL;
bool client_info_exists = false;
list_for_each_entry(client_info,
&chip->sensor[dt_index].thr_list, list) {
if (client_info->param == param) {
client_info->low_thr_requested = param->low_thr;
client_info->high_thr_requested = param->high_thr;
client_info->state_request = param->state_request;
client_info->notify_low_thr = false;
client_info->notify_high_thr = false;
client_info_exists = true;
pr_debug("client found\n");
}
}
if (!client_info_exists) {
client_info = devm_kzalloc(chip->dev,
sizeof(struct adc_tm_client_info), GFP_KERNEL);
if (!client_info)
return -ENOMEM;
pr_debug("new client\n");
client_info->param = param;
client_info->low_thr_requested = param->low_thr;
client_info->high_thr_requested = param->high_thr;
client_info->state_request = param->state_request;
list_add_tail(&client_info->list,
&chip->sensor[dt_index].thr_list);
}
return 0;
}
static int32_t adc_tm5_thr_update(struct adc_tm_sensor *sensor,
int32_t high_thr, int32_t low_thr)
{
int ret = 0;
u8 trip_low_thr[2], trip_high_thr[2];
uint16_t reg_low_thr_lsb, reg_high_thr_lsb;
uint32_t scale_type = 0, mask = 0, btm_chan_idx = 0;
struct adc_tm_config tm_config;
struct adc_tm_chip *chip;
chip = sensor->chip;
ret = adc_tm5_get_btm_idx(chip,
sensor->btm_ch, &btm_chan_idx);
if (ret < 0) {
pr_err("Invalid btm channel idx\n");
return ret;
}
tm_config.high_thr_voltage = (int64_t)high_thr;
tm_config.low_thr_voltage = (int64_t)low_thr;
tm_config.prescal = sensor->prescaling;
scale_type = sensor->adc_rscale_fn;
if (scale_type >= SCALE_RSCALE_NONE) {
ret = -EBADF;
return ret;
}
adc_tm_rscale_fn[scale_type].chan(chip->data, &tm_config);
mask = lower_32_bits(tm_config.high_thr_voltage);
trip_high_thr[0] = ADC_TM_LOWER_MASK(mask);
trip_high_thr[1] = ADC_TM_UPPER_MASK(mask);
mask = lower_32_bits(tm_config.low_thr_voltage);
trip_low_thr[0] = ADC_TM_LOWER_MASK(mask);
trip_low_thr[1] = ADC_TM_UPPER_MASK(mask);
pr_debug("high_thr:0x%llx, low_thr:0x%llx\n",
tm_config.high_thr_voltage, tm_config.low_thr_voltage);
reg_low_thr_lsb = ADC_TM_Mn_LOW_THR0(btm_chan_idx);
reg_high_thr_lsb = ADC_TM_Mn_HIGH_THR0(btm_chan_idx);
if (low_thr != INT_MIN) {
ret = adc_tm5_write_reg(chip, reg_low_thr_lsb,
trip_low_thr, 2);
if (ret) {
pr_err("Low set threshold err\n");
return ret;
}
}
if (high_thr != INT_MAX) {
ret = adc_tm5_write_reg(chip, reg_high_thr_lsb,
trip_high_thr, 2);
if (ret) {
pr_err("High set threshold err\n");
return ret;
}
}
return ret;
}
static int32_t adc_tm5_manage_thresholds(struct adc_tm_sensor *sensor)
{
int ret = 0, high_thr = INT_MAX, low_thr = INT_MIN;
struct adc_tm_client_info *client_info = NULL;
struct list_head *thr_list;
uint32_t btm_chan_idx = 0;
struct adc_tm_chip *chip = sensor->chip;
ret = adc_tm5_get_btm_idx(chip, sensor->btm_ch, &btm_chan_idx);
if (ret < 0) {
pr_err("Invalid btm channel idx with %d\n", ret);
return ret;
}
/*
* Reset the high_thr_set and low_thr_set of all
* clients since the thresholds will be recomputed.
*/
list_for_each(thr_list, &sensor->thr_list) {
client_info = list_entry(thr_list,
struct adc_tm_client_info, list);
client_info->high_thr_set = false;
client_info->low_thr_set = false;
}
/* Find the min of high_thr and max of low_thr */
list_for_each(thr_list, &sensor->thr_list) {
client_info = list_entry(thr_list,
struct adc_tm_client_info, list);
if ((client_info->state_request == ADC_TM_HIGH_THR_ENABLE) ||
(client_info->state_request ==
ADC_TM_HIGH_LOW_THR_ENABLE))
if (client_info->high_thr_requested < high_thr)
high_thr = client_info->high_thr_requested;
if ((client_info->state_request == ADC_TM_LOW_THR_ENABLE) ||
(client_info->state_request ==
ADC_TM_HIGH_LOW_THR_ENABLE))
if (client_info->low_thr_requested > low_thr)
low_thr = client_info->low_thr_requested;
pr_debug("threshold compared is high:%d and low:%d\n",
client_info->high_thr_requested,
client_info->low_thr_requested);
pr_debug("current threshold is high:%d and low:%d\n",
high_thr, low_thr);
}
/* Check which of the high_thr and low_thr got set */
list_for_each(thr_list, &sensor->thr_list) {
client_info = list_entry(thr_list,
struct adc_tm_client_info, list);
if ((client_info->state_request == ADC_TM_HIGH_THR_ENABLE) ||
(client_info->state_request ==
ADC_TM_HIGH_LOW_THR_ENABLE))
if (high_thr == client_info->high_thr_requested)
client_info->high_thr_set = true;
if ((client_info->state_request == ADC_TM_LOW_THR_ENABLE) ||
(client_info->state_request ==
ADC_TM_HIGH_LOW_THR_ENABLE))
if (low_thr == client_info->low_thr_requested)
client_info->low_thr_set = true;
}
ret = adc_tm5_thr_update(sensor, high_thr, low_thr);
if (ret < 0)
pr_err("setting chan:%d threshold failed\n", btm_chan_idx);
pr_debug("threshold written is high:%d and low:%d\n",
high_thr, low_thr);
return 0;
}
void notify_adc_tm_fn(struct work_struct *work)
{
struct adc_tm_client_info *client_info = NULL;
struct adc_tm_chip *chip;
struct list_head *thr_list;
uint32_t btm_chan_num = 0, btm_chan_idx = 0;
int ret = 0;
struct adc_tm_sensor *adc_tm = container_of(work,
struct adc_tm_sensor, work);
chip = adc_tm->chip;
btm_chan_num = adc_tm->btm_ch;
ret = adc_tm5_get_btm_idx(chip, btm_chan_num, &btm_chan_idx);
if (ret < 0) {
pr_err("Invalid btm channel idx\n");
return;
}
mutex_lock(&chip->adc_mutex_lock);
if (adc_tm->low_thr_triggered) {
/* adjust thr, calling manage_thr */
list_for_each(thr_list, &adc_tm->thr_list) {
client_info = list_entry(thr_list,
struct adc_tm_client_info, list);
if (client_info->low_thr_set) {
client_info->low_thr_set = false;
client_info->notify_low_thr = true;
if (client_info->state_request ==
ADC_TM_HIGH_LOW_THR_ENABLE)
client_info->state_request =
ADC_TM_HIGH_THR_ENABLE;
else
client_info->state_request =
ADC_TM_LOW_THR_DISABLE;
}
}
adc_tm5_manage_thresholds(adc_tm);
adc_tm->low_thr_triggered = false;
}
if (adc_tm->high_thr_triggered) {
/* adjust thr, calling manage_thr */
list_for_each(thr_list, &adc_tm->thr_list) {
client_info = list_entry(thr_list,
struct adc_tm_client_info, list);
if (client_info->high_thr_set) {
client_info->high_thr_set = false;
client_info->notify_high_thr = true;
if (client_info->state_request ==
ADC_TM_HIGH_LOW_THR_ENABLE)
client_info->state_request =
ADC_TM_LOW_THR_ENABLE;
else
client_info->state_request =
ADC_TM_HIGH_THR_DISABLE;
}
}
adc_tm5_manage_thresholds(adc_tm);
adc_tm->high_thr_triggered = false;
}
mutex_unlock(&chip->adc_mutex_lock);
list_for_each_entry(client_info, &adc_tm->thr_list, list) {
if (client_info->notify_low_thr) {
if (client_info->param->threshold_notification
!= NULL) {
pr_debug("notify kernel with low state\n");
client_info->param->threshold_notification(
ADC_TM_LOW_STATE,
client_info->param->btm_ctx);
client_info->notify_low_thr = false;
}
}
if (client_info->notify_high_thr) {
if (client_info->param->threshold_notification
!= NULL) {
pr_debug("notify kernel with high state\n");
client_info->param->threshold_notification(
ADC_TM_HIGH_STATE,
client_info->param->btm_ctx);
client_info->notify_high_thr = false;
}
}
}
}
int32_t adc_tm5_channel_measure(struct adc_tm_chip *chip,
struct adc_tm_param *param)
{
int ret = 0, i = 0;
uint32_t channel, dt_index = 0, btm_chan_idx = 0;
bool chan_found = false, high_thr_set = false, low_thr_set = false;
struct adc_tm_client_info *client_info = NULL;
ret = adc_tm_is_valid(chip);
if (ret || (param == NULL))
return -EINVAL;
if (param->threshold_notification == NULL) {
pr_debug("No notification for high/low temp\n");
return -EINVAL;
}
mutex_lock(&chip->adc_mutex_lock);
channel = param->channel;
while (i < chip->dt_channels) {
if (chip->sensor[i].adc_ch == channel) {
dt_index = i;
chan_found = true;
break;
}
i++;
}
if (!chan_found) {
pr_err("not a valid ADC_TM channel\n");
ret = -EINVAL;
goto fail_unlock;
}
ret = adc_tm5_get_btm_idx(chip,
chip->sensor[dt_index].btm_ch, &btm_chan_idx);
if (ret < 0) {
pr_err("Invalid btm channel idx with %d\n", ret);
goto fail_unlock;
}
/* add channel client to channel list */
adc_tm_add_to_list(chip, dt_index, param);
/* set right thresholds for the sensor */
adc_tm5_manage_thresholds(&chip->sensor[dt_index]);
/* enable low/high irqs */
list_for_each_entry(client_info,
&chip->sensor[dt_index].thr_list, list) {
if (client_info->high_thr_set == true)
high_thr_set = true;
if (client_info->low_thr_set == true)
low_thr_set = true;
}
if (low_thr_set) {
/* Enable low threshold's interrupt */
pr_debug("low sensor:%x with state:%d\n",
dt_index, param->state_request);
ret = adc_tm5_reg_update(chip,
ADC_TM_Mn_EN(btm_chan_idx),
ADC_TM_Mn_LOW_THR_INT_EN, true);
if (ret < 0) {
pr_err("low thr enable err:%d\n",
chip->sensor[dt_index].btm_ch);
goto fail_unlock;
}
}
if (high_thr_set) {
/* Enable high threshold's interrupt */
pr_debug("high sensor mask:%x with state:%d\n",
dt_index, param->state_request);
ret = adc_tm5_reg_update(chip,
ADC_TM_Mn_EN(btm_chan_idx),
ADC_TM_Mn_HIGH_THR_INT_EN, true);
if (ret < 0) {
pr_err("high thr enable err:%d\n",
chip->sensor[dt_index].btm_ch);
goto fail_unlock;
}
}
/* configure channel */
ret = adc_tm5_configure(&chip->sensor[dt_index], btm_chan_idx);
if (ret < 0) {
pr_err("Error during adc-tm configure:%d\n", ret);
goto fail_unlock;
}
ret = adc_tm5_enable(chip);
if (ret < 0)
pr_err("Error enabling adc-tm with %d\n", ret);
fail_unlock:
mutex_unlock(&chip->adc_mutex_lock);
return ret;
}
EXPORT_SYMBOL(adc_tm5_channel_measure);
int32_t adc_tm5_disable_chan_meas(struct adc_tm_chip *chip,
struct adc_tm_param *param)
{
int ret = 0, i = 0;
uint32_t channel, dt_index = 0, btm_chan_idx = 0;
unsigned long flags;
ret = adc_tm_is_valid(chip);
if (ret || (param == NULL))
return -EINVAL;
channel = param->channel;
while (i < chip->dt_channels) {
if (chip->sensor[i].adc_ch == channel) {
dt_index = i;
break;
}
i++;
}
if (i == chip->dt_channels) {
pr_err("not a valid ADC_TM channel\n");
return -EINVAL;
}
ret = adc_tm5_get_btm_idx(chip,
chip->sensor[dt_index].btm_ch, &btm_chan_idx);
if (ret < 0) {
pr_err("Invalid btm channel idx with %d\n", ret);
return ret;
}
spin_lock_irqsave(&chip->adc_tm_lock, flags);
ret = adc_tm5_reg_update(chip, ADC_TM_Mn_EN(btm_chan_idx),
ADC_TM_Mn_HIGH_THR_INT_EN, false);
if (ret < 0) {
pr_err("high thr disable err\n");
goto fail;
}
ret = adc_tm5_reg_update(chip, ADC_TM_Mn_EN(btm_chan_idx),
ADC_TM_Mn_LOW_THR_INT_EN, false);
if (ret < 0) {
pr_err("low thr disable err\n");
goto fail;
}
ret = adc_tm5_reg_update(chip, ADC_TM_Mn_EN(btm_chan_idx),
ADC_TM_Mn_MEAS_EN, false);
if (ret < 0)
pr_err("multi measurement disable failed\n");
fail:
spin_unlock_irqrestore(&chip->adc_tm_lock, flags);
return ret;
}
EXPORT_SYMBOL(adc_tm5_disable_chan_meas);
static int adc_tm5_set_mode(struct adc_tm_sensor *sensor,
enum thermal_device_mode mode)
{
struct adc_tm_chip *chip = sensor->chip;
int ret = 0;
uint32_t btm_chan_idx = 0;
ret = adc_tm5_get_btm_idx(chip, sensor->btm_ch, &btm_chan_idx);
if (ret < 0) {
pr_err("Invalid btm channel idx with %d\n", ret);
return ret;
}
if (mode == THERMAL_DEVICE_ENABLED) {
ret = adc_tm5_configure(sensor, btm_chan_idx);
if (ret < 0) {
pr_err("Error during adc-tm configure:%d\n", ret);
return ret;
}
ret = adc_tm5_enable(chip);
if (ret < 0)
pr_err("Error enabling adc-tm with %d\n", ret);
} else if (mode == THERMAL_DEVICE_DISABLED) {
ret = adc_tm5_reg_update(chip,
ADC_TM_Mn_EN(btm_chan_idx),
ADC_TM_Mn_MEAS_EN, false);
if (ret < 0)
pr_err("Disable failed for ch:%d\n", btm_chan_idx);
}
return ret;
}
static int adc_tm5_activate_trip_type(struct adc_tm_sensor *adc_tm,
int trip, enum thermal_device_mode mode)
{
struct adc_tm_chip *chip = adc_tm->chip;
int ret = 0;
bool state = false;
uint32_t btm_chan_idx = 0, btm_chan = 0;
if (mode == THERMAL_DEVICE_ENABLED)
state = true;
btm_chan = adc_tm->btm_ch;
ret = adc_tm5_get_btm_idx(chip, btm_chan, &btm_chan_idx);
if (ret < 0) {
pr_err("Invalid btm channel idx\n");
return ret;
}
switch (trip) {
case THERMAL_TRIP_CONFIGURABLE_HI:
/* low_thr (lower voltage) for higher temp */
ret = adc_tm5_reg_update(chip,
ADC_TM_Mn_EN(btm_chan_idx),
ADC_TM_Mn_LOW_THR_INT_EN, state);
if (ret)
pr_err("channel:%x failed\n", btm_chan);
break;
case THERMAL_TRIP_CONFIGURABLE_LOW:
/* high_thr (higher voltage) for cooler temp */
ret = adc_tm5_reg_update(chip,
ADC_TM_Mn_EN(btm_chan_idx),
ADC_TM_Mn_HIGH_THR_INT_EN, state);
if (ret)
pr_err("channel:%x failed\n", btm_chan);
break;
default:
return -EINVAL;
}
return ret;
}
static int adc_tm5_set_trip_temp(struct adc_tm_sensor *sensor,
int low_temp, int high_temp)
{
struct adc_tm_chip *chip;
struct adc_tm_config tm_config;
u8 trip_low_thr[2], trip_high_thr[2];
uint16_t reg_low_thr_lsb, reg_high_thr_lsb;
int ret;
uint32_t btm_chan = 0, btm_chan_idx = 0, mask = 0;
unsigned long flags;
int emul_temp;
if (!sensor)
return -EINVAL;
pr_debug("%s:low_temp(mdegC):%d, high_temp(mdegC):%d\n", __func__,
low_temp, high_temp);
chip = sensor->chip;
tm_config.channel = sensor->adc_ch;
tm_config.high_thr_temp = tm_config.low_thr_temp = 0;
if (high_temp != INT_MAX)
tm_config.high_thr_temp = high_temp;
if (low_temp != INT_MIN)
tm_config.low_thr_temp = low_temp;
if ((high_temp == INT_MAX) && (low_temp == INT_MIN)) {
pr_err("No trips to set\n");
return -EINVAL;
}
/* Set threshold to extremely value while emul temp set */
emul_temp = sensor->emul_temperature;
if (emul_temp) {
high_temp = INT_MAX;
low_temp = INT_MIN;
}
pr_debug("requested a low temp- %d and high temp- %d\n",
tm_config.low_thr_temp, tm_config.high_thr_temp);
adc_tm_scale_therm_voltage_100k(&tm_config, chip->data);
/* Cool temperature corresponds to high voltage threshold */
mask = lower_32_bits(tm_config.high_thr_voltage);
trip_high_thr[0] = ADC_TM_LOWER_MASK(mask);
trip_high_thr[1] = ADC_TM_UPPER_MASK(mask);
/* Warm temperature corresponds to low voltage threshold */
mask = lower_32_bits(tm_config.low_thr_voltage);
trip_low_thr[0] = ADC_TM_LOWER_MASK(mask);
trip_low_thr[1] = ADC_TM_UPPER_MASK(mask);
pr_debug("high_thr:0x%llx, low_thr:0x%llx\n",
tm_config.high_thr_voltage, tm_config.low_thr_voltage);
btm_chan = sensor->btm_ch;
ret = adc_tm5_get_btm_idx(chip, btm_chan, &btm_chan_idx);
if (ret < 0) {
pr_err("Invalid btm channel idx\n");
return ret;
}
spin_lock_irqsave(&chip->adc_tm_lock, flags);
reg_low_thr_lsb = ADC_TM_Mn_LOW_THR0(btm_chan_idx);
reg_high_thr_lsb = ADC_TM_Mn_HIGH_THR0(btm_chan_idx);
if (high_temp != INT_MAX) {
ret = adc_tm5_write_reg(chip, reg_low_thr_lsb,
trip_low_thr, 2);
if (ret) {
pr_err("Warm set threshold err\n");
goto fail;
}
ret = adc_tm5_activate_trip_type(sensor,
THERMAL_TRIP_CONFIGURABLE_HI,
THERMAL_DEVICE_ENABLED);
if (ret) {
pr_err("adc-tm warm activation failed\n");
goto fail;
}
} else {
ret = adc_tm5_activate_trip_type(sensor,
THERMAL_TRIP_CONFIGURABLE_HI,
THERMAL_DEVICE_DISABLED);
if (ret) {
pr_err("adc-tm warm deactivation failed\n");
goto fail;
}
}
if (low_temp != INT_MIN) {
ret = adc_tm5_write_reg(chip, reg_high_thr_lsb,
trip_high_thr, 2);
if (ret) {
pr_err("adc-tm cool temp set threshold err\n");
goto fail;
}
ret = adc_tm5_activate_trip_type(sensor,
THERMAL_TRIP_CONFIGURABLE_LOW,
THERMAL_DEVICE_ENABLED);
if (ret) {
pr_err("adc-tm cool activation failed\n");
goto fail;
}
} else {
ret = adc_tm5_activate_trip_type(sensor,
THERMAL_TRIP_CONFIGURABLE_LOW,
THERMAL_DEVICE_DISABLED);
if (ret) {
pr_err("adc-tm cool deactivation failed\n");
goto fail;
}
}
if ((high_temp != INT_MAX) || (low_temp != INT_MIN)) {
ret = adc_tm5_set_mode(sensor, THERMAL_DEVICE_ENABLED);
if (ret)
pr_err("sensor enabled failed\n");
} else {
ret = adc_tm5_set_mode(sensor, THERMAL_DEVICE_DISABLED);
if (ret)
pr_err("sensor disable failed\n");
}
fail:
spin_unlock_irqrestore(&chip->adc_tm_lock, flags);
return ret;
}
static irqreturn_t adc_tm5_handler(int irq, void *data)
{
struct adc_tm_chip *chip = data;
u8 status_low, status_high, ctl;
int ret = 0, i = 0;
unsigned long flags;
ret = adc_tm5_read_reg(chip, ADC_TM_STATUS_LOW, &status_low, 1);
if (ret < 0) {
pr_err("adc-tm-tm read status low failed with %d\n", ret);
return IRQ_HANDLED;
}
ret = adc_tm5_read_reg(chip, ADC_TM_STATUS_HIGH, &status_high, 1);
if (ret < 0) {
pr_err("adc-tm-tm read status high failed with %d\n", ret);
return IRQ_HANDLED;
}
while (i < chip->dt_channels) {
bool upper_set = false, lower_set = false;
u8 data_low = 0, data_high = 0;
u16 code = 0;
int temp;
if (!chip->sensor[i].non_thermal &&
IS_ERR(chip->sensor[i].tzd)) {
pr_err("thermal device not found\n");
i++;
continue;
}
if (!chip->sensor[i].non_thermal) {
ret = adc_tm5_get_temp(&chip->sensor[i], &temp);
if (ret < 0) {
i++;
continue;
}
ret = adc_tm5_read_reg(chip, ADC_TM_Mn_DATA0(i),
&data_low, 1);
if (ret)
pr_err("adc_tm data_low read failed with %d\n",
ret);
ret = adc_tm5_read_reg(chip, ADC_TM_Mn_DATA1(i),
&data_high, 1);
if (ret)
pr_err("adc_tm data_high read failed with %d\n",
ret);
code = ((data_high << ADC_TM_DATA_SHIFT) | data_low);
/* skip it since voltage reading is zero */
if (!code) {
pr_err("Sensor:%s voltage is zero\n",
chip->sensor[i].tzd->type);
i++;
continue;
}
}
spin_lock_irqsave(&chip->adc_tm_lock, flags);
ret = adc_tm5_read_reg(chip, ADC_TM_Mn_EN(i), &ctl, 1);
if (ret) {
pr_err("ctl read failed with %d\n", ret);
goto fail;
}
if ((status_low & 0x1) && (ctl & ADC_TM_Mn_MEAS_EN)
&& (ctl & ADC_TM_Mn_LOW_THR_INT_EN))
lower_set = true;
if ((status_high & 0x1) && (ctl & ADC_TM_Mn_MEAS_EN) &&
(ctl & ADC_TM_Mn_HIGH_THR_INT_EN))
upper_set = true;
fail:
status_low >>= 1;
status_high >>= 1;
spin_unlock_irqrestore(&chip->adc_tm_lock, flags);
if (!(upper_set || lower_set)) {
i++;
continue;
}
if (!chip->sensor[i].non_thermal) {
/*
* Expected behavior is while notifying
* of_thermal, thermal core will call set_trips
* with new thresholds and activate/disable
* the appropriate trips.
*/
pr_debug("notifying of_thermal\n");
temp = therm_fwd_scale((int64_t)code,
ADC_HC_VDD_REF, chip->data);
of_thermal_handle_trip_temp(chip->sensor[i].tzd,
temp);
} else {
if (lower_set) {
ret = adc_tm5_reg_update(chip,
ADC_TM_Mn_EN(i),
ADC_TM_Mn_LOW_THR_INT_EN,
false);
if (ret < 0) {
pr_err("low thr disable failed\n");
return IRQ_HANDLED;
}
chip->sensor[i].low_thr_triggered
= true;
queue_work(chip->sensor[i].req_wq,
&chip->sensor[i].work);
}
if (upper_set) {
ret = adc_tm5_reg_update(chip,
ADC_TM_Mn_EN(i),
ADC_TM_Mn_HIGH_THR_INT_EN,
false);
if (ret < 0) {
pr_err("high thr disable failed\n");
return IRQ_HANDLED;
}
chip->sensor[i].high_thr_triggered = true;
queue_work(chip->sensor[i].req_wq,
&chip->sensor[i].work);
}
}
i++;
}
return IRQ_HANDLED;
}
static int adc_tm5_register_interrupts(struct adc_tm_chip *chip)
{
struct platform_device *pdev;
int ret, irq;
if (!chip)
return -EINVAL;
pdev = to_platform_device(chip->dev);
irq = platform_get_irq_byname(pdev, "thr-int-en");
if (irq < 0) {
dev_err(&pdev->dev, "failed to get irq %s\n",
"thr-int-en");
return irq;
}
ret = devm_request_threaded_irq(&pdev->dev, irq, NULL,
adc_tm5_handler,
IRQF_TRIGGER_RISING | IRQF_ONESHOT,
"thr-int-en", chip);
if (ret) {
dev_err(&pdev->dev, "failed to get irq %s\n",
"thr-int-en");
return ret;
}
enable_irq_wake(irq);
return ret;
}
static int adc_tm5_init(struct adc_tm_chip *chip, uint32_t dt_chans)
{
u8 buf[4], channels_available, meas_int_timer_2_3 = 0;
int ret;
int dig_param_len = 4;
int pmic_subtype_660 = 0;
unsigned int offset_btm_idx = 0, i;
if ((chip->pmic_rev_id) &&
(chip->pmic_rev_id->pmic_subtype == PM660_SUBTYPE)) {
dig_param_len = 2;
pmic_subtype_660 = 1;
} else {
ret = adc_tm5_read_reg(chip, ADC_TM_NUM_BTM,
&channels_available, 1);
if (ret < 0) {
pr_err("read failed for BTM channels\n");
return ret;
}
if (dt_chans > channels_available) {
pr_err("More nodes than channels supported:%d\n",
channels_available);
return -EINVAL;
}
}
ret = adc_tm5_read_reg(chip,
ADC_TM_ADC_DIG_PARAM, buf, dig_param_len);
if (ret < 0) {
pr_err("adc-tm block read failed with %d\n", ret);
return ret;
}
/* Select decimation */
buf[0] = chip->prop.decimation;
/* Select number of samples in fast average mode */
buf[1] = chip->prop.fast_avg_samples | ADC_TM_FAST_AVG_EN;
/* Select timer1 */
buf[2] = chip->prop.timer1;
/* Select timer2 and timer3 */
meas_int_timer_2_3 |= chip->prop.timer2 <<
ADC_TM_MEAS_INTERVAL_CTL2_SHIFT;
meas_int_timer_2_3 |= chip->prop.timer3;
buf[3] = meas_int_timer_2_3;
ret = adc_tm5_write_reg(chip,
ADC_TM_ADC_DIG_PARAM, buf, dig_param_len);
if (ret < 0)
pr_err("adc-tm block write failed with %d\n", ret);
if (pmic_subtype_660) {
ret = adc_tm5_write_reg(chip,
ADC_TM_MEAS_INTERVAL_CTL_660, &buf[2], 2);
if (ret < 0)
pr_err("adc-tm block write failed with %d\n", ret);
}
spin_lock_init(&chip->adc_tm_lock);
mutex_init(&chip->adc_mutex_lock);
if (chip->pmic_rev_id) {
switch (chip->pmic_rev_id->pmic_subtype)
/*
* PM8150B 1.0 CH_0 and CH_1 is already used.
* Therefore configure to use CH_2 onwards.
*/
case PM8150B_SUBTYPE:
if (chip->pmic_rev_id->rev4 == PM8150B_V1P0_REV4)
offset_btm_idx = ADC_TM_CHAN2;
}
for (i = 0; i < dt_chans; i++) {
if ((i + offset_btm_idx) > ADC_TM_CHAN7) {
pr_err("Invalid BTM index %d\n", (i + offset_btm_idx));
return -EINVAL;
}
chip->sensor[i].btm_ch =
adc_tm_ch_data[i + offset_btm_idx].btm_amux_ch;
}
return ret;
}
static const struct adc_tm_ops ops_adc_tm5 = {
.init = adc_tm5_init,
.set_trips = adc_tm5_set_trip_temp,
.interrupts_reg = adc_tm5_register_interrupts,
.get_temp = adc_tm5_get_temp,
.set_emul_temp = adc_tm5_set_emul_temp,
};
const struct adc_tm_data data_adc_tm5 = {
.ops = &ops_adc_tm5,
.full_scale_code_volt = 0x70e4,
.decimation = (unsigned int []) {250, 420, 840},
.hw_settle = (unsigned int []) {15, 100, 200, 300, 400, 500, 600, 700,
1, 2, 4, 8, 16, 32, 64, 128},
};
const struct adc_tm_data data_adc_tm_rev2 = {
.ops = &ops_adc_tm5,
.full_scale_code_volt = 0x4000,
.decimation = (unsigned int []) {256, 512, 1024},
.hw_settle = (unsigned int []) {0, 100, 200, 300, 400, 500, 600, 700,
800, 900, 1, 2, 4, 6, 8, 10},
};