18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only
28c2ecf20Sopenharmony_ci/*
38c2ecf20Sopenharmony_ci * Copyright (c) 2015, Sony Mobile Communications Inc.
48c2ecf20Sopenharmony_ci * Copyright (c) 2013, The Linux Foundation. All rights reserved.
58c2ecf20Sopenharmony_ci */
68c2ecf20Sopenharmony_ci
78c2ecf20Sopenharmony_ci#include <linux/module.h>
88c2ecf20Sopenharmony_ci#include <linux/skbuff.h>
98c2ecf20Sopenharmony_ci#include <linux/rpmsg.h>
108c2ecf20Sopenharmony_ci
118c2ecf20Sopenharmony_ci#include "qrtr.h"
128c2ecf20Sopenharmony_ci
138c2ecf20Sopenharmony_cistruct qrtr_smd_dev {
148c2ecf20Sopenharmony_ci	struct qrtr_endpoint ep;
158c2ecf20Sopenharmony_ci	struct rpmsg_endpoint *channel;
168c2ecf20Sopenharmony_ci	struct device *dev;
178c2ecf20Sopenharmony_ci};
188c2ecf20Sopenharmony_ci
198c2ecf20Sopenharmony_ci/* from smd to qrtr */
208c2ecf20Sopenharmony_cistatic int qcom_smd_qrtr_callback(struct rpmsg_device *rpdev,
218c2ecf20Sopenharmony_ci				  void *data, int len, void *priv, u32 addr)
228c2ecf20Sopenharmony_ci{
238c2ecf20Sopenharmony_ci	struct qrtr_smd_dev *qdev = dev_get_drvdata(&rpdev->dev);
248c2ecf20Sopenharmony_ci	int rc;
258c2ecf20Sopenharmony_ci
268c2ecf20Sopenharmony_ci	if (!qdev)
278c2ecf20Sopenharmony_ci		return -EAGAIN;
288c2ecf20Sopenharmony_ci
298c2ecf20Sopenharmony_ci	rc = qrtr_endpoint_post(&qdev->ep, data, len);
308c2ecf20Sopenharmony_ci	if (rc == -EINVAL) {
318c2ecf20Sopenharmony_ci		dev_err(qdev->dev, "invalid ipcrouter packet\n");
328c2ecf20Sopenharmony_ci		/* return 0 to let smd drop the packet */
338c2ecf20Sopenharmony_ci		rc = 0;
348c2ecf20Sopenharmony_ci	}
358c2ecf20Sopenharmony_ci
368c2ecf20Sopenharmony_ci	return rc;
378c2ecf20Sopenharmony_ci}
388c2ecf20Sopenharmony_ci
398c2ecf20Sopenharmony_ci/* from qrtr to smd */
408c2ecf20Sopenharmony_cistatic int qcom_smd_qrtr_send(struct qrtr_endpoint *ep, struct sk_buff *skb)
418c2ecf20Sopenharmony_ci{
428c2ecf20Sopenharmony_ci	struct qrtr_smd_dev *qdev = container_of(ep, struct qrtr_smd_dev, ep);
438c2ecf20Sopenharmony_ci	int rc;
448c2ecf20Sopenharmony_ci
458c2ecf20Sopenharmony_ci	rc = skb_linearize(skb);
468c2ecf20Sopenharmony_ci	if (rc)
478c2ecf20Sopenharmony_ci		goto out;
488c2ecf20Sopenharmony_ci
498c2ecf20Sopenharmony_ci	rc = rpmsg_send(qdev->channel, skb->data, skb->len);
508c2ecf20Sopenharmony_ci
518c2ecf20Sopenharmony_ciout:
528c2ecf20Sopenharmony_ci	if (rc)
538c2ecf20Sopenharmony_ci		kfree_skb(skb);
548c2ecf20Sopenharmony_ci	else
558c2ecf20Sopenharmony_ci		consume_skb(skb);
568c2ecf20Sopenharmony_ci	return rc;
578c2ecf20Sopenharmony_ci}
588c2ecf20Sopenharmony_ci
598c2ecf20Sopenharmony_cistatic int qcom_smd_qrtr_probe(struct rpmsg_device *rpdev)
608c2ecf20Sopenharmony_ci{
618c2ecf20Sopenharmony_ci	struct qrtr_smd_dev *qdev;
628c2ecf20Sopenharmony_ci	int rc;
638c2ecf20Sopenharmony_ci
648c2ecf20Sopenharmony_ci	qdev = devm_kzalloc(&rpdev->dev, sizeof(*qdev), GFP_KERNEL);
658c2ecf20Sopenharmony_ci	if (!qdev)
668c2ecf20Sopenharmony_ci		return -ENOMEM;
678c2ecf20Sopenharmony_ci
688c2ecf20Sopenharmony_ci	qdev->channel = rpdev->ept;
698c2ecf20Sopenharmony_ci	qdev->dev = &rpdev->dev;
708c2ecf20Sopenharmony_ci	qdev->ep.xmit = qcom_smd_qrtr_send;
718c2ecf20Sopenharmony_ci
728c2ecf20Sopenharmony_ci	rc = qrtr_endpoint_register(&qdev->ep, QRTR_EP_NID_AUTO);
738c2ecf20Sopenharmony_ci	if (rc)
748c2ecf20Sopenharmony_ci		return rc;
758c2ecf20Sopenharmony_ci
768c2ecf20Sopenharmony_ci	dev_set_drvdata(&rpdev->dev, qdev);
778c2ecf20Sopenharmony_ci
788c2ecf20Sopenharmony_ci	dev_dbg(&rpdev->dev, "Qualcomm SMD QRTR driver probed\n");
798c2ecf20Sopenharmony_ci
808c2ecf20Sopenharmony_ci	return 0;
818c2ecf20Sopenharmony_ci}
828c2ecf20Sopenharmony_ci
838c2ecf20Sopenharmony_cistatic void qcom_smd_qrtr_remove(struct rpmsg_device *rpdev)
848c2ecf20Sopenharmony_ci{
858c2ecf20Sopenharmony_ci	struct qrtr_smd_dev *qdev = dev_get_drvdata(&rpdev->dev);
868c2ecf20Sopenharmony_ci
878c2ecf20Sopenharmony_ci	qrtr_endpoint_unregister(&qdev->ep);
888c2ecf20Sopenharmony_ci
898c2ecf20Sopenharmony_ci	dev_set_drvdata(&rpdev->dev, NULL);
908c2ecf20Sopenharmony_ci}
918c2ecf20Sopenharmony_ci
928c2ecf20Sopenharmony_cistatic const struct rpmsg_device_id qcom_smd_qrtr_smd_match[] = {
938c2ecf20Sopenharmony_ci	{ "IPCRTR" },
948c2ecf20Sopenharmony_ci	{}
958c2ecf20Sopenharmony_ci};
968c2ecf20Sopenharmony_ci
978c2ecf20Sopenharmony_cistatic struct rpmsg_driver qcom_smd_qrtr_driver = {
988c2ecf20Sopenharmony_ci	.probe = qcom_smd_qrtr_probe,
998c2ecf20Sopenharmony_ci	.remove = qcom_smd_qrtr_remove,
1008c2ecf20Sopenharmony_ci	.callback = qcom_smd_qrtr_callback,
1018c2ecf20Sopenharmony_ci	.id_table = qcom_smd_qrtr_smd_match,
1028c2ecf20Sopenharmony_ci	.drv = {
1038c2ecf20Sopenharmony_ci		.name = "qcom_smd_qrtr",
1048c2ecf20Sopenharmony_ci	},
1058c2ecf20Sopenharmony_ci};
1068c2ecf20Sopenharmony_ci
1078c2ecf20Sopenharmony_cimodule_rpmsg_driver(qcom_smd_qrtr_driver);
1088c2ecf20Sopenharmony_ci
1098c2ecf20Sopenharmony_ciMODULE_ALIAS("rpmsg:IPCRTR");
1108c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("Qualcomm IPC-Router SMD interface driver");
1118c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL v2");
112