mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-01 10:43:43 +00:00
45e7d78ef5
The .remove() callback for a platform driver returns an int which makes many driver authors wrongly assume it's possible to do error handling by returning an error code. However the value returned is ignored (apart from emitting a warning) and this typically results in resource leaks. To improve here there is a quest to make the remove callback return void. In the first step of this quest all drivers are converted to .remove_new(), which already returns void. Eventually after all drivers are converted, .remove_new() will be renamed to .remove(). Trivially convert this driver from always returning zero in the remove callback to the void returning variant. Link: https://lore.kernel.org/r/d30beb557e0e97ea194028f62d3c4c10841d3e7c.1709886922.git.u.kleine-koenig@pengutronix.de Signed-off-by: Uwe Kleine-König <u.kleine-koenig@pengutronix.de>
621 lines
14 KiB
C
621 lines
14 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* Sample in-kernel QMI client driver
|
|
*
|
|
* Copyright (c) 2013-2014, The Linux Foundation. All rights reserved.
|
|
* Copyright (C) 2017 Linaro Ltd.
|
|
*/
|
|
#include <linux/kernel.h>
|
|
#include <linux/module.h>
|
|
#include <linux/debugfs.h>
|
|
#include <linux/device.h>
|
|
#include <linux/platform_device.h>
|
|
#include <linux/qrtr.h>
|
|
#include <linux/net.h>
|
|
#include <linux/completion.h>
|
|
#include <linux/idr.h>
|
|
#include <linux/string.h>
|
|
#include <net/sock.h>
|
|
#include <linux/soc/qcom/qmi.h>
|
|
|
|
#define PING_REQ1_TLV_TYPE 0x1
|
|
#define PING_RESP1_TLV_TYPE 0x2
|
|
#define PING_OPT1_TLV_TYPE 0x10
|
|
#define PING_OPT2_TLV_TYPE 0x11
|
|
|
|
#define DATA_REQ1_TLV_TYPE 0x1
|
|
#define DATA_RESP1_TLV_TYPE 0x2
|
|
#define DATA_OPT1_TLV_TYPE 0x10
|
|
#define DATA_OPT2_TLV_TYPE 0x11
|
|
|
|
#define TEST_MED_DATA_SIZE_V01 8192
|
|
#define TEST_MAX_NAME_SIZE_V01 255
|
|
|
|
#define TEST_PING_REQ_MSG_ID_V01 0x20
|
|
#define TEST_DATA_REQ_MSG_ID_V01 0x21
|
|
|
|
#define TEST_PING_REQ_MAX_MSG_LEN_V01 266
|
|
#define TEST_DATA_REQ_MAX_MSG_LEN_V01 8456
|
|
|
|
struct test_name_type_v01 {
|
|
u32 name_len;
|
|
char name[TEST_MAX_NAME_SIZE_V01];
|
|
};
|
|
|
|
static const struct qmi_elem_info test_name_type_v01_ei[] = {
|
|
{
|
|
.data_type = QMI_DATA_LEN,
|
|
.elem_len = 1,
|
|
.elem_size = sizeof(u8),
|
|
.array_type = NO_ARRAY,
|
|
.tlv_type = QMI_COMMON_TLV_TYPE,
|
|
.offset = offsetof(struct test_name_type_v01,
|
|
name_len),
|
|
},
|
|
{
|
|
.data_type = QMI_UNSIGNED_1_BYTE,
|
|
.elem_len = TEST_MAX_NAME_SIZE_V01,
|
|
.elem_size = sizeof(char),
|
|
.array_type = VAR_LEN_ARRAY,
|
|
.tlv_type = QMI_COMMON_TLV_TYPE,
|
|
.offset = offsetof(struct test_name_type_v01,
|
|
name),
|
|
},
|
|
{}
|
|
};
|
|
|
|
struct test_ping_req_msg_v01 {
|
|
char ping[4];
|
|
|
|
u8 client_name_valid;
|
|
struct test_name_type_v01 client_name;
|
|
};
|
|
|
|
static const struct qmi_elem_info test_ping_req_msg_v01_ei[] = {
|
|
{
|
|
.data_type = QMI_UNSIGNED_1_BYTE,
|
|
.elem_len = 4,
|
|
.elem_size = sizeof(char),
|
|
.array_type = STATIC_ARRAY,
|
|
.tlv_type = PING_REQ1_TLV_TYPE,
|
|
.offset = offsetof(struct test_ping_req_msg_v01,
|
|
ping),
|
|
},
|
|
{
|
|
.data_type = QMI_OPT_FLAG,
|
|
.elem_len = 1,
|
|
.elem_size = sizeof(u8),
|
|
.array_type = NO_ARRAY,
|
|
.tlv_type = PING_OPT1_TLV_TYPE,
|
|
.offset = offsetof(struct test_ping_req_msg_v01,
|
|
client_name_valid),
|
|
},
|
|
{
|
|
.data_type = QMI_STRUCT,
|
|
.elem_len = 1,
|
|
.elem_size = sizeof(struct test_name_type_v01),
|
|
.array_type = NO_ARRAY,
|
|
.tlv_type = PING_OPT1_TLV_TYPE,
|
|
.offset = offsetof(struct test_ping_req_msg_v01,
|
|
client_name),
|
|
.ei_array = test_name_type_v01_ei,
|
|
},
|
|
{}
|
|
};
|
|
|
|
struct test_ping_resp_msg_v01 {
|
|
struct qmi_response_type_v01 resp;
|
|
|
|
u8 pong_valid;
|
|
char pong[4];
|
|
|
|
u8 service_name_valid;
|
|
struct test_name_type_v01 service_name;
|
|
};
|
|
|
|
static const struct qmi_elem_info test_ping_resp_msg_v01_ei[] = {
|
|
{
|
|
.data_type = QMI_STRUCT,
|
|
.elem_len = 1,
|
|
.elem_size = sizeof(struct qmi_response_type_v01),
|
|
.array_type = NO_ARRAY,
|
|
.tlv_type = PING_RESP1_TLV_TYPE,
|
|
.offset = offsetof(struct test_ping_resp_msg_v01,
|
|
resp),
|
|
.ei_array = qmi_response_type_v01_ei,
|
|
},
|
|
{
|
|
.data_type = QMI_OPT_FLAG,
|
|
.elem_len = 1,
|
|
.elem_size = sizeof(u8),
|
|
.array_type = NO_ARRAY,
|
|
.tlv_type = PING_OPT1_TLV_TYPE,
|
|
.offset = offsetof(struct test_ping_resp_msg_v01,
|
|
pong_valid),
|
|
},
|
|
{
|
|
.data_type = QMI_UNSIGNED_1_BYTE,
|
|
.elem_len = 4,
|
|
.elem_size = sizeof(char),
|
|
.array_type = STATIC_ARRAY,
|
|
.tlv_type = PING_OPT1_TLV_TYPE,
|
|
.offset = offsetof(struct test_ping_resp_msg_v01,
|
|
pong),
|
|
},
|
|
{
|
|
.data_type = QMI_OPT_FLAG,
|
|
.elem_len = 1,
|
|
.elem_size = sizeof(u8),
|
|
.array_type = NO_ARRAY,
|
|
.tlv_type = PING_OPT2_TLV_TYPE,
|
|
.offset = offsetof(struct test_ping_resp_msg_v01,
|
|
service_name_valid),
|
|
},
|
|
{
|
|
.data_type = QMI_STRUCT,
|
|
.elem_len = 1,
|
|
.elem_size = sizeof(struct test_name_type_v01),
|
|
.array_type = NO_ARRAY,
|
|
.tlv_type = PING_OPT2_TLV_TYPE,
|
|
.offset = offsetof(struct test_ping_resp_msg_v01,
|
|
service_name),
|
|
.ei_array = test_name_type_v01_ei,
|
|
},
|
|
{}
|
|
};
|
|
|
|
struct test_data_req_msg_v01 {
|
|
u32 data_len;
|
|
u8 data[TEST_MED_DATA_SIZE_V01];
|
|
|
|
u8 client_name_valid;
|
|
struct test_name_type_v01 client_name;
|
|
};
|
|
|
|
static const struct qmi_elem_info test_data_req_msg_v01_ei[] = {
|
|
{
|
|
.data_type = QMI_DATA_LEN,
|
|
.elem_len = 1,
|
|
.elem_size = sizeof(u32),
|
|
.array_type = NO_ARRAY,
|
|
.tlv_type = DATA_REQ1_TLV_TYPE,
|
|
.offset = offsetof(struct test_data_req_msg_v01,
|
|
data_len),
|
|
},
|
|
{
|
|
.data_type = QMI_UNSIGNED_1_BYTE,
|
|
.elem_len = TEST_MED_DATA_SIZE_V01,
|
|
.elem_size = sizeof(u8),
|
|
.array_type = VAR_LEN_ARRAY,
|
|
.tlv_type = DATA_REQ1_TLV_TYPE,
|
|
.offset = offsetof(struct test_data_req_msg_v01,
|
|
data),
|
|
},
|
|
{
|
|
.data_type = QMI_OPT_FLAG,
|
|
.elem_len = 1,
|
|
.elem_size = sizeof(u8),
|
|
.array_type = NO_ARRAY,
|
|
.tlv_type = DATA_OPT1_TLV_TYPE,
|
|
.offset = offsetof(struct test_data_req_msg_v01,
|
|
client_name_valid),
|
|
},
|
|
{
|
|
.data_type = QMI_STRUCT,
|
|
.elem_len = 1,
|
|
.elem_size = sizeof(struct test_name_type_v01),
|
|
.array_type = NO_ARRAY,
|
|
.tlv_type = DATA_OPT1_TLV_TYPE,
|
|
.offset = offsetof(struct test_data_req_msg_v01,
|
|
client_name),
|
|
.ei_array = test_name_type_v01_ei,
|
|
},
|
|
{}
|
|
};
|
|
|
|
struct test_data_resp_msg_v01 {
|
|
struct qmi_response_type_v01 resp;
|
|
|
|
u8 data_valid;
|
|
u32 data_len;
|
|
u8 data[TEST_MED_DATA_SIZE_V01];
|
|
|
|
u8 service_name_valid;
|
|
struct test_name_type_v01 service_name;
|
|
};
|
|
|
|
static const struct qmi_elem_info test_data_resp_msg_v01_ei[] = {
|
|
{
|
|
.data_type = QMI_STRUCT,
|
|
.elem_len = 1,
|
|
.elem_size = sizeof(struct qmi_response_type_v01),
|
|
.array_type = NO_ARRAY,
|
|
.tlv_type = DATA_RESP1_TLV_TYPE,
|
|
.offset = offsetof(struct test_data_resp_msg_v01,
|
|
resp),
|
|
.ei_array = qmi_response_type_v01_ei,
|
|
},
|
|
{
|
|
.data_type = QMI_OPT_FLAG,
|
|
.elem_len = 1,
|
|
.elem_size = sizeof(u8),
|
|
.array_type = NO_ARRAY,
|
|
.tlv_type = DATA_OPT1_TLV_TYPE,
|
|
.offset = offsetof(struct test_data_resp_msg_v01,
|
|
data_valid),
|
|
},
|
|
{
|
|
.data_type = QMI_DATA_LEN,
|
|
.elem_len = 1,
|
|
.elem_size = sizeof(u32),
|
|
.array_type = NO_ARRAY,
|
|
.tlv_type = DATA_OPT1_TLV_TYPE,
|
|
.offset = offsetof(struct test_data_resp_msg_v01,
|
|
data_len),
|
|
},
|
|
{
|
|
.data_type = QMI_UNSIGNED_1_BYTE,
|
|
.elem_len = TEST_MED_DATA_SIZE_V01,
|
|
.elem_size = sizeof(u8),
|
|
.array_type = VAR_LEN_ARRAY,
|
|
.tlv_type = DATA_OPT1_TLV_TYPE,
|
|
.offset = offsetof(struct test_data_resp_msg_v01,
|
|
data),
|
|
},
|
|
{
|
|
.data_type = QMI_OPT_FLAG,
|
|
.elem_len = 1,
|
|
.elem_size = sizeof(u8),
|
|
.array_type = NO_ARRAY,
|
|
.tlv_type = DATA_OPT2_TLV_TYPE,
|
|
.offset = offsetof(struct test_data_resp_msg_v01,
|
|
service_name_valid),
|
|
},
|
|
{
|
|
.data_type = QMI_STRUCT,
|
|
.elem_len = 1,
|
|
.elem_size = sizeof(struct test_name_type_v01),
|
|
.array_type = NO_ARRAY,
|
|
.tlv_type = DATA_OPT2_TLV_TYPE,
|
|
.offset = offsetof(struct test_data_resp_msg_v01,
|
|
service_name),
|
|
.ei_array = test_name_type_v01_ei,
|
|
},
|
|
{}
|
|
};
|
|
|
|
/*
|
|
* ping_write() - ping_pong debugfs file write handler
|
|
* @file: debugfs file context
|
|
* @user_buf: reference to the user data (ignored)
|
|
* @count: number of bytes in @user_buf
|
|
* @ppos: offset in @file to write
|
|
*
|
|
* This function allows user space to send out a ping_pong QMI encoded message
|
|
* to the associated remote test service and will return with the result of the
|
|
* transaction. It serves as an example of how to provide a custom response
|
|
* handler.
|
|
*
|
|
* Return: @count, or negative errno on failure.
|
|
*/
|
|
static ssize_t ping_write(struct file *file, const char __user *user_buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
struct qmi_handle *qmi = file->private_data;
|
|
struct test_ping_req_msg_v01 req = {};
|
|
struct qmi_txn txn;
|
|
int ret;
|
|
|
|
memcpy(req.ping, "ping", sizeof(req.ping));
|
|
|
|
ret = qmi_txn_init(qmi, &txn, NULL, NULL);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
ret = qmi_send_request(qmi, NULL, &txn,
|
|
TEST_PING_REQ_MSG_ID_V01,
|
|
TEST_PING_REQ_MAX_MSG_LEN_V01,
|
|
test_ping_req_msg_v01_ei, &req);
|
|
if (ret < 0) {
|
|
qmi_txn_cancel(&txn);
|
|
return ret;
|
|
}
|
|
|
|
ret = qmi_txn_wait(&txn, 5 * HZ);
|
|
if (ret < 0)
|
|
count = ret;
|
|
|
|
return count;
|
|
}
|
|
|
|
static const struct file_operations ping_fops = {
|
|
.open = simple_open,
|
|
.write = ping_write,
|
|
};
|
|
|
|
static void ping_pong_cb(struct qmi_handle *qmi, struct sockaddr_qrtr *sq,
|
|
struct qmi_txn *txn, const void *data)
|
|
{
|
|
const struct test_ping_resp_msg_v01 *resp = data;
|
|
|
|
if (!txn) {
|
|
pr_err("spurious ping response\n");
|
|
return;
|
|
}
|
|
|
|
if (resp->resp.result == QMI_RESULT_FAILURE_V01)
|
|
txn->result = -ENXIO;
|
|
else if (!resp->pong_valid || memcmp(resp->pong, "pong", 4))
|
|
txn->result = -EINVAL;
|
|
|
|
complete(&txn->completion);
|
|
}
|
|
|
|
/*
|
|
* data_write() - data debugfs file write handler
|
|
* @file: debugfs file context
|
|
* @user_buf: reference to the user data
|
|
* @count: number of bytes in @user_buf
|
|
* @ppos: offset in @file to write
|
|
*
|
|
* This function allows user space to send out a data QMI encoded message to
|
|
* the associated remote test service and will return with the result of the
|
|
* transaction. It serves as an example of how to have the QMI helpers decode a
|
|
* transaction response into a provided object automatically.
|
|
*
|
|
* Return: @count, or negative errno on failure.
|
|
*/
|
|
static ssize_t data_write(struct file *file, const char __user *user_buf,
|
|
size_t count, loff_t *ppos)
|
|
|
|
{
|
|
struct qmi_handle *qmi = file->private_data;
|
|
struct test_data_resp_msg_v01 *resp;
|
|
struct test_data_req_msg_v01 *req;
|
|
struct qmi_txn txn;
|
|
int ret;
|
|
|
|
req = kzalloc(sizeof(*req), GFP_KERNEL);
|
|
if (!req)
|
|
return -ENOMEM;
|
|
|
|
resp = kzalloc(sizeof(*resp), GFP_KERNEL);
|
|
if (!resp) {
|
|
kfree(req);
|
|
return -ENOMEM;
|
|
}
|
|
|
|
req->data_len = min_t(size_t, sizeof(req->data), count);
|
|
if (copy_from_user(req->data, user_buf, req->data_len)) {
|
|
ret = -EFAULT;
|
|
goto out;
|
|
}
|
|
|
|
ret = qmi_txn_init(qmi, &txn, test_data_resp_msg_v01_ei, resp);
|
|
if (ret < 0)
|
|
goto out;
|
|
|
|
ret = qmi_send_request(qmi, NULL, &txn,
|
|
TEST_DATA_REQ_MSG_ID_V01,
|
|
TEST_DATA_REQ_MAX_MSG_LEN_V01,
|
|
test_data_req_msg_v01_ei, req);
|
|
if (ret < 0) {
|
|
qmi_txn_cancel(&txn);
|
|
goto out;
|
|
}
|
|
|
|
ret = qmi_txn_wait(&txn, 5 * HZ);
|
|
if (ret < 0) {
|
|
goto out;
|
|
} else if (!resp->data_valid ||
|
|
resp->data_len != req->data_len ||
|
|
memcmp(resp->data, req->data, req->data_len)) {
|
|
pr_err("response data doesn't match expectation\n");
|
|
ret = -EINVAL;
|
|
goto out;
|
|
}
|
|
|
|
ret = count;
|
|
|
|
out:
|
|
kfree(resp);
|
|
kfree(req);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static const struct file_operations data_fops = {
|
|
.open = simple_open,
|
|
.write = data_write,
|
|
};
|
|
|
|
static const struct qmi_msg_handler qmi_sample_handlers[] = {
|
|
{
|
|
.type = QMI_RESPONSE,
|
|
.msg_id = TEST_PING_REQ_MSG_ID_V01,
|
|
.ei = test_ping_resp_msg_v01_ei,
|
|
.decoded_size = sizeof(struct test_ping_req_msg_v01),
|
|
.fn = ping_pong_cb
|
|
},
|
|
{}
|
|
};
|
|
|
|
struct qmi_sample {
|
|
struct qmi_handle qmi;
|
|
|
|
struct dentry *de_dir;
|
|
struct dentry *de_data;
|
|
struct dentry *de_ping;
|
|
};
|
|
|
|
static struct dentry *qmi_debug_dir;
|
|
|
|
static int qmi_sample_probe(struct platform_device *pdev)
|
|
{
|
|
struct sockaddr_qrtr *sq;
|
|
struct qmi_sample *sample;
|
|
char path[20];
|
|
int ret;
|
|
|
|
sample = devm_kzalloc(&pdev->dev, sizeof(*sample), GFP_KERNEL);
|
|
if (!sample)
|
|
return -ENOMEM;
|
|
|
|
ret = qmi_handle_init(&sample->qmi, TEST_DATA_REQ_MAX_MSG_LEN_V01,
|
|
NULL,
|
|
qmi_sample_handlers);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
sq = dev_get_platdata(&pdev->dev);
|
|
ret = kernel_connect(sample->qmi.sock, (struct sockaddr *)sq,
|
|
sizeof(*sq), 0);
|
|
if (ret < 0) {
|
|
pr_err("failed to connect to remote service port\n");
|
|
goto err_release_qmi_handle;
|
|
}
|
|
|
|
snprintf(path, sizeof(path), "%d:%d", sq->sq_node, sq->sq_port);
|
|
|
|
sample->de_dir = debugfs_create_dir(path, qmi_debug_dir);
|
|
if (IS_ERR(sample->de_dir)) {
|
|
ret = PTR_ERR(sample->de_dir);
|
|
goto err_release_qmi_handle;
|
|
}
|
|
|
|
sample->de_data = debugfs_create_file("data", 0600, sample->de_dir,
|
|
sample, &data_fops);
|
|
if (IS_ERR(sample->de_data)) {
|
|
ret = PTR_ERR(sample->de_data);
|
|
goto err_remove_de_dir;
|
|
}
|
|
|
|
sample->de_ping = debugfs_create_file("ping", 0600, sample->de_dir,
|
|
sample, &ping_fops);
|
|
if (IS_ERR(sample->de_ping)) {
|
|
ret = PTR_ERR(sample->de_ping);
|
|
goto err_remove_de_data;
|
|
}
|
|
|
|
platform_set_drvdata(pdev, sample);
|
|
|
|
return 0;
|
|
|
|
err_remove_de_data:
|
|
debugfs_remove(sample->de_data);
|
|
err_remove_de_dir:
|
|
debugfs_remove(sample->de_dir);
|
|
err_release_qmi_handle:
|
|
qmi_handle_release(&sample->qmi);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void qmi_sample_remove(struct platform_device *pdev)
|
|
{
|
|
struct qmi_sample *sample = platform_get_drvdata(pdev);
|
|
|
|
debugfs_remove(sample->de_ping);
|
|
debugfs_remove(sample->de_data);
|
|
debugfs_remove(sample->de_dir);
|
|
|
|
qmi_handle_release(&sample->qmi);
|
|
}
|
|
|
|
static struct platform_driver qmi_sample_driver = {
|
|
.probe = qmi_sample_probe,
|
|
.remove_new = qmi_sample_remove,
|
|
.driver = {
|
|
.name = "qmi_sample_client",
|
|
},
|
|
};
|
|
|
|
static int qmi_sample_new_server(struct qmi_handle *qmi,
|
|
struct qmi_service *service)
|
|
{
|
|
struct platform_device *pdev;
|
|
struct sockaddr_qrtr sq = { AF_QIPCRTR, service->node, service->port };
|
|
int ret;
|
|
|
|
pdev = platform_device_alloc("qmi_sample_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_sample_del_server(struct qmi_handle *qmi,
|
|
struct qmi_service *service)
|
|
{
|
|
struct platform_device *pdev = service->priv;
|
|
|
|
platform_device_unregister(pdev);
|
|
}
|
|
|
|
static struct qmi_handle lookup_client;
|
|
|
|
static const struct qmi_ops lookup_ops = {
|
|
.new_server = qmi_sample_new_server,
|
|
.del_server = qmi_sample_del_server,
|
|
};
|
|
|
|
static int qmi_sample_init(void)
|
|
{
|
|
int ret;
|
|
|
|
qmi_debug_dir = debugfs_create_dir("qmi_sample", NULL);
|
|
if (IS_ERR(qmi_debug_dir)) {
|
|
pr_err("failed to create qmi_sample dir\n");
|
|
return PTR_ERR(qmi_debug_dir);
|
|
}
|
|
|
|
ret = platform_driver_register(&qmi_sample_driver);
|
|
if (ret)
|
|
goto err_remove_debug_dir;
|
|
|
|
ret = qmi_handle_init(&lookup_client, 0, &lookup_ops, NULL);
|
|
if (ret < 0)
|
|
goto err_unregister_driver;
|
|
|
|
qmi_add_lookup(&lookup_client, 15, 0, 0);
|
|
|
|
return 0;
|
|
|
|
err_unregister_driver:
|
|
platform_driver_unregister(&qmi_sample_driver);
|
|
err_remove_debug_dir:
|
|
debugfs_remove(qmi_debug_dir);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void qmi_sample_exit(void)
|
|
{
|
|
qmi_handle_release(&lookup_client);
|
|
|
|
platform_driver_unregister(&qmi_sample_driver);
|
|
|
|
debugfs_remove(qmi_debug_dir);
|
|
}
|
|
|
|
module_init(qmi_sample_init);
|
|
module_exit(qmi_sample_exit);
|
|
|
|
MODULE_DESCRIPTION("Sample QMI client driver");
|
|
MODULE_LICENSE("GPL v2");
|