smd.c 2.7 KB
Newer Older
C
Courtney Cavin 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
/*
 * Copyright (c) 2015, Sony Mobile Communications Inc.
 * Copyright (c) 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/skbuff.h>
#include <linux/soc/qcom/smd.h>

#include "qrtr.h"

struct qrtr_smd_dev {
	struct qrtr_endpoint ep;
	struct qcom_smd_channel *channel;
24
	struct device *dev;
C
Courtney Cavin 已提交
25 26 27
};

/* from smd to qrtr */
28
static int qcom_smd_qrtr_callback(struct qcom_smd_channel *channel,
C
Courtney Cavin 已提交
29 30
				  const void *data, size_t len)
{
31
	struct qrtr_smd_dev *qdev = qcom_smd_get_drvdata(channel);
C
Courtney Cavin 已提交
32 33 34 35 36 37 38
	int rc;

	if (!qdev)
		return -EAGAIN;

	rc = qrtr_endpoint_post(&qdev->ep, data, len);
	if (rc == -EINVAL) {
39
		dev_err(qdev->dev, "invalid ipcrouter packet\n");
C
Courtney Cavin 已提交
40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76
		/* return 0 to let smd drop the packet */
		rc = 0;
	}

	return rc;
}

/* from qrtr to smd */
static int qcom_smd_qrtr_send(struct qrtr_endpoint *ep, struct sk_buff *skb)
{
	struct qrtr_smd_dev *qdev = container_of(ep, struct qrtr_smd_dev, ep);
	int rc;

	rc = skb_linearize(skb);
	if (rc)
		goto out;

	rc = qcom_smd_send(qdev->channel, skb->data, skb->len);

out:
	if (rc)
		kfree_skb(skb);
	else
		consume_skb(skb);
	return rc;
}

static int qcom_smd_qrtr_probe(struct qcom_smd_device *sdev)
{
	struct qrtr_smd_dev *qdev;
	int rc;

	qdev = devm_kzalloc(&sdev->dev, sizeof(*qdev), GFP_KERNEL);
	if (!qdev)
		return -ENOMEM;

	qdev->channel = sdev->channel;
77
	qdev->dev = &sdev->dev;
C
Courtney Cavin 已提交
78 79 80 81 82 83
	qdev->ep.xmit = qcom_smd_qrtr_send;

	rc = qrtr_endpoint_register(&qdev->ep, QRTR_EP_NID_AUTO);
	if (rc)
		return rc;

84
	qcom_smd_set_drvdata(sdev->channel, qdev);
C
Courtney Cavin 已提交
85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120
	dev_set_drvdata(&sdev->dev, qdev);

	dev_dbg(&sdev->dev, "Qualcomm SMD QRTR driver probed\n");

	return 0;
}

static void qcom_smd_qrtr_remove(struct qcom_smd_device *sdev)
{
	struct qrtr_smd_dev *qdev = dev_get_drvdata(&sdev->dev);

	qrtr_endpoint_unregister(&qdev->ep);

	dev_set_drvdata(&sdev->dev, NULL);
}

static const struct qcom_smd_id qcom_smd_qrtr_smd_match[] = {
	{ "IPCRTR" },
	{}
};

static struct qcom_smd_driver qcom_smd_qrtr_driver = {
	.probe = qcom_smd_qrtr_probe,
	.remove = qcom_smd_qrtr_remove,
	.callback = qcom_smd_qrtr_callback,
	.smd_match_table = qcom_smd_qrtr_smd_match,
	.driver = {
		.name = "qcom_smd_qrtr",
		.owner = THIS_MODULE,
	},
};

module_qcom_smd_driver(qcom_smd_qrtr_driver);

MODULE_DESCRIPTION("Qualcomm IPC-Router SMD interface driver");
MODULE_LICENSE("GPL v2");