cregit-Linux how code gets into the kernel

Release 4.17 samples/qmi/qmi_sample_client.c

Directory: samples/qmi
// 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 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 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 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 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 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; }

Contributors

PersonTokensPropCommitsCommitProp
Björn Andersson148100.00%1100.00%
Total148100.00%1100.00%

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); }

Contributors

PersonTokensPropCommitsCommitProp
Björn Andersson98100.00%1100.00%
Total98100.00%1100.00%

/* * 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; }

Contributors

PersonTokensPropCommitsCommitProp
Björn Andersson292100.00%1100.00%
Total292100.00%1100.00%

static const struct file_operations data_fops = { .open = simple_open, .write = data_write, }; static 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; }

Contributors

PersonTokensPropCommitsCommitProp
Björn Andersson315100.00%1100.00%
Total315100.00%1100.00%


static int 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); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Björn Andersson53100.00%1100.00%
Total53100.00%1100.00%

static struct platform_driver qmi_sample_driver = { .probe = qmi_sample_probe, .remove = 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; }

Contributors

PersonTokensPropCommitsCommitProp
Björn Andersson113100.00%1100.00%
Total113100.00%1100.00%


static void qmi_sample_del_server(struct qmi_handle *qmi, struct qmi_service *service) { struct platform_device *pdev = service->priv; platform_device_unregister(pdev); }

Contributors

PersonTokensPropCommitsCommitProp
Björn Andersson30100.00%1100.00%
Total30100.00%1100.00%

static struct qmi_handle lookup_client; static 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; }

Contributors

PersonTokensPropCommitsCommitProp
Björn Andersson112100.00%1100.00%
Total112100.00%1100.00%


static void qmi_sample_exit(void) { qmi_handle_release(&lookup_client); platform_driver_unregister(&qmi_sample_driver); debugfs_remove(qmi_debug_dir); }

Contributors

PersonTokensPropCommitsCommitProp
Björn Andersson25100.00%1100.00%
Total25100.00%1100.00%

module_init(qmi_sample_init); module_exit(qmi_sample_exit); MODULE_DESCRIPTION("Sample QMI client driver"); MODULE_LICENSE("GPL v2");

Overall Contributors

PersonTokensPropCommitsCommitProp
Björn Andersson2470100.00%1100.00%
Total2470100.00%1100.00%
Directory: samples/qmi
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.