Merge "spmi: msm: Add virtual SPMI PMIC front end driver"
This commit is contained in:
commit
8be37176ca
4 changed files with 403 additions and 0 deletions
|
@ -0,0 +1,24 @@
|
||||||
|
QTI Virtual SPMI controller (Virtual PMIC Arbiter)
|
||||||
|
|
||||||
|
The Virtual SPMI PMIC Arbiter is a frontend proxy based on backend virtual device.
|
||||||
|
|
||||||
|
Required properties:
|
||||||
|
- compatible : should be "qcom,virtspmi-pmic-arb".
|
||||||
|
- reg-names : must contain:
|
||||||
|
"core" - core registers
|
||||||
|
|
||||||
|
- reg : address + size pairs describing the Virtual PMIC arb register sets;
|
||||||
|
order must correspond with the order of entries in reg-names
|
||||||
|
- #address-cells : must be set to 2
|
||||||
|
- #size-cells : must be set to 0
|
||||||
|
|
||||||
|
Example Virtual PMIC-Arbiter:
|
||||||
|
|
||||||
|
spmi {
|
||||||
|
compatible = "qcom,virtspmi-pmic-arb";
|
||||||
|
reg-names = "core";
|
||||||
|
reg = <0xfc4cf000 0x1000>;
|
||||||
|
|
||||||
|
#address-cells = <2>;
|
||||||
|
#size-cells = <0>;
|
||||||
|
};
|
|
@ -24,4 +24,13 @@ config SPMI_MSM_PMIC_ARB
|
||||||
This is required for communicating with Qualcomm PMICs and
|
This is required for communicating with Qualcomm PMICs and
|
||||||
other devices that have the SPMI interface.
|
other devices that have the SPMI interface.
|
||||||
|
|
||||||
|
config VIRTSPMI_MSM_PMIC_ARB
|
||||||
|
tristate "QTI MSM Virtual SPMI Controller (Virtual PMIC Arbiter)"
|
||||||
|
depends on ARCH_QCOM || COMPILE_TEST
|
||||||
|
depends on HAS_IOMEM
|
||||||
|
default ARCH_QCOM
|
||||||
|
help
|
||||||
|
This is a virtual SPMI frontend driver for QTI MSM virtual
|
||||||
|
platform.
|
||||||
|
|
||||||
endif
|
endif
|
||||||
|
|
|
@ -4,3 +4,5 @@
|
||||||
obj-$(CONFIG_SPMI) += spmi.o
|
obj-$(CONFIG_SPMI) += spmi.o
|
||||||
|
|
||||||
obj-$(CONFIG_SPMI_MSM_PMIC_ARB) += spmi-pmic-arb.o
|
obj-$(CONFIG_SPMI_MSM_PMIC_ARB) += spmi-pmic-arb.o
|
||||||
|
|
||||||
|
obj-$(CONFIG_VIRTSPMI_MSM_PMIC_ARB) += virtspmi-pmic-arb.o
|
||||||
|
|
368
drivers/spmi/virtspmi-pmic-arb.c
Normal file
368
drivers/spmi/virtspmi-pmic-arb.c
Normal file
|
@ -0,0 +1,368 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2018, 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/bitmap.h>
|
||||||
|
#include <linux/delay.h>
|
||||||
|
#include <linux/err.h>
|
||||||
|
#include <linux/interrupt.h>
|
||||||
|
#include <linux/io.h>
|
||||||
|
#include <linux/kernel.h>
|
||||||
|
#include <linux/module.h>
|
||||||
|
#include <linux/of.h>
|
||||||
|
#include <linux/platform_device.h>
|
||||||
|
#include <linux/slab.h>
|
||||||
|
#include <linux/spmi.h>
|
||||||
|
|
||||||
|
/* PMIC Arbiter configuration registers */
|
||||||
|
#define VPMIC_ARB_VERSION 0x0000
|
||||||
|
|
||||||
|
/* Virtual PMIC Arbiter registers offset*/
|
||||||
|
#define VPMIC_ARB_CMD 0x00
|
||||||
|
#define VPMIC_ARB_STATUS 0x04
|
||||||
|
#define VPMIC_ARB_DATA0 0x08
|
||||||
|
#define VPMIC_ARB_DATA1 0x10
|
||||||
|
|
||||||
|
/* Channel Status fields */
|
||||||
|
enum pmic_arb_chnl_status {
|
||||||
|
PMIC_ARB_STATUS_DONE = BIT(0),
|
||||||
|
PMIC_ARB_STATUS_FAILURE = BIT(1),
|
||||||
|
PMIC_ARB_STATUS_DENIED = BIT(2),
|
||||||
|
PMIC_ARB_STATUS_DROPPED = BIT(3),
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Command register fields */
|
||||||
|
#define PMIC_ARB_CMD_MAX_BYTE_COUNT 8
|
||||||
|
|
||||||
|
/* Command Opcodes */
|
||||||
|
enum pmic_arb_cmd_op_code {
|
||||||
|
PMIC_ARB_OP_EXT_WRITEL = 0,
|
||||||
|
PMIC_ARB_OP_EXT_READL = 1,
|
||||||
|
PMIC_ARB_OP_EXT_WRITE = 2,
|
||||||
|
PMIC_ARB_OP_RESET = 3,
|
||||||
|
PMIC_ARB_OP_SLEEP = 4,
|
||||||
|
PMIC_ARB_OP_SHUTDOWN = 5,
|
||||||
|
PMIC_ARB_OP_WAKEUP = 6,
|
||||||
|
PMIC_ARB_OP_AUTHENTICATE = 7,
|
||||||
|
PMIC_ARB_OP_MSTR_READ = 8,
|
||||||
|
PMIC_ARB_OP_MSTR_WRITE = 9,
|
||||||
|
PMIC_ARB_OP_EXT_READ = 13,
|
||||||
|
PMIC_ARB_OP_WRITE = 14,
|
||||||
|
PMIC_ARB_OP_READ = 15,
|
||||||
|
PMIC_ARB_OP_ZERO_WRITE = 16,
|
||||||
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* PMIC arbiter version 5 uses different register offsets for read/write vs
|
||||||
|
* observer channels.
|
||||||
|
*/
|
||||||
|
enum pmic_arb_channel {
|
||||||
|
PMIC_ARB_CHANNEL_RW,
|
||||||
|
PMIC_ARB_CHANNEL_OBS,
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Maximum number of support PMIC peripherals */
|
||||||
|
#define PMIC_ARB_MAX_PERIPHS 512
|
||||||
|
#define PMIC_ARB_TIMEOUT_US 100
|
||||||
|
#define PMIC_ARB_MAX_TRANS_BYTES (8)
|
||||||
|
|
||||||
|
struct vspmi_backend_driver_ver_ops;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vspmi_pmic_arb - Virtual SPMI PMIC Arbiter object
|
||||||
|
*
|
||||||
|
* @lock: lock to synchronize accesses.
|
||||||
|
* @spmic: SPMI controller object
|
||||||
|
* @ver_ops: backend version dependent operations.
|
||||||
|
*/
|
||||||
|
struct vspmi_pmic_arb {
|
||||||
|
void __iomem *core;
|
||||||
|
resource_size_t core_size;
|
||||||
|
raw_spinlock_t lock;
|
||||||
|
struct spmi_controller *spmic;
|
||||||
|
const struct vspmi_backend_driver_ver_ops *ver_ops;
|
||||||
|
};
|
||||||
|
static struct vspmi_pmic_arb *the_pa;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* pmic_arb_ver: version dependent functionality.
|
||||||
|
*
|
||||||
|
* @ver_str: version string.
|
||||||
|
* @fmt_cmd: formats a GENI/SPMI command.
|
||||||
|
*/
|
||||||
|
struct vspmi_backend_driver_ver_ops {
|
||||||
|
const char *ver_str;
|
||||||
|
u32 (*fmt_cmd)(u8 opc, u8 sid, u16 addr, u8 bc);
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vspmi_pa_read_data: reads vspmi backend's register and copy 1..4 bytes to buf
|
||||||
|
* @bc: byte count -1. range: 0..3
|
||||||
|
* @reg: register's address
|
||||||
|
* @buf: output parameter, length must be bc + 1
|
||||||
|
*/
|
||||||
|
static void
|
||||||
|
vspmi_pa_read_data(struct vspmi_pmic_arb *pa, u8 *buf, u32 reg, u8 bc)
|
||||||
|
{
|
||||||
|
u32 data = __raw_readl(pa->core + reg);
|
||||||
|
|
||||||
|
memcpy(buf, &data, (bc & 3) + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vspmi_pa_write_data: write 1..4 bytes from buf to vspmi backend's register
|
||||||
|
* @bc: byte-count -1. range: 0..3.
|
||||||
|
* @reg: register's address.
|
||||||
|
* @buf: buffer to write. length must be bc + 1.
|
||||||
|
*/
|
||||||
|
static void
|
||||||
|
vspmi_pa_write_data(struct vspmi_pmic_arb *pa, const u8 *buf, u32 reg, u8 bc)
|
||||||
|
{
|
||||||
|
u32 data = 0;
|
||||||
|
|
||||||
|
memcpy(&data, buf, (bc & 3) + 1);
|
||||||
|
writel_relaxed(data, pa->core + reg);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int vspmi_pmic_arb_wait_for_done(struct spmi_controller *ctrl,
|
||||||
|
void __iomem *base, u8 sid, u16 addr,
|
||||||
|
enum pmic_arb_channel ch_type)
|
||||||
|
{
|
||||||
|
u32 status = 0;
|
||||||
|
u32 timeout = PMIC_ARB_TIMEOUT_US;
|
||||||
|
u32 offset;
|
||||||
|
|
||||||
|
offset = VPMIC_ARB_STATUS;
|
||||||
|
|
||||||
|
while (timeout--) {
|
||||||
|
status = readl_relaxed(base + offset);
|
||||||
|
|
||||||
|
if (status & PMIC_ARB_STATUS_DONE) {
|
||||||
|
if (status & PMIC_ARB_STATUS_DENIED) {
|
||||||
|
dev_err(&ctrl->dev,
|
||||||
|
"%s: transaction denied (0x%x)\n",
|
||||||
|
__func__, status);
|
||||||
|
return -EPERM;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (status & PMIC_ARB_STATUS_FAILURE) {
|
||||||
|
dev_err(&ctrl->dev,
|
||||||
|
"%s: transaction failed (0x%x)\n",
|
||||||
|
__func__, status);
|
||||||
|
return -EIO;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (status & PMIC_ARB_STATUS_DROPPED) {
|
||||||
|
dev_err(&ctrl->dev,
|
||||||
|
"%s: transaction dropped (0x%x)\n",
|
||||||
|
__func__, status);
|
||||||
|
return -EIO;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
udelay(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
dev_err(&ctrl->dev,
|
||||||
|
"%s: timeout, status 0x%x\n",
|
||||||
|
__func__, status);
|
||||||
|
return -ETIMEDOUT;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int vspmi_pmic_arb_read_cmd(struct spmi_controller *ctrl, u8 opc, u8 sid,
|
||||||
|
u16 addr, u8 *buf, size_t len)
|
||||||
|
{
|
||||||
|
struct vspmi_pmic_arb *pa = spmi_controller_get_drvdata(ctrl);
|
||||||
|
unsigned long flags;
|
||||||
|
u8 bc = len;
|
||||||
|
u32 cmd;
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
if (bc >= PMIC_ARB_MAX_TRANS_BYTES) {
|
||||||
|
dev_err(&ctrl->dev,
|
||||||
|
"pmic-arb supports 1..%d bytes per trans, but:%zu requested",
|
||||||
|
PMIC_ARB_MAX_TRANS_BYTES, len);
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Check the opcode */
|
||||||
|
if (opc >= 0x60 && opc <= 0x7F)
|
||||||
|
opc = PMIC_ARB_OP_READ;
|
||||||
|
else if (opc >= 0x20 && opc <= 0x2F)
|
||||||
|
opc = PMIC_ARB_OP_EXT_READ;
|
||||||
|
else if (opc >= 0x38 && opc <= 0x3F)
|
||||||
|
opc = PMIC_ARB_OP_EXT_READL;
|
||||||
|
else
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
cmd = pa->ver_ops->fmt_cmd(opc, sid, addr, bc);
|
||||||
|
|
||||||
|
raw_spin_lock_irqsave(&pa->lock, flags);
|
||||||
|
writel_relaxed(cmd, pa->core + VPMIC_ARB_CMD);
|
||||||
|
rc = vspmi_pmic_arb_wait_for_done(ctrl, pa->core, sid, addr,
|
||||||
|
PMIC_ARB_CHANNEL_OBS);
|
||||||
|
if (rc)
|
||||||
|
goto done;
|
||||||
|
|
||||||
|
vspmi_pa_read_data(pa, buf, VPMIC_ARB_DATA0, min_t(u8, bc, 3));
|
||||||
|
|
||||||
|
if (bc > 3)
|
||||||
|
vspmi_pa_read_data(pa, buf + 4, VPMIC_ARB_DATA1, bc - 4);
|
||||||
|
|
||||||
|
done:
|
||||||
|
raw_spin_unlock_irqrestore(&pa->lock, flags);
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int vspmi_pmic_arb_write_cmd(struct spmi_controller *ctrl, u8 opc,
|
||||||
|
u8 sid, u16 addr, const u8 *buf, size_t len)
|
||||||
|
{
|
||||||
|
struct vspmi_pmic_arb *pa = spmi_controller_get_drvdata(ctrl);
|
||||||
|
unsigned long flags;
|
||||||
|
u8 bc = len;
|
||||||
|
u32 cmd;
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
if (bc >= PMIC_ARB_MAX_TRANS_BYTES) {
|
||||||
|
dev_err(&ctrl->dev,
|
||||||
|
"pmic-arb supports 1..%d bytes per trans, but:%zu requested",
|
||||||
|
PMIC_ARB_MAX_TRANS_BYTES, len);
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Check the opcode */
|
||||||
|
if (opc >= 0x40 && opc <= 0x5F)
|
||||||
|
opc = PMIC_ARB_OP_WRITE;
|
||||||
|
else if (opc <= 0x0F)
|
||||||
|
opc = PMIC_ARB_OP_EXT_WRITE;
|
||||||
|
else if (opc >= 0x30 && opc <= 0x37)
|
||||||
|
opc = PMIC_ARB_OP_EXT_WRITEL;
|
||||||
|
else if (opc >= 0x80)
|
||||||
|
opc = PMIC_ARB_OP_ZERO_WRITE;
|
||||||
|
else
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
cmd = pa->ver_ops->fmt_cmd(opc, sid, addr, bc);
|
||||||
|
|
||||||
|
/* Write data to FIFOs */
|
||||||
|
raw_spin_lock_irqsave(&pa->lock, flags);
|
||||||
|
vspmi_pa_write_data(pa, buf, VPMIC_ARB_DATA0, min_t(u8, bc, 3));
|
||||||
|
if (bc > 3)
|
||||||
|
vspmi_pa_write_data(pa, buf + 4, VPMIC_ARB_DATA1, bc - 4);
|
||||||
|
|
||||||
|
/* Start the transaction */
|
||||||
|
writel_relaxed(cmd, pa->core + VPMIC_ARB_CMD);
|
||||||
|
rc = vspmi_pmic_arb_wait_for_done(ctrl, pa->core, sid, addr,
|
||||||
|
PMIC_ARB_CHANNEL_RW);
|
||||||
|
raw_spin_unlock_irqrestore(&pa->lock, flags);
|
||||||
|
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
static u32 vspmi_pmic_arb_fmt_cmd_v1(u8 opc, u8 sid, u16 addr, u8 bc)
|
||||||
|
{
|
||||||
|
return (opc << 27) | ((sid & 0xf) << 20) | (addr << 4) | (bc & 0x7);
|
||||||
|
}
|
||||||
|
|
||||||
|
static const struct vspmi_backend_driver_ver_ops pmic_arb_v1 = {
|
||||||
|
.ver_str = "v1",
|
||||||
|
.fmt_cmd = vspmi_pmic_arb_fmt_cmd_v1,
|
||||||
|
};
|
||||||
|
|
||||||
|
static int vspmi_pmic_arb_probe(struct platform_device *pdev)
|
||||||
|
{
|
||||||
|
struct vspmi_pmic_arb *pa;
|
||||||
|
struct spmi_controller *ctrl;
|
||||||
|
struct resource *res;
|
||||||
|
u32 backend_ver;
|
||||||
|
int err;
|
||||||
|
|
||||||
|
ctrl = spmi_controller_alloc(&pdev->dev, sizeof(*pa));
|
||||||
|
if (!ctrl)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
pa = spmi_controller_get_drvdata(ctrl);
|
||||||
|
pa->spmic = ctrl;
|
||||||
|
|
||||||
|
res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "core");
|
||||||
|
if (!res) {
|
||||||
|
dev_err(&pdev->dev, "vdev resource not specified\n");
|
||||||
|
err = -EINVAL;
|
||||||
|
goto err_put_ctrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
pa->core = devm_ioremap_resource(&ctrl->dev, res);
|
||||||
|
if (IS_ERR(pa->core)) {
|
||||||
|
err = PTR_ERR(pa->core);
|
||||||
|
goto err_put_ctrl;
|
||||||
|
}
|
||||||
|
pa->core_size = resource_size(res);
|
||||||
|
|
||||||
|
backend_ver = VPMIC_ARB_VERSION;
|
||||||
|
|
||||||
|
if (backend_ver == VPMIC_ARB_VERSION)
|
||||||
|
pa->ver_ops = &pmic_arb_v1;
|
||||||
|
|
||||||
|
dev_info(&ctrl->dev, "PMIC arbiter version %s (0x%x)\n",
|
||||||
|
pa->ver_ops->ver_str, backend_ver);
|
||||||
|
|
||||||
|
platform_set_drvdata(pdev, ctrl);
|
||||||
|
raw_spin_lock_init(&pa->lock);
|
||||||
|
|
||||||
|
ctrl->read_cmd = vspmi_pmic_arb_read_cmd;
|
||||||
|
ctrl->write_cmd = vspmi_pmic_arb_write_cmd;
|
||||||
|
|
||||||
|
err = spmi_controller_add(ctrl);
|
||||||
|
if (err)
|
||||||
|
goto err_put_ctrl;
|
||||||
|
|
||||||
|
the_pa = pa;
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
err_put_ctrl:
|
||||||
|
spmi_controller_put(ctrl);
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int vspmi_pmic_arb_remove(struct platform_device *pdev)
|
||||||
|
{
|
||||||
|
struct spmi_controller *ctrl = platform_get_drvdata(pdev);
|
||||||
|
|
||||||
|
spmi_controller_remove(ctrl);
|
||||||
|
the_pa = NULL;
|
||||||
|
spmi_controller_put(ctrl);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const struct of_device_id vspmi_pmic_arb_match_table[] = {
|
||||||
|
{ .compatible = "qcom,virtspmi-pmic-arb", },
|
||||||
|
{},
|
||||||
|
};
|
||||||
|
MODULE_DEVICE_TABLE(of, vspmi_pmic_arb_match_table);
|
||||||
|
|
||||||
|
static struct platform_driver vspmi_pmic_arb_driver = {
|
||||||
|
.probe = vspmi_pmic_arb_probe,
|
||||||
|
.remove = vspmi_pmic_arb_remove,
|
||||||
|
.driver = {
|
||||||
|
.name = "virtspmi_pmic_arb",
|
||||||
|
.of_match_table = vspmi_pmic_arb_match_table,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
static int __init vspmi_pmic_arb_init(void)
|
||||||
|
{
|
||||||
|
return platform_driver_register(&vspmi_pmic_arb_driver);
|
||||||
|
}
|
||||||
|
arch_initcall(vspmi_pmic_arb_init);
|
||||||
|
|
||||||
|
MODULE_LICENSE("GPL v2");
|
||||||
|
MODULE_ALIAS("platform:virtspmi_pmic_arb");
|
Loading…
Add table
Reference in a new issue