You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
kernel_samsung_sm7125/drivers/soc/qcom/adsp_vote_qmi.c

259 lines
5.9 KiB

/*
* Copyright (c) 2019, 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/kernel.h>
#include <linux/module.h>
#include <linux/device.h>
#include <linux/platform_device.h>
#include <linux/workqueue.h>
#include <linux/soc/qcom/qmi.h>
#include <linux/slab.h>
#include <linux/iommu.h>
#include <linux/cpu_pm.h>
#include <linux/pm.h>
#include "adsp_lpm_voting_v01.h"
#define PGS_TIMEOUT msecs_to_jiffies(3000)
static struct sockaddr_qrtr sq;
static int pgs_vote_send_sync_msg(struct qmi_handle *dev, int vote)
{
int ret;
struct prod_set_lpm_vote_req_msg_v01 *req;
struct prod_set_lpm_vote_resp_msg_v01 *resp;
struct qmi_txn txn;
if (!dev)
return -ENODEV;
req = kzalloc(sizeof(struct prod_set_lpm_vote_req_msg_v01), GFP_KERNEL);
if (!req)
return -ENOMEM;
resp = kzalloc(sizeof(struct prod_set_lpm_vote_resp_msg_v01),
GFP_KERNEL);
if (!resp) {
kfree(req);
return -ENOMEM;
}
req->keep_adsp_out_of_lpm = (u8) vote;
ret = qmi_txn_init(dev, &txn, prod_set_lpm_vote_resp_msg_v01_ei, resp);
if (ret < 0) {
pr_err("Failed Init txn for Mode resp %d\n", ret);
goto out;
}
ret = qmi_send_request(dev, &sq, &txn,
PROD_SET_LPM_VOTE_REQ_V01,
PROD_SET_LPM_VOTE_REQ_MSG_V01_MAX_MSG_LEN,
prod_set_lpm_vote_req_msg_v01_ei, req);
if (ret < 0) {
qmi_txn_cancel(&txn);
pr_err("Fail to send Mode req %d\n", ret);
goto out;
}
ret = qmi_txn_wait(&txn, PGS_TIMEOUT);
if (ret < 0) {
pr_err("Mode resp wait failed with ret %d\n", ret);
goto out;
}
if (resp->resp.result != QMI_RESULT_SUCCESS_V01) {
pr_err("QMI Mode request rejected, result:%d error:%d\n",
resp->resp.result, resp->resp.error);
ret = -resp->resp.result;
goto out;
}
if (!vote)
pr_info("ADSP manual unvoting is successful\n");
else
pr_info("ADSP manual voting is successful\n");
out:
kfree(req);
kfree(resp);
return ret;
}
static struct qmi_handle adsp_qmi_client;
static int send_adsp_manual_unvote(struct device *dev)
{
int ret = 0;
struct qmi_handle *qmi_dev = (struct qmi_handle *) dev_get_drvdata(dev);
pr_info("ADSP unvoting initiated\n");
ret = pgs_vote_send_sync_msg(qmi_dev, 0);
if (ret < 0)
pr_err("ADSP Unvoting is failed\n");
return 0;
}
static int send_adsp_manual_vote(struct device *dev)
{
int ret = 0;
struct qmi_handle *qmi_dev = (struct qmi_handle *) dev_get_drvdata(dev);
pr_info("ADSP voting initiated\n");
ret = pgs_vote_send_sync_msg(qmi_dev, 1);
if (ret < 0)
pr_err("ADSP voting is failed\n");
return 0;
}
static int qmi_adsp_manual_vote_new_server(struct qmi_handle *qmi,
struct qmi_service *service)
{
struct platform_device *pdev;
int ret;
sq.sq_family = AF_QIPCRTR;
sq.sq_node = service->node;
sq.sq_port = service->port;
pdev = platform_device_alloc("qmi_adsp_vote_client",
PLATFORM_DEVID_AUTO);
if (!pdev)
return -ENOMEM;
ret = platform_device_add_data(pdev, &sq, sizeof(sq));
if (ret)
goto err_put_device;
ret = platform_device_add(pdev);
if (ret)
goto err_put_device;
service->priv = pdev;
return 0;
err_put_device:
platform_device_put(pdev);
return ret;
}
static void qmi_adsp_manual_vote_del_server(struct qmi_handle *qmi,
struct qmi_service *service)
{
struct platform_device *pdev = service->priv;
platform_device_unregister(pdev);
}
static struct qmi_ops adsp_qmi_ops = {
.new_server = qmi_adsp_manual_vote_new_server,
.del_server = qmi_adsp_manual_vote_del_server,
};
static const struct dev_pm_ops adsp_manual_vote_dev_pm_ops = {
SET_NOIRQ_SYSTEM_SLEEP_PM_OPS(send_adsp_manual_unvote,
send_adsp_manual_vote)
};
static int adsp_manual_vote_driver_probe(struct platform_device *pdev)
{
int ret = 0;
struct qmi_handle *dev;
dev = kzalloc(sizeof(*dev), GFP_KERNEL);
if (!dev)
return -ENOMEM;
ret = qmi_handle_init(dev,
PROD_SET_LPM_VOTE_REQ_MSG_V01_MAX_MSG_LEN,
&adsp_qmi_ops, NULL);
if (ret < 0) {
pr_err("Successfully got the qmi_handle for the client\n");
goto err_handle;
}
//Register a new lookup with the service PGS_SERVICE_ID_V01
ret = qmi_add_lookup(dev, PGS_SERVICE_ID_V01,
PGS_SERVICE_VERS_V01, 0);
if (ret < 0)
goto err_handle;
platform_set_drvdata(pdev, (void *) dev);
pr_info("ADSP vote device is registered successfully\n");
return ret;
err_handle:
kfree(dev);
return ret;
}
static int adsp_manual_vote_driver_remove(struct platform_device *pdev)
{
struct qmi_handle *dev = (struct qmi_handle *)
platform_get_drvdata(pdev);
qmi_handle_release(dev);
return 0;
}
static struct platform_device adsp_manual_vote_device = {
.name = "adsp_manual_vote",
.id = -1,
};
static struct platform_driver adsp_manual_vote_driver = {
.probe = adsp_manual_vote_driver_probe,
.remove = adsp_manual_vote_driver_remove,
.driver = {
.name = "adsp_manual_vote",
.owner = THIS_MODULE,
.pm = &adsp_manual_vote_dev_pm_ops,
},
};
static int __init adsp_manual_vote_qmi_init(void)
{
int ret = 0;
ret = platform_device_register(&adsp_manual_vote_device);
if (ret)
return -ENODEV;
pr_info("ADSP vote device is registered successfully\n");
ret = platform_driver_register(&adsp_manual_vote_driver);
if (ret) {
pr_err("%s: fail to register ADSP manual vote device driver\n",
__func__);
goto out;
}
out:
return ret;
}
static void __exit adsp_manual_vote_qmi_deinit(void)
{
qmi_handle_release(&adsp_qmi_client);
platform_driver_unregister(&adsp_manual_vote_driver);
platform_device_unregister(&adsp_manual_vote_device);
}
module_init(adsp_manual_vote_qmi_init);
module_exit(adsp_manual_vote_qmi_deinit);
MODULE_LICENSE("GPL v2");
MODULE_DESCRIPTION("ADSP Manual vote QMI client driver");