blob: 8a5957c71244b2229b2e9a4ea2061520924762ab [file] [log] [blame]
Arun Kumar Neelakantamcb1ee8d2013-05-10 13:05:37 +05301/* Copyright (c) 2012-2013, The Linux Foundation. All rights reserved.
Karthikeyan Ramasubramaniane1f4f732011-08-08 13:34:47 -06002 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 and
5 * only version 2 as published by the Free Software Foundation.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 *
12 */
13
14#include <linux/slab.h>
15#include <linux/uaccess.h>
16#include <linux/module.h>
17#include <linux/fs.h>
18#include <linux/kernel.h>
19#include <linux/errno.h>
20#include <linux/jiffies.h>
21#include <linux/debugfs.h>
22#include <linux/io.h>
23#include <linux/idr.h>
24#include <linux/string.h>
25#include <linux/sched.h>
26#include <linux/wait.h>
27#include <linux/delay.h>
28#include <linux/completion.h>
29
30#include <mach/msm_ipc_logging.h>
31
32#include "ipc_logging.h"
33
34static DEFINE_MUTEX(ipc_log_debugfs_init_lock);
35static struct dentry *root_dent;
Karthikeyan Ramasubramaniane1f4f732011-08-08 13:34:47 -060036
37static int debug_log(struct ipc_log_context *ilctxt,
38 char *buff, int size, int cont)
39{
40 int i = 0;
41
42 if (size < MAX_MSG_DECODED_SIZE) {
43 pr_err("%s: buffer size %d < %d\n", __func__, size,
44 MAX_MSG_DECODED_SIZE);
45 return -ENOMEM;
46 }
47 do {
Eric Holmbergdd734f92013-06-14 17:17:47 -060048 i = ipc_log_extract(ilctxt, buff, size - 1);
Karthikeyan Ramasubramaniane1f4f732011-08-08 13:34:47 -060049 if (cont && i == 0) {
50 wait_for_completion_interruptible(&ilctxt->read_avail);
51 if (signal_pending(current))
52 break;
53 }
54 } while (cont && i == 0);
55
56 return i;
57}
58
59/*
60 * VFS Read operation helper which dispatches the call to the debugfs
61 * read command stored in file->private_data.
62 *
63 * @file File structure
64 * @buff user buffer
65 * @count size of user buffer
66 * @ppos file position to read from (only a value of 0 is accepted)
67 * @cont 1 = continuous mode (don't return 0 to signal end-of-file)
68 *
69 * @returns ==0 end of file
70 * >0 number of bytes read
71 * <0 error
72 */
73static ssize_t debug_read_helper(struct file *file, char __user *buff,
74 size_t count, loff_t *ppos, int cont)
75{
76 struct ipc_log_context *ilctxt = file->private_data;
77 char *buffer;
78 int bsize;
79
80 buffer = kmalloc(count, GFP_KERNEL);
81 if (!buffer)
82 return -ENOMEM;
83
84 bsize = debug_log(ilctxt, buffer, count, cont);
85 if (bsize > 0) {
86 if (copy_to_user(buff, buffer, bsize)) {
87 kfree(buffer);
88 return -EFAULT;
89 }
90 *ppos += bsize;
91 }
92 kfree(buffer);
93 return bsize;
94}
95
96static ssize_t debug_read(struct file *file, char __user *buff,
97 size_t count, loff_t *ppos)
98{
99 return debug_read_helper(file, buff, count, ppos, 0);
100}
101
102static ssize_t debug_read_cont(struct file *file, char __user *buff,
103 size_t count, loff_t *ppos)
104{
105 return debug_read_helper(file, buff, count, ppos, 1);
106}
107
108static int debug_open(struct inode *inode, struct file *file)
109{
110 file->private_data = inode->i_private;
111 return 0;
112}
113
114static const struct file_operations debug_ops = {
115 .read = debug_read,
116 .open = debug_open,
117};
118
119static const struct file_operations debug_ops_cont = {
120 .read = debug_read_cont,
121 .open = debug_open,
122};
123
124static void debug_create(const char *name, mode_t mode,
125 struct dentry *dent,
126 struct ipc_log_context *ilctxt,
127 const struct file_operations *fops)
128{
129 debugfs_create_file(name, mode, dent, ilctxt, fops);
130}
131
132static void dfunc_string(struct encode_context *ectxt,
133 struct decode_context *dctxt)
134{
135 tsv_timestamp_read(ectxt, dctxt, " ");
136 tsv_byte_array_read(ectxt, dctxt, "");
Eric Holmberg06ed50a2012-08-07 13:23:31 -0600137
138 /* add trailing \n if necessary */
139 if (*(dctxt->buff - 1) != '\n') {
140 if (dctxt->size) {
141 ++dctxt->buff;
142 --dctxt->size;
143 }
144 *(dctxt->buff - 1) = '\n';
145 }
Karthikeyan Ramasubramaniane1f4f732011-08-08 13:34:47 -0600146}
147
148void check_and_create_debugfs(void)
149{
150 mutex_lock(&ipc_log_debugfs_init_lock);
151 if (!root_dent) {
152 root_dent = debugfs_create_dir("ipc_logging", 0);
153
154 if (IS_ERR(root_dent)) {
155 pr_err("%s: unable to create debugfs %ld\n",
156 __func__, IS_ERR(root_dent));
157 root_dent = NULL;
158 }
159 }
160 mutex_unlock(&ipc_log_debugfs_init_lock);
161}
162EXPORT_SYMBOL(check_and_create_debugfs);
163
164void create_ctx_debugfs(struct ipc_log_context *ctxt,
165 const char *mod_name)
166{
167 if (!root_dent)
168 check_and_create_debugfs();
169
170 if (root_dent) {
171 ctxt->dent = debugfs_create_dir(mod_name, root_dent);
172 if (!IS_ERR(ctxt->dent)) {
173 debug_create("log", 0444, ctxt->dent,
174 ctxt, &debug_ops);
175 debug_create("log_cont", 0444, ctxt->dent,
176 ctxt, &debug_ops_cont);
177 }
178 }
179 add_deserialization_func((void *)ctxt,
180 TSV_TYPE_STRING, dfunc_string);
181}
182EXPORT_SYMBOL(create_ctx_debugfs);