soc: qcom: Driver for the Qualcomm RPM over SMD
Driver for the Resource Power Manager (RPM) found in Qualcomm 8974 based devices. The driver exposes resources that child drivers can operate on; to implementing regulator, clock and bus frequency drivers. Signed-off-by: Bjorn Andersson <bjorn.andersson@sonymobile.com> Signed-off-by: Andy Gross <agross@codeaurora.org>
This commit is contained in:
parent
f2ab3298fb
commit
936f14cf4e
|
@ -27,6 +27,20 @@ config QCOM_SMD
|
||||||
providing communication channels to remote processors in Qualcomm
|
providing communication channels to remote processors in Qualcomm
|
||||||
platforms.
|
platforms.
|
||||||
|
|
||||||
|
config QCOM_SMD_RPM
|
||||||
|
tristate "Qualcomm Resource Power Manager (RPM) over SMD"
|
||||||
|
depends on QCOM_SMD && OF
|
||||||
|
help
|
||||||
|
If you say yes to this option, support will be included for the
|
||||||
|
Resource Power Manager system found in the Qualcomm 8974 based
|
||||||
|
devices.
|
||||||
|
|
||||||
|
This is required to access many regulators, clocks and bus
|
||||||
|
frequencies controlled by the RPM on these devices.
|
||||||
|
|
||||||
|
Say M here if you want to include support for the Qualcomm RPM as a
|
||||||
|
module. This will build a module called "qcom-smd-rpm".
|
||||||
|
|
||||||
config QCOM_SMEM
|
config QCOM_SMEM
|
||||||
tristate "Qualcomm Shared Memory Manager (SMEM)"
|
tristate "Qualcomm Shared Memory Manager (SMEM)"
|
||||||
depends on ARCH_QCOM
|
depends on ARCH_QCOM
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
obj-$(CONFIG_QCOM_GSBI) += qcom_gsbi.o
|
obj-$(CONFIG_QCOM_GSBI) += qcom_gsbi.o
|
||||||
obj-$(CONFIG_QCOM_PM) += spm.o
|
obj-$(CONFIG_QCOM_PM) += spm.o
|
||||||
obj-$(CONFIG_QCOM_SMD) += smd.o
|
obj-$(CONFIG_QCOM_SMD) += smd.o
|
||||||
|
obj-$(CONFIG_QCOM_SMD_RPM) += smd-rpm.o
|
||||||
obj-$(CONFIG_QCOM_SMEM) += smem.o
|
obj-$(CONFIG_QCOM_SMEM) += smem.o
|
||||||
|
|
|
@ -0,0 +1,244 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2015, Sony Mobile Communications AB.
|
||||||
|
* Copyright (c) 2012-2013, 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/platform_device.h>
|
||||||
|
#include <linux/of_platform.h>
|
||||||
|
#include <linux/io.h>
|
||||||
|
#include <linux/interrupt.h>
|
||||||
|
|
||||||
|
#include <linux/soc/qcom/smd.h>
|
||||||
|
#include <linux/soc/qcom/smd-rpm.h>
|
||||||
|
|
||||||
|
#define RPM_REQUEST_TIMEOUT (5 * HZ)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct qcom_smd_rpm - state of the rpm device driver
|
||||||
|
* @rpm_channel: reference to the smd channel
|
||||||
|
* @ack: completion for acks
|
||||||
|
* @lock: mutual exclusion around the send/complete pair
|
||||||
|
* @ack_status: result of the rpm request
|
||||||
|
*/
|
||||||
|
struct qcom_smd_rpm {
|
||||||
|
struct qcom_smd_channel *rpm_channel;
|
||||||
|
|
||||||
|
struct completion ack;
|
||||||
|
struct mutex lock;
|
||||||
|
int ack_status;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct qcom_rpm_header - header for all rpm requests and responses
|
||||||
|
* @service_type: identifier of the service
|
||||||
|
* @length: length of the payload
|
||||||
|
*/
|
||||||
|
struct qcom_rpm_header {
|
||||||
|
u32 service_type;
|
||||||
|
u32 length;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct qcom_rpm_request - request message to the rpm
|
||||||
|
* @msg_id: identifier of the outgoing message
|
||||||
|
* @flags: active/sleep state flags
|
||||||
|
* @type: resource type
|
||||||
|
* @id: resource id
|
||||||
|
* @data_len: length of the payload following this header
|
||||||
|
*/
|
||||||
|
struct qcom_rpm_request {
|
||||||
|
u32 msg_id;
|
||||||
|
u32 flags;
|
||||||
|
u32 type;
|
||||||
|
u32 id;
|
||||||
|
u32 data_len;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct qcom_rpm_message - response message from the rpm
|
||||||
|
* @msg_type: indicator of the type of message
|
||||||
|
* @length: the size of this message, including the message header
|
||||||
|
* @msg_id: message id
|
||||||
|
* @message: textual message from the rpm
|
||||||
|
*
|
||||||
|
* Multiple of these messages can be stacked in an rpm message.
|
||||||
|
*/
|
||||||
|
struct qcom_rpm_message {
|
||||||
|
u32 msg_type;
|
||||||
|
u32 length;
|
||||||
|
union {
|
||||||
|
u32 msg_id;
|
||||||
|
u8 message[0];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
#define RPM_SERVICE_TYPE_REQUEST 0x00716572 /* "req\0" */
|
||||||
|
|
||||||
|
#define RPM_MSG_TYPE_ERR 0x00727265 /* "err\0" */
|
||||||
|
#define RPM_MSG_TYPE_MSG_ID 0x2367736d /* "msg#" */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* qcom_rpm_smd_write - write @buf to @type:@id
|
||||||
|
* @rpm: rpm handle
|
||||||
|
* @type: resource type
|
||||||
|
* @id: resource identifier
|
||||||
|
* @buf: the data to be written
|
||||||
|
* @count: number of bytes in @buf
|
||||||
|
*/
|
||||||
|
int qcom_rpm_smd_write(struct qcom_smd_rpm *rpm,
|
||||||
|
int state,
|
||||||
|
u32 type, u32 id,
|
||||||
|
void *buf,
|
||||||
|
size_t count)
|
||||||
|
{
|
||||||
|
static unsigned msg_id = 1;
|
||||||
|
int left;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
struct {
|
||||||
|
struct qcom_rpm_header hdr;
|
||||||
|
struct qcom_rpm_request req;
|
||||||
|
u8 payload[count];
|
||||||
|
} pkt;
|
||||||
|
|
||||||
|
/* SMD packets to the RPM may not exceed 256 bytes */
|
||||||
|
if (WARN_ON(sizeof(pkt) >= 256))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
mutex_lock(&rpm->lock);
|
||||||
|
|
||||||
|
pkt.hdr.service_type = RPM_SERVICE_TYPE_REQUEST;
|
||||||
|
pkt.hdr.length = sizeof(struct qcom_rpm_request) + count;
|
||||||
|
|
||||||
|
pkt.req.msg_id = msg_id++;
|
||||||
|
pkt.req.flags = BIT(state);
|
||||||
|
pkt.req.type = type;
|
||||||
|
pkt.req.id = id;
|
||||||
|
pkt.req.data_len = count;
|
||||||
|
memcpy(pkt.payload, buf, count);
|
||||||
|
|
||||||
|
ret = qcom_smd_send(rpm->rpm_channel, &pkt, sizeof(pkt));
|
||||||
|
if (ret)
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
left = wait_for_completion_timeout(&rpm->ack, RPM_REQUEST_TIMEOUT);
|
||||||
|
if (!left)
|
||||||
|
ret = -ETIMEDOUT;
|
||||||
|
else
|
||||||
|
ret = rpm->ack_status;
|
||||||
|
|
||||||
|
out:
|
||||||
|
mutex_unlock(&rpm->lock);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(qcom_rpm_smd_write);
|
||||||
|
|
||||||
|
static int qcom_smd_rpm_callback(struct qcom_smd_device *qsdev,
|
||||||
|
const void *data,
|
||||||
|
size_t count)
|
||||||
|
{
|
||||||
|
const struct qcom_rpm_header *hdr = data;
|
||||||
|
const struct qcom_rpm_message *msg;
|
||||||
|
struct qcom_smd_rpm *rpm = dev_get_drvdata(&qsdev->dev);
|
||||||
|
const u8 *buf = data + sizeof(struct qcom_rpm_header);
|
||||||
|
const u8 *end = buf + hdr->length;
|
||||||
|
char msgbuf[32];
|
||||||
|
int status = 0;
|
||||||
|
u32 len;
|
||||||
|
|
||||||
|
if (hdr->service_type != RPM_SERVICE_TYPE_REQUEST ||
|
||||||
|
hdr->length < sizeof(struct qcom_rpm_message)) {
|
||||||
|
dev_err(&qsdev->dev, "invalid request\n");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (buf < end) {
|
||||||
|
msg = (struct qcom_rpm_message *)buf;
|
||||||
|
switch (msg->msg_type) {
|
||||||
|
case RPM_MSG_TYPE_MSG_ID:
|
||||||
|
break;
|
||||||
|
case RPM_MSG_TYPE_ERR:
|
||||||
|
len = min_t(u32, ALIGN(msg->length, 4), sizeof(msgbuf));
|
||||||
|
memcpy_fromio(msgbuf, msg->message, len);
|
||||||
|
msgbuf[len - 1] = 0;
|
||||||
|
|
||||||
|
if (!strcmp(msgbuf, "resource does not exist"))
|
||||||
|
status = -ENXIO;
|
||||||
|
else
|
||||||
|
status = -EINVAL;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
buf = PTR_ALIGN(buf + 2 * sizeof(u32) + msg->length, 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
rpm->ack_status = status;
|
||||||
|
complete(&rpm->ack);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int qcom_smd_rpm_probe(struct qcom_smd_device *sdev)
|
||||||
|
{
|
||||||
|
struct qcom_smd_rpm *rpm;
|
||||||
|
|
||||||
|
rpm = devm_kzalloc(&sdev->dev, sizeof(*rpm), GFP_KERNEL);
|
||||||
|
if (!rpm)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
mutex_init(&rpm->lock);
|
||||||
|
init_completion(&rpm->ack);
|
||||||
|
|
||||||
|
rpm->rpm_channel = sdev->channel;
|
||||||
|
|
||||||
|
dev_set_drvdata(&sdev->dev, rpm);
|
||||||
|
|
||||||
|
return of_platform_populate(sdev->dev.of_node, NULL, NULL, &sdev->dev);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void qcom_smd_rpm_remove(struct qcom_smd_device *sdev)
|
||||||
|
{
|
||||||
|
of_platform_depopulate(&sdev->dev);
|
||||||
|
}
|
||||||
|
|
||||||
|
static const struct of_device_id qcom_smd_rpm_of_match[] = {
|
||||||
|
{ .compatible = "qcom,rpm-msm8974" },
|
||||||
|
{}
|
||||||
|
};
|
||||||
|
MODULE_DEVICE_TABLE(of, qcom_smd_rpm_of_match);
|
||||||
|
|
||||||
|
static struct qcom_smd_driver qcom_smd_rpm_driver = {
|
||||||
|
.probe = qcom_smd_rpm_probe,
|
||||||
|
.remove = qcom_smd_rpm_remove,
|
||||||
|
.callback = qcom_smd_rpm_callback,
|
||||||
|
.driver = {
|
||||||
|
.name = "qcom_smd_rpm",
|
||||||
|
.owner = THIS_MODULE,
|
||||||
|
.of_match_table = qcom_smd_rpm_of_match,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
static int __init qcom_smd_rpm_init(void)
|
||||||
|
{
|
||||||
|
return qcom_smd_driver_register(&qcom_smd_rpm_driver);
|
||||||
|
}
|
||||||
|
arch_initcall(qcom_smd_rpm_init);
|
||||||
|
|
||||||
|
static void __exit qcom_smd_rpm_exit(void)
|
||||||
|
{
|
||||||
|
qcom_smd_driver_unregister(&qcom_smd_rpm_driver);
|
||||||
|
}
|
||||||
|
module_exit(qcom_smd_rpm_exit);
|
||||||
|
|
||||||
|
MODULE_AUTHOR("Bjorn Andersson <bjorn.andersson@sonymobile.com>");
|
||||||
|
MODULE_DESCRIPTION("Qualcomm SMD backed RPM driver");
|
||||||
|
MODULE_LICENSE("GPL v2");
|
|
@ -0,0 +1,35 @@
|
||||||
|
#ifndef __QCOM_SMD_RPM_H__
|
||||||
|
#define __QCOM_SMD_RPM_H__
|
||||||
|
|
||||||
|
struct qcom_smd_rpm;
|
||||||
|
|
||||||
|
#define QCOM_SMD_RPM_ACTIVE_STATE 0
|
||||||
|
#define QCOM_SMD_RPM_SLEEP_STATE 1
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Constants used for addressing resources in the RPM.
|
||||||
|
*/
|
||||||
|
#define QCOM_SMD_RPM_BOOST 0x61747362
|
||||||
|
#define QCOM_SMD_RPM_BUS_CLK 0x316b6c63
|
||||||
|
#define QCOM_SMD_RPM_BUS_MASTER 0x73616d62
|
||||||
|
#define QCOM_SMD_RPM_BUS_SLAVE 0x766c7362
|
||||||
|
#define QCOM_SMD_RPM_CLK_BUF_A 0x616B6C63
|
||||||
|
#define QCOM_SMD_RPM_LDOA 0x616f646c
|
||||||
|
#define QCOM_SMD_RPM_LDOB 0x626F646C
|
||||||
|
#define QCOM_SMD_RPM_MEM_CLK 0x326b6c63
|
||||||
|
#define QCOM_SMD_RPM_MISC_CLK 0x306b6c63
|
||||||
|
#define QCOM_SMD_RPM_NCPA 0x6170636E
|
||||||
|
#define QCOM_SMD_RPM_NCPB 0x6270636E
|
||||||
|
#define QCOM_SMD_RPM_OCMEM_PWR 0x706d636f
|
||||||
|
#define QCOM_SMD_RPM_QPIC_CLK 0x63697071
|
||||||
|
#define QCOM_SMD_RPM_SMPA 0x61706d73
|
||||||
|
#define QCOM_SMD_RPM_SMPB 0x62706d73
|
||||||
|
#define QCOM_SMD_RPM_SPDM 0x63707362
|
||||||
|
#define QCOM_SMD_RPM_VSA 0x00617376
|
||||||
|
|
||||||
|
int qcom_rpm_smd_write(struct qcom_smd_rpm *rpm,
|
||||||
|
int state,
|
||||||
|
u32 resource_type, u32 resource_id,
|
||||||
|
void *buf, size_t count);
|
||||||
|
|
||||||
|
#endif
|
Loading…
Reference in New Issue