android_kernel_xiaomi_sm8350/drivers/hwmon/qti_amoled_ecm.c
Fenglin Wu ba6fb533f2 hwmon: qti_amoled_ecm: return avg_current as 0 upon error exceptions
Avg_current is an unsigned 16-bit data and assigning it a negative
value would make it being shown as a very big positive value which
doesn't make sense for ECM current measurement. Instead, assign it
to 0 upon error exceptions and print out corresponding messages.

Change-Id: I438667e40d63e4142860b7de4e7cf7495d3c424e
Signed-off-by: Fenglin Wu <fenglinw@codeaurora.org>
2021-03-09 09:35:43 +08:00

952 lines
22 KiB
C

// SPDX-License-Identifier: GPL-2.0-only
/*
* Copyright (c) 2020-2021, The Linux Foundation. All rights reserved.
*/
#define pr_fmt(fmt) "AMOLED_ECM: %s: " fmt, __func__
#include <linux/device.h>
#include <linux/errno.h>
#include <linux/hwmon.h>
#include <linux/hwmon-sysfs.h>
#include <linux/interrupt.h>
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/nvmem-consumer.h>
#include <linux/of.h>
#include <linux/of_irq.h>
#include <linux/platform_device.h>
#include <linux/regmap.h>
#include <linux/workqueue.h>
#include <drm/drm_panel.h>
/* AMOLED AB register definitions */
#define AB_REVISION2 0x01
/* AMOLED ECM register definitions */
#define AB_ECM_EN_CTL 0xA0
#define ECM_EN BIT(7)
#define AB_ECM_COUNTER_CTL 0xA1
#define ECM_COUNTER_START BIT(7)
/* AMOLED ECM SDAM Offsets */
#define ECM_SDAM_START_BASE 0x40
#define ECM_SDAMX_SAMPLE_START_ADDR 0x46
#define ECM_FAULT_LOG 0x48
#define ECM_ROUTINE_LOG 0x49
#define ECM_ACTIVE_SDAM 0x4D
#define ECM_SDAM0_ACTIVE BIT(0)
#define ECM_SDAM1_ACTIVE BIT(1)
#define ECM_SAMPLE_CNT_LSB 0x4E
#define ECM_SAMPLE_CNT_MSB 0x4F
#define ECM_STATUS_SET 0x50
#define ECM_STATUS_CLR 0x51
#define ECM_ONGOING BIT(0)
#define ECM_DONE BIT(1)
#define ECM_ABORT BIT(2)
#define ECM_SDAM0_FULL BIT(3)
#define ECM_SDAM1_FULL BIT(4)
#define ECM_SDAM0_INDEX 0x52
#define ECM_SDAM1_INDEX 0x53
#define ECM_MODE 0x54
#define ECM_CONTINUOUS 0
#define ECM_N_ESWIRE 1
#define ECM_M_ASWIRE 2
#define ECM_ESWIRE_ASWIRE 3
#define ECM_USE_TIMER 4
#define ECM_N_ESWIRE_COUNT_LSB 0x55
#define ECM_N_ESWIRE_COUNT_MSB 0x56
#define ECM_M_ASWIRE_COUNT_LSB 0x57
#define ECM_M_ASWIRE_COUNT_MSB 0x58
#define ECM_ESWIRE_ASWIRE_SKIP_COUNT_LSB 0x59
#define ECM_ESWIRE_ASWIRE_SKIP_COUNT_MSB 0x5A
#define ECM_TIMER_LSB 0x5B
#define ECM_TIMER_MSB 0x5C
#define ECM_TIMER_SKIP_LSB 0x5D
#define ECM_TIMER_SKIP_MSB 0x5E
#define ECM_SEND_IRQ 0x5F
#define SEND_SDAM0_IRQ BIT(0)
#define SEND_SDAM1_IRQ BIT(1)
#define ECM_WRITE_TO_SDAM 0x60
#define WRITE_SDAM0_DATA BIT(0)
#define WRITE_SDAM1_DATA BIT(1)
#define OVERWRITE_SDAM0_DATA BIT(4)
#define OVERWRITE_SDAM1_DATA BIT(5)
#define ECM_AVERAGE_LSB 0x61
#define ECM_AVERAGE_MSB 0x62
#define ECM_MIN_LSB 0x63
#define ECM_MIN_MSB 0x64
#define ECM_MAX_LSB 0x65
#define ECM_MAX_MSB 0x66
#define ECM_SDAM0_SAMPLE_START_ADDR 0x6C
#define ECM_SDAM_SAMPLE_END_ADDR 0xBF
/* ECM specific definitions */
#define ECM_SAMPLE_GAIN_V1 15
#define ECM_SAMPLE_GAIN_V2 16
#define ECM_MIN_M_SAMPLES 10
#define AMOLED_AB_REVISION_1P0 0
#define AMOLED_AB_REVISION_2P0 1
enum amoled_ecm_mode {
ECM_MODE_CONTINUOUS = 0,
ECM_MODE_MULTI_FRAMES,
ECM_MODE_IDLE,
};
struct amoled_ecm_sdam_config {
u8 reg;
u8 reset_val;
};
/**
* struct amoled_ecm_sdam - AMOLED ECM sdam data structure
* @nvmem: Pointer to nvmem device
* @start_addr: Start address of ECM samples in SDAM
* @irq_name: Interrupt name for SDAM
* @irq: Interrupt associated with the SDAM
*/
struct amoled_ecm_sdam {
struct nvmem_device *nvmem;
u32 start_addr;
const char *irq_name;
int irq;
};
/**
* struct amoled_ecm_data - Structure for AMOLED ECM data
* @m_cumulative: Cumulative of M sample values
* @num_m_samples: Number of M samples available
* @time_period_ms: Time period(in milli seconds) for ECM request
* @frames: Number of frames for ECM request
* @avg_current: AMOLED ECM average calculated
* @mode: AMOLED ECM mode of operation
*/
struct amoled_ecm_data {
unsigned long long m_cumulative;
u32 num_m_samples;
u32 time_period_ms;
u16 frames;
u16 avg_current;
enum amoled_ecm_mode mode;
};
/**
* struct amoled ecm - Structure for AMOLED ECM device
* @regmap: Pointer for regmap structure
* @dev: Pointer for AMOLED ECM device
* @data: AMOLED ECM data structure
* @sdam: Pointer for array of ECM sdams
* @sdam_lock: Locking for mutual exclusion
* @average_work: Delayed work to calculate ECM average
* @drm_notifier: The notifier block for receiving DRM notifications
* @active_panel: Active DRM panel which sends DRM notifications
* @num_sdams: Number of SDAMs used for AMOLED ECM
* @base: Base address of the AMOLED ECM module
* @ab_revision: Revision of the AMOLED AB module
* @enable: Flag to enable/disable AMOLED ECM
* @abort: Flag to indicated AMOLED ECM has aborted
* @reenable: Flag to reenable ECM when display goes unblank
*/
struct amoled_ecm {
struct regmap *regmap;
struct device *dev;
struct amoled_ecm_data data;
struct amoled_ecm_sdam *sdam;
struct mutex sdam_lock;
struct delayed_work average_work;
struct notifier_block drm_notifier;
struct drm_panel *active_panel;
u32 num_sdams;
u32 base;
u8 ab_revision;
bool enable;
bool abort;
bool reenable;
};
static struct amoled_ecm_sdam_config ecm_reset_config[] = {
{ ECM_FAULT_LOG, 0x00 },
{ ECM_ROUTINE_LOG, 0x00 },
{ ECM_ACTIVE_SDAM, 0x01 },
{ ECM_SAMPLE_CNT_LSB, 0x00 },
{ ECM_SAMPLE_CNT_MSB, 0x00 },
{ ECM_STATUS_SET, 0x00 },
{ ECM_STATUS_CLR, 0xFF },
{ ECM_SDAM0_INDEX, 0x6C },
{ ECM_SDAM1_INDEX, 0x46 },
{ ECM_MODE, 0x00 },
/* Valid only when ECM uses 2 SDAMs */
{ ECM_SEND_IRQ, 0x03 },
{ ECM_WRITE_TO_SDAM, 0x03 }
};
static int ecm_nvmem_device_write(struct nvmem_device *nvmem,
unsigned int offset,
size_t bytes, void *buf)
{
size_t i;
u8 *ptr = buf;
for (i = 0; i < bytes; i++)
pr_debug("Wrote %#x to %#x\n", *ptr++, offset + i);
return nvmem_device_write(nvmem, offset, bytes, buf);
}
static int ecm_reset_sdam_config(struct amoled_ecm *ecm)
{
int rc, i;
for (i = 0; i < ARRAY_SIZE(ecm_reset_config); i++) {
rc = ecm_nvmem_device_write(ecm->sdam[0].nvmem,
ecm_reset_config[i].reg,
1, &ecm_reset_config[i].reset_val);
if (rc < 0) {
pr_err("Failed to write %u to SDAM, rc=%d\n",
ecm_reset_config[i].reg, rc);
return rc;
}
}
usleep_range(10000, 12000);
return rc;
}
static int amoled_ecm_enable(struct amoled_ecm *ecm)
{
struct amoled_ecm_data *data = &ecm->data;
int rc;
if (data->frames) {
rc = ecm_nvmem_device_write(ecm->sdam[0].nvmem,
ECM_N_ESWIRE_COUNT_LSB, 2, &data->frames);
if (rc < 0) {
pr_err("Failed to write swire count to SDAM, rc=%d\n",
rc);
return rc;
}
data->mode = ECM_MODE_MULTI_FRAMES;
} else {
if (!data->time_period_ms)
return -EINVAL;
data->mode = ECM_MODE_CONTINUOUS;
}
if ((ecm->ab_revision != AMOLED_AB_REVISION_1P0) &&
(ecm->ab_revision != AMOLED_AB_REVISION_2P0)) {
pr_err("ECM is not supported for AB version %u\n",
ecm->ab_revision);
return -ENODEV;
}
rc = ecm_reset_sdam_config(ecm);
if (rc < 0) {
pr_err("Failed to reset ECM SDAM configuration, rc=%d\n", rc);
return rc;
}
rc = ecm_nvmem_device_write(ecm->sdam[0].nvmem, ECM_MODE, 1,
&data->mode);
if (rc < 0) {
pr_err("Failed to write ECM mode to SDAM, rc=%d\n", rc);
return rc;
}
rc = regmap_write(ecm->regmap, ecm->base + AB_ECM_EN_CTL, ECM_EN);
if (rc < 0) {
pr_err("Failed to enable ECM, rc=%d\n", rc);
return rc;
}
rc = regmap_write(ecm->regmap, ecm->base + AB_ECM_COUNTER_CTL,
ECM_COUNTER_START);
if (rc < 0) {
pr_err("Failed to enable ECM counter, rc=%d\n", rc);
return rc;
}
if (data->mode == ECM_MODE_CONTINUOUS)
schedule_delayed_work(&ecm->average_work,
msecs_to_jiffies(data->time_period_ms));
ecm->enable = true;
return rc;
}
static int amoled_ecm_disable(struct amoled_ecm *ecm)
{
int rc;
rc = regmap_write(ecm->regmap, ecm->base + AB_ECM_COUNTER_CTL, 0);
if (rc < 0) {
pr_err("Failed to disable ECM counter, rc=%d\n", rc);
return rc;
}
rc = regmap_write(ecm->regmap, ecm->base + AB_ECM_EN_CTL, 0);
if (rc < 0) {
pr_err("Failed to disable ECM, rc=%d\n", rc);
return rc;
}
rc = ecm_nvmem_device_write(ecm->sdam[0].nvmem, ECM_AVERAGE_LSB, 2,
&ecm->data.avg_current);
if (rc < 0) {
pr_err("Failed to write ECM average to SDAM, rc=%d\n", rc);
return rc;
}
pr_debug("ECM_AVERAGE:%u\n", ecm->data.avg_current);
cancel_delayed_work(&ecm->average_work);
ecm->data.avg_current = 0;
ecm->data.m_cumulative = 0;
ecm->data.num_m_samples = 0;
ecm->data.mode = ECM_MODE_IDLE;
ecm->abort = false;
ecm->enable = false;
return rc;
}
static void ecm_average_work(struct work_struct *work)
{
struct amoled_ecm *ecm = container_of(work, struct amoled_ecm,
average_work.work);
struct amoled_ecm_data *data = &ecm->data;
mutex_lock(&ecm->sdam_lock);
if (!data->num_m_samples || !data->m_cumulative) {
pr_warn_ratelimited("Invalid data, num_m_samples=%u m_cumulative:%u\n",
data->num_m_samples, data->m_cumulative);
data->avg_current = 0;
} else {
data->avg_current = data->m_cumulative / data->num_m_samples;
pr_debug("avg_current=%u mA\n", data->avg_current);
}
data->m_cumulative = 0;
data->num_m_samples = 0;
mutex_unlock(&ecm->sdam_lock);
/*
* If ECM is not aborted and still enabled, run it one more time
*/
if (!ecm->abort && ecm->enable)
schedule_delayed_work(&ecm->average_work,
msecs_to_jiffies(ecm->data.time_period_ms));
}
static ssize_t enable_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
struct amoled_ecm *ecm = dev_get_drvdata(dev);
return scnprintf(buf, PAGE_SIZE, "%d\n", ecm->enable);
}
static ssize_t enable_store(struct device *dev,
struct device_attribute *attr,
const char *buf, size_t count)
{
struct amoled_ecm *ecm = dev_get_drvdata(dev);
bool val;
int rc;
rc = kstrtobool(buf, &val);
if (rc < 0)
return rc;
if (ecm->enable == val) {
pr_err("AMOLED ECM is already %s\n",
val ? "enabled" : "disabled");
return -EINVAL;
}
if (val) {
rc = amoled_ecm_enable(ecm);
if (rc < 0) {
pr_err("Failed to enable AMOLED ECM, rc=%d\n", rc);
return rc;
}
} else {
rc = amoled_ecm_disable(ecm);
if (rc < 0) {
pr_err("Failed to disable AMOLED ECM, rc=%d\n", rc);
return rc;
}
ecm->data.frames = 0;
ecm->data.time_period_ms = 0;
}
return count;
}
static DEVICE_ATTR_RW(enable);
static ssize_t frames_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
struct amoled_ecm *ecm = dev_get_drvdata(dev);
return scnprintf(buf, PAGE_SIZE, "%u\n", ecm->data.frames);
}
static ssize_t frames_store(struct device *dev,
struct device_attribute *attr,
const char *buf, size_t count)
{
struct amoled_ecm *ecm = dev_get_drvdata(dev);
u16 val;
int rc;
if (ecm->enable) {
pr_err("Failed to configure frames, ECM is already running\n");
return -EINVAL;
}
rc = kstrtou16(buf, 0, &val);
if ((rc < 0) || !val)
return -EINVAL;
ecm->data.frames = val;
return count;
}
static DEVICE_ATTR_RW(frames);
static ssize_t time_period_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
struct amoled_ecm *ecm = dev_get_drvdata(dev);
return scnprintf(buf, PAGE_SIZE, "%u\n", ecm->data.time_period_ms);
}
static ssize_t time_period_store(struct device *dev,
struct device_attribute *attr,
const char *buf, size_t count)
{
struct amoled_ecm *ecm = dev_get_drvdata(dev);
u32 val;
int rc;
if (ecm->enable) {
pr_err("Failed to configure time_period, ECM is already running\n");
return -EINVAL;
}
rc = kstrtou32(buf, 0, &val);
if ((rc < 0) || !val)
return -EINVAL;
ecm->data.time_period_ms = val;
return count;
}
static DEVICE_ATTR_RW(time_period);
static ssize_t avg_current_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
struct amoled_ecm *ecm = dev_get_drvdata(dev);
return scnprintf(buf, PAGE_SIZE, "%u\n", ecm->data.avg_current);
}
static DEVICE_ATTR_RO(avg_current);
static struct attribute *amoled_ecm_attrs[] = {
&dev_attr_enable.attr,
&dev_attr_frames.attr,
&dev_attr_time_period.attr,
&dev_attr_avg_current.attr,
NULL,
};
static const struct attribute_group amoled_ecm_group = {
.name = "amoled_ecm",
.attrs = amoled_ecm_attrs,
};
__ATTRIBUTE_GROUPS(amoled_ecm);
static int get_sdam_from_irq(struct amoled_ecm *ecm, int irq)
{
int i;
for (i = 0; i < ecm->num_sdams; i++)
if (ecm->sdam[i].irq == irq)
return i;
return -ENOENT;
}
static int handle_ecm_abort(struct amoled_ecm *ecm)
{
struct amoled_ecm_data *data = &ecm->data;
int rc;
u8 mode = data->mode;
switch (mode) {
case ECM_MODE_MULTI_FRAMES:
pr_warn_ratelimited("Multiple frames mode is not supported\n");
data->avg_current = 0;
break;
case ECM_MODE_CONTINUOUS:
if (data->num_m_samples < ECM_MIN_M_SAMPLES) {
pr_warn_ratelimited("Too few samples %u for continuous mode\n",
data->num_m_samples);
data->avg_current = 0;
break;
}
ecm->abort = true;
schedule_delayed_work(&ecm->average_work, 0);
break;
default:
pr_err_ratelimited("Invalid ECM operation mode: %u\n", mode);
data->avg_current = 0;
return -EINVAL;
}
rc = amoled_ecm_disable(ecm);
if (rc < 0)
pr_err("Failed to disable AMOLED ECM, rc=%d\n", rc);
return rc;
}
static irqreturn_t sdam_full_irq_handler(int irq, void *_ecm)
{
struct amoled_ecm *ecm = _ecm;
struct amoled_ecm_data *data = &ecm->data;
u64 cumulative = 0, m_sample;
int rc, i, sdam_num, sdam_start, num_ecm_samples, max_samples;
u16 ecm_sample, gain;
u8 buf[2], int_status, sdam_index, overwrite;
sdam_num = get_sdam_from_irq(ecm, irq);
if (sdam_num < 0) {
pr_err("Invalid SDAM interrupt, err=%d\n", sdam_num);
return IRQ_HANDLED;
}
rc = nvmem_device_read(ecm->sdam[0].nvmem, ECM_STATUS_SET, 1,
&int_status);
if (rc < 0) {
pr_err("Failed to read interrupt status from SDAM, rc=%d\n",
rc);
return IRQ_HANDLED;
}
pr_debug("ECM_STATUS_SET: %#x\n", int_status);
if (data->mode != ECM_MODE_CONTINUOUS &&
data->mode != ECM_MODE_MULTI_FRAMES)
return IRQ_HANDLED;
if (int_status & ECM_ABORT) {
rc = handle_ecm_abort(ecm);
if (rc < 0) {
pr_err("Failed to handle ECM_ABORT interrupt, rc=%d\n",
rc);
return IRQ_HANDLED;
}
}
rc = nvmem_device_read(ecm->sdam[0].nvmem,
(ECM_SDAM0_INDEX + sdam_num), 1, &sdam_index);
if (rc < 0) {
pr_err("Failed to read SDAM index, rc=%d\n", rc);
goto irq_exit;
}
pr_debug("sdam_num:%d sdam_index:%#x\n", sdam_num, sdam_index);
sdam_start = ecm->sdam[sdam_num].start_addr;
max_samples = (ECM_SDAM_SAMPLE_END_ADDR + 1 - sdam_start) / 2;
num_ecm_samples = (sdam_index + 1 - sdam_start) / 2;
if (!num_ecm_samples || (num_ecm_samples > max_samples)) {
pr_err("Incorrect number of ECM samples, num_ecm_samples:%d max_samples:%d\n",
num_ecm_samples, max_samples);
return IRQ_HANDLED;
}
mutex_lock(&ecm->sdam_lock);
rc = nvmem_device_read(ecm->sdam[0].nvmem, ECM_WRITE_TO_SDAM, 1,
&overwrite);
if (rc < 0) {
pr_err("Failed to read ECM_WRITE_TO_SDAM from SDAM, rc=%d\n",
rc);
goto irq_exit;
}
overwrite &= ~(OVERWRITE_SDAM0_DATA << sdam_num);
rc = ecm_nvmem_device_write(ecm->sdam[0].nvmem, ECM_WRITE_TO_SDAM,
1, &overwrite);
if (rc < 0) {
pr_err("Failed to write ECM_WRITE_TO_SDAM to SDAM, rc=%d\n",
rc);
goto irq_exit;
}
/*
* For AMOLED AB peripheral,
* Revision 1.0:
* ECM measured current = 15 times of each LSB
*
* Revision 2.0:
* ECM measured current = 16 times of each LSB
*/
if (ecm->ab_revision == AMOLED_AB_REVISION_1P0)
gain = ECM_SAMPLE_GAIN_V1;
else
gain = ECM_SAMPLE_GAIN_V2;
for (i = sdam_start; i < sdam_index; i += 2) {
rc = nvmem_device_read(ecm->sdam[sdam_num].nvmem, i, 2, buf);
if (rc < 0) {
pr_err("Failed to read SDAM sample, rc=%d\n", rc);
goto irq_exit;
}
ecm_sample = (buf[1] << 8) | buf[0];
cumulative += ((ecm_sample * 1000) / gain) / 1000;
}
overwrite |= (OVERWRITE_SDAM0_DATA << sdam_num);
rc = ecm_nvmem_device_write(ecm->sdam[0].nvmem, ECM_WRITE_TO_SDAM,
1, &overwrite);
if (rc < 0) {
pr_err("Failed to write ECM_WRITE_TO_SDAM to SDAM, rc=%d\n",
rc);
goto irq_exit;
}
if (!cumulative) {
pr_err("Error, No ECM samples captured. Cumulative:%lu\n",
cumulative);
goto irq_exit;
}
m_sample = cumulative / num_ecm_samples;
data->m_cumulative += m_sample;
data->num_m_samples++;
buf[0] = (ECM_SDAM0_FULL << sdam_num);
rc = ecm_nvmem_device_write(ecm->sdam[0].nvmem, ECM_STATUS_CLR, 1,
&buf[0]);
if (rc < 0) {
pr_err("Failed to clear interrupt status in SDAM, rc=%d\n",
rc);
goto irq_exit;
}
if ((data->mode == ECM_MODE_MULTI_FRAMES) &&
(sdam_index < max_samples))
schedule_delayed_work(&ecm->average_work, 0);
irq_exit:
mutex_unlock(&ecm->sdam_lock);
return IRQ_HANDLED;
}
#ifdef CONFIG_DRM
static int amoled_ecm_parse_panel_dt(struct amoled_ecm *ecm)
{
struct device_node *np = ecm->dev->of_node;
struct device_node *pnode;
struct drm_panel *panel;
int i, count;
count = of_count_phandle_with_args(np, "display-panels", NULL);
if (count <= 0)
return 0;
for (i = 0; i < count; i++) {
pnode = of_parse_phandle(np, "display-panels", i);
panel = of_drm_find_panel(pnode);
of_node_put(pnode);
if (!IS_ERR(panel)) {
ecm->active_panel = panel;
return 0;
}
}
return PTR_ERR(panel);
}
#else
static inline int amoled_ecm_parse_panel_dt(struct amoled_ecm *ecm)
{
return 0;
}
#endif
static int amoled_ecm_parse_dt(struct amoled_ecm *ecm)
{
int rc = 0, i;
u32 val;
u8 buf[20];
rc = of_property_read_u32(ecm->dev->of_node, "reg", &val);
if (rc < 0) {
pr_err("Failed to get reg, rc = %d\n", rc);
return rc;
}
ecm->base = val;
rc = of_property_count_strings(ecm->dev->of_node, "nvmem-names");
if (rc < 0) {
pr_err("Could not find nvmem device\n");
return rc;
}
ecm->num_sdams = rc;
ecm->sdam = devm_kcalloc(ecm->dev, ecm->num_sdams,
sizeof(*ecm->sdam), GFP_KERNEL);
if (!ecm->sdam)
return -ENOMEM;
for (i = 0; i < ecm->num_sdams; i++) {
scnprintf(buf, sizeof(buf), "ecm-sdam%d", i);
rc = of_irq_get_byname(ecm->dev->of_node, buf);
if (rc < 0) {
pr_err("Failed to get irq for ecm sdam, err=%d\n", rc);
return -EINVAL;
}
ecm->sdam[i].irq_name = devm_kstrdup(ecm->dev, buf,
GFP_KERNEL);
if (!ecm->sdam[i].irq_name)
return -ENOMEM;
ecm->sdam[i].irq = rc;
scnprintf(buf, sizeof(buf), "amoled-ecm-sdam%d", i);
ecm->sdam[i].nvmem = devm_nvmem_device_get(ecm->dev, buf);
if (IS_ERR(ecm->sdam[i].nvmem)) {
rc = PTR_ERR(ecm->sdam[i].nvmem);
if (rc != -EPROBE_DEFER)
pr_err("Failed to get nvmem device, rc=%d\n",
rc);
ecm->sdam[i].nvmem = NULL;
return rc;
}
}
rc = amoled_ecm_parse_panel_dt(ecm);
if (rc && rc != -EPROBE_DEFER)
pr_err("failed to get active panel, rc=%d\n", rc);
return rc;
}
#ifdef CONFIG_DRM
static int drm_notifier_callback(struct notifier_block *nb,
unsigned long event, void *data)
{
struct amoled_ecm *ecm = container_of(nb,
struct amoled_ecm, drm_notifier);
struct drm_panel_notifier *evdata = data;
int blank, rc;
pr_debug("DRM event received: %d\n", event);
if (event != DRM_PANEL_EVENT_BLANK)
return NOTIFY_DONE;
blank = *(int *)evdata->data;
if (blank == DRM_PANEL_BLANK_POWERDOWN && ecm->enable) {
rc = amoled_ecm_disable(ecm);
if (rc < 0) {
pr_err("Failed to disable ECM for display BLANK, rc=%d\n",
rc);
return rc;
}
ecm->reenable = true;
pr_debug("Disabled ECM for display BLANK\n");
} else if (blank == DRM_PANEL_BLANK_UNBLANK && ecm->reenable) {
rc = amoled_ecm_enable(ecm);
if (rc < 0) {
pr_err("Failed to reenable ECM for display UNBLANK, rc=%d\n",
rc);
return rc;
}
ecm->reenable = false;
pr_debug("Enabled ECM for display UNBLANK\n");
}
return NOTIFY_DONE;
}
static int qti_amoled_register_drm_notifier(struct amoled_ecm *ecm)
{
int rc = 0;
if (ecm->active_panel) {
ecm->drm_notifier.notifier_call = drm_notifier_callback;
rc = drm_panel_notifier_register(ecm->active_panel,
&ecm->drm_notifier);
if (rc < 0)
pr_err("failed to register DRM notifier, rc=%d\n", rc);
}
return rc;
}
static int qti_amoled_unregister_drm_notifier(struct amoled_ecm *ecm)
{
if (ecm->active_panel)
return drm_panel_notifier_unregister(ecm->active_panel,
&ecm->drm_notifier);
return 0;
}
#else
static inline int drm_notifier_callback(struct notifier_block *nb,
unsigned long event, void *data)
{
return NOTIFY_DONE;
}
static inline int qti_amoled_register_drm_notifier(struct amoled_ecm *ecm)
{
return 0;
}
static inline int qti_amoled_unregister_drm_notifier(struct amoled_ecm *ecm)
{
return 0;
}
#endif
static int qti_amoled_ecm_probe(struct platform_device *pdev)
{
struct device *hwmon_dev;
struct amoled_ecm *ecm;
int rc, i;
unsigned int temp;
ecm = devm_kzalloc(&pdev->dev, sizeof(*ecm), GFP_KERNEL);
if (!ecm)
return -ENOMEM;
ecm->dev = &pdev->dev;
ecm->regmap = dev_get_regmap(pdev->dev.parent, NULL);
if (!ecm->regmap) {
dev_err(&pdev->dev, "Failed to get regmap\n");
return -EINVAL;
}
rc = amoled_ecm_parse_dt(ecm);
if (rc < 0) {
if (rc != -EPROBE_DEFER)
dev_err(&pdev->dev, "Failed to parse AMOLED ECM rc=%d\n",
rc);
return rc;
}
rc = regmap_read(ecm->regmap, ecm->base + AB_REVISION2, &temp);
if (rc < 0) {
dev_err(&pdev->dev, "Failed to read AB revision, rc=%d\n", rc);
return rc;
}
ecm->ab_revision = temp;
ecm->enable = false;
ecm->abort = false;
ecm->data.m_cumulative = 0;
ecm->data.num_m_samples = 0;
ecm->data.time_period_ms = 0;
ecm->data.frames = 0;
ecm->data.avg_current = 0;
ecm->data.mode = ECM_MODE_IDLE;
INIT_DELAYED_WORK(&ecm->average_work, ecm_average_work);
mutex_init(&ecm->sdam_lock);
dev_set_drvdata(ecm->dev, ecm);
for (i = 0; i < ecm->num_sdams; i++) {
rc = devm_request_threaded_irq(ecm->dev, ecm->sdam[i].irq,
NULL, sdam_full_irq_handler,
IRQF_ONESHOT, ecm->sdam[i].irq_name, ecm);
if (rc < 0) {
dev_err(&pdev->dev, "Failed to request IRQ(%s), rc=%d\n",
ecm->sdam[i].irq_name, rc);
return rc;
}
ecm->sdam[i].start_addr = i ? ECM_SDAMX_SAMPLE_START_ADDR
: ECM_SDAM0_SAMPLE_START_ADDR;
}
hwmon_dev = devm_hwmon_device_register_with_groups(&pdev->dev,
"amoled_ecm", ecm, amoled_ecm_groups);
if (IS_ERR_OR_NULL(hwmon_dev)) {
rc = PTR_ERR(hwmon_dev);
pr_err("failed to register hwmon device for amoled-ecm, rc=%d\n",
rc);
return rc;
}
return qti_amoled_register_drm_notifier(ecm);
}
static int qti_amoled_ecm_remove(struct platform_device *pdev)
{
struct amoled_ecm *ecm = dev_get_drvdata(&pdev->dev);
return qti_amoled_unregister_drm_notifier(ecm);
}
static const struct of_device_id amoled_ecm_match_table[] = {
{ .compatible = "qcom,amoled-ecm", },
{ },
};
static struct platform_driver qti_amoled_ecm_driver = {
.driver = {
.name = "qti_amoled_ecm",
.of_match_table = amoled_ecm_match_table,
},
.probe = qti_amoled_ecm_probe,
.remove = qti_amoled_ecm_remove,
};
module_platform_driver(qti_amoled_ecm_driver);
MODULE_DESCRIPTION("QTI AMOLED ECM driver");
MODULE_LICENSE("GPL v2");