blob: d070e37c9142643417e874eb8eba746edac923a5 [file] [log] [blame]
Karthikeyan Ramasubramaniana9859e82012-07-12 13:10:42 -06001/* Copyright (c) 2013, The Linux Foundation. All rights reserved.
2 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 and
5 * only version 2 as published by the Free Software Foundation.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 *
12 */
13
14#include <linux/module.h>
15#include <linux/slab.h>
16#include <linux/errno.h>
17#include <linux/delay.h>
18#include <linux/debugfs.h>
19#include <linux/qmi_encdec.h>
20
21#include <asm/uaccess.h>
22
23#include <mach/msm_qmi_interface.h>
24
25#include "kernel_test_service_v01.h"
26
27#define TEST_SERVICE_SVC_ID 0x0000000f
28#define TEST_SERVICE_INS_ID 1
29
30static int test_rep_cnt = 10;
31module_param_named(rep_cnt, test_rep_cnt, int, S_IRUGO | S_IWUSR | S_IWGRP);
32
33static int test_data_sz = 50;
34module_param_named(data_sz, test_data_sz, int, S_IRUGO | S_IWUSR | S_IWGRP);
35
36static int test_clnt_debug_mask;
37module_param_named(debug_mask, test_clnt_debug_mask,
38 int, S_IRUGO | S_IWUSR | S_IWGRP);
39
40#define D(x...) do { \
41 if (test_clnt_debug_mask) \
42 pr_debug(x); \
43} while (0)
44
45/* Variable to initiate the test through debugfs interface */
46static struct dentry *test_dent;
47
48/* Test client port for IPC Router */
49static struct qmi_handle *test_clnt;
50static int test_clnt_reset;
51
52/* Reader thread to receive responses & indications */
53static void test_clnt_recv_msg(struct work_struct *work);
54static DECLARE_DELAYED_WORK(work_recv_msg, test_clnt_recv_msg);
55static void test_clnt_svc_arrive(struct work_struct *work);
56static DECLARE_DELAYED_WORK(work_svc_arrive, test_clnt_svc_arrive);
57static void test_clnt_svc_exit(struct work_struct *work);
58static DECLARE_DELAYED_WORK(work_svc_exit, test_clnt_svc_exit);
59static struct workqueue_struct *test_clnt_workqueue;
60
61/* Variable to hold the test result */
62static int test_res;
63
64static int test_qmi_ping_pong_send_sync_msg(void)
65{
66 struct test_ping_req_msg_v01 req;
67 struct test_ping_resp_msg_v01 resp;
68 struct msg_desc req_desc, resp_desc;
69 int rc;
70
71 memcpy(req.ping, "ping", sizeof(req.ping));
72 req.client_name_valid = 0;
73
74 req_desc.max_msg_len = TEST_PING_REQ_MAX_MSG_LEN_V01;
75 req_desc.msg_id = TEST_PING_REQ_MSG_ID_V01;
76 req_desc.ei_array = test_ping_req_msg_v01_ei;
77
78 resp_desc.max_msg_len = TEST_PING_REQ_MAX_MSG_LEN_V01;
79 resp_desc.msg_id = TEST_PING_REQ_MSG_ID_V01;
80 resp_desc.ei_array = test_ping_resp_msg_v01_ei;
81
82 rc = qmi_send_req_wait(test_clnt, &req_desc, &req, sizeof(req),
83 &resp_desc, &resp, sizeof(resp), 0);
84 if (rc < 0) {
85 pr_err("%s: send req failed %d\n", __func__, rc);
86 return rc;
87 }
88
89 D("%s: Received %s response\n", __func__, resp.pong);
90 return rc;
91}
92
93static int test_qmi_data_send_sync_msg(unsigned int data_len)
94{
95 struct test_data_req_msg_v01 *req;
96 struct test_data_resp_msg_v01 *resp;
97 struct msg_desc req_desc, resp_desc;
98 int rc, i;
99
100 req = kzalloc(sizeof(struct test_data_req_msg_v01), GFP_KERNEL);
101 if (!req) {
102 pr_err("%s: Data req msg alloc failed\n", __func__);
103 return -ENOMEM;
104 }
105
106 resp = kzalloc(sizeof(struct test_data_resp_msg_v01), GFP_KERNEL);
107 if (!resp) {
108 pr_err("%s: Data resp msg alloc failed\n", __func__);
109 kfree(req);
110 return -ENOMEM;
111 }
112
113 req->data_len = data_len;
114 for (i = 0; i < data_len; i = i + sizeof(int))
115 memcpy(req->data + i, (uint8_t *)&i, sizeof(int));
116 req->client_name_valid = 0;
117
118 req_desc.max_msg_len = TEST_DATA_REQ_MAX_MSG_LEN_V01;
119 req_desc.msg_id = TEST_DATA_REQ_MSG_ID_V01;
120 req_desc.ei_array = test_data_req_msg_v01_ei;
121
122 resp_desc.max_msg_len = TEST_DATA_REQ_MAX_MSG_LEN_V01;
123 resp_desc.msg_id = TEST_DATA_REQ_MSG_ID_V01;
124 resp_desc.ei_array = test_data_resp_msg_v01_ei;
125
126 rc = qmi_send_req_wait(test_clnt, &req_desc, req, sizeof(*req),
127 &resp_desc, resp, sizeof(*resp), 0);
128 if (rc < 0) {
129 pr_err("%s: send req failed\n", __func__);
130 goto data_send_err;
131 }
132
133 D("%s: data_valid %d\n", __func__, resp->data_valid);
134 D("%s: data_len %d\n", __func__, resp->data_len);
135data_send_err:
136 kfree(resp);
137 kfree(req);
138 return rc;
139}
140
141static void test_clnt_recv_msg(struct work_struct *work)
142{
143 int rc;
144
145 rc = qmi_recv_msg(test_clnt);
146 if (rc < 0)
147 pr_err("%s: Error receiving message\n", __func__);
148}
149
150static void test_clnt_notify(struct qmi_handle *handle,
151 enum qmi_event_type event, void *notify_priv)
152{
153 switch (event) {
154 case QMI_RECV_MSG:
155 queue_delayed_work(test_clnt_workqueue,
156 &work_recv_msg, 0);
157 break;
158 default:
159 break;
160 }
161}
162
163static void test_clnt_svc_arrive(struct work_struct *work)
164{
165 int rc;
166
167 D("%s begins\n", __func__);
168
169 /* Create a Local client port for QMI communication */
170 test_clnt = qmi_handle_create(test_clnt_notify, NULL);
171 if (!test_clnt) {
172 pr_err("%s: QMI client handle alloc failed\n", __func__);
173 return;
174 }
175
176 D("%s: Lookup server name\n", __func__);
177 rc = qmi_connect_to_service(test_clnt, TEST_SERVICE_SVC_ID,
178 TEST_SERVICE_INS_ID);
179 if (rc < 0) {
180 pr_err("%s: Server not found\n", __func__);
181 qmi_handle_destroy(test_clnt);
182 test_clnt = NULL;
183 return;
184 }
185 test_clnt_reset = 0;
186 D("%s complete\n", __func__);
187}
188
189static void test_clnt_svc_exit(struct work_struct *work)
190{
191 D("%s begins\n", __func__);
192
193 qmi_handle_destroy(test_clnt);
194 test_clnt_reset = 1;
195 test_clnt = NULL;
196
197 D("%s complete\n", __func__);
198}
199
200static int test_clnt_svc_event_notify(struct notifier_block *this,
201 unsigned long code,
202 void *_cmd)
203{
204 D("%s: event %ld\n", __func__, code);
205 switch (code) {
206 case QMI_SERVER_ARRIVE:
207 queue_delayed_work(test_clnt_workqueue,
208 &work_svc_arrive, 0);
209 break;
210 case QMI_SERVER_EXIT:
211 queue_delayed_work(test_clnt_workqueue,
212 &work_svc_exit, 0);
213 break;
214 default:
215 break;
216 }
217 return 0;
218}
219
220static int test_qmi_open(struct inode *ip, struct file *fp)
221{
222 if (!test_clnt) {
223 pr_err("%s Test client is not initialized\n", __func__);
224 return -ENODEV;
225 }
226 return 0;
227}
228
229static ssize_t test_qmi_read(struct file *fp, char __user *buf,
230 size_t count, loff_t *pos)
231{
232 char _buf[16];
233 snprintf(_buf, sizeof(_buf), "%d\n", test_res);
234 test_res = 0;
235 return simple_read_from_buffer(buf, count, pos,
236 _buf, strnlen(_buf, 16));
237}
238
239static int test_qmi_release(struct inode *ip, struct file *fp)
240{
241 return 0;
242}
243
244static ssize_t test_qmi_write(struct file *fp, const char __user *buf,
245 size_t count, loff_t *pos)
246{
247 unsigned char cmd[64];
248 int len;
249 int i;
250
251 if (count < 1)
252 return 0;
253
254 len = min(count, (sizeof(cmd) - 1));
255
256 if (copy_from_user(cmd, buf, len))
257 return -EFAULT;
258
259 cmd[len] = 0;
260 if (cmd[len-1] == '\n') {
261 cmd[len-1] = 0;
262 len--;
263 }
264
265 if (!strncmp(cmd, "ping_pong", sizeof(cmd))) {
266 for (i = 0; i < test_rep_cnt; i++) {
267 test_res = test_qmi_ping_pong_send_sync_msg();
268 if (test_res == -ENETRESET || test_clnt_reset) {
269 do {
270 msleep(50);
271 } while (test_clnt_reset);
272 }
273 }
274 } else if (!strncmp(cmd, "data", sizeof(cmd))) {
275 for (i = 0; i < test_rep_cnt; i++) {
276 test_res = test_qmi_data_send_sync_msg(test_data_sz);
277 if (test_res == -ENETRESET || test_clnt_reset) {
278 do {
279 msleep(50);
280 } while (test_clnt_reset);
281 }
282 }
283 } else {
284 test_res = -EINVAL;
285 }
286 return count;
287}
288
289static struct notifier_block test_clnt_nb = {
290 .notifier_call = test_clnt_svc_event_notify,
291};
292
293static const struct file_operations debug_ops = {
294 .owner = THIS_MODULE,
295 .open = test_qmi_open,
296 .read = test_qmi_read,
297 .write = test_qmi_write,
298 .release = test_qmi_release,
299};
300
301static int __init test_qmi_init(void)
302{
303 int rc;
304
305 test_clnt_workqueue = create_singlethread_workqueue("test_clnt");
306 if (!test_clnt_workqueue)
307 return -EFAULT;
308
309 rc = qmi_svc_event_notifier_register(TEST_SERVICE_SVC_ID,
310 TEST_SERVICE_INS_ID, &test_clnt_nb);
311 if (rc < 0) {
312 pr_err("%s: notifier register failed\n", __func__);
313 destroy_workqueue(test_clnt_workqueue);
314 return rc;
315 }
316
317 test_dent = debugfs_create_file("test_qmi_client", 0444, 0,
318 NULL, &debug_ops);
319 if (IS_ERR(test_dent)) {
320 pr_err("%s: unable to create debugfs %ld\n",
321 __func__, IS_ERR(test_dent));
322 test_dent = NULL;
323 qmi_svc_event_notifier_unregister(TEST_SERVICE_SVC_ID,
324 TEST_SERVICE_INS_ID, &test_clnt_nb);
325 destroy_workqueue(test_clnt_workqueue);
326 return -EFAULT;
327 }
328
329 return 0;
330}
331
332static void __exit test_qmi_exit(void)
333{
334 qmi_svc_event_notifier_unregister(TEST_SERVICE_SVC_ID,
335 TEST_SERVICE_INS_ID, &test_clnt_nb);
336 destroy_workqueue(test_clnt_workqueue);
337 debugfs_remove(test_dent);
338}
339
340module_init(test_qmi_init);
341module_exit(test_qmi_exit);
342
343MODULE_DESCRIPTION("TEST QMI Client Driver");
344MODULE_LICENSE("GPL v2");