blob: 8aa4758478a384d40f7d628ce926b8aa051f78b9 [file] [log] [blame]
Mona Hossaina5f1aab2012-03-29 10:18:07 -07001
Mona Hossainacea1022012-04-09 13:37:27 -07002
Mona Hossaind44a3842012-10-15 09:41:35 -07003/*Qualcomm Secure Execution Environment Communicator (QSEECOM) driver
Mona Hossain2892b6b2012-02-17 13:53:11 -08004 *
Hariprasad Dhalinarasimhaa3b96442013-01-02 10:40:40 -08005 * Copyright (c) 2012-2013, The Linux Foundation. All rights reserved.
Mona Hossain2892b6b2012-02-17 13:53:11 -08006 *
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License version 2 and
9 * only version 2 as published by the Free Software Foundation.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 */
16
17#define pr_fmt(fmt) "QSEECOM: %s: " fmt, __func__
18
19#include <linux/kernel.h>
20#include <linux/slab.h>
21#include <linux/module.h>
22#include <linux/fs.h>
23#include <linux/platform_device.h>
24#include <linux/debugfs.h>
25#include <linux/cdev.h>
26#include <linux/uaccess.h>
27#include <linux/sched.h>
28#include <linux/list.h>
29#include <linux/mutex.h>
30#include <linux/io.h>
Mitchel Humpherys4f8be2e2012-09-06 10:41:41 -070031#include <linux/msm_ion.h>
Mona Hossain2892b6b2012-02-17 13:53:11 -080032#include <linux/types.h>
33#include <linux/clk.h>
34#include <linux/qseecom.h>
Mona Hossaind44a3842012-10-15 09:41:35 -070035#include <linux/elf.h>
36#include <linux/firmware.h>
Mona Hossainacea1022012-04-09 13:37:27 -070037#include <linux/freezer.h>
Mona Hossainf1f2ed62012-11-15 19:51:33 -080038#include <linux/scatterlist.h>
Ramesh Masavarapua26cce72012-04-09 12:32:25 -070039#include <mach/board.h>
Mona Hossain2892b6b2012-02-17 13:53:11 -080040#include <mach/msm_bus.h>
41#include <mach/msm_bus_board.h>
42#include <mach/scm.h>
Stephen Boyd77db8bb2012-06-27 15:15:16 -070043#include <mach/subsystem_restart.h>
Ramesh Masavarapuff377032012-09-14 12:11:32 -070044#include <mach/socinfo.h>
Mona Hossain803c3d92012-11-21 13:33:42 -080045#include <mach/qseecomi.h>
Mona Hossain2892b6b2012-02-17 13:53:11 -080046#include "qseecom_legacy.h"
Mona Hossaind44a3842012-10-15 09:41:35 -070047#include "qseecom_kernel.h"
Mona Hossain2892b6b2012-02-17 13:53:11 -080048
49#define QSEECOM_DEV "qseecom"
50#define QSEOS_VERSION_13 0x13
51#define QSEOS_VERSION_14 0x14
Mona Hossain05c73562012-10-29 17:49:01 -070052#define QSEEE_VERSION_00 0x400000
Mona Hossain5b76a622012-11-15 20:09:08 -080053#define QSEE_VERSION_01 0x401000
54#define QSEE_VERSION_02 0x402000
55
Mona Hossain05c73562012-10-29 17:49:01 -070056
57#define QSEOS_CHECK_VERSION_CMD 0x00001803
Mona Hossain2892b6b2012-02-17 13:53:11 -080058
Mona Hossaind39e33b2012-11-05 13:36:40 -080059#define QSEE_CE_CLK_100MHZ 100000000
60#define QSEE_CE_CLK_50MHZ 50000000
61
Mona Hossain13dd8922013-01-03 06:11:09 -080062#define QSEECOM_MAX_SG_ENTRY 512
Mona Hossainf1f2ed62012-11-15 19:51:33 -080063
Ramesh Masavarapua26cce72012-04-09 12:32:25 -070064enum qseecom_clk_definitions {
65 CLK_DFAB = 0,
66 CLK_SFPB,
67};
68
Mona Hossain2892b6b2012-02-17 13:53:11 -080069static struct class *driver_class;
70static dev_t qseecom_device_no;
71static struct cdev qseecom_cdev;
72
73/* Data structures used in legacy support */
74static void *pil;
75static uint32_t pil_ref_cnt;
76static DEFINE_MUTEX(pil_access_lock);
77
Mona Hossain2892b6b2012-02-17 13:53:11 -080078static DEFINE_MUTEX(qsee_bw_mutex);
79static DEFINE_MUTEX(app_access_lock);
80
81static int qsee_bw_count;
Ramesh Masavarapua26cce72012-04-09 12:32:25 -070082static int qsee_sfpb_bw_count;
Mona Hossain2892b6b2012-02-17 13:53:11 -080083static uint32_t qsee_perf_client;
84
85struct qseecom_registered_listener_list {
86 struct list_head list;
87 struct qseecom_register_listener_req svc;
88 u8 *sb_reg_req;
89 u8 *sb_virt;
90 s32 sb_phys;
91 size_t sb_length;
92 struct ion_handle *ihandle; /* Retrieve phy addr */
93
94 wait_queue_head_t rcv_req_wq;
95 int rcv_req_flag;
96};
97
98struct qseecom_registered_app_list {
99 struct list_head list;
100 u32 app_id;
101 u32 ref_cnt;
102};
103
Mona Hossaind44a3842012-10-15 09:41:35 -0700104struct qseecom_registered_kclient_list {
105 struct list_head list;
106 struct qseecom_handle *handle;
107};
108
Mona Hossain2892b6b2012-02-17 13:53:11 -0800109struct qseecom_control {
110 struct ion_client *ion_clnt; /* Ion client */
111 struct list_head registered_listener_list_head;
112 spinlock_t registered_listener_list_lock;
113
114 struct list_head registered_app_list_head;
115 spinlock_t registered_app_list_lock;
116
Mona Hossaind44a3842012-10-15 09:41:35 -0700117 struct list_head registered_kclient_list_head;
118 spinlock_t registered_kclient_list_lock;
119
Mona Hossain2892b6b2012-02-17 13:53:11 -0800120 wait_queue_head_t send_resp_wq;
121 int send_resp_flag;
122
123 uint32_t qseos_version;
Mona Hossain05c73562012-10-29 17:49:01 -0700124 uint32_t qsee_version;
Ramesh Masavarapuff377032012-09-14 12:11:32 -0700125 struct device *pdev;
Mona Hossain05c73562012-10-29 17:49:01 -0700126 bool commonlib_loaded;
Mona Hossain2892b6b2012-02-17 13:53:11 -0800127};
128
129struct qseecom_client_handle {
130 u32 app_id;
131 u8 *sb_virt;
132 s32 sb_phys;
133 uint32_t user_virt_sb_base;
134 size_t sb_length;
135 struct ion_handle *ihandle; /* Retrieve phy addr */
Mona Hossain04d3fac2012-12-03 10:10:37 -0800136 bool perf_enabled;
137 bool fast_load_enabled;
Mona Hossain2892b6b2012-02-17 13:53:11 -0800138};
139
140struct qseecom_listener_handle {
141 u32 id;
142};
143
144static struct qseecom_control qseecom;
145
146struct qseecom_dev_handle {
147 bool service;
148 union {
149 struct qseecom_client_handle client;
150 struct qseecom_listener_handle listener;
151 };
152 bool released;
153 int abort;
154 wait_queue_head_t abort_wq;
155 atomic_t ioctl_count;
156};
157
Ramesh Masavarapuff377032012-09-14 12:11:32 -0700158struct clk *ce_core_clk;
159struct clk *ce_clk;
160struct clk *ce_core_src_clk;
161struct clk *ce_bus_clk;
162
Mona Hossainf1f2ed62012-11-15 19:51:33 -0800163struct qseecom_sg_entry {
164 uint32_t phys_addr;
165 uint32_t len;
166};
167
Ramesh Masavarapua26cce72012-04-09 12:32:25 -0700168/* Function proto types */
Mona Hossain04d3fac2012-12-03 10:10:37 -0800169static int qsee_vote_for_clock(struct qseecom_dev_handle *, int32_t);
170static void qsee_disable_clock_vote(struct qseecom_dev_handle *, int32_t);
Ramesh Masavarapua26cce72012-04-09 12:32:25 -0700171
Mona Hossain2892b6b2012-02-17 13:53:11 -0800172static int __qseecom_is_svc_unique(struct qseecom_dev_handle *data,
Mona Hossain0af10ab2012-02-28 18:26:41 -0800173 struct qseecom_register_listener_req *svc)
Mona Hossain2892b6b2012-02-17 13:53:11 -0800174{
175 struct qseecom_registered_listener_list *ptr;
176 int unique = 1;
177 unsigned long flags;
178
179 spin_lock_irqsave(&qseecom.registered_listener_list_lock, flags);
180 list_for_each_entry(ptr, &qseecom.registered_listener_list_head, list) {
Mona Hossain0af10ab2012-02-28 18:26:41 -0800181 if (ptr->svc.listener_id == svc->listener_id) {
Mona Hossain2892b6b2012-02-17 13:53:11 -0800182 pr_err("Service id: %u is already registered\n",
183 ptr->svc.listener_id);
184 unique = 0;
185 break;
186 }
187 }
188 spin_unlock_irqrestore(&qseecom.registered_listener_list_lock, flags);
189 return unique;
190}
191
192static struct qseecom_registered_listener_list *__qseecom_find_svc(
193 int32_t listener_id)
194{
195 struct qseecom_registered_listener_list *entry = NULL;
196 unsigned long flags;
197
198 spin_lock_irqsave(&qseecom.registered_listener_list_lock, flags);
199 list_for_each_entry(entry, &qseecom.registered_listener_list_head, list)
200 {
201 if (entry->svc.listener_id == listener_id)
202 break;
203 }
204 spin_unlock_irqrestore(&qseecom.registered_listener_list_lock, flags);
205 return entry;
206}
207
208static int __qseecom_set_sb_memory(struct qseecom_registered_listener_list *svc,
209 struct qseecom_dev_handle *handle,
210 struct qseecom_register_listener_req *listener)
211{
212 int ret = 0;
Mona Hossain2892b6b2012-02-17 13:53:11 -0800213 struct qseecom_register_listener_ireq req;
214 struct qseecom_command_scm_resp resp;
215 ion_phys_addr_t pa;
216
217 /* Get the handle of the shared fd */
Laura Abbottb14ed962012-01-30 14:18:08 -0800218 svc->ihandle = ion_import_dma_buf(qseecom.ion_clnt,
219 listener->ifd_data_fd);
Mona Hossain2892b6b2012-02-17 13:53:11 -0800220 if (svc->ihandle == NULL) {
221 pr_err("Ion client could not retrieve the handle\n");
222 return -ENOMEM;
223 }
224
225 /* Get the physical address of the ION BUF */
226 ret = ion_phys(qseecom.ion_clnt, svc->ihandle, &pa, &svc->sb_length);
227
228 /* Populate the structure for sending scm call to load image */
Mitchel Humpherys911b4b72012-09-12 14:42:50 -0700229 svc->sb_virt = (char *) ion_map_kernel(qseecom.ion_clnt, svc->ihandle);
Mona Hossain2892b6b2012-02-17 13:53:11 -0800230 svc->sb_phys = pa;
231
232 if (qseecom.qseos_version == QSEOS_VERSION_14) {
233 req.qsee_cmd_id = QSEOS_REGISTER_LISTENER;
234 req.listener_id = svc->svc.listener_id;
235 req.sb_len = svc->sb_length;
236 req.sb_ptr = (void *)svc->sb_phys;
237
238 resp.result = QSEOS_RESULT_INCOMPLETE;
239
240 ret = scm_call(SCM_SVC_TZSCHEDULER, 1, &req,
241 sizeof(req), &resp, sizeof(resp));
242 if (ret) {
243 pr_err("qseecom_scm_call failed with err: %d\n", ret);
244 return -EINVAL;
245 }
246
247 if (resp.result != QSEOS_RESULT_SUCCESS) {
248 pr_err("Error SB registration req: resp.result = %d\n",
249 resp.result);
250 return -EPERM;
251 }
252 } else {
253 struct qseecom_command cmd;
254 struct qseecom_response resp;
255 struct qse_pr_init_sb_req_s sb_init_req;
256 struct qse_pr_init_sb_rsp_s sb_init_rsp;
257
258 svc->sb_reg_req = kzalloc((sizeof(sb_init_req) +
259 sizeof(sb_init_rsp)), GFP_KERNEL);
260
261 sb_init_req.pr_cmd = TZ_SCHED_CMD_ID_REGISTER_LISTENER;
262 sb_init_req.listener_id = svc->svc.listener_id;
263 sb_init_req.sb_len = svc->sb_length;
264 sb_init_req.sb_ptr = svc->sb_phys;
265
266 memcpy(svc->sb_reg_req, &sb_init_req, sizeof(sb_init_req));
267
268 /* It will always be a new cmd from this method */
269 cmd.cmd_type = TZ_SCHED_CMD_NEW;
270 cmd.sb_in_cmd_addr = (u8 *)(virt_to_phys(svc->sb_reg_req));
271 cmd.sb_in_cmd_len = sizeof(sb_init_req);
272
273 resp.cmd_status = TZ_SCHED_STATUS_INCOMPLETE;
274
275 ret = scm_call(SCM_SVC_TZSCHEDULER, 1, &cmd, sizeof(cmd)
276 , &resp, sizeof(resp));
277
278 if (ret) {
279 pr_err("qseecom_scm_call failed with err: %d\n", ret);
280 return -EINVAL;
281 }
282
283 if (resp.cmd_status != TZ_SCHED_STATUS_COMPLETE) {
284 pr_err("SB registration fail resp.cmd_status %d\n",
285 resp.cmd_status);
286 return -EINVAL;
287 }
288 memset(svc->sb_virt, 0, svc->sb_length);
289 }
290 return 0;
291}
292
293static int qseecom_register_listener(struct qseecom_dev_handle *data,
294 void __user *argp)
295{
296 int ret = 0;
297 unsigned long flags;
298 struct qseecom_register_listener_req rcvd_lstnr;
299 struct qseecom_registered_listener_list *new_entry;
300
301 ret = copy_from_user(&rcvd_lstnr, argp, sizeof(rcvd_lstnr));
302 if (ret) {
303 pr_err("copy_from_user failed\n");
304 return ret;
305 }
Mona Hossain0af10ab2012-02-28 18:26:41 -0800306 data->listener.id = 0;
307 data->service = true;
308 if (!__qseecom_is_svc_unique(data, &rcvd_lstnr)) {
Mona Hossain2892b6b2012-02-17 13:53:11 -0800309 pr_err("Service is not unique and is already registered\n");
Mona Hossain0af10ab2012-02-28 18:26:41 -0800310 data->released = true;
311 return -EBUSY;
Mona Hossain2892b6b2012-02-17 13:53:11 -0800312 }
313
314 new_entry = kmalloc(sizeof(*new_entry), GFP_KERNEL);
315 if (!new_entry) {
316 pr_err("kmalloc failed\n");
317 return -ENOMEM;
318 }
319 memcpy(&new_entry->svc, &rcvd_lstnr, sizeof(rcvd_lstnr));
320 new_entry->rcv_req_flag = 0;
321
322 new_entry->svc.listener_id = rcvd_lstnr.listener_id;
323 new_entry->sb_length = rcvd_lstnr.sb_size;
324 if (__qseecom_set_sb_memory(new_entry, data, &rcvd_lstnr)) {
325 pr_err("qseecom_set_sb_memoryfailed\n");
326 kzfree(new_entry);
327 return -ENOMEM;
328 }
Mona Hossain0af10ab2012-02-28 18:26:41 -0800329
Mona Hossain2892b6b2012-02-17 13:53:11 -0800330 data->listener.id = rcvd_lstnr.listener_id;
Mona Hossain2892b6b2012-02-17 13:53:11 -0800331 init_waitqueue_head(&new_entry->rcv_req_wq);
332
333 spin_lock_irqsave(&qseecom.registered_listener_list_lock, flags);
334 list_add_tail(&new_entry->list, &qseecom.registered_listener_list_head);
335 spin_unlock_irqrestore(&qseecom.registered_listener_list_lock, flags);
Mona Hossain0af10ab2012-02-28 18:26:41 -0800336
Mona Hossain2892b6b2012-02-17 13:53:11 -0800337 return ret;
338}
339
340static int qseecom_unregister_listener(struct qseecom_dev_handle *data)
341{
342 int ret = 0;
343 unsigned long flags;
344 uint32_t unmap_mem = 0;
345 struct qseecom_register_listener_ireq req;
346 struct qseecom_registered_listener_list *ptr_svc = NULL;
347 struct qseecom_command_scm_resp resp;
348 struct ion_handle *ihandle = NULL; /* Retrieve phy addr */
349
350 if (qseecom.qseos_version == QSEOS_VERSION_14) {
351 req.qsee_cmd_id = QSEOS_DEREGISTER_LISTENER;
352 req.listener_id = data->listener.id;
353 resp.result = QSEOS_RESULT_INCOMPLETE;
354
355 ret = scm_call(SCM_SVC_TZSCHEDULER, 1, &req,
356 sizeof(req), &resp, sizeof(resp));
357 if (ret) {
358 pr_err("qseecom_scm_call failed with err: %d\n", ret);
359 return ret;
360 }
361
362 if (resp.result != QSEOS_RESULT_SUCCESS) {
363 pr_err("SB deregistartion: result=%d\n", resp.result);
364 return -EPERM;
365 }
366 } else {
367 struct qse_pr_init_sb_req_s sb_init_req;
368 struct qseecom_command cmd;
369 struct qseecom_response resp;
370 struct qseecom_registered_listener_list *svc;
371
372 svc = __qseecom_find_svc(data->listener.id);
373 sb_init_req.pr_cmd = TZ_SCHED_CMD_ID_REGISTER_LISTENER;
374 sb_init_req.listener_id = data->listener.id;
375 sb_init_req.sb_len = 0;
376 sb_init_req.sb_ptr = 0;
377
378 memcpy(svc->sb_reg_req, &sb_init_req, sizeof(sb_init_req));
379
380 /* It will always be a new cmd from this method */
381 cmd.cmd_type = TZ_SCHED_CMD_NEW;
382 cmd.sb_in_cmd_addr = (u8 *)(virt_to_phys(svc->sb_reg_req));
383 cmd.sb_in_cmd_len = sizeof(sb_init_req);
384 resp.cmd_status = TZ_SCHED_STATUS_INCOMPLETE;
385
386 ret = scm_call(SCM_SVC_TZSCHEDULER, 1, &cmd, sizeof(cmd),
387 &resp, sizeof(resp));
388 if (ret) {
389 pr_err("qseecom_scm_call failed with err: %d\n", ret);
390 return ret;
391 }
392 kzfree(svc->sb_reg_req);
393 if (resp.cmd_status != TZ_SCHED_STATUS_COMPLETE) {
394 pr_err("Error with SB initialization\n");
395 return -EPERM;
396 }
397 }
398 data->abort = 1;
399 spin_lock_irqsave(&qseecom.registered_listener_list_lock, flags);
400 list_for_each_entry(ptr_svc, &qseecom.registered_listener_list_head,
401 list) {
402 if (ptr_svc->svc.listener_id == data->listener.id) {
403 wake_up_all(&ptr_svc->rcv_req_wq);
404 break;
405 }
406 }
407 spin_unlock_irqrestore(&qseecom.registered_listener_list_lock, flags);
408
409 while (atomic_read(&data->ioctl_count) > 1) {
Mona Hossainacea1022012-04-09 13:37:27 -0700410 if (wait_event_freezable(data->abort_wq,
Mona Hossain2892b6b2012-02-17 13:53:11 -0800411 atomic_read(&data->ioctl_count) <= 1)) {
412 pr_err("Interrupted from abort\n");
413 ret = -ERESTARTSYS;
414 break;
415 }
416 }
417
418 spin_lock_irqsave(&qseecom.registered_listener_list_lock, flags);
419 list_for_each_entry(ptr_svc,
420 &qseecom.registered_listener_list_head,
421 list)
422 {
423 if (ptr_svc->svc.listener_id == data->listener.id) {
424 if (ptr_svc->sb_virt) {
425 unmap_mem = 1;
426 ihandle = ptr_svc->ihandle;
427 }
428 list_del(&ptr_svc->list);
429 kzfree(ptr_svc);
430 break;
431 }
432 }
433 spin_unlock_irqrestore(&qseecom.registered_listener_list_lock, flags);
434
435 /* Unmap the memory */
436 if (unmap_mem) {
437 if (!IS_ERR_OR_NULL(ihandle)) {
438 ion_unmap_kernel(qseecom.ion_clnt, ihandle);
439 ion_free(qseecom.ion_clnt, ihandle);
440 }
441 }
442 data->released = true;
443 return ret;
444}
445
446static int qseecom_set_client_mem_param(struct qseecom_dev_handle *data,
447 void __user *argp)
448{
449 ion_phys_addr_t pa;
450 int32_t ret;
Mona Hossain2892b6b2012-02-17 13:53:11 -0800451 struct qseecom_set_sb_mem_param_req req;
452 uint32_t len;
453
454 /* Copy the relevant information needed for loading the image */
455 if (__copy_from_user(&req, (void __user *)argp, sizeof(req)))
456 return -EFAULT;
457
Mona Hossain2892b6b2012-02-17 13:53:11 -0800458 /* Get the handle of the shared fd */
Laura Abbottb14ed962012-01-30 14:18:08 -0800459 data->client.ihandle = ion_import_dma_buf(qseecom.ion_clnt,
460 req.ifd_data_fd);
Mona Hossain2892b6b2012-02-17 13:53:11 -0800461 if (IS_ERR_OR_NULL(data->client.ihandle)) {
462 pr_err("Ion client could not retrieve the handle\n");
463 return -ENOMEM;
464 }
465 /* Get the physical address of the ION BUF */
466 ret = ion_phys(qseecom.ion_clnt, data->client.ihandle, &pa, &len);
467 /* Populate the structure for sending scm call to load image */
468 data->client.sb_virt = (char *) ion_map_kernel(qseecom.ion_clnt,
Mitchel Humpherys911b4b72012-09-12 14:42:50 -0700469 data->client.ihandle);
Mona Hossain2892b6b2012-02-17 13:53:11 -0800470 data->client.sb_phys = pa;
471 data->client.sb_length = req.sb_len;
472 data->client.user_virt_sb_base = req.virt_sb_base;
473 return 0;
474}
475
Mona Hossain2892b6b2012-02-17 13:53:11 -0800476static int __qseecom_listener_has_sent_rsp(struct qseecom_dev_handle *data)
477{
478 int ret;
479 ret = (qseecom.send_resp_flag != 0);
480 return ret || data->abort;
481}
482
483static int __qseecom_process_incomplete_cmd(struct qseecom_dev_handle *data,
484 struct qseecom_command_scm_resp *resp)
485{
486 int ret = 0;
Mona Hossain0b3a2792013-02-05 17:38:55 -0800487 int rc = 0;
Mona Hossain2892b6b2012-02-17 13:53:11 -0800488 uint32_t lstnr;
489 unsigned long flags;
490 struct qseecom_client_listener_data_irsp send_data_rsp;
491 struct qseecom_registered_listener_list *ptr_svc = NULL;
492
493
494 while (resp->result == QSEOS_RESULT_INCOMPLETE) {
495 lstnr = resp->data;
496 /*
497 * Wake up blocking lsitener service with the lstnr id
498 */
499 spin_lock_irqsave(&qseecom.registered_listener_list_lock,
500 flags);
501 list_for_each_entry(ptr_svc,
502 &qseecom.registered_listener_list_head, list) {
503 if (ptr_svc->svc.listener_id == lstnr) {
504 ptr_svc->rcv_req_flag = 1;
505 wake_up_interruptible(&ptr_svc->rcv_req_wq);
506 break;
507 }
508 }
509 spin_unlock_irqrestore(&qseecom.registered_listener_list_lock,
510 flags);
511 if (ptr_svc->svc.listener_id != lstnr) {
512 pr_warning("Service requested for does on exist\n");
513 return -ERESTARTSYS;
514 }
515 pr_debug("waking up rcv_req_wq and "
516 "waiting for send_resp_wq\n");
Mona Hossainacea1022012-04-09 13:37:27 -0700517 if (wait_event_freezable(qseecom.send_resp_wq,
Mona Hossain2892b6b2012-02-17 13:53:11 -0800518 __qseecom_listener_has_sent_rsp(data))) {
519 pr_warning("Interrupted: exiting send_cmd loop\n");
520 return -ERESTARTSYS;
521 }
522
523 if (data->abort) {
Mona Hossaina5f1aab2012-03-29 10:18:07 -0700524 pr_err("Aborting listener service %d\n",
525 data->listener.id);
Mona Hossain0b3a2792013-02-05 17:38:55 -0800526 rc = -ENODEV;
527 send_data_rsp.status = QSEOS_RESULT_FAILURE;
528 } else {
529 send_data_rsp.status = QSEOS_RESULT_SUCCESS;
Mona Hossain2892b6b2012-02-17 13:53:11 -0800530 }
Mona Hossain0b3a2792013-02-05 17:38:55 -0800531
Mona Hossain2892b6b2012-02-17 13:53:11 -0800532 qseecom.send_resp_flag = 0;
533 send_data_rsp.qsee_cmd_id = QSEOS_LISTENER_DATA_RSP_COMMAND;
534 send_data_rsp.listener_id = lstnr ;
535
536 ret = scm_call(SCM_SVC_TZSCHEDULER, 1,
537 (const void *)&send_data_rsp,
538 sizeof(send_data_rsp), resp,
539 sizeof(*resp));
540 if (ret) {
541 pr_err("qseecom_scm_call failed with err: %d\n", ret);
542 return ret;
543 }
Mona Hossainbb0bca12012-04-12 11:47:45 -0700544 if (resp->result == QSEOS_RESULT_FAILURE) {
545 pr_err("Response result %d not supported\n",
546 resp->result);
547 return -EINVAL;
548 }
Mona Hossain2892b6b2012-02-17 13:53:11 -0800549 }
Mona Hossain0b3a2792013-02-05 17:38:55 -0800550 if (rc)
551 return rc;
552
Mona Hossain2892b6b2012-02-17 13:53:11 -0800553 return ret;
554}
555
Ramesh Masavarapu1ee4ac82012-06-22 15:38:34 -0700556static int __qseecom_check_app_exists(struct qseecom_check_app_ireq req)
557{
558 int32_t ret;
559 struct qseecom_command_scm_resp resp;
560
561 /* SCM_CALL to check if app_id for the mentioned app exists */
562 ret = scm_call(SCM_SVC_TZSCHEDULER, 1, &req,
563 sizeof(struct qseecom_check_app_ireq),
564 &resp, sizeof(resp));
565 if (ret) {
566 pr_err("scm_call to check if app is already loaded failed\n");
567 return -EINVAL;
568 }
569
Ramesh Masavarapu13d99672012-07-17 11:05:15 -0700570 if (resp.result == QSEOS_RESULT_FAILURE) {
571 return 0;
572 } else {
573 switch (resp.resp_type) {
574 /*qsee returned listener type response */
575 case QSEOS_LISTENER_ID:
576 pr_err("resp type is of listener type instead of app");
577 return -EINVAL;
578 break;
579 case QSEOS_APP_ID:
580 return resp.data;
581 default:
582 pr_err("invalid resp type (%d) from qsee",
583 resp.resp_type);
584 return -ENODEV;
585 break;
586 }
587 }
Ramesh Masavarapu1ee4ac82012-06-22 15:38:34 -0700588}
589
Mona Hossain2892b6b2012-02-17 13:53:11 -0800590static int qseecom_load_app(struct qseecom_dev_handle *data, void __user *argp)
591{
592 struct qseecom_registered_app_list *entry = NULL;
593 unsigned long flags = 0;
594 u32 app_id = 0;
595 struct ion_handle *ihandle; /* Ion handle */
596 struct qseecom_load_img_req load_img_req;
597 int32_t ret;
598 ion_phys_addr_t pa = 0;
599 uint32_t len;
600 struct qseecom_command_scm_resp resp;
Mona Hossain436b75f2012-11-20 17:10:40 -0800601 struct qseecom_check_app_ireq req;
Ramesh Masavarapu13d99672012-07-17 11:05:15 -0700602 struct qseecom_load_app_ireq load_req;
603
Mona Hossain2892b6b2012-02-17 13:53:11 -0800604 /* Copy the relevant information needed for loading the image */
605 if (__copy_from_user(&load_img_req,
606 (void __user *)argp,
607 sizeof(struct qseecom_load_img_req))) {
608 pr_err("copy_from_user failed\n");
609 return -EFAULT;
610 }
Ramesh Masavarapua26cce72012-04-09 12:32:25 -0700611 /* Vote for the SFPB clock */
Mona Hossain04d3fac2012-12-03 10:10:37 -0800612 ret = qsee_vote_for_clock(data, CLK_SFPB);
Ramesh Masavarapua26cce72012-04-09 12:32:25 -0700613 if (ret)
614 pr_warning("Unable to vote for SFPB clock");
Mona Hossain436b75f2012-11-20 17:10:40 -0800615 req.qsee_cmd_id = QSEOS_APP_LOOKUP_COMMAND;
616 memcpy(req.app_name, load_img_req.img_name, MAX_APP_NAME_SIZE);
Mona Hossain2892b6b2012-02-17 13:53:11 -0800617
Mona Hossain436b75f2012-11-20 17:10:40 -0800618 ret = __qseecom_check_app_exists(req);
619 if (ret < 0)
620 return ret;
621 else
622 app_id = ret;
623
624 if (app_id) {
625 pr_warn("App id %d (%s) already exists\n", app_id,
626 (char *)(req.app_name));
627 spin_lock_irqsave(&qseecom.registered_app_list_lock, flags);
628 list_for_each_entry(entry,
629 &qseecom.registered_app_list_head, list){
630 if (entry->app_id == app_id) {
631 entry->ref_cnt++;
632 break;
633 }
634 }
635 spin_unlock_irqrestore(
636 &qseecom.registered_app_list_lock, flags);
637 } else {
638 pr_warn("App (%s) does'nt exist, loading apps for first time\n",
Mona Hossaind44a3842012-10-15 09:41:35 -0700639 (char *)(load_img_req.img_name));
Mona Hossain436b75f2012-11-20 17:10:40 -0800640 /* Get the handle of the shared fd */
641 ihandle = ion_import_dma_buf(qseecom.ion_clnt,
Mona Hossain2892b6b2012-02-17 13:53:11 -0800642 load_img_req.ifd_data_fd);
Mona Hossain436b75f2012-11-20 17:10:40 -0800643 if (IS_ERR_OR_NULL(ihandle)) {
644 pr_err("Ion client could not retrieve the handle\n");
Mona Hossain04d3fac2012-12-03 10:10:37 -0800645 qsee_disable_clock_vote(data, CLK_SFPB);
Mona Hossain436b75f2012-11-20 17:10:40 -0800646 return -ENOMEM;
647 }
Mona Hossain2892b6b2012-02-17 13:53:11 -0800648
Mona Hossain436b75f2012-11-20 17:10:40 -0800649 /* Get the physical address of the ION BUF */
650 ret = ion_phys(qseecom.ion_clnt, ihandle, &pa, &len);
Mona Hossain2892b6b2012-02-17 13:53:11 -0800651
Mona Hossain436b75f2012-11-20 17:10:40 -0800652 /* Populate the structure for sending scm call to load image */
653 memcpy(load_req.app_name, load_img_req.img_name,
654 MAX_APP_NAME_SIZE);
655 load_req.qsee_cmd_id = QSEOS_APP_START_COMMAND;
656 load_req.mdt_len = load_img_req.mdt_len;
657 load_req.img_len = load_img_req.img_len;
658 load_req.phy_addr = pa;
Mona Hossain2892b6b2012-02-17 13:53:11 -0800659
Mona Hossain436b75f2012-11-20 17:10:40 -0800660 /* SCM_CALL to load the app and get the app_id back */
661 ret = scm_call(SCM_SVC_TZSCHEDULER, 1, &load_req,
Ramesh Masavarapu13d99672012-07-17 11:05:15 -0700662 sizeof(struct qseecom_load_app_ireq),
663 &resp, sizeof(resp));
Ramesh Masavarapu13d99672012-07-17 11:05:15 -0700664 if (ret) {
Mona Hossain436b75f2012-11-20 17:10:40 -0800665 pr_err("scm_call to load app failed\n");
666 return -EINVAL;
667 }
668
669 if (resp.result == QSEOS_RESULT_FAILURE) {
670 pr_err("scm_call rsp.result is QSEOS_RESULT_FAILURE\n");
Ramesh Masavarapu13d99672012-07-17 11:05:15 -0700671 if (!IS_ERR_OR_NULL(ihandle))
672 ion_free(qseecom.ion_clnt, ihandle);
Mona Hossain04d3fac2012-12-03 10:10:37 -0800673 qsee_disable_clock_vote(data, CLK_SFPB);
Mona Hossain436b75f2012-11-20 17:10:40 -0800674 return -EFAULT;
Ramesh Masavarapu13d99672012-07-17 11:05:15 -0700675 }
Ramesh Masavarapu13d99672012-07-17 11:05:15 -0700676
Mona Hossain436b75f2012-11-20 17:10:40 -0800677 if (resp.result == QSEOS_RESULT_INCOMPLETE) {
678 ret = __qseecom_process_incomplete_cmd(data, &resp);
679 if (ret) {
680 pr_err("process_incomplete_cmd failed err: %d\n",
681 ret);
682 if (!IS_ERR_OR_NULL(ihandle))
683 ion_free(qseecom.ion_clnt, ihandle);
Mona Hossain04d3fac2012-12-03 10:10:37 -0800684 qsee_disable_clock_vote(data, CLK_SFPB);
Mona Hossain436b75f2012-11-20 17:10:40 -0800685 return ret;
686 }
687 }
688
689 if (resp.result != QSEOS_RESULT_SUCCESS) {
690 pr_err("scm_call failed resp.result unknown, %d\n",
Ramesh Masavarapu13d99672012-07-17 11:05:15 -0700691 resp.result);
Mona Hossain436b75f2012-11-20 17:10:40 -0800692 if (!IS_ERR_OR_NULL(ihandle))
693 ion_free(qseecom.ion_clnt, ihandle);
Mona Hossain04d3fac2012-12-03 10:10:37 -0800694 qsee_disable_clock_vote(data, CLK_SFPB);
Mona Hossain436b75f2012-11-20 17:10:40 -0800695 return -EFAULT;
696 }
697
698 app_id = resp.data;
699
700 entry = kmalloc(sizeof(*entry), GFP_KERNEL);
701 if (!entry) {
702 pr_err("kmalloc failed\n");
Mona Hossain04d3fac2012-12-03 10:10:37 -0800703 qsee_disable_clock_vote(data, CLK_SFPB);
Mona Hossain436b75f2012-11-20 17:10:40 -0800704 return -ENOMEM;
705 }
706 entry->app_id = app_id;
707 entry->ref_cnt = 1;
708
709 /* Deallocate the handle */
Ramesh Masavarapu13d99672012-07-17 11:05:15 -0700710 if (!IS_ERR_OR_NULL(ihandle))
711 ion_free(qseecom.ion_clnt, ihandle);
Ramesh Masavarapu13d99672012-07-17 11:05:15 -0700712
Mona Hossain436b75f2012-11-20 17:10:40 -0800713 spin_lock_irqsave(&qseecom.registered_app_list_lock, flags);
714 list_add_tail(&entry->list, &qseecom.registered_app_list_head);
715 spin_unlock_irqrestore(&qseecom.registered_app_list_lock,
716 flags);
Ramesh Masavarapu13d99672012-07-17 11:05:15 -0700717
Mona Hossain436b75f2012-11-20 17:10:40 -0800718 pr_warn("App with id %d (%s) now loaded\n", app_id,
Mona Hossaind44a3842012-10-15 09:41:35 -0700719 (char *)(load_img_req.img_name));
Mona Hossain436b75f2012-11-20 17:10:40 -0800720 }
Mona Hossain2892b6b2012-02-17 13:53:11 -0800721 data->client.app_id = app_id;
722 load_img_req.app_id = app_id;
723 if (copy_to_user(argp, &load_img_req, sizeof(load_img_req))) {
724 pr_err("copy_to_user failed\n");
725 kzfree(entry);
Mona Hossain04d3fac2012-12-03 10:10:37 -0800726 qsee_disable_clock_vote(data, CLK_SFPB);
Mona Hossain2892b6b2012-02-17 13:53:11 -0800727 return -EFAULT;
728 }
Mona Hossain04d3fac2012-12-03 10:10:37 -0800729 qsee_disable_clock_vote(data, CLK_SFPB);
Mona Hossain2892b6b2012-02-17 13:53:11 -0800730 return 0;
731}
732
733static int __qseecom_cleanup_app(struct qseecom_dev_handle *data)
734{
735 wake_up_all(&qseecom.send_resp_wq);
736 while (atomic_read(&data->ioctl_count) > 1) {
Mona Hossainacea1022012-04-09 13:37:27 -0700737 if (wait_event_freezable(data->abort_wq,
Mona Hossain2892b6b2012-02-17 13:53:11 -0800738 atomic_read(&data->ioctl_count) <= 1)) {
739 pr_err("Interrupted from abort\n");
740 return -ERESTARTSYS;
741 break;
742 }
743 }
744 /* Set unload app */
745 return 1;
746}
747
748static int qseecom_unload_app(struct qseecom_dev_handle *data)
749{
750 unsigned long flags;
751 int ret = 0;
752 struct qseecom_command_scm_resp resp;
753 struct qseecom_registered_app_list *ptr_app;
Mona Hossain340dba82012-08-07 19:54:46 -0700754 bool unload = false;
755 bool found_app = false;
Mona Hossain2892b6b2012-02-17 13:53:11 -0800756
Mona Hossain1fb538f2012-08-30 16:19:38 -0700757 if ((qseecom.qseos_version == QSEOS_VERSION_14) &&
758 (data->client.app_id > 0)) {
Mona Hossain2892b6b2012-02-17 13:53:11 -0800759 spin_lock_irqsave(&qseecom.registered_app_list_lock, flags);
760 list_for_each_entry(ptr_app, &qseecom.registered_app_list_head,
761 list) {
762 if (ptr_app->app_id == data->client.app_id) {
Mona Hossain340dba82012-08-07 19:54:46 -0700763 found_app = true;
Mona Hossain2892b6b2012-02-17 13:53:11 -0800764 if (ptr_app->ref_cnt == 1) {
Mona Hossain340dba82012-08-07 19:54:46 -0700765 unload = true;
Mona Hossain2892b6b2012-02-17 13:53:11 -0800766 break;
767 } else {
768 ptr_app->ref_cnt--;
Mona Hossain1fb538f2012-08-30 16:19:38 -0700769 pr_warn("Can't unload app(%d) inuse\n",
Mona Hossaina5f1aab2012-03-29 10:18:07 -0700770 ptr_app->app_id);
Mona Hossain2892b6b2012-02-17 13:53:11 -0800771 break;
772 }
773 }
774 }
775 spin_unlock_irqrestore(&qseecom.registered_app_list_lock,
776 flags);
Mona Hossain1fb538f2012-08-30 16:19:38 -0700777 if (found_app == false) {
778 pr_err("Cannot find app with id = %d\n",
779 data->client.app_id);
780 return -EINVAL;
781 }
Mona Hossain2892b6b2012-02-17 13:53:11 -0800782 }
783
784 if ((unload) && (qseecom.qseos_version == QSEOS_VERSION_14)) {
785 struct qseecom_unload_app_ireq req;
786
Mona Hossain340dba82012-08-07 19:54:46 -0700787 __qseecom_cleanup_app(data);
788 spin_lock_irqsave(&qseecom.registered_app_list_lock, flags);
789 list_del(&ptr_app->list);
790 kzfree(ptr_app);
791 spin_unlock_irqrestore(&qseecom.registered_app_list_lock,
792 flags);
Mona Hossain2892b6b2012-02-17 13:53:11 -0800793 /* Populate the structure for sending scm call to load image */
794 req.qsee_cmd_id = QSEOS_APP_SHUTDOWN_COMMAND;
795 req.app_id = data->client.app_id;
796
797 /* SCM_CALL to unload the app */
798 ret = scm_call(SCM_SVC_TZSCHEDULER, 1, &req,
799 sizeof(struct qseecom_unload_app_ireq),
800 &resp, sizeof(resp));
801 if (ret) {
Mona Hossainbb0bca12012-04-12 11:47:45 -0700802 pr_err("scm_call to unload app (id = %d) failed\n",
803 req.app_id);
Mona Hossain2892b6b2012-02-17 13:53:11 -0800804 return -EFAULT;
Mona Hossainbb0bca12012-04-12 11:47:45 -0700805 } else {
806 pr_warn("App id %d now unloaded\n", req.app_id);
Mona Hossain2892b6b2012-02-17 13:53:11 -0800807 }
808 if (resp.result == QSEOS_RESULT_INCOMPLETE) {
809 ret = __qseecom_process_incomplete_cmd(data, &resp);
810 if (ret) {
811 pr_err("process_incomplete_cmd fail err: %d\n",
812 ret);
813 return ret;
814 }
815 }
816 }
817
818 if (qseecom.qseos_version == QSEOS_VERSION_13) {
819 data->abort = 1;
820 wake_up_all(&qseecom.send_resp_wq);
821 while (atomic_read(&data->ioctl_count) > 0) {
Mona Hossainacea1022012-04-09 13:37:27 -0700822 if (wait_event_freezable(data->abort_wq,
Mona Hossain2892b6b2012-02-17 13:53:11 -0800823 atomic_read(&data->ioctl_count) <= 0)) {
824 pr_err("Interrupted from abort\n");
825 ret = -ERESTARTSYS;
826 break;
827 }
828 }
Mona Hossain2892b6b2012-02-17 13:53:11 -0800829 }
Mona Hossain340dba82012-08-07 19:54:46 -0700830 if (!IS_ERR_OR_NULL(data->client.ihandle)) {
831 ion_unmap_kernel(qseecom.ion_clnt, data->client.ihandle);
832 ion_free(qseecom.ion_clnt, data->client.ihandle);
833 data->client.ihandle = NULL;
834 }
Mona Hossain2892b6b2012-02-17 13:53:11 -0800835 data->released = true;
836 return ret;
837}
838
839static uint32_t __qseecom_uvirt_to_kphys(struct qseecom_dev_handle *data,
840 uint32_t virt)
841{
842 return data->client.sb_phys + (virt - data->client.user_virt_sb_base);
843}
844
845static int __qseecom_send_cmd_legacy(struct qseecom_dev_handle *data,
846 struct qseecom_send_cmd_req *req)
847{
848 int ret = 0;
849 unsigned long flags;
850 u32 reqd_len_sb_in = 0;
851 struct qseecom_command cmd;
852 struct qseecom_response resp;
853
854
855 if (req->cmd_req_buf == NULL || req->resp_buf == NULL) {
856 pr_err("cmd buffer or response buffer is null\n");
857 return -EINVAL;
858 }
859
860 if (req->cmd_req_len <= 0 ||
861 req->resp_len <= 0 ||
862 req->cmd_req_len > data->client.sb_length ||
863 req->resp_len > data->client.sb_length) {
864 pr_err("cmd buffer length or "
865 "response buffer length not valid\n");
866 return -EINVAL;
867 }
868
869 reqd_len_sb_in = req->cmd_req_len + req->resp_len;
870 if (reqd_len_sb_in > data->client.sb_length) {
871 pr_debug("Not enough memory to fit cmd_buf and "
872 "resp_buf. Required: %u, Available: %u\n",
873 reqd_len_sb_in, data->client.sb_length);
874 return -ENOMEM;
875 }
876 cmd.cmd_type = TZ_SCHED_CMD_NEW;
877 cmd.sb_in_cmd_addr = (u8 *) data->client.sb_phys;
878 cmd.sb_in_cmd_len = req->cmd_req_len;
879
880 resp.cmd_status = TZ_SCHED_STATUS_INCOMPLETE;
881 resp.sb_in_rsp_addr = (u8 *)data->client.sb_phys + req->cmd_req_len;
882 resp.sb_in_rsp_len = req->resp_len;
883
884 ret = scm_call(SCM_SVC_TZSCHEDULER, 1, (const void *)&cmd,
885 sizeof(cmd), &resp, sizeof(resp));
886
887 if (ret) {
888 pr_err("qseecom_scm_call_legacy failed with err: %d\n", ret);
889 return ret;
890 }
891
892 while (resp.cmd_status != TZ_SCHED_STATUS_COMPLETE) {
893 /*
894 * If cmd is incomplete, get the callback cmd out from SB out
895 * and put it on the list
896 */
897 struct qseecom_registered_listener_list *ptr_svc = NULL;
898 /*
899 * We don't know which service can handle the command. so we
900 * wake up all blocking services and let them figure out if
901 * they can handle the given command.
902 */
903 spin_lock_irqsave(&qseecom.registered_listener_list_lock,
904 flags);
905 list_for_each_entry(ptr_svc,
906 &qseecom.registered_listener_list_head, list) {
907 ptr_svc->rcv_req_flag = 1;
908 wake_up_interruptible(&ptr_svc->rcv_req_wq);
909 }
910 spin_unlock_irqrestore(&qseecom.registered_listener_list_lock,
911 flags);
912
913 pr_debug("waking up rcv_req_wq and "
914 "waiting for send_resp_wq\n");
Mona Hossainacea1022012-04-09 13:37:27 -0700915 if (wait_event_freezable(qseecom.send_resp_wq,
Mona Hossain2892b6b2012-02-17 13:53:11 -0800916 __qseecom_listener_has_sent_rsp(data))) {
917 pr_warning("qseecom Interrupted: exiting send_cmd loop\n");
918 return -ERESTARTSYS;
919 }
920
921 if (data->abort) {
922 pr_err("Aborting driver\n");
923 return -ENODEV;
924 }
925 qseecom.send_resp_flag = 0;
926 cmd.cmd_type = TZ_SCHED_CMD_PENDING;
927 ret = scm_call(SCM_SVC_TZSCHEDULER, 1, (const void *)&cmd,
928 sizeof(cmd), &resp, sizeof(resp));
929 if (ret) {
930 pr_err("qseecom_scm_call failed with err: %d\n", ret);
931 return ret;
932 }
933 }
934 return ret;
935}
936
937static int __qseecom_send_cmd(struct qseecom_dev_handle *data,
938 struct qseecom_send_cmd_req *req)
939{
940 int ret = 0;
941 u32 reqd_len_sb_in = 0;
942 struct qseecom_client_send_data_ireq send_data_req;
943 struct qseecom_command_scm_resp resp;
944
945 if (req->cmd_req_buf == NULL || req->resp_buf == NULL) {
946 pr_err("cmd buffer or response buffer is null\n");
947 return -EINVAL;
948 }
949
950 if (req->cmd_req_len <= 0 ||
951 req->resp_len <= 0 ||
952 req->cmd_req_len > data->client.sb_length ||
953 req->resp_len > data->client.sb_length) {
954 pr_err("cmd buffer length or "
955 "response buffer length not valid\n");
956 return -EINVAL;
957 }
958
959 reqd_len_sb_in = req->cmd_req_len + req->resp_len;
960 if (reqd_len_sb_in > data->client.sb_length) {
961 pr_debug("Not enough memory to fit cmd_buf and "
962 "resp_buf. Required: %u, Available: %u\n",
963 reqd_len_sb_in, data->client.sb_length);
964 return -ENOMEM;
965 }
966
967 send_data_req.qsee_cmd_id = QSEOS_CLIENT_SEND_DATA_COMMAND;
968 send_data_req.app_id = data->client.app_id;
969 send_data_req.req_ptr = (void *)(__qseecom_uvirt_to_kphys(data,
970 (uint32_t)req->cmd_req_buf));
971 send_data_req.req_len = req->cmd_req_len;
972 send_data_req.rsp_ptr = (void *)(__qseecom_uvirt_to_kphys(data,
973 (uint32_t)req->resp_buf));
974 send_data_req.rsp_len = req->resp_len;
975
976 ret = scm_call(SCM_SVC_TZSCHEDULER, 1, (const void *) &send_data_req,
977 sizeof(send_data_req),
978 &resp, sizeof(resp));
979 if (ret) {
980 pr_err("qseecom_scm_call failed with err: %d\n", ret);
981 return ret;
982 }
983
984 if (resp.result == QSEOS_RESULT_INCOMPLETE) {
985 ret = __qseecom_process_incomplete_cmd(data, &resp);
986 if (ret) {
987 pr_err("process_incomplete_cmd failed err: %d\n", ret);
988 return ret;
989 }
Mona Hossainbb0bca12012-04-12 11:47:45 -0700990 } else {
991 if (resp.result != QSEOS_RESULT_SUCCESS) {
992 pr_err("Response result %d not supported\n",
993 resp.result);
994 ret = -EINVAL;
995 }
Mona Hossain2892b6b2012-02-17 13:53:11 -0800996 }
997 return ret;
998}
999
1000
1001static int qseecom_send_cmd(struct qseecom_dev_handle *data, void __user *argp)
1002{
1003 int ret = 0;
1004 struct qseecom_send_cmd_req req;
1005
1006 ret = copy_from_user(&req, argp, sizeof(req));
1007 if (ret) {
1008 pr_err("copy_from_user failed\n");
1009 return ret;
1010 }
1011 if (qseecom.qseos_version == QSEOS_VERSION_14)
1012 ret = __qseecom_send_cmd(data, &req);
1013 else
1014 ret = __qseecom_send_cmd_legacy(data, &req);
1015 if (ret)
1016 return ret;
1017
1018 pr_debug("sending cmd_req->rsp size: %u, ptr: 0x%p\n",
1019 req.resp_len, req.resp_buf);
1020 return ret;
1021}
1022
1023static int __qseecom_send_cmd_req_clean_up(
1024 struct qseecom_send_modfd_cmd_req *req)
1025{
1026 char *field;
1027 uint32_t *update;
1028 int ret = 0;
1029 int i = 0;
1030
1031 for (i = 0; i < MAX_ION_FD; i++) {
Mona Hossaina5f1aab2012-03-29 10:18:07 -07001032 if (req->ifd_data[i].fd > 0) {
Mona Hossain2892b6b2012-02-17 13:53:11 -08001033 field = (char *)req->cmd_req_buf +
1034 req->ifd_data[i].cmd_buf_offset;
1035 update = (uint32_t *) field;
1036 *update = 0;
1037 }
1038 }
1039 return ret;
1040}
1041
1042static int __qseecom_update_with_phy_addr(
1043 struct qseecom_send_modfd_cmd_req *req)
1044{
1045 struct ion_handle *ihandle;
1046 char *field;
Mona Hossain2892b6b2012-02-17 13:53:11 -08001047 int ret = 0;
1048 int i = 0;
Mona Hossain2892b6b2012-02-17 13:53:11 -08001049
1050 for (i = 0; i < MAX_ION_FD; i++) {
Mona Hossainf1f2ed62012-11-15 19:51:33 -08001051 struct sg_table *sg_ptr = NULL;
Mona Hossaina5f1aab2012-03-29 10:18:07 -07001052 if (req->ifd_data[i].fd > 0) {
Mona Hossain2892b6b2012-02-17 13:53:11 -08001053 /* Get the handle of the shared fd */
Laura Abbottb14ed962012-01-30 14:18:08 -08001054 ihandle = ion_import_dma_buf(qseecom.ion_clnt,
Mona Hossain2892b6b2012-02-17 13:53:11 -08001055 req->ifd_data[i].fd);
1056 if (IS_ERR_OR_NULL(ihandle)) {
1057 pr_err("Ion client can't retrieve the handle\n");
1058 return -ENOMEM;
1059 }
1060 field = (char *) req->cmd_req_buf +
1061 req->ifd_data[i].cmd_buf_offset;
Mona Hossain2892b6b2012-02-17 13:53:11 -08001062
1063 /* Populate the cmd data structure with the phys_addr */
Mona Hossainf1f2ed62012-11-15 19:51:33 -08001064 sg_ptr = ion_sg_table(qseecom.ion_clnt, ihandle);
1065 if (sg_ptr == NULL) {
1066 pr_err("IOn client could not retrieve sg table\n");
1067 goto err;
1068 }
1069 if (sg_ptr->nents == 0) {
1070 pr_err("Num of scattered entries is 0\n");
1071 goto err;
1072 }
1073 if (sg_ptr->nents > QSEECOM_MAX_SG_ENTRY) {
1074 pr_err("Num of scattered entries");
1075 pr_err(" (%d) is greater than max supported %d\n",
1076 sg_ptr->nents, QSEECOM_MAX_SG_ENTRY);
1077 goto err;
1078 }
1079 if (sg_ptr->nents == 1) {
1080 uint32_t *update;
1081 update = (uint32_t *) field;
1082 *update = (uint32_t)sg_dma_address(sg_ptr->sgl);
1083 } else {
1084 struct qseecom_sg_entry *update;
1085 struct scatterlist *sg;
1086 int j = 0;
1087 update = (struct qseecom_sg_entry *) field;
1088 sg = sg_ptr->sgl;
1089 for (j = 0; j < sg_ptr->nents; j++) {
1090 update->phys_addr = (uint32_t)
1091 sg_dma_address(sg);
1092 update->len = (uint32_t)sg->length;
1093 update++;
1094 sg = sg_next(sg);
1095 }
1096 }
Mona Hossain2892b6b2012-02-17 13:53:11 -08001097 /* Deallocate the handle */
1098 if (!IS_ERR_OR_NULL(ihandle))
1099 ion_free(qseecom.ion_clnt, ihandle);
1100 }
1101 }
1102 return ret;
Mona Hossainf1f2ed62012-11-15 19:51:33 -08001103err:
1104 if (!IS_ERR_OR_NULL(ihandle))
1105 ion_free(qseecom.ion_clnt, ihandle);
1106 return -ENOMEM;
Mona Hossain2892b6b2012-02-17 13:53:11 -08001107}
1108
1109static int qseecom_send_modfd_cmd(struct qseecom_dev_handle *data,
1110 void __user *argp)
1111{
1112 int ret = 0;
1113 struct qseecom_send_modfd_cmd_req req;
1114 struct qseecom_send_cmd_req send_cmd_req;
1115
1116 ret = copy_from_user(&req, argp, sizeof(req));
1117 if (ret) {
1118 pr_err("copy_from_user failed\n");
1119 return ret;
1120 }
1121 send_cmd_req.cmd_req_buf = req.cmd_req_buf;
1122 send_cmd_req.cmd_req_len = req.cmd_req_len;
1123 send_cmd_req.resp_buf = req.resp_buf;
1124 send_cmd_req.resp_len = req.resp_len;
1125
1126 ret = __qseecom_update_with_phy_addr(&req);
1127 if (ret)
1128 return ret;
1129 if (qseecom.qseos_version == QSEOS_VERSION_14)
1130 ret = __qseecom_send_cmd(data, &send_cmd_req);
1131 else
1132 ret = __qseecom_send_cmd_legacy(data, &send_cmd_req);
1133 __qseecom_send_cmd_req_clean_up(&req);
1134
1135 if (ret)
1136 return ret;
1137
1138 pr_debug("sending cmd_req->rsp size: %u, ptr: 0x%p\n",
1139 req.resp_len, req.resp_buf);
1140 return ret;
1141}
1142
1143static int __qseecom_listener_has_rcvd_req(struct qseecom_dev_handle *data,
1144 struct qseecom_registered_listener_list *svc)
1145{
1146 int ret;
1147 ret = (svc->rcv_req_flag != 0);
1148 return ret || data->abort;
1149}
1150
1151static int qseecom_receive_req(struct qseecom_dev_handle *data)
1152{
1153 int ret = 0;
1154 struct qseecom_registered_listener_list *this_lstnr;
1155
1156 this_lstnr = __qseecom_find_svc(data->listener.id);
1157 while (1) {
Mona Hossainacea1022012-04-09 13:37:27 -07001158 if (wait_event_freezable(this_lstnr->rcv_req_wq,
Mona Hossain2892b6b2012-02-17 13:53:11 -08001159 __qseecom_listener_has_rcvd_req(data,
1160 this_lstnr))) {
1161 pr_warning("Interrupted: exiting wait_rcv_req loop\n");
1162 /* woken up for different reason */
1163 return -ERESTARTSYS;
1164 }
1165
1166 if (data->abort) {
1167 pr_err("Aborting driver!\n");
1168 return -ENODEV;
1169 }
1170 this_lstnr->rcv_req_flag = 0;
1171 if (qseecom.qseos_version == QSEOS_VERSION_13) {
1172 if (*((uint32_t *)this_lstnr->sb_virt) != 0)
1173 break;
1174 } else {
1175 break;
1176 }
1177 }
1178 return ret;
1179}
1180
Mona Hossaind44a3842012-10-15 09:41:35 -07001181static bool __qseecom_is_fw_image_valid(const struct firmware *fw_entry)
1182{
1183 struct elf32_hdr *ehdr;
1184
1185 if (fw_entry->size < sizeof(*ehdr)) {
1186 pr_err("%s: Not big enough to be an elf header\n",
1187 qseecom.pdev->init_name);
1188 return false;
1189 }
1190 ehdr = (struct elf32_hdr *)fw_entry->data;
1191 if (memcmp(ehdr->e_ident, ELFMAG, SELFMAG)) {
1192 pr_err("%s: Not an elf header\n",
1193 qseecom.pdev->init_name);
1194 return false;
1195 }
1196
1197 if (ehdr->e_phnum == 0) {
1198 pr_err("%s: No loadable segments\n",
1199 qseecom.pdev->init_name);
1200 return false;
1201 }
1202 if (sizeof(struct elf32_phdr) * ehdr->e_phnum +
1203 sizeof(struct elf32_hdr) > fw_entry->size) {
1204 pr_err("%s: Program headers not within mdt\n",
1205 qseecom.pdev->init_name);
1206 return false;
1207 }
1208 return true;
1209}
1210
1211static int __qseecom_get_fw_size(char *appname, uint32_t *fw_size)
1212{
1213 int ret = -1;
1214 int i = 0, rc = 0;
1215 const struct firmware *fw_entry = NULL;
1216 struct elf32_phdr *phdr;
1217 char fw_name[MAX_APP_NAME_SIZE];
1218 struct elf32_hdr *ehdr;
1219 int num_images = 0;
1220
1221 snprintf(fw_name, sizeof(fw_name), "%s.mdt", appname);
1222 rc = request_firmware(&fw_entry, fw_name, qseecom.pdev);
1223 if (rc) {
1224 pr_err("error with request_firmware\n");
1225 ret = -EIO;
1226 goto err;
1227 }
1228 if (!__qseecom_is_fw_image_valid(fw_entry)) {
1229 ret = -EIO;
1230 goto err;
1231 }
1232 *fw_size = fw_entry->size;
1233 phdr = (struct elf32_phdr *)(fw_entry->data + sizeof(struct elf32_hdr));
1234 ehdr = (struct elf32_hdr *)fw_entry->data;
1235 num_images = ehdr->e_phnum;
1236 release_firmware(fw_entry);
1237 for (i = 0; i < num_images; i++, phdr++) {
1238 memset(fw_name, 0, sizeof(fw_name));
1239 snprintf(fw_name, ARRAY_SIZE(fw_name), "%s.b%02d", appname, i);
1240 ret = request_firmware(&fw_entry, fw_name, qseecom.pdev);
1241 if (ret)
1242 goto err;
1243 *fw_size += fw_entry->size;
1244 release_firmware(fw_entry);
1245 }
1246 return ret;
1247err:
1248 if (fw_entry)
1249 release_firmware(fw_entry);
1250 *fw_size = 0;
1251 return ret;
1252}
1253
1254static int __qseecom_get_fw_data(char *appname, u8 *img_data,
1255 struct qseecom_load_app_ireq *load_req)
1256{
1257 int ret = -1;
1258 int i = 0, rc = 0;
1259 const struct firmware *fw_entry = NULL;
1260 char fw_name[MAX_APP_NAME_SIZE];
1261 u8 *img_data_ptr = img_data;
1262 struct elf32_hdr *ehdr;
1263 int num_images = 0;
1264
1265 snprintf(fw_name, sizeof(fw_name), "%s.mdt", appname);
1266 rc = request_firmware(&fw_entry, fw_name, qseecom.pdev);
1267 if (rc) {
1268 ret = -EIO;
1269 goto err;
1270 }
1271 load_req->img_len = fw_entry->size;
1272 memcpy(img_data_ptr, fw_entry->data, fw_entry->size);
1273 img_data_ptr = img_data_ptr + fw_entry->size;
1274 load_req->mdt_len = fw_entry->size; /*Get MDT LEN*/
1275 ehdr = (struct elf32_hdr *)fw_entry->data;
1276 num_images = ehdr->e_phnum;
1277 release_firmware(fw_entry);
1278 for (i = 0; i < num_images; i++) {
1279 snprintf(fw_name, ARRAY_SIZE(fw_name), "%s.b%02d", appname, i);
1280 ret = request_firmware(&fw_entry, fw_name, qseecom.pdev);
1281 if (ret) {
1282 pr_err("Failed to locate blob %s\n", fw_name);
1283 goto err;
1284 }
1285 memcpy(img_data_ptr, fw_entry->data, fw_entry->size);
1286 img_data_ptr = img_data_ptr + fw_entry->size;
1287 load_req->img_len += fw_entry->size;
1288 release_firmware(fw_entry);
1289 }
1290 load_req->phy_addr = virt_to_phys(img_data);
1291 return ret;
1292err:
1293 release_firmware(fw_entry);
1294 return ret;
1295}
1296
1297static int __qseecom_load_fw(struct qseecom_dev_handle *data, char *appname)
1298{
1299 int ret = -1;
1300 uint32_t fw_size = 0;
1301 struct qseecom_load_app_ireq load_req = {0, 0, 0, 0};
1302 struct qseecom_command_scm_resp resp;
1303 u8 *img_data = NULL;
1304
1305 if (__qseecom_get_fw_size(appname, &fw_size))
1306 return -EIO;
1307
1308 img_data = kzalloc(fw_size, GFP_KERNEL);
1309 if (!img_data) {
1310 pr_err("Failied to allocate memory for copying image data\n");
1311 return -ENOMEM;
1312 }
1313 ret = __qseecom_get_fw_data(appname, img_data, &load_req);
1314 if (ret) {
1315 kzfree(img_data);
1316 return -EIO;
1317 }
1318
1319 /* Populate the remaining parameters */
1320 load_req.qsee_cmd_id = QSEOS_APP_START_COMMAND;
1321 memcpy(load_req.app_name, appname, MAX_APP_NAME_SIZE);
Mona Hossain04d3fac2012-12-03 10:10:37 -08001322 ret = qsee_vote_for_clock(data, CLK_SFPB);
Mona Hossain60f9fb02012-11-05 13:51:50 -08001323 if (ret) {
1324 kzfree(img_data);
1325 pr_warning("Unable to vote for SFPB clock");
Mona Hossain60f9fb02012-11-05 13:51:50 -08001326 return -EIO;
1327 }
1328
Mona Hossaind44a3842012-10-15 09:41:35 -07001329 /* SCM_CALL to load the image */
1330 ret = scm_call(SCM_SVC_TZSCHEDULER, 1, &load_req,
1331 sizeof(struct qseecom_load_app_ireq),
1332 &resp, sizeof(resp));
1333 kzfree(img_data);
1334 if (ret) {
1335 pr_err("scm_call to load failed : ret %d\n", ret);
Mona Hossain04d3fac2012-12-03 10:10:37 -08001336 qsee_disable_clock_vote(data, CLK_SFPB);
Mona Hossaind44a3842012-10-15 09:41:35 -07001337 return -EIO;
1338 }
1339
1340 switch (resp.result) {
1341 case QSEOS_RESULT_SUCCESS:
1342 ret = resp.data;
1343 break;
1344 case QSEOS_RESULT_INCOMPLETE:
1345 ret = __qseecom_process_incomplete_cmd(data, &resp);
1346 if (ret)
1347 pr_err("process_incomplete_cmd FAILED\n");
1348 else
1349 ret = resp.data;
1350 break;
1351 case QSEOS_RESULT_FAILURE:
1352 pr_err("scm call failed with response QSEOS_RESULT FAILURE\n");
1353 break;
1354 default:
1355 pr_err("scm call return unknown response %d\n", resp.result);
1356 ret = -EINVAL;
1357 break;
1358 }
Mona Hossain04d3fac2012-12-03 10:10:37 -08001359 qsee_disable_clock_vote(data, CLK_SFPB);
Mona Hossain60f9fb02012-11-05 13:51:50 -08001360
Mona Hossaind44a3842012-10-15 09:41:35 -07001361 return ret;
1362}
1363
Mona Hossain9498f5e2013-01-23 18:08:45 -08001364static int qseecom_load_commonlib_image(struct qseecom_dev_handle *data)
Mona Hossain05c73562012-10-29 17:49:01 -07001365{
1366 int32_t ret = 0;
1367 uint32_t fw_size = 0;
1368 struct qseecom_load_app_ireq load_req = {0, 0, 0, 0};
1369 struct qseecom_command_scm_resp resp;
1370 u8 *img_data = NULL;
1371
Hariprasad Dhalinarasimhaa3b96442013-01-02 10:40:40 -08001372 if (__qseecom_get_fw_size("cmnlib", &fw_size))
Mona Hossain05c73562012-10-29 17:49:01 -07001373 return -EIO;
1374
1375 img_data = kzalloc(fw_size, GFP_KERNEL);
1376 if (!img_data) {
1377 pr_err("Mem allocation for lib image data failed\n");
1378 return -ENOMEM;
1379 }
Hariprasad Dhalinarasimhaa3b96442013-01-02 10:40:40 -08001380 ret = __qseecom_get_fw_data("cmnlib", img_data, &load_req);
Mona Hossain05c73562012-10-29 17:49:01 -07001381 if (ret) {
1382 kzfree(img_data);
1383 return -EIO;
1384 }
1385 /* Populate the remaining parameters */
1386 load_req.qsee_cmd_id = QSEOS_LOAD_SERV_IMAGE_COMMAND;
1387 /* SCM_CALL to load the image */
1388 ret = scm_call(SCM_SVC_TZSCHEDULER, 1, &load_req,
1389 sizeof(struct qseecom_load_lib_image_ireq),
1390 &resp, sizeof(resp));
Mona Hossain05c73562012-10-29 17:49:01 -07001391 if (ret) {
1392 pr_err("scm_call to load failed : ret %d\n", ret);
1393 ret = -EIO;
1394 } else {
1395 switch (resp.result) {
1396 case QSEOS_RESULT_SUCCESS:
1397 break;
1398 case QSEOS_RESULT_FAILURE:
1399 pr_err("scm call failed w/response result%d\n",
1400 resp.result);
1401 ret = -EINVAL;
1402 break;
Mona Hossain9498f5e2013-01-23 18:08:45 -08001403 case QSEOS_RESULT_INCOMPLETE:
1404 ret = __qseecom_process_incomplete_cmd(data, &resp);
1405 if (ret)
1406 pr_err("process_incomplete_cmd failed err: %d\n",
1407 ret);
1408 break;
Mona Hossain05c73562012-10-29 17:49:01 -07001409 default:
1410 pr_err("scm call return unknown response %d\n",
1411 resp.result);
1412 ret = -EINVAL;
1413 break;
1414 }
1415 }
Hariprasad Dhalinarasimha1a81ca32013-01-31 18:32:32 -08001416 kzfree(img_data);
Mona Hossain05c73562012-10-29 17:49:01 -07001417 return ret;
1418}
1419
1420static int qseecom_unload_commonlib_image(void)
1421{
1422 int ret = -EINVAL;
1423 struct qseecom_unload_lib_image_ireq unload_req = {0};
1424 struct qseecom_command_scm_resp resp;
1425
1426 /* Populate the remaining parameters */
1427 unload_req.qsee_cmd_id = QSEOS_UNLOAD_SERV_IMAGE_COMMAND;
1428 /* SCM_CALL to load the image */
1429 ret = scm_call(SCM_SVC_TZSCHEDULER, 1, &unload_req,
1430 sizeof(struct qseecom_unload_lib_image_ireq),
1431 &resp, sizeof(resp));
1432 if (ret) {
1433 pr_err("scm_call to unload lib failed : ret %d\n", ret);
1434 ret = -EIO;
1435 } else {
1436 switch (resp.result) {
1437 case QSEOS_RESULT_SUCCESS:
1438 break;
1439 case QSEOS_RESULT_FAILURE:
1440 pr_err("scm fail resp.result QSEOS_RESULT FAILURE\n");
1441 break;
1442 default:
1443 pr_err("scm call return unknown response %d\n",
1444 resp.result);
1445 ret = -EINVAL;
1446 break;
1447 }
1448 }
1449 return ret;
1450}
1451
Mona Hossaind44a3842012-10-15 09:41:35 -07001452int qseecom_start_app(struct qseecom_handle **handle,
1453 char *app_name, uint32_t size)
1454{
Mona Hossain05c73562012-10-29 17:49:01 -07001455 int32_t ret = 0;
Mona Hossaind44a3842012-10-15 09:41:35 -07001456 unsigned long flags = 0;
1457 struct qseecom_dev_handle *data = NULL;
1458 struct qseecom_check_app_ireq app_ireq;
1459 struct qseecom_registered_app_list *entry = NULL;
1460 struct qseecom_registered_kclient_list *kclient_entry = NULL;
1461 bool found_app = false;
1462 uint32_t len;
1463 ion_phys_addr_t pa;
1464
1465 if (qseecom.qseos_version == QSEOS_VERSION_13) {
1466 pr_err("This functionality is UNSUPPORTED in version 1.3\n");
1467 return -EINVAL;
1468 }
1469
Mona Hossain823f9882012-11-23 14:42:20 -08001470 *handle = kzalloc(sizeof(struct qseecom_handle), GFP_KERNEL);
1471 if (!(*handle)) {
1472 pr_err("failed to allocate memory for kernel client handle\n");
1473 return -ENOMEM;
1474 }
1475
Mona Hossaind44a3842012-10-15 09:41:35 -07001476 data = kzalloc(sizeof(*data), GFP_KERNEL);
1477 if (!data) {
1478 pr_err("kmalloc failed\n");
1479 if (ret == 0) {
1480 kfree(*handle);
1481 *handle = NULL;
1482 }
1483 return -ENOMEM;
1484 }
1485 data->abort = 0;
1486 data->service = false;
1487 data->released = false;
1488 data->client.app_id = ret;
1489 data->client.sb_length = size;
1490 data->client.user_virt_sb_base = 0;
1491 data->client.ihandle = NULL;
1492
1493 init_waitqueue_head(&data->abort_wq);
1494 atomic_set(&data->ioctl_count, 0);
1495
1496 data->client.ihandle = ion_alloc(qseecom.ion_clnt, size, 4096,
1497 ION_HEAP(ION_QSECOM_HEAP_ID), 0);
1498 if (IS_ERR_OR_NULL(data->client.ihandle)) {
1499 pr_err("Ion client could not retrieve the handle\n");
1500 kfree(data);
1501 kfree(*handle);
1502 *handle = NULL;
1503 return -EINVAL;
1504 }
1505
Mona Hossain9498f5e2013-01-23 18:08:45 -08001506 if (qseecom.qsee_version > QSEEE_VERSION_00) {
1507 mutex_lock(&app_access_lock);
1508 if (qseecom.commonlib_loaded == false) {
1509 ret = qseecom_load_commonlib_image(data);
1510 if (ret == 0)
1511 qseecom.commonlib_loaded = true;
1512 }
1513 mutex_unlock(&app_access_lock);
1514 }
1515
1516 if (ret) {
1517 pr_err("Failed to loadd commonlib image\n");
1518 kfree(data);
1519 kfree(*handle);
1520 *handle = NULL;
1521 return -EIO;
1522 }
1523
1524 app_ireq.qsee_cmd_id = QSEOS_APP_LOOKUP_COMMAND;
1525 memcpy(app_ireq.app_name, app_name, MAX_APP_NAME_SIZE);
1526 ret = __qseecom_check_app_exists(app_ireq);
1527 if (ret < 0)
1528 return -EINVAL;
1529
Mona Hossaind44a3842012-10-15 09:41:35 -07001530 if (ret > 0) {
1531 pr_warn("App id %d for [%s] app exists\n", ret,
1532 (char *)app_ireq.app_name);
1533 spin_lock_irqsave(&qseecom.registered_app_list_lock, flags);
1534 list_for_each_entry(entry,
1535 &qseecom.registered_app_list_head, list){
1536 if (entry->app_id == ret) {
1537 entry->ref_cnt++;
1538 found_app = true;
1539 break;
1540 }
1541 }
1542 spin_unlock_irqrestore(
1543 &qseecom.registered_app_list_lock, flags);
1544 if (!found_app)
1545 pr_warn("App_id %d [%s] was loaded but not registered\n",
1546 ret, (char *)app_ireq.app_name);
1547 } else {
1548 /* load the app and get the app_id */
1549 pr_debug("%s: Loading app for the first time'\n",
1550 qseecom.pdev->init_name);
1551 mutex_lock(&app_access_lock);
1552 ret = __qseecom_load_fw(data, app_name);
1553 mutex_unlock(&app_access_lock);
1554
1555 if (ret < 0) {
1556 kfree(*handle);
1557 *handle = NULL;
1558 return ret;
1559 }
1560 data->client.app_id = ret;
1561 }
1562 if (!found_app) {
1563 entry = kmalloc(sizeof(*entry), GFP_KERNEL);
1564 if (!entry) {
1565 pr_err("kmalloc failed\n");
1566 return -ENOMEM;
1567 }
1568 entry->app_id = ret;
1569 entry->ref_cnt = 1;
1570
1571 spin_lock_irqsave(&qseecom.registered_app_list_lock, flags);
1572 list_add_tail(&entry->list, &qseecom.registered_app_list_head);
1573 spin_unlock_irqrestore(&qseecom.registered_app_list_lock,
1574 flags);
1575 }
1576
1577 /* Get the physical address of the ION BUF */
1578 ret = ion_phys(qseecom.ion_clnt, data->client.ihandle, &pa, &len);
1579 /* Populate the structure for sending scm call to load image */
1580 data->client.sb_virt = (char *) ion_map_kernel(qseecom.ion_clnt,
1581 data->client.ihandle);
Hariprasad Dhalinarasimhaacfb09c2013-01-10 13:16:15 -08001582 data->client.user_virt_sb_base = (uint32_t)data->client.sb_virt;
Mona Hossaind44a3842012-10-15 09:41:35 -07001583 data->client.sb_phys = pa;
1584 (*handle)->dev = (void *)data;
1585 (*handle)->sbuf = (unsigned char *)data->client.sb_virt;
1586 (*handle)->sbuf_len = data->client.sb_length;
1587
1588 kclient_entry = kzalloc(sizeof(*kclient_entry), GFP_KERNEL);
1589 if (!kclient_entry) {
1590 pr_err("kmalloc failed\n");
1591 return -ENOMEM;
1592 }
1593 kclient_entry->handle = *handle;
1594
1595 spin_lock_irqsave(&qseecom.registered_kclient_list_lock, flags);
1596 list_add_tail(&kclient_entry->list,
1597 &qseecom.registered_kclient_list_head);
1598 spin_unlock_irqrestore(&qseecom.registered_kclient_list_lock, flags);
1599
1600 return 0;
1601}
1602EXPORT_SYMBOL(qseecom_start_app);
1603
1604int qseecom_shutdown_app(struct qseecom_handle **handle)
1605{
1606 int ret = -EINVAL;
Mona Hossain33824022013-02-25 09:32:33 -08001607 struct qseecom_dev_handle *data;
1608
Mona Hossaind44a3842012-10-15 09:41:35 -07001609 struct qseecom_registered_kclient_list *kclient = NULL;
1610 unsigned long flags = 0;
1611 bool found_handle = false;
1612
1613 if (qseecom.qseos_version == QSEOS_VERSION_13) {
1614 pr_err("This functionality is UNSUPPORTED in version 1.3\n");
1615 return -EINVAL;
1616 }
Mona Hossain33824022013-02-25 09:32:33 -08001617 if ((handle == NULL) || (*handle == NULL)) {
Mona Hossaind44a3842012-10-15 09:41:35 -07001618 pr_err("Handle is not initialized\n");
1619 return -EINVAL;
1620 }
Mona Hossain33824022013-02-25 09:32:33 -08001621 data = (struct qseecom_dev_handle *) ((*handle)->dev);
Mona Hossaind44a3842012-10-15 09:41:35 -07001622 spin_lock_irqsave(&qseecom.registered_kclient_list_lock, flags);
1623 list_for_each_entry(kclient, &qseecom.registered_kclient_list_head,
1624 list) {
1625 if (kclient->handle == (*handle)) {
1626 list_del(&kclient->list);
1627 found_handle = true;
1628 break;
1629 }
1630 }
1631 spin_unlock_irqrestore(&qseecom.registered_kclient_list_lock, flags);
1632 if (!found_handle)
1633 pr_err("Unable to find the handle, exiting\n");
1634 else
1635 ret = qseecom_unload_app(data);
Mona Hossain04d3fac2012-12-03 10:10:37 -08001636 if (data->client.fast_load_enabled == true)
1637 qsee_disable_clock_vote(data, CLK_SFPB);
1638 if (data->client.perf_enabled == true)
1639 qsee_disable_clock_vote(data, CLK_DFAB);
Mona Hossaind44a3842012-10-15 09:41:35 -07001640 if (ret == 0) {
1641 kzfree(data);
1642 kzfree(*handle);
1643 kzfree(kclient);
1644 *handle = NULL;
1645 }
1646 return ret;
1647}
1648EXPORT_SYMBOL(qseecom_shutdown_app);
1649
1650int qseecom_send_command(struct qseecom_handle *handle, void *send_buf,
1651 uint32_t sbuf_len, void *resp_buf, uint32_t rbuf_len)
1652{
1653 int ret = 0;
1654 struct qseecom_send_cmd_req req = {0, 0, 0, 0};
1655 struct qseecom_dev_handle *data;
1656
1657 if (qseecom.qseos_version == QSEOS_VERSION_13) {
1658 pr_err("This functionality is UNSUPPORTED in version 1.3\n");
1659 return -EINVAL;
1660 }
1661
1662 if (handle == NULL) {
1663 pr_err("Handle is not initialized\n");
1664 return -EINVAL;
1665 }
1666 data = handle->dev;
1667
1668 req.cmd_req_len = sbuf_len;
1669 req.resp_len = rbuf_len;
1670 req.cmd_req_buf = send_buf;
1671 req.resp_buf = resp_buf;
1672
1673 mutex_lock(&app_access_lock);
1674 atomic_inc(&data->ioctl_count);
1675
1676 ret = __qseecom_send_cmd(data, &req);
1677
1678 atomic_dec(&data->ioctl_count);
1679 mutex_unlock(&app_access_lock);
1680
1681 if (ret)
1682 return ret;
1683
1684 pr_debug("sending cmd_req->rsp size: %u, ptr: 0x%p\n",
1685 req.resp_len, req.resp_buf);
1686 return ret;
1687}
1688EXPORT_SYMBOL(qseecom_send_command);
1689
Mona Hossain91a8fc92012-11-07 19:58:30 -08001690int qseecom_set_bandwidth(struct qseecom_handle *handle, bool high)
1691{
Mona Hossainfca6f422013-01-12 13:00:35 -08001692 int ret = 0;
Mona Hossain91a8fc92012-11-07 19:58:30 -08001693 if ((handle == NULL) || (handle->dev == NULL)) {
1694 pr_err("No valid kernel client\n");
1695 return -EINVAL;
1696 }
Mona Hossainfca6f422013-01-12 13:00:35 -08001697 if (high) {
1698 ret = qsee_vote_for_clock(handle->dev, CLK_DFAB);
1699 if (ret)
1700 pr_err("Failed to vote for DFAB clock%d\n", ret);
1701 ret = qsee_vote_for_clock(handle->dev, CLK_SFPB);
1702 if (ret) {
1703 pr_err("Failed to vote for SFPB clock%d\n", ret);
1704 qsee_disable_clock_vote(handle->dev, CLK_DFAB);
1705 }
1706 } else {
Mona Hossain04d3fac2012-12-03 10:10:37 -08001707 qsee_disable_clock_vote(handle->dev, CLK_DFAB);
Mona Hossainfca6f422013-01-12 13:00:35 -08001708 qsee_disable_clock_vote(handle->dev, CLK_SFPB);
Mona Hossain91a8fc92012-11-07 19:58:30 -08001709 }
Mona Hossainfca6f422013-01-12 13:00:35 -08001710 return ret;
Mona Hossain91a8fc92012-11-07 19:58:30 -08001711}
1712EXPORT_SYMBOL(qseecom_set_bandwidth);
1713
Mona Hossain2892b6b2012-02-17 13:53:11 -08001714static int qseecom_send_resp(void)
1715{
1716 qseecom.send_resp_flag = 1;
1717 wake_up_interruptible(&qseecom.send_resp_wq);
1718 return 0;
1719}
1720
1721static int qseecom_get_qseos_version(struct qseecom_dev_handle *data,
1722 void __user *argp)
1723{
1724 struct qseecom_qseos_version_req req;
1725
1726 if (copy_from_user(&req, argp, sizeof(req))) {
1727 pr_err("copy_from_user failed");
1728 return -EINVAL;
1729 }
1730 req.qseos_version = qseecom.qseos_version;
1731 if (copy_to_user(argp, &req, sizeof(req))) {
1732 pr_err("copy_to_user failed");
1733 return -EINVAL;
1734 }
1735 return 0;
1736}
1737
Mona Hossain04d3fac2012-12-03 10:10:37 -08001738static int qsee_vote_for_clock(struct qseecom_dev_handle *data,
1739 int32_t clk_type)
Mona Hossain2892b6b2012-02-17 13:53:11 -08001740{
1741 int ret = 0;
1742
1743 if (!qsee_perf_client)
Ramesh Masavarapue640e842012-04-03 11:21:54 -07001744 return ret;
Mona Hossain2892b6b2012-02-17 13:53:11 -08001745
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07001746 switch (clk_type) {
1747 case CLK_DFAB:
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07001748 mutex_lock(&qsee_bw_mutex);
1749 if (!qsee_bw_count) {
Ramesh Masavarapu8d756582012-10-03 10:18:06 -07001750 if (qsee_sfpb_bw_count > 0)
1751 ret = msm_bus_scale_client_update_request(
1752 qsee_perf_client, 3);
Mona Hossaind39e33b2012-11-05 13:36:40 -08001753 else {
1754 if (ce_core_src_clk != NULL)
1755 clk_set_rate(ce_core_src_clk,
1756 QSEE_CE_CLK_100MHZ);
Ramesh Masavarapu8d756582012-10-03 10:18:06 -07001757 ret = msm_bus_scale_client_update_request(
1758 qsee_perf_client, 1);
Mona Hossaind39e33b2012-11-05 13:36:40 -08001759 }
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07001760 if (ret)
1761 pr_err("DFAB Bandwidth req failed (%d)\n",
1762 ret);
Mona Hossain04d3fac2012-12-03 10:10:37 -08001763 else {
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07001764 qsee_bw_count++;
Mona Hossain04d3fac2012-12-03 10:10:37 -08001765 data->client.perf_enabled = true;
1766 }
Ramesh Masavarapu8d756582012-10-03 10:18:06 -07001767 } else {
1768 qsee_bw_count++;
Mona Hossain04d3fac2012-12-03 10:10:37 -08001769 data->client.perf_enabled = true;
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07001770 }
1771 mutex_unlock(&qsee_bw_mutex);
1772 break;
1773 case CLK_SFPB:
Ramesh Masavarapu8d756582012-10-03 10:18:06 -07001774 mutex_lock(&qsee_bw_mutex);
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07001775 if (!qsee_sfpb_bw_count) {
Ramesh Masavarapu8d756582012-10-03 10:18:06 -07001776 if (qsee_bw_count > 0)
1777 ret = msm_bus_scale_client_update_request(
1778 qsee_perf_client, 3);
Mona Hossaind39e33b2012-11-05 13:36:40 -08001779 else {
1780 if (ce_core_src_clk != NULL)
1781 clk_set_rate(ce_core_src_clk,
1782 QSEE_CE_CLK_100MHZ);
Ramesh Masavarapu8d756582012-10-03 10:18:06 -07001783 ret = msm_bus_scale_client_update_request(
1784 qsee_perf_client, 2);
Mona Hossaind39e33b2012-11-05 13:36:40 -08001785 }
Ramesh Masavarapu8d756582012-10-03 10:18:06 -07001786
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07001787 if (ret)
1788 pr_err("SFPB Bandwidth req failed (%d)\n",
1789 ret);
Mona Hossain04d3fac2012-12-03 10:10:37 -08001790 else {
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07001791 qsee_sfpb_bw_count++;
Mona Hossain04d3fac2012-12-03 10:10:37 -08001792 data->client.fast_load_enabled = true;
1793 }
Ramesh Masavarapu8d756582012-10-03 10:18:06 -07001794 } else {
1795 qsee_sfpb_bw_count++;
Mona Hossain04d3fac2012-12-03 10:10:37 -08001796 data->client.fast_load_enabled = true;
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07001797 }
Ramesh Masavarapu8d756582012-10-03 10:18:06 -07001798 mutex_unlock(&qsee_bw_mutex);
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07001799 break;
1800 default:
1801 pr_err("Clock type not defined\n");
1802 break;
Mona Hossain2892b6b2012-02-17 13:53:11 -08001803 }
Mona Hossain2892b6b2012-02-17 13:53:11 -08001804 return ret;
1805}
1806
Mona Hossain04d3fac2012-12-03 10:10:37 -08001807static void qsee_disable_clock_vote(struct qseecom_dev_handle *data,
1808 int32_t clk_type)
Mona Hossain2892b6b2012-02-17 13:53:11 -08001809{
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07001810 int32_t ret = 0;
Ramesh Masavarapua1bc0e42012-03-05 07:42:48 -08001811
Mona Hossain2892b6b2012-02-17 13:53:11 -08001812 if (!qsee_perf_client)
Ramesh Masavarapua1bc0e42012-03-05 07:42:48 -08001813 return;
1814
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07001815 switch (clk_type) {
1816 case CLK_DFAB:
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07001817 mutex_lock(&qsee_bw_mutex);
Ramesh Masavarapu8d756582012-10-03 10:18:06 -07001818 if (qsee_bw_count == 0) {
1819 pr_err("Client error.Extra call to disable DFAB clk\n");
1820 mutex_unlock(&qsee_bw_mutex);
1821 return;
1822 }
1823
Mona Hossain04d3fac2012-12-03 10:10:37 -08001824 if (qsee_bw_count == 1) {
Ramesh Masavarapu8d756582012-10-03 10:18:06 -07001825 if (qsee_sfpb_bw_count > 0)
1826 ret = msm_bus_scale_client_update_request(
1827 qsee_perf_client, 2);
Mona Hossaind39e33b2012-11-05 13:36:40 -08001828 else {
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07001829 ret = msm_bus_scale_client_update_request(
1830 qsee_perf_client, 0);
Mona Hossaind39e33b2012-11-05 13:36:40 -08001831 if (ce_core_src_clk != NULL)
1832 clk_set_rate(ce_core_src_clk,
1833 QSEE_CE_CLK_50MHZ);
1834 }
Ramesh Masavarapu8d756582012-10-03 10:18:06 -07001835 if (ret)
1836 pr_err("SFPB Bandwidth req fail (%d)\n",
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07001837 ret);
Mona Hossain04d3fac2012-12-03 10:10:37 -08001838 else {
1839 qsee_bw_count--;
1840 data->client.perf_enabled = false;
1841 }
1842 } else {
1843 qsee_bw_count--;
1844 data->client.perf_enabled = false;
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07001845 }
1846 mutex_unlock(&qsee_bw_mutex);
1847 break;
1848 case CLK_SFPB:
Ramesh Masavarapu8d756582012-10-03 10:18:06 -07001849 mutex_lock(&qsee_bw_mutex);
1850 if (qsee_sfpb_bw_count == 0) {
1851 pr_err("Client error.Extra call to disable SFPB clk\n");
1852 mutex_unlock(&qsee_bw_mutex);
1853 return;
1854 }
Mona Hossain04d3fac2012-12-03 10:10:37 -08001855 if (qsee_sfpb_bw_count == 1) {
Ramesh Masavarapu8d756582012-10-03 10:18:06 -07001856 if (qsee_bw_count > 0)
1857 ret = msm_bus_scale_client_update_request(
1858 qsee_perf_client, 1);
Mona Hossaind39e33b2012-11-05 13:36:40 -08001859 else {
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07001860 ret = msm_bus_scale_client_update_request(
1861 qsee_perf_client, 0);
Mona Hossaind39e33b2012-11-05 13:36:40 -08001862 if (ce_core_src_clk != NULL)
1863 clk_set_rate(ce_core_src_clk,
1864 QSEE_CE_CLK_50MHZ);
1865 }
Ramesh Masavarapu8d756582012-10-03 10:18:06 -07001866 if (ret)
1867 pr_err("SFPB Bandwidth req fail (%d)\n",
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07001868 ret);
Mona Hossain04d3fac2012-12-03 10:10:37 -08001869 else {
1870 qsee_sfpb_bw_count--;
1871 data->client.fast_load_enabled = false;
1872 }
1873 } else {
1874 qsee_sfpb_bw_count--;
1875 data->client.fast_load_enabled = false;
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07001876 }
Ramesh Masavarapu8d756582012-10-03 10:18:06 -07001877 mutex_unlock(&qsee_bw_mutex);
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07001878 break;
1879 default:
1880 pr_err("Clock type not defined\n");
1881 break;
Ramesh Masavarapue640e842012-04-03 11:21:54 -07001882 }
Mona Hossain2892b6b2012-02-17 13:53:11 -08001883
Mona Hossain2892b6b2012-02-17 13:53:11 -08001884}
1885
Mona Hossain5ab9d772012-04-11 21:00:40 -07001886static int qseecom_load_external_elf(struct qseecom_dev_handle *data,
1887 void __user *argp)
1888{
1889 struct ion_handle *ihandle; /* Ion handle */
1890 struct qseecom_load_img_req load_img_req;
Ramesh Masavarapu41af8ab2012-05-21 11:08:00 -07001891 int ret;
1892 int set_cpu_ret = 0;
Mona Hossain5ab9d772012-04-11 21:00:40 -07001893 ion_phys_addr_t pa = 0;
1894 uint32_t len;
Ramesh Masavarapu41af8ab2012-05-21 11:08:00 -07001895 struct cpumask mask;
Mona Hossain5ab9d772012-04-11 21:00:40 -07001896 struct qseecom_load_app_ireq load_req;
1897 struct qseecom_command_scm_resp resp;
1898
1899 /* Copy the relevant information needed for loading the image */
1900 if (__copy_from_user(&load_img_req,
1901 (void __user *)argp,
1902 sizeof(struct qseecom_load_img_req))) {
1903 pr_err("copy_from_user failed\n");
1904 return -EFAULT;
1905 }
1906
1907 /* Get the handle of the shared fd */
Laura Abbottb14ed962012-01-30 14:18:08 -08001908 ihandle = ion_import_dma_buf(qseecom.ion_clnt,
Mona Hossain5ab9d772012-04-11 21:00:40 -07001909 load_img_req.ifd_data_fd);
1910 if (IS_ERR_OR_NULL(ihandle)) {
1911 pr_err("Ion client could not retrieve the handle\n");
1912 return -ENOMEM;
1913 }
1914
1915 /* Get the physical address of the ION BUF */
1916 ret = ion_phys(qseecom.ion_clnt, ihandle, &pa, &len);
1917
1918 /* Populate the structure for sending scm call to load image */
1919 load_req.qsee_cmd_id = QSEOS_LOAD_EXTERNAL_ELF_COMMAND;
1920 load_req.mdt_len = load_img_req.mdt_len;
1921 load_req.img_len = load_img_req.img_len;
1922 load_req.phy_addr = pa;
1923
Ramesh Masavarapu41af8ab2012-05-21 11:08:00 -07001924 /* SCM_CALL tied to Core0 */
1925 mask = CPU_MASK_CPU0;
1926 set_cpu_ret = set_cpus_allowed_ptr(current, &mask);
1927 if (set_cpu_ret) {
1928 pr_err("set_cpus_allowed_ptr failed : ret %d\n",
1929 set_cpu_ret);
1930 ret = -EFAULT;
1931 goto qseecom_load_external_elf_set_cpu_err;
1932 }
1933
Mona Hossain5ab9d772012-04-11 21:00:40 -07001934 /* SCM_CALL to load the external elf */
1935 ret = scm_call(SCM_SVC_TZSCHEDULER, 1, &load_req,
1936 sizeof(struct qseecom_load_app_ireq),
1937 &resp, sizeof(resp));
1938 if (ret) {
Ramesh Masavarapu41af8ab2012-05-21 11:08:00 -07001939 pr_err("scm_call to load failed : ret %d\n",
Mona Hossain5ab9d772012-04-11 21:00:40 -07001940 ret);
1941 ret = -EFAULT;
Ramesh Masavarapu41af8ab2012-05-21 11:08:00 -07001942 goto qseecom_load_external_elf_scm_err;
Mona Hossain5ab9d772012-04-11 21:00:40 -07001943 }
1944
1945 if (resp.result == QSEOS_RESULT_INCOMPLETE) {
1946 ret = __qseecom_process_incomplete_cmd(data, &resp);
1947 if (ret)
1948 pr_err("process_incomplete_cmd failed err: %d\n",
1949 ret);
1950 } else {
1951 if (resp.result != QSEOS_RESULT_SUCCESS) {
1952 pr_err("scm_call to load image failed resp.result =%d\n",
1953 resp.result);
1954 ret = -EFAULT;
1955 }
1956 }
Ramesh Masavarapu41af8ab2012-05-21 11:08:00 -07001957
1958qseecom_load_external_elf_scm_err:
1959 /* Restore the CPU mask */
1960 mask = CPU_MASK_ALL;
1961 set_cpu_ret = set_cpus_allowed_ptr(current, &mask);
1962 if (set_cpu_ret) {
1963 pr_err("set_cpus_allowed_ptr failed to restore mask: ret %d\n",
1964 set_cpu_ret);
1965 ret = -EFAULT;
1966 }
1967
1968qseecom_load_external_elf_set_cpu_err:
Mona Hossain5ab9d772012-04-11 21:00:40 -07001969 /* Deallocate the handle */
1970 if (!IS_ERR_OR_NULL(ihandle))
1971 ion_free(qseecom.ion_clnt, ihandle);
1972
1973 return ret;
1974}
1975
1976static int qseecom_unload_external_elf(struct qseecom_dev_handle *data)
1977{
1978 int ret = 0;
Ramesh Masavarapu41af8ab2012-05-21 11:08:00 -07001979 int set_cpu_ret = 0;
Mona Hossain5ab9d772012-04-11 21:00:40 -07001980 struct qseecom_command_scm_resp resp;
1981 struct qseecom_unload_app_ireq req;
Ramesh Masavarapu41af8ab2012-05-21 11:08:00 -07001982 struct cpumask mask;
Mona Hossain5ab9d772012-04-11 21:00:40 -07001983
1984 /* Populate the structure for sending scm call to unload image */
1985 req.qsee_cmd_id = QSEOS_UNLOAD_EXTERNAL_ELF_COMMAND;
Ramesh Masavarapu41af8ab2012-05-21 11:08:00 -07001986
1987 /* SCM_CALL tied to Core0 */
1988 mask = CPU_MASK_CPU0;
1989 ret = set_cpus_allowed_ptr(current, &mask);
1990 if (ret) {
1991 pr_err("set_cpus_allowed_ptr failed : ret %d\n",
1992 ret);
1993 return -EFAULT;
1994 }
1995
Mona Hossain5ab9d772012-04-11 21:00:40 -07001996 /* SCM_CALL to unload the external elf */
1997 ret = scm_call(SCM_SVC_TZSCHEDULER, 1, &req,
1998 sizeof(struct qseecom_unload_app_ireq),
1999 &resp, sizeof(resp));
2000 if (ret) {
2001 pr_err("scm_call to unload failed : ret %d\n",
2002 ret);
Ramesh Masavarapu41af8ab2012-05-21 11:08:00 -07002003 ret = -EFAULT;
2004 goto qseecom_unload_external_elf_scm_err;
Mona Hossain5ab9d772012-04-11 21:00:40 -07002005 }
2006 if (resp.result == QSEOS_RESULT_INCOMPLETE) {
2007 ret = __qseecom_process_incomplete_cmd(data, &resp);
2008 if (ret)
2009 pr_err("process_incomplete_cmd fail err: %d\n",
2010 ret);
2011 } else {
2012 if (resp.result != QSEOS_RESULT_SUCCESS) {
2013 pr_err("scm_call to unload image failed resp.result =%d\n",
2014 resp.result);
2015 ret = -EFAULT;
2016 }
2017 }
Ramesh Masavarapu41af8ab2012-05-21 11:08:00 -07002018
2019qseecom_unload_external_elf_scm_err:
2020 /* Restore the CPU mask */
2021 mask = CPU_MASK_ALL;
2022 set_cpu_ret = set_cpus_allowed_ptr(current, &mask);
2023 if (set_cpu_ret) {
2024 pr_err("set_cpus_allowed_ptr failed to restore mask: ret %d\n",
2025 set_cpu_ret);
2026 ret = -EFAULT;
2027 }
2028
Mona Hossain5ab9d772012-04-11 21:00:40 -07002029 return ret;
2030}
Mona Hossain2892b6b2012-02-17 13:53:11 -08002031
Ramesh Masavarapu1ee4ac82012-06-22 15:38:34 -07002032static int qseecom_query_app_loaded(struct qseecom_dev_handle *data,
2033 void __user *argp)
2034{
2035
2036 int32_t ret;
2037 struct qseecom_qseos_app_load_query query_req;
2038 struct qseecom_check_app_ireq req;
Ramesh Masavarapu13d99672012-07-17 11:05:15 -07002039 struct qseecom_registered_app_list *entry = NULL;
2040 unsigned long flags = 0;
Ramesh Masavarapu1ee4ac82012-06-22 15:38:34 -07002041
2042 /* Copy the relevant information needed for loading the image */
2043 if (__copy_from_user(&query_req,
2044 (void __user *)argp,
2045 sizeof(struct qseecom_qseos_app_load_query))) {
2046 pr_err("copy_from_user failed\n");
2047 return -EFAULT;
2048 }
2049
2050 req.qsee_cmd_id = QSEOS_APP_LOOKUP_COMMAND;
2051 memcpy(req.app_name, query_req.app_name, MAX_APP_NAME_SIZE);
2052
2053 ret = __qseecom_check_app_exists(req);
Ramesh Masavarapu13d99672012-07-17 11:05:15 -07002054
2055 if ((ret == -EINVAL) || (ret == -ENODEV)) {
Ramesh Masavarapu1ee4ac82012-06-22 15:38:34 -07002056 pr_err(" scm call to check if app is loaded failed");
2057 return ret; /* scm call failed */
2058 } else if (ret > 0) {
Ramesh Masavarapu13d99672012-07-17 11:05:15 -07002059 pr_warn("App id %d (%s) already exists\n", ret,
2060 (char *)(req.app_name));
2061 spin_lock_irqsave(&qseecom.registered_app_list_lock, flags);
2062 list_for_each_entry(entry,
2063 &qseecom.registered_app_list_head, list){
2064 if (entry->app_id == ret) {
2065 entry->ref_cnt++;
2066 break;
2067 }
2068 }
2069 spin_unlock_irqrestore(
2070 &qseecom.registered_app_list_lock, flags);
2071 data->client.app_id = ret;
2072 query_req.app_id = ret;
2073
2074 if (copy_to_user(argp, &query_req, sizeof(query_req))) {
2075 pr_err("copy_to_user failed\n");
2076 return -EFAULT;
2077 }
Ramesh Masavarapu1ee4ac82012-06-22 15:38:34 -07002078 return -EEXIST; /* app already loaded */
2079 } else {
2080 return 0; /* app not loaded */
2081 }
2082}
2083
Mona Hossain2892b6b2012-02-17 13:53:11 -08002084static long qseecom_ioctl(struct file *file, unsigned cmd,
2085 unsigned long arg)
2086{
2087 int ret = 0;
2088 struct qseecom_dev_handle *data = file->private_data;
2089 void __user *argp = (void __user *) arg;
2090
2091 if (data->abort) {
2092 pr_err("Aborting qseecom driver\n");
2093 return -ENODEV;
2094 }
2095
2096 switch (cmd) {
2097 case QSEECOM_IOCTL_REGISTER_LISTENER_REQ: {
2098 pr_debug("ioctl register_listener_req()\n");
2099 atomic_inc(&data->ioctl_count);
2100 ret = qseecom_register_listener(data, argp);
2101 atomic_dec(&data->ioctl_count);
2102 wake_up_all(&data->abort_wq);
2103 if (ret)
2104 pr_err("failed qseecom_register_listener: %d\n", ret);
2105 break;
2106 }
2107 case QSEECOM_IOCTL_UNREGISTER_LISTENER_REQ: {
2108 pr_debug("ioctl unregister_listener_req()\n");
2109 atomic_inc(&data->ioctl_count);
2110 ret = qseecom_unregister_listener(data);
2111 atomic_dec(&data->ioctl_count);
2112 wake_up_all(&data->abort_wq);
2113 if (ret)
2114 pr_err("failed qseecom_unregister_listener: %d\n", ret);
2115 break;
2116 }
2117 case QSEECOM_IOCTL_SEND_CMD_REQ: {
2118 /* Only one client allowed here at a time */
Ramesh Masavarapud8610112012-06-26 15:32:52 -07002119 mutex_lock(&app_access_lock);
Mona Hossain2892b6b2012-02-17 13:53:11 -08002120 atomic_inc(&data->ioctl_count);
2121 ret = qseecom_send_cmd(data, argp);
2122 atomic_dec(&data->ioctl_count);
2123 wake_up_all(&data->abort_wq);
Ramesh Masavarapud8610112012-06-26 15:32:52 -07002124 mutex_unlock(&app_access_lock);
Mona Hossain2892b6b2012-02-17 13:53:11 -08002125 if (ret)
2126 pr_err("failed qseecom_send_cmd: %d\n", ret);
2127 break;
2128 }
2129 case QSEECOM_IOCTL_SEND_MODFD_CMD_REQ: {
2130 /* Only one client allowed here at a time */
Ramesh Masavarapud8610112012-06-26 15:32:52 -07002131 mutex_lock(&app_access_lock);
Mona Hossain2892b6b2012-02-17 13:53:11 -08002132 atomic_inc(&data->ioctl_count);
2133 ret = qseecom_send_modfd_cmd(data, argp);
2134 atomic_dec(&data->ioctl_count);
2135 wake_up_all(&data->abort_wq);
Ramesh Masavarapud8610112012-06-26 15:32:52 -07002136 mutex_unlock(&app_access_lock);
Mona Hossain2892b6b2012-02-17 13:53:11 -08002137 if (ret)
2138 pr_err("failed qseecom_send_cmd: %d\n", ret);
2139 break;
2140 }
2141 case QSEECOM_IOCTL_RECEIVE_REQ: {
2142 atomic_inc(&data->ioctl_count);
2143 ret = qseecom_receive_req(data);
2144 atomic_dec(&data->ioctl_count);
2145 wake_up_all(&data->abort_wq);
2146 if (ret)
2147 pr_err("failed qseecom_receive_req: %d\n", ret);
2148 break;
2149 }
2150 case QSEECOM_IOCTL_SEND_RESP_REQ: {
2151 atomic_inc(&data->ioctl_count);
2152 ret = qseecom_send_resp();
2153 atomic_dec(&data->ioctl_count);
2154 wake_up_all(&data->abort_wq);
2155 if (ret)
2156 pr_err("failed qseecom_send_resp: %d\n", ret);
2157 break;
2158 }
2159 case QSEECOM_IOCTL_SET_MEM_PARAM_REQ: {
2160 ret = qseecom_set_client_mem_param(data, argp);
2161 if (ret)
2162 pr_err("failed Qqseecom_set_mem_param request: %d\n",
2163 ret);
2164 break;
2165 }
2166 case QSEECOM_IOCTL_LOAD_APP_REQ: {
2167 mutex_lock(&app_access_lock);
2168 atomic_inc(&data->ioctl_count);
Mona Hossain05c73562012-10-29 17:49:01 -07002169 if (qseecom.qsee_version > QSEEE_VERSION_00) {
2170 if (qseecom.commonlib_loaded == false) {
Mona Hossain9498f5e2013-01-23 18:08:45 -08002171 ret = qseecom_load_commonlib_image(data);
Mona Hossain05c73562012-10-29 17:49:01 -07002172 if (ret == 0)
2173 qseecom.commonlib_loaded = true;
2174 }
2175 }
2176 if (ret == 0)
2177 ret = qseecom_load_app(data, argp);
Mona Hossain2892b6b2012-02-17 13:53:11 -08002178 atomic_dec(&data->ioctl_count);
2179 mutex_unlock(&app_access_lock);
2180 if (ret)
2181 pr_err("failed load_app request: %d\n", ret);
2182 break;
2183 }
2184 case QSEECOM_IOCTL_UNLOAD_APP_REQ: {
2185 mutex_lock(&app_access_lock);
2186 atomic_inc(&data->ioctl_count);
2187 ret = qseecom_unload_app(data);
2188 atomic_dec(&data->ioctl_count);
2189 mutex_unlock(&app_access_lock);
2190 if (ret)
2191 pr_err("failed unload_app request: %d\n", ret);
2192 break;
2193 }
2194 case QSEECOM_IOCTL_GET_QSEOS_VERSION_REQ: {
2195 atomic_inc(&data->ioctl_count);
2196 ret = qseecom_get_qseos_version(data, argp);
2197 if (ret)
2198 pr_err("qseecom_get_qseos_version: %d\n", ret);
2199 atomic_dec(&data->ioctl_count);
2200 break;
2201 }
2202 case QSEECOM_IOCTL_PERF_ENABLE_REQ:{
2203 atomic_inc(&data->ioctl_count);
Mona Hossain04d3fac2012-12-03 10:10:37 -08002204 ret = qsee_vote_for_clock(data, CLK_DFAB);
Mona Hossain2892b6b2012-02-17 13:53:11 -08002205 if (ret)
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07002206 pr_err("Failed to vote for DFAB clock%d\n", ret);
Mona Hossain8e2d73a2013-01-10 04:30:04 -08002207 ret = qsee_vote_for_clock(data, CLK_SFPB);
2208 if (ret)
2209 pr_err("Failed to vote for SFPB clock%d\n", ret);
Mona Hossain2892b6b2012-02-17 13:53:11 -08002210 atomic_dec(&data->ioctl_count);
2211 break;
2212 }
2213 case QSEECOM_IOCTL_PERF_DISABLE_REQ:{
2214 atomic_inc(&data->ioctl_count);
Mona Hossain04d3fac2012-12-03 10:10:37 -08002215 qsee_disable_clock_vote(data, CLK_DFAB);
Mona Hossain8e2d73a2013-01-10 04:30:04 -08002216 qsee_disable_clock_vote(data, CLK_SFPB);
Mona Hossain2892b6b2012-02-17 13:53:11 -08002217 atomic_dec(&data->ioctl_count);
2218 break;
2219 }
Mona Hossain5ab9d772012-04-11 21:00:40 -07002220 case QSEECOM_IOCTL_LOAD_EXTERNAL_ELF_REQ: {
2221 data->released = true;
2222 if (qseecom.qseos_version == QSEOS_VERSION_13) {
2223 pr_err("Loading External elf image unsupported in rev 0x13\n");
2224 ret = -EINVAL;
2225 break;
2226 }
2227 mutex_lock(&app_access_lock);
2228 atomic_inc(&data->ioctl_count);
2229 ret = qseecom_load_external_elf(data, argp);
2230 atomic_dec(&data->ioctl_count);
2231 mutex_unlock(&app_access_lock);
2232 if (ret)
2233 pr_err("failed load_external_elf request: %d\n", ret);
2234 break;
2235 }
2236 case QSEECOM_IOCTL_UNLOAD_EXTERNAL_ELF_REQ: {
2237 data->released = true;
2238 if (qseecom.qseos_version == QSEOS_VERSION_13) {
2239 pr_err("Unloading External elf image unsupported in rev 0x13\n");
2240 ret = -EINVAL;
2241 break;
2242 }
2243 mutex_lock(&app_access_lock);
2244 atomic_inc(&data->ioctl_count);
2245 ret = qseecom_unload_external_elf(data);
2246 atomic_dec(&data->ioctl_count);
2247 mutex_unlock(&app_access_lock);
2248 if (ret)
2249 pr_err("failed unload_app request: %d\n", ret);
2250 break;
2251 }
Ramesh Masavarapu1ee4ac82012-06-22 15:38:34 -07002252 case QSEECOM_IOCTL_APP_LOADED_QUERY_REQ: {
2253 mutex_lock(&app_access_lock);
2254 atomic_inc(&data->ioctl_count);
2255 ret = qseecom_query_app_loaded(data, argp);
2256 atomic_dec(&data->ioctl_count);
2257 mutex_unlock(&app_access_lock);
2258 break;
2259 }
Mona Hossain2892b6b2012-02-17 13:53:11 -08002260 default:
2261 return -EINVAL;
2262 }
2263 return ret;
2264}
2265
2266static int qseecom_open(struct inode *inode, struct file *file)
2267{
2268 int ret = 0;
2269 struct qseecom_dev_handle *data;
2270
2271 data = kzalloc(sizeof(*data), GFP_KERNEL);
2272 if (!data) {
2273 pr_err("kmalloc failed\n");
2274 return -ENOMEM;
2275 }
2276 file->private_data = data;
2277 data->abort = 0;
2278 data->service = false;
2279 data->released = false;
2280 init_waitqueue_head(&data->abort_wq);
2281 atomic_set(&data->ioctl_count, 0);
Mona Hossain7cdbfaf2012-03-15 18:56:10 -07002282 if (qseecom.qseos_version == QSEOS_VERSION_13) {
2283 int pil_error;
2284 mutex_lock(&pil_access_lock);
2285 if (pil_ref_cnt == 0) {
Stephen Boyd77db8bb2012-06-27 15:15:16 -07002286 pil = subsystem_get("tzapps");
Mona Hossain7cdbfaf2012-03-15 18:56:10 -07002287 if (IS_ERR(pil)) {
2288 pr_err("Playready PIL image load failed\n");
2289 pil_error = PTR_ERR(pil);
2290 pil = NULL;
2291 pr_debug("tzapps image load FAILED\n");
2292 mutex_unlock(&pil_access_lock);
2293 return pil_error;
2294 }
2295 }
2296 pil_ref_cnt++;
2297 mutex_unlock(&pil_access_lock);
2298 }
Mona Hossain2892b6b2012-02-17 13:53:11 -08002299 return ret;
2300}
2301
2302static int qseecom_release(struct inode *inode, struct file *file)
2303{
2304 struct qseecom_dev_handle *data = file->private_data;
2305 int ret = 0;
2306
2307 if (data->released == false) {
2308 pr_warn("data->released == false\n");
2309 if (data->service)
2310 ret = qseecom_unregister_listener(data);
2311 else
2312 ret = qseecom_unload_app(data);
2313 if (ret) {
2314 pr_err("Close failed\n");
2315 return ret;
2316 }
2317 }
Mona Hossain04d3fac2012-12-03 10:10:37 -08002318 if (data->client.fast_load_enabled == true)
2319 qsee_disable_clock_vote(data, CLK_SFPB);
2320 if (data->client.perf_enabled == true)
2321 qsee_disable_clock_vote(data, CLK_DFAB);
2322
Mona Hossain7cdbfaf2012-03-15 18:56:10 -07002323 if (qseecom.qseos_version == QSEOS_VERSION_13) {
2324 mutex_lock(&pil_access_lock);
2325 if (pil_ref_cnt == 1)
Stephen Boyd77db8bb2012-06-27 15:15:16 -07002326 subsystem_put(pil);
Mona Hossain7cdbfaf2012-03-15 18:56:10 -07002327 pil_ref_cnt--;
2328 mutex_unlock(&pil_access_lock);
2329 }
Mona Hossain2892b6b2012-02-17 13:53:11 -08002330 kfree(data);
Ramesh Masavarapua1bc0e42012-03-05 07:42:48 -08002331
Mona Hossain2892b6b2012-02-17 13:53:11 -08002332 return ret;
2333}
2334
Mona Hossain2892b6b2012-02-17 13:53:11 -08002335static const struct file_operations qseecom_fops = {
2336 .owner = THIS_MODULE,
2337 .unlocked_ioctl = qseecom_ioctl,
2338 .open = qseecom_open,
2339 .release = qseecom_release
2340};
2341
Mona Hossaind39e33b2012-11-05 13:36:40 -08002342static int __qseecom_enable_clk(void)
2343{
2344 int rc = 0;
2345
2346 /* Enable CE core clk */
2347 rc = clk_prepare_enable(ce_core_clk);
2348 if (rc) {
2349 pr_err("Unable to enable/prepare CE core clk\n");
2350 return -EIO;
2351 } else {
2352 /* Enable CE clk */
2353 rc = clk_prepare_enable(ce_clk);
2354 if (rc) {
2355 pr_err("Unable to enable/prepare CE iface clk\n");
2356 clk_disable_unprepare(ce_core_clk);
2357 return -EIO;
2358 } else {
2359 /* Enable AXI clk */
2360 rc = clk_prepare_enable(ce_bus_clk);
2361 if (rc) {
2362 pr_err("Unable to enable/prepare CE iface clk\n");
2363 clk_disable_unprepare(ce_core_clk);
2364 clk_disable_unprepare(ce_clk);
2365 return -EIO;
2366 }
2367 }
2368 }
2369 return rc;
2370}
2371
2372static void __qseecom_disable_clk(void)
2373{
2374 if (ce_clk != NULL)
2375 clk_disable_unprepare(ce_clk);
2376 if (ce_core_clk != NULL)
2377 clk_disable_unprepare(ce_core_clk);
2378 if (ce_bus_clk != NULL)
2379 clk_disable_unprepare(ce_bus_clk);
2380}
2381
2382static int __qseecom_init_clk(void)
Ramesh Masavarapuff377032012-09-14 12:11:32 -07002383{
2384 int rc = 0;
2385 struct device *pdev;
2386
2387 pdev = qseecom.pdev;
2388 /* Get CE3 src core clk. */
2389 ce_core_src_clk = clk_get(pdev, "core_clk_src");
2390 if (!IS_ERR(ce_core_src_clk)) {
Mona Hossaind39e33b2012-11-05 13:36:40 -08002391 /* Set the core src clk @50Mhz */
2392 rc = clk_set_rate(ce_core_src_clk, QSEE_CE_CLK_50MHZ);
Ramesh Masavarapuff377032012-09-14 12:11:32 -07002393 if (rc) {
2394 clk_put(ce_core_src_clk);
2395 pr_err("Unable to set the core src clk @100Mhz.\n");
Mona Hossaind39e33b2012-11-05 13:36:40 -08002396 return -EIO;
Ramesh Masavarapuff377032012-09-14 12:11:32 -07002397 }
2398 } else {
2399 pr_warn("Unable to get CE core src clk, set to NULL\n");
2400 ce_core_src_clk = NULL;
2401 }
2402
2403 /* Get CE core clk */
2404 ce_core_clk = clk_get(pdev, "core_clk");
2405 if (IS_ERR(ce_core_clk)) {
2406 rc = PTR_ERR(ce_core_clk);
2407 pr_err("Unable to get CE core clk\n");
2408 if (ce_core_src_clk != NULL)
2409 clk_put(ce_core_src_clk);
Mona Hossaind39e33b2012-11-05 13:36:40 -08002410 return -EIO;
Ramesh Masavarapuff377032012-09-14 12:11:32 -07002411 }
2412
2413 /* Get CE Interface clk */
2414 ce_clk = clk_get(pdev, "iface_clk");
2415 if (IS_ERR(ce_clk)) {
2416 rc = PTR_ERR(ce_clk);
2417 pr_err("Unable to get CE interface clk\n");
2418 if (ce_core_src_clk != NULL)
2419 clk_put(ce_core_src_clk);
2420 clk_put(ce_core_clk);
Mona Hossaind39e33b2012-11-05 13:36:40 -08002421 return -EIO;
Ramesh Masavarapuff377032012-09-14 12:11:32 -07002422 }
2423
2424 /* Get CE AXI clk */
2425 ce_bus_clk = clk_get(pdev, "bus_clk");
2426 if (IS_ERR(ce_bus_clk)) {
2427 rc = PTR_ERR(ce_bus_clk);
2428 pr_err("Unable to get CE BUS interface clk\n");
2429 if (ce_core_src_clk != NULL)
2430 clk_put(ce_core_src_clk);
2431 clk_put(ce_core_clk);
2432 clk_put(ce_clk);
Mona Hossaind39e33b2012-11-05 13:36:40 -08002433 return -EIO;
Ramesh Masavarapuff377032012-09-14 12:11:32 -07002434 }
Ramesh Masavarapuff377032012-09-14 12:11:32 -07002435 return rc;
2436}
2437
Mona Hossaind39e33b2012-11-05 13:36:40 -08002438static void __qseecom_deinit_clk(void)
Ramesh Masavarapuff377032012-09-14 12:11:32 -07002439{
Mona Hossaind39e33b2012-11-05 13:36:40 -08002440 if (ce_clk != NULL) {
2441 clk_put(ce_clk);
2442 ce_clk = NULL;
2443 }
2444 if (ce_core_clk != NULL) {
2445 clk_put(ce_core_clk);
2446 ce_clk = NULL;
2447 }
2448 if (ce_bus_clk != NULL) {
2449 clk_put(ce_bus_clk);
2450 ce_clk = NULL;
2451 }
2452 if (ce_core_src_clk != NULL) {
Ramesh Masavarapuff377032012-09-14 12:11:32 -07002453 clk_put(ce_core_src_clk);
Mona Hossaind39e33b2012-11-05 13:36:40 -08002454 ce_core_src_clk = NULL;
2455 }
Ramesh Masavarapuff377032012-09-14 12:11:32 -07002456}
2457
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07002458static int __devinit qseecom_probe(struct platform_device *pdev)
Mona Hossain2892b6b2012-02-17 13:53:11 -08002459{
2460 int rc;
Mona Hossaind39e33b2012-11-05 13:36:40 -08002461 int ret = 0;
Mona Hossain2892b6b2012-02-17 13:53:11 -08002462 struct device *class_dev;
2463 char qsee_not_legacy = 0;
Mona Hossaind44a3842012-10-15 09:41:35 -07002464 struct msm_bus_scale_pdata *qseecom_platform_support = NULL;
Mona Hossain2892b6b2012-02-17 13:53:11 -08002465 uint32_t system_call_id = QSEOS_CHECK_VERSION_CMD;
2466
Ramesh Masavarapue640e842012-04-03 11:21:54 -07002467 qsee_bw_count = 0;
Ramesh Masavarapue640e842012-04-03 11:21:54 -07002468 qsee_perf_client = 0;
Mona Hossaind39e33b2012-11-05 13:36:40 -08002469 qsee_sfpb_bw_count = 0;
2470
2471 ce_core_clk = NULL;
2472 ce_clk = NULL;
2473 ce_core_src_clk = NULL;
2474 ce_bus_clk = NULL;
Ramesh Masavarapue640e842012-04-03 11:21:54 -07002475
Mona Hossain2892b6b2012-02-17 13:53:11 -08002476 rc = alloc_chrdev_region(&qseecom_device_no, 0, 1, QSEECOM_DEV);
2477 if (rc < 0) {
2478 pr_err("alloc_chrdev_region failed %d\n", rc);
2479 return rc;
2480 }
2481
2482 driver_class = class_create(THIS_MODULE, QSEECOM_DEV);
2483 if (IS_ERR(driver_class)) {
2484 rc = -ENOMEM;
2485 pr_err("class_create failed %d\n", rc);
2486 goto unregister_chrdev_region;
2487 }
2488
2489 class_dev = device_create(driver_class, NULL, qseecom_device_no, NULL,
2490 QSEECOM_DEV);
2491 if (!class_dev) {
2492 pr_err("class_device_create failed %d\n", rc);
2493 rc = -ENOMEM;
2494 goto class_destroy;
2495 }
2496
2497 cdev_init(&qseecom_cdev, &qseecom_fops);
2498 qseecom_cdev.owner = THIS_MODULE;
2499
2500 rc = cdev_add(&qseecom_cdev, MKDEV(MAJOR(qseecom_device_no), 0), 1);
2501 if (rc < 0) {
2502 pr_err("cdev_add failed %d\n", rc);
2503 goto err;
2504 }
2505
2506 INIT_LIST_HEAD(&qseecom.registered_listener_list_head);
2507 spin_lock_init(&qseecom.registered_listener_list_lock);
2508 INIT_LIST_HEAD(&qseecom.registered_app_list_head);
2509 spin_lock_init(&qseecom.registered_app_list_lock);
Mona Hossaind44a3842012-10-15 09:41:35 -07002510 INIT_LIST_HEAD(&qseecom.registered_kclient_list_head);
2511 spin_lock_init(&qseecom.registered_kclient_list_lock);
Mona Hossain2892b6b2012-02-17 13:53:11 -08002512 init_waitqueue_head(&qseecom.send_resp_wq);
2513 qseecom.send_resp_flag = 0;
2514
2515 rc = scm_call(6, 1, &system_call_id, sizeof(system_call_id),
2516 &qsee_not_legacy, sizeof(qsee_not_legacy));
2517 if (rc) {
Mona Hossain05c73562012-10-29 17:49:01 -07002518 pr_err("Failed to retrieve QSEOS version information %d\n", rc);
Mona Hossain2892b6b2012-02-17 13:53:11 -08002519 goto err;
2520 }
Mona Hossain05c73562012-10-29 17:49:01 -07002521 if (qsee_not_legacy) {
2522 uint32_t feature = 10;
2523
2524 qseecom.qsee_version = QSEEE_VERSION_00;
2525 rc = scm_call(6, 3, &feature, sizeof(feature),
2526 &qseecom.qsee_version, sizeof(qseecom.qsee_version));
2527 if (rc) {
2528 pr_err("Failed to get QSEE version info %d\n", rc);
2529 goto err;
2530 }
Mona Hossain2892b6b2012-02-17 13:53:11 -08002531 qseecom.qseos_version = QSEOS_VERSION_14;
Mona Hossain05c73562012-10-29 17:49:01 -07002532 } else {
Mona Hossain2892b6b2012-02-17 13:53:11 -08002533 qseecom.qseos_version = QSEOS_VERSION_13;
Mona Hossain05c73562012-10-29 17:49:01 -07002534 qseecom.qsee_version = 0;
Mona Hossain2892b6b2012-02-17 13:53:11 -08002535 pil = NULL;
2536 pil_ref_cnt = 0;
2537 }
Mona Hossain05c73562012-10-29 17:49:01 -07002538 qseecom.commonlib_loaded = false;
Ramesh Masavarapuff377032012-09-14 12:11:32 -07002539 qseecom.pdev = class_dev;
Mona Hossain2892b6b2012-02-17 13:53:11 -08002540 /* Create ION msm client */
Mona Hossaind44a3842012-10-15 09:41:35 -07002541 qseecom.ion_clnt = msm_ion_client_create(-1, "qseecom-kernel");
Mona Hossain2892b6b2012-02-17 13:53:11 -08002542 if (qseecom.ion_clnt == NULL) {
2543 pr_err("Ion client cannot be created\n");
2544 rc = -ENOMEM;
2545 goto err;
2546 }
2547
2548 /* register client for bus scaling */
Ramesh Masavarapuff377032012-09-14 12:11:32 -07002549 if (pdev->dev.of_node) {
2550 ret = __qseecom_init_clk();
2551 if (ret)
2552 goto err;
Mona Hossaind39e33b2012-11-05 13:36:40 -08002553 ret = __qseecom_enable_clk();
2554 if (ret) {
2555 __qseecom_deinit_clk();
2556 goto err;
2557 }
Ramesh Masavarapuff377032012-09-14 12:11:32 -07002558 qseecom_platform_support = (struct msm_bus_scale_pdata *)
2559 msm_bus_cl_get_pdata(pdev);
Mona Hossain5b76a622012-11-15 20:09:08 -08002560 if (qseecom.qsee_version >= (QSEE_VERSION_02)) {
2561 struct resource *resource = NULL;
2562 struct qsee_apps_region_info_ireq req;
2563 struct qseecom_command_scm_resp resp;
2564
2565 resource = platform_get_resource_byname(pdev,
2566 IORESOURCE_MEM, "secapp-region");
2567 if (resource) {
2568 req.qsee_cmd_id = QSEOS_APP_REGION_NOTIFICATION;
2569 req.addr = resource->start;
2570 req.size = resource_size(resource);
2571 pr_warn("secure app region addr=0x%x size=0x%x",
2572 req.addr, req.size);
2573 } else {
2574 pr_err("Fail to get secure app region info\n");
2575 rc = -EINVAL;
2576 goto err;
2577 }
2578 rc = scm_call(SCM_SVC_TZSCHEDULER, 1, &req, sizeof(req),
2579 &resp, sizeof(resp));
2580 if (rc) {
2581 pr_err("Failed to send secapp region info %d\n",
2582 rc);
2583 goto err;
2584 }
2585 }
Ramesh Masavarapuff377032012-09-14 12:11:32 -07002586 } else {
Ramesh Masavarapufb1f01e2012-06-14 09:40:40 -07002587 qseecom_platform_support = (struct msm_bus_scale_pdata *)
2588 pdev->dev.platform_data;
Mona Hossain2892b6b2012-02-17 13:53:11 -08002589 }
Ramesh Masavarapua1bc0e42012-03-05 07:42:48 -08002590
Ramesh Masavarapuff377032012-09-14 12:11:32 -07002591 qsee_perf_client = msm_bus_scale_register_client(
2592 qseecom_platform_support);
2593
2594 if (!qsee_perf_client)
2595 pr_err("Unable to register bus client\n");
2596 return 0;
Mona Hossain2892b6b2012-02-17 13:53:11 -08002597err:
2598 device_destroy(driver_class, qseecom_device_no);
2599class_destroy:
2600 class_destroy(driver_class);
2601unregister_chrdev_region:
2602 unregister_chrdev_region(qseecom_device_no, 1);
2603 return rc;
2604}
2605
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07002606static int __devinit qseecom_remove(struct platform_device *pdev)
2607{
Mona Hossaind44a3842012-10-15 09:41:35 -07002608 struct qseecom_registered_kclient_list *kclient = NULL;
2609 unsigned long flags = 0;
2610 int ret = 0;
2611
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07002612 if (pdev->dev.platform_data != NULL)
2613 msm_bus_scale_unregister_client(qsee_perf_client);
Mona Hossaind44a3842012-10-15 09:41:35 -07002614
2615 spin_lock_irqsave(&qseecom.registered_kclient_list_lock, flags);
2616 kclient = list_entry((&qseecom.registered_kclient_list_head)->next,
2617 struct qseecom_registered_kclient_list, list);
2618 if (list_empty(&kclient->list)) {
2619 spin_unlock_irqrestore(&qseecom.registered_kclient_list_lock,
2620 flags);
2621 return 0;
2622 }
2623 list_for_each_entry(kclient, &qseecom.registered_kclient_list_head,
2624 list) {
2625 if (kclient)
2626 list_del(&kclient->list);
2627 break;
2628 }
2629 spin_unlock_irqrestore(&qseecom.registered_kclient_list_lock, flags);
2630
2631
2632 while (kclient->handle != NULL) {
2633 ret = qseecom_unload_app(kclient->handle->dev);
2634 if (ret == 0) {
2635 kzfree(kclient->handle->dev);
2636 kzfree(kclient->handle);
2637 kzfree(kclient);
2638 }
2639 spin_lock_irqsave(&qseecom.registered_kclient_list_lock, flags);
2640 kclient = list_entry(
2641 (&qseecom.registered_kclient_list_head)->next,
2642 struct qseecom_registered_kclient_list, list);
2643 if (list_empty(&kclient->list)) {
2644 spin_unlock_irqrestore(
2645 &qseecom.registered_kclient_list_lock, flags);
2646 return 0;
2647 }
2648 list_for_each_entry(kclient,
2649 &qseecom.registered_kclient_list_head, list) {
2650 if (kclient)
2651 list_del(&kclient->list);
2652 break;
2653 }
2654 spin_unlock_irqrestore(&qseecom.registered_kclient_list_lock,
2655 flags);
2656 if (!kclient) {
2657 ret = 0;
2658 break;
2659 }
2660 }
Mona Hossain05c73562012-10-29 17:49:01 -07002661 if (qseecom.qseos_version > QSEEE_VERSION_00)
2662 qseecom_unload_commonlib_image();
Mona Hossaind39e33b2012-11-05 13:36:40 -08002663
2664 if (qsee_perf_client)
2665 msm_bus_scale_client_update_request(qsee_perf_client, 0);
2666 /* register client for bus scaling */
2667 if (pdev->dev.of_node) {
2668 __qseecom_disable_clk();
2669 __qseecom_deinit_clk();
2670 }
Mona Hossaind44a3842012-10-15 09:41:35 -07002671 return ret;
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07002672};
2673
Ramesh Masavarapufb1f01e2012-06-14 09:40:40 -07002674static struct of_device_id qseecom_match[] = {
2675 {
2676 .compatible = "qcom,qseecom",
2677 },
2678 {}
2679};
2680
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07002681static struct platform_driver qseecom_plat_driver = {
2682 .probe = qseecom_probe,
2683 .remove = qseecom_remove,
2684 .driver = {
2685 .name = "qseecom",
2686 .owner = THIS_MODULE,
Ramesh Masavarapufb1f01e2012-06-14 09:40:40 -07002687 .of_match_table = qseecom_match,
Ramesh Masavarapua26cce72012-04-09 12:32:25 -07002688 },
2689};
2690
2691static int __devinit qseecom_init(void)
2692{
2693 return platform_driver_register(&qseecom_plat_driver);
2694}
2695
2696static void __devexit qseecom_exit(void)
Mona Hossain2892b6b2012-02-17 13:53:11 -08002697{
Mona Hossain2892b6b2012-02-17 13:53:11 -08002698 device_destroy(driver_class, qseecom_device_no);
2699 class_destroy(driver_class);
2700 unregister_chrdev_region(qseecom_device_no, 1);
2701 ion_client_destroy(qseecom.ion_clnt);
2702}
2703
2704MODULE_LICENSE("GPL v2");
2705MODULE_DESCRIPTION("Qualcomm Secure Execution Environment Communicator");
2706
2707module_init(qseecom_init);
2708module_exit(qseecom_exit);