blob: a613ac1cd891eb3fabcaf89440809116aeba78a6 [file] [log] [blame]
/* SPDX-License-Identifier: GPL-2.0 */
/*
* Copyright (C) 2020, Google Inc
*
* MAX77759 MAXQ opcode management.
*/
#include <linux/completion.h>
#include <linux/delay.h>
#include <linux/gpio.h>
#include <linux/gpio/consumer.h>
#include <linux/interrupt.h>
#include <linux/i2c.h>
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/mutex.h>
#include <linux/of_gpio.h>
#include <linux/regmap.h>
#include <misc/logbuffer.h>
#include "max77759_maxq.h"
#include "max77759_regs.h"
#include "gbms_storage.h"
#define MAX_GPIO_TRIG_RISING 0
#define MAX_GPIO_TRIG_FALLING 1
#define MAX_GPIO5_TRIG_MASK BIT(0)
#define MAX_GPIO5_TRIG(x) ((x) << 0)
#define MAX_GPIO6_TRIG_MASK BIT(1)
#define MAX_GPIO6_TRIG(x) ((x) << 1)
#define PAYLOAD_REQUEST_LENGTH_BYTES 33
#define PAYLOAD_RESPONSE_LENGTH_BYTES 3
#define OPCODE_GPIO_TRIGGER_READ 0x21
#define OPCODE_GPIO_TRIGGER_R_REQ_LEN 1
#define OPCODE_GPIO_TRIGGER_R_RES_LEN 2
#define OPCODE_GPIO_TRIGGER_R_RES_OFFSET 1
#define OPCODE_GPIO_TRIGGER_WRITE 0x22
#define OPCODE_GPIO_TRIGGER_W_REQ_LEN 2
#define OPCODE_GPIO_TRIGGER_W_REQ_OFFSET 1
#define OPCODE_GPIO_TRIGGER_W_RES_LEN 1
#define OPCODE_GPIO_CONTROL_READ 0x23
#define OPCODE_GPIO_CONTROL_R_REQ_LEN 1
#define OPCODE_GPIO_CONTROL_R_RES_LEN 2
#define OPCODE_GPIO_CONTROL_R_RES_OFFSET 1
#define OPCODE_GPIO_CONTROL_WRITE 0x24
#define OPCODE_GPIO_CONTROL_W_REQ_LEN 2
#define OPCODE_GPIO_CONTROL_W_REQ_OFFSET 1
#define OPCODE_GPIO_CONTROL_W_RES_LEN 1
#define OPCODE_CHECK_CC_AND_SBU 0x85
#define OPCODE_CHECK_CC_AND_SBU_REQ_LEN 9
#define OPCODE_CHECK_CC_AND_SBU_RES_LEN 5
#define OPCODE_USER_SPACE_MAX_ADDR 31
#define OPCODE_USER_SPACE_MAX_LEN 30
#define OPCODE_USER_SPACE_READ 0x81
#define OPCODE_USER_SPACE_R_REQ_LEN 3
#define OPCODE_USER_SPACE_R_RES_LEN 32
#define OPCODE_USER_SPACE_WRITE 0x82
#define OPCODE_USER_SPACE_W_REQ_LEN 32
#define OPCODE_USER_SPACE_W_RES_LEN 32
#define MAXQ_AP_DATAOUT0 0x81
#define MAXQ_AP_DATAOUT32 0xA1
#define MAXQ_AP_DATAIN0 0xB1
#define MAXQ_REPLY_TIMEOUT_MS 200
#define RSBM_ADDR 0
#define RSBR_ADDR 4
#define RS_TAG_LENGTH 4
#define RS_TAG_OFFSET_ADDR 0
#define RS_TAG_OFFSET_LENGTH 1
#define RS_TAG_OFFSET_DATA 2
#define LOG_BUFFER_SIZE 256
struct max77759_maxq {
struct completion reply_done;
/* Denotes the current request in progress. */
unsigned int req_no;
/* Updated by the irq handler. */
unsigned int req_done;
/* Protects req_no and req_done variables. */
struct mutex req_lock;
struct logbuffer *log;
bool init_done;
struct regmap *regmap;
/* To make sure that there is only one request in progress. */
struct mutex maxq_lock;
u8 request_opcode;
bool poll;
};
enum write_userspace_offset {
ADDR_OPCODE,
START_ADDR,
DATA_LEN,
DATA_START
};
enum request_payload_offset {
REQUEST_OPCODE,
/* Following are specific to CHECK_CC_AND_SBU */
REQUEST_TYPE,
CC1RD,
CC2RD,
CCADCSKIPPED,
CC1ADC,
CC2ADC,
SBU1ADC,
SBU2ADC,
WRITE_BYTES
};
enum response_payload_offset {
RESPONSE_OPCODE,
RESPONSE_TYPE,
RESULT,
CC_THRESH,
SBU_THRESH
};
/* Caller holds req_lock */
static int maxq_read_response_locked(struct max77759_maxq *maxq,
u8 *response, u8 response_len)
{
int ret = 0;
ret = regmap_bulk_read(maxq->regmap, MAXQ_AP_DATAIN0, response,
response_len);
if (ret) {
logbuffer_log(maxq->log, "I2C request failed ret:%d", ret);
return -EINVAL;
}
if (response[RESPONSE_OPCODE] != maxq->request_opcode) {
logbuffer_log(maxq->log,
"OPCODE does not match request:%u response:%u",
maxq->request_opcode, response[RESPONSE_OPCODE]);
return -EINVAL;
}
return ret;
}
/* Caller holds req_lock */
static int maxq_wait_for_response_locked(struct max77759_maxq *maxq,
unsigned int current_request,
u8 *response, u8 response_len)
{
int ret = 0;
unsigned long timeout = MAXQ_REPLY_TIMEOUT_MS;
/* Wait for response from Maxq */
if (maxq->poll) {
msleep(timeout);
ret = maxq_read_response_locked(maxq, response, response_len);
goto exit;
}
/* IRQ from MAXQ */
while (timeout) {
mutex_unlock(&maxq->req_lock);
timeout = wait_for_completion_timeout(&maxq->reply_done,
timeout);
mutex_lock(&maxq->req_lock);
if (current_request == maxq->req_done) {
ret = maxq_read_response_locked(maxq, response,
response_len);
break;
} else if (!timeout) {
logbuffer_log(maxq->log,
"Timeout or Request number does not match current:req:%u req_done:%u",
current_request, maxq->req_done);
ret = -EINVAL;
}
}
exit:
logbuffer_log(maxq->log, "MAXQ DONE: current_req: %u ret:%d",
current_request, ret);
return ret;
}
static int maxq_issue_opcode_command(struct max77759_maxq *maxq, u8 *request,
u8 request_length, u8 *response,
u8 response_length)
{
unsigned int current_request;
int ret = -ENODEV;
if (!maxq->init_done)
return ret;
mutex_lock(&maxq->maxq_lock);
maxq->request_opcode = request[REQUEST_OPCODE];
mutex_lock(&maxq->req_lock);
current_request = ++maxq->req_no;
logbuffer_log(maxq->log, "MAXQ REQ:current_req: %u opcode:%u",
current_request, maxq->request_opcode);
ret = regmap_bulk_write(maxq->regmap, MAXQ_AP_DATAOUT0, request,
request_length);
if (ret) {
logbuffer_log(maxq->log, "I2C request write failed");
goto req_unlock;
}
if (request_length < PAYLOAD_REQUEST_LENGTH_BYTES) {
ret = regmap_write(maxq->regmap, MAXQ_AP_DATAOUT32, 0);
if (ret) {
logbuffer_log(maxq->log,
"I2C request write DATAOUT32 failed");
goto req_unlock;
}
}
ret = maxq_wait_for_response_locked(maxq, current_request,
response, response_length);
req_unlock:
mutex_unlock(&maxq->req_lock);
mutex_unlock(&maxq->maxq_lock);
return ret;
}
static void maxq_log_array(struct max77759_maxq *maxq, char *title,
u8 *data, int length)
{
char buf[LOG_BUFFER_SIZE];
int i, len = 0;
for (i = 0; i < length; i++) {
len += scnprintf(&buf[len], LOG_BUFFER_SIZE - len,
"%#x ", data[i]);
if (len >= LOG_BUFFER_SIZE)
break;
}
logbuffer_log(maxq->log, "%s: %s", title, buf);
}
static int maxq_user_space_read(struct max77759_maxq *maxq, u8 *data)
{
int ret, i;
u8 request[OPCODE_USER_SPACE_R_REQ_LEN];
u8 response[OPCODE_USER_SPACE_R_RES_LEN];
request[ADDR_OPCODE] = OPCODE_USER_SPACE_READ;
request[START_ADDR] = data[0];
request[DATA_LEN] = data[1];
if (request[START_ADDR] > OPCODE_USER_SPACE_MAX_ADDR)
return -EINVAL;
if (request[DATA_LEN] > OPCODE_USER_SPACE_MAX_LEN)
return -EINVAL;
logbuffer_log(maxq->log, "MAXQ user space read opcode:%#x, start:%#x, length:%#x",
request[ADDR_OPCODE], request[START_ADDR], request[DATA_LEN]);
ret = maxq_issue_opcode_command(maxq, request,
OPCODE_USER_SPACE_R_REQ_LEN,
response,
OPCODE_USER_SPACE_R_RES_LEN);
if (ret < 0)
return ret;
for (i = DATA_START; i < DATA_START + request[DATA_LEN]; i++)
data[i - DATA_START] = response[i];
maxq_log_array(maxq, "MAXQ user space read result",
&response[DATA_START], request[DATA_LEN]);
return ret;
}
static int maxq_user_space_write(struct max77759_maxq *maxq, u8 *data)
{
int ret, i;
u8 request[OPCODE_USER_SPACE_W_REQ_LEN];
u8 response[OPCODE_USER_SPACE_W_RES_LEN];
request[ADDR_OPCODE] = OPCODE_USER_SPACE_WRITE;
request[START_ADDR] = data[0];
request[DATA_LEN] = data[1];
if (request[START_ADDR] > OPCODE_USER_SPACE_MAX_ADDR)
return -EINVAL;
if (request[DATA_LEN] > OPCODE_USER_SPACE_MAX_LEN)
return -EINVAL;
logbuffer_log(maxq->log, "MAXQ user space write opcode:%#x, start:%#x, length:%#x",
request[ADDR_OPCODE], request[START_ADDR], request[DATA_LEN]);
for (i = DATA_START; i < DATA_START + request[DATA_LEN]; i++)
request[i] = data[i - 1];
ret = maxq_issue_opcode_command(maxq, request,
OPCODE_USER_SPACE_W_REQ_LEN,
response,
OPCODE_USER_SPACE_W_RES_LEN);
if (ret < 0)
return ret;
maxq_log_array(maxq, "MAXQ user space write data",
&response[DATA_START], request[DATA_LEN]);
return ret;
}
static int maxq_rs_read(struct max77759_maxq *maxq, gbms_tag_t tag, u8 *data)
{
int ret;
u8 buff[OPCODE_USER_SPACE_R_RES_LEN];
if (tag == GBMS_TAG_RSBM)
buff[RS_TAG_OFFSET_ADDR] = RSBM_ADDR;
else if (tag == GBMS_TAG_RSBR)
buff[RS_TAG_OFFSET_ADDR] = RSBM_ADDR;
else
return -EINVAL;
buff[RS_TAG_OFFSET_LENGTH] = RS_TAG_LENGTH;
ret = maxq_user_space_read(maxq, buff);
if (ret < 0)
return ret;
memcpy(data, buff, RS_TAG_LENGTH);
return ret;
}
static int maxq_rs_write(struct max77759_maxq *maxq, gbms_tag_t tag, u8 *data)
{
int ret;
u8 buff[OPCODE_USER_SPACE_W_REQ_LEN];
if (tag == GBMS_TAG_RSBM)
buff[RS_TAG_OFFSET_ADDR] = RSBM_ADDR;
else if (tag == GBMS_TAG_RSBR)
buff[RS_TAG_OFFSET_ADDR] = RSBR_ADDR;
else
return -EINVAL;
buff[RS_TAG_OFFSET_LENGTH] = RS_TAG_LENGTH;
memcpy(&buff[RS_TAG_OFFSET_DATA], data, RS_TAG_LENGTH);
ret = maxq_user_space_write(maxq, buff);
return ret;
}
static int maxq_storage_read(gbms_tag_t tag, void *buff, size_t size,
void *ptr)
{
int ret;
struct max77759_maxq *maxq = ptr;
switch (tag) {
case GBMS_TAG_RS32:
if (size && size > OPCODE_USER_SPACE_R_RES_LEN)
return -EINVAL;
ret = maxq_user_space_read(maxq, buff);
break;
case GBMS_TAG_RSBM:
case GBMS_TAG_RSBR:
if (size && size > RS_TAG_LENGTH)
return -EINVAL;
ret = maxq_rs_read(maxq, tag, buff);
break;
default:
ret = -ENOENT;
break;
}
return ret;
}
static int maxq_storage_write(gbms_tag_t tag, const void *buff, size_t size,
void *ptr)
{
int ret;
struct max77759_maxq *maxq = ptr;
switch (tag) {
case GBMS_TAG_RS32:
if (size && size > OPCODE_USER_SPACE_W_RES_LEN)
return -EINVAL;
ret = maxq_user_space_write(maxq, (void *)buff);
break;
case GBMS_TAG_RSBM:
case GBMS_TAG_RSBR:
if (size && size > RS_TAG_LENGTH)
return -EINVAL;
ret = maxq_rs_write(maxq, tag, (void *)buff);
break;
default:
ret = -ENOENT;
break;
}
return ret;
}
static struct gbms_storage_desc maxq_storage_dsc = {
.read = maxq_storage_read,
.write = maxq_storage_write,
};
void maxq_irq(struct max77759_maxq *maxq)
{
mutex_lock(&maxq->req_lock);
maxq->req_done = maxq->req_no;
logbuffer_log(maxq->log, "MAXQ IRQ: req_done: %u", maxq->req_done);
complete(&maxq->reply_done);
mutex_unlock(&maxq->req_lock);
}
EXPORT_SYMBOL_GPL(maxq_irq);
int maxq_query_contaminant(struct max77759_maxq *maxq, u8 cc1_raw,
u8 cc2_raw, u8 sbu1_raw, u8 sbu2_raw, u8 cc1_rd,
u8 cc2_rd, u8 type, u8 cc_adc_skipped,
u8 *response, u8 response_len)
{
int ret;
u8 payload[OPCODE_CHECK_CC_AND_SBU_REQ_LEN];
payload[REQUEST_OPCODE] = OPCODE_CHECK_CC_AND_SBU;
payload[REQUEST_TYPE] = type;
payload[CC1RD] = cc1_rd;
payload[CC2RD] = cc2_rd;
payload[CCADCSKIPPED] = cc_adc_skipped;
payload[CC1ADC] = cc1_raw;
payload[CC2ADC] = cc2_raw;
payload[SBU1ADC] = sbu1_raw;
payload[SBU2ADC] = sbu2_raw;
logbuffer_log(maxq->log,
"MAXQ opcode:%#x type:%#x cc1_rd:%#x cc2_rd:%#x ADCSKIPPED:%#x cc1adc:%#x cc2adc:%#x sbu1adc:%#x sbu2adc:%#x",
OPCODE_CHECK_CC_AND_SBU, type, cc1_rd, cc2_rd,
cc_adc_skipped, cc1_raw, cc2_raw, sbu1_raw, sbu2_raw);
ret = maxq_issue_opcode_command(maxq, payload,
OPCODE_CHECK_CC_AND_SBU_REQ_LEN,
response, response_len);
if (!ret)
logbuffer_log(maxq->log, "MAXQ Contaminant response:%u",
response[RESULT]);
return ret;
}
EXPORT_SYMBOL_GPL(maxq_query_contaminant);
int maxq_gpio_control_read(struct max77759_maxq *maxq, u8 *gpio)
{
int ret;
u8 request = OPCODE_GPIO_CONTROL_READ;
u8 response[OPCODE_GPIO_CONTROL_R_RES_LEN];
logbuffer_log(maxq->log, "MAXQ gpio read opcode:%#x", request);
ret = maxq_issue_opcode_command(maxq, &request,
OPCODE_GPIO_CONTROL_R_REQ_LEN,
response,
OPCODE_GPIO_CONTROL_R_RES_LEN);
if (!ret)
*gpio = response[OPCODE_GPIO_CONTROL_R_RES_OFFSET];
logbuffer_log(maxq->log, "MAXQ GPIO:%#x", *gpio);
return ret;
}
EXPORT_SYMBOL_GPL(maxq_gpio_control_read);
int maxq_gpio_control_write(struct max77759_maxq *maxq, u8 gpio)
{
int ret;
u8 request[OPCODE_GPIO_CONTROL_W_REQ_LEN];
u8 response[OPCODE_GPIO_CONTROL_W_RES_LEN];
request[REQUEST_OPCODE] = OPCODE_GPIO_CONTROL_WRITE;
request[OPCODE_GPIO_CONTROL_W_REQ_OFFSET] = gpio;
logbuffer_log(maxq->log, "MAXQ gpio write opcode:%#x val:%#x",
request[REQUEST_OPCODE],
request[OPCODE_GPIO_CONTROL_W_REQ_OFFSET]);
ret = maxq_issue_opcode_command(maxq, request,
OPCODE_GPIO_CONTROL_W_REQ_LEN,
response,
OPCODE_GPIO_CONTROL_W_RES_LEN);
return ret;
}
EXPORT_SYMBOL_GPL(maxq_gpio_control_write);
int maxq_gpio_trigger_read(struct max77759_maxq *maxq, u8 gpio, bool *trigger_falling)
{
int ret;
u8 request = OPCODE_GPIO_TRIGGER_READ;
u8 response[OPCODE_GPIO_TRIGGER_R_RES_LEN];
/* Only GPIO5 and GPIO6 supported */
if (gpio != 5 && gpio != 6) {
logbuffer_log(maxq->log, "MAXQ gpio trigger read invalid gpio %d", gpio);
return -EINVAL;
}
logbuffer_log(maxq->log, "MAXQ gpio trigger read opcode:%#x", request);
ret = maxq_issue_opcode_command(maxq, &request,
OPCODE_GPIO_TRIGGER_R_REQ_LEN,
response,
OPCODE_GPIO_TRIGGER_R_RES_LEN);
if (!ret) {
*trigger_falling = response[OPCODE_GPIO_TRIGGER_R_RES_OFFSET] & ((gpio == 5) ?
MAX_GPIO5_TRIG_MASK : MAX_GPIO6_TRIG_MASK);
logbuffer_log(maxq->log, "MAXQ GPIO%d TRIGGER:%s", gpio,
*trigger_falling ? "falling" : "rising");
} else {
logbuffer_log(maxq->log, "MAXQ GPIO%d TRIGGER read failed", gpio);
}
return ret;
}
EXPORT_SYMBOL_GPL(maxq_gpio_trigger_read);
int maxq_gpio_trigger_write(struct max77759_maxq *maxq, u8 gpio, bool trigger_falling)
{
int ret;
u8 request[OPCODE_GPIO_TRIGGER_W_REQ_LEN], response[OPCODE_GPIO_TRIGGER_W_RES_LEN];
u8 trigger_val;
bool trigger_falling_other;
/* Only GPIO5 and GPIO6 supported */
if (gpio != 5 && gpio != 6) {
logbuffer_log(maxq->log, "MAXQ gpio trigger read invalid gpio %d", gpio);
return -EINVAL;
}
/* Read the other GPIO */
ret = maxq_gpio_trigger_read(maxq, gpio == 5 ? 6 : 5, &trigger_falling_other);
if (ret < 0)
return ret;
/* Compose trigger write val */
if (gpio == 5) {
trigger_val = MAX_GPIO6_TRIG(trigger_falling_other ? MAX_GPIO_TRIG_FALLING :
MAX_GPIO_TRIG_RISING);
trigger_val |= MAX_GPIO5_TRIG(trigger_falling ? MAX_GPIO_TRIG_FALLING :
MAX_GPIO_TRIG_RISING);
} else {
trigger_val = MAX_GPIO5_TRIG(trigger_falling_other ? MAX_GPIO_TRIG_FALLING :
MAX_GPIO_TRIG_RISING);
trigger_val |= MAX_GPIO6_TRIG(trigger_falling ? MAX_GPIO_TRIG_FALLING :
MAX_GPIO_TRIG_RISING);
}
request[REQUEST_OPCODE] = OPCODE_GPIO_TRIGGER_WRITE;
request[OPCODE_GPIO_TRIGGER_W_REQ_OFFSET] = trigger_val;
logbuffer_log(maxq->log, "MAXQ gpio trigger write opcode:%#x val:%#x",
request[REQUEST_OPCODE],
request[OPCODE_GPIO_TRIGGER_W_REQ_OFFSET]);
ret = maxq_issue_opcode_command(maxq, request,
OPCODE_GPIO_TRIGGER_W_REQ_LEN,
response,
OPCODE_GPIO_TRIGGER_W_RES_LEN);
return ret;
}
EXPORT_SYMBOL_GPL(maxq_gpio_trigger_write);
struct max77759_maxq *maxq_init(struct device *dev, struct regmap *regmap,
bool poll)
{
struct max77759_maxq *maxq;
int ret;
maxq = devm_kzalloc(dev, sizeof(*maxq), GFP_KERNEL);
if (IS_ERR_OR_NULL(maxq))
return ERR_PTR(-ENOMEM);
maxq->log = logbuffer_register("maxq");
if (IS_ERR_OR_NULL(maxq->log)) {
dev_err(dev, "MAXQ logbuffer register failed\n");
maxq->log = NULL;
}
maxq->regmap = regmap;
init_completion(&maxq->reply_done);
mutex_init(&maxq->maxq_lock);
mutex_init(&maxq->req_lock);
maxq->poll = poll;
ret = gbms_storage_register(&maxq_storage_dsc,
"max77759_maxq", maxq);
if (ret < 0)
dev_err(dev, "MAXQ gbms_storage_register failed, ret:%d\n", ret);
maxq->init_done = true;
logbuffer_log(maxq->log, "MAXQ: probe done");
return maxq;
}
EXPORT_SYMBOL_GPL(maxq_init);
void maxq_remove(struct max77759_maxq *maxq)
{
maxq->init_done = false;
logbuffer_unregister(maxq->log);
}
EXPORT_SYMBOL_GPL(maxq_remove);
MODULE_AUTHOR("Badhri Jagan Sridharan <[email protected]>");
MODULE_DESCRIPTION("Maxim 77759 MAXQ OPCDOE management");
MODULE_LICENSE("GPL");