blob: 6e73a60a5492f94d52e47979d002d02fb38434ea [file] [log] [blame]
Duy Truong790f06d2013-02-13 16:38:12 -08001/* Copyright (c) 2011, The Linux Foundation. All rights reserved.
Laxminath Kasam1a461112010-11-15 12:17:26 +05302 *
3 * This software is licensed under the terms of the GNU General Public
4 * License version 2, as published by the Free Software Foundation, and
5 * may be copied, distributed, and modified under those terms.
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 <asm/atomic.h>
15#include <linux/debugfs.h>
16#include <linux/delay.h>
17#include <linux/fs.h>
18#include <linux/io.h>
19#include <linux/miscdevice.h>
20#include <linux/module.h>
21#include <linux/sched.h>
22#include <linux/uaccess.h>
23#include <linux/wait.h>
24
25#include <mach/debug_mm.h>
26#include <mach/msm_iomap.h>
27
28#include "dsp_debug.h"
29
30static wait_queue_head_t dsp_wait;
31static int dsp_has_crashed;
32static int dsp_wait_count;
33
34static atomic_t dsp_crash_count = ATOMIC_INIT(0);
35static dsp_state_cb cb_ptr;
36
37#define MAX_LEN 64
38#define HDR_LEN 20
39#define NUM_DSP_RAM_BANKS 3
40
41static char l_buf[MAX_LEN];
42#ifdef CONFIG_DEBUG_FS
43static struct dentry *dsp_dentry;
44#endif
45
46void q5audio_dsp_not_responding(void)
47{
48 if (cb_ptr)
49 cb_ptr(DSP_STATE_CRASHED);
50
51 MM_DBG("entered q5audio_dsp_not_responding\n");
52 if (atomic_add_return(1, &dsp_crash_count) != 1) {
53 MM_ERR("q5audio_dsp_not_responding() \
54 - parking additional crasher...\n");
55 for (;;)
56 msleep(1000);
57 }
58 if (dsp_wait_count) {
59 dsp_has_crashed = 1;
60 wake_up(&dsp_wait);
61
62 while (dsp_has_crashed != 2)
63 wait_event(dsp_wait, dsp_has_crashed == 2);
64 } else {
65 MM_ERR("q5audio_dsp_not_responding() - no waiter?\n");
66 }
67
68 if (cb_ptr)
69 cb_ptr(DSP_STATE_CRASH_DUMP_DONE);
70}
71
72static int dsp_open(struct inode *inode, struct file *file)
73{
74 return 0;
75}
76
77static ssize_t dsp_write(struct file *file, const char __user *buf,
78 size_t count, loff_t *pos)
79{
80 char cmd[32];
81
82 if (count >= sizeof(cmd))
83 return -EINVAL;
84 if (copy_from_user(cmd, buf, count))
85 return -EFAULT;
86 cmd[count] = 0;
87
88 if ((count > 1) && (cmd[count-1] == '\n'))
89 cmd[count-1] = 0;
90
91 if (!strncmp(cmd, "wait-for-crash", sizeof("wait-for-crash"))) {
92 while (!dsp_has_crashed) {
93 int res;
94 dsp_wait_count++;
95 res = wait_event_interruptible(dsp_wait,
96 dsp_has_crashed);
97 if (res < 0) {
98 dsp_wait_count--;
99 return res;
100 }
101 }
102 } else if (!strncmp(cmd, "boom", sizeof("boom"))) {
103 q5audio_dsp_not_responding();
104 } else if (!strncmp(cmd, "continue-crash", sizeof("continue-crash"))) {
105 dsp_has_crashed = 2;
106 wake_up(&dsp_wait);
107 } else {
108 MM_ERR("[%s:%s] unknown dsp_debug command: %s\n", __MM_FILE__,
109 __func__, cmd);
110 }
111
112 return count;
113}
114
115static ssize_t dsp_read(struct file *file, char __user *buf,
116 size_t count, loff_t *pos)
117{
118 size_t actual = 0;
119 static void *dsp_addr;
120 static unsigned copy_ok_count;
121
122 MM_INFO("pos = %lld\n", *pos);
123 if (*pos >= DSP_RAM_SIZE * NUM_DSP_RAM_BANKS)
124 return 0;
125
126 if (*pos == 0)
127 dsp_addr = (*pos + RAMA_BASE);
128 else if (*pos == DSP_RAM_SIZE)
129 dsp_addr = RAMB_BASE;
130 else if (*pos >= DSP_RAM_SIZE * 2)
131 dsp_addr = RAMC_BASE;
132
133 MM_INFO("dsp_addr = %p\n", dsp_addr);
134 while (count >= PAGE_SIZE) {
135 if (copy_to_user(buf, dsp_addr, PAGE_SIZE)) {
136 MM_ERR("[%s:%s] copy error @ %p\n", __MM_FILE__,
137 __func__, buf);
138 return -EFAULT;
139 }
140 copy_ok_count += PAGE_SIZE;
141 dsp_addr = (char *)dsp_addr + PAGE_SIZE;
142 buf += PAGE_SIZE;
143 actual += PAGE_SIZE;
144 count -= PAGE_SIZE;
145 }
146
147 *pos += actual;
148 return actual;
149}
150
151static int dsp_release(struct inode *inode, struct file *file)
152{
153 return 0;
154}
155
156int dsp_debug_register(dsp_state_cb ptr)
157{
158 if (ptr == NULL)
159 return -EINVAL;
160
161 cb_ptr = ptr;
162
163 return 0;
164}
165
166static const struct file_operations dsp_fops = {
167 .owner = THIS_MODULE,
168 .open = dsp_open,
169 .read = dsp_read,
170 .write = dsp_write,
171 .release = dsp_release,
172};
173
174#ifdef CONFIG_DEBUG_FS
175static struct miscdevice dsp_misc = {
176 .minor = MISC_DYNAMIC_MINOR,
177 .name = "dsp_debug",
178 .fops = &dsp_fops,
179};
180#endif
181
182static ssize_t dsp_debug_open(struct inode *inode, struct file *file)
183{
184 file->private_data = inode->i_private;
185 MM_DBG("adsp debugfs opened\n");
186 return 0;
187}
188
189static ssize_t dsp_debug_write(struct file *file, const char __user *buf,
190 size_t count, loff_t *ppos)
191{
192 int len;
193
194 if (count < 0)
195 return 0;
196 len = count > (MAX_LEN - 1) ? (MAX_LEN - 1) : count;
197 if (copy_from_user(l_buf + HDR_LEN, buf, len)) {
198 MM_ERR("Unable to copy data from user space\n");
199 return -EFAULT;
200 }
201 l_buf[len + HDR_LEN] = 0;
202 if (l_buf[len + HDR_LEN - 1] == '\n') {
203 l_buf[len + HDR_LEN - 1] = 0;
204 len--;
205 }
206 if (!strncmp(l_buf + HDR_LEN, "boom", 64)) {
207 q5audio_dsp_not_responding();
208 } else if (!strncmp(l_buf + HDR_LEN, "continue-crash",
209 sizeof("continue-crash"))) {
210 dsp_has_crashed = 2;
211 wake_up(&dsp_wait);
212 } else
213 MM_ERR("Unknown command\n");
214
215 return count;
216}
217static const struct file_operations dsp_debug_fops = {
218 .write = dsp_debug_write,
219 .open = dsp_debug_open,
220};
221
222static int __init dsp_init(void)
223{
224 init_waitqueue_head(&dsp_wait);
225#ifdef CONFIG_DEBUG_FS
226 dsp_dentry = debugfs_create_file("dsp_debug", S_IFREG | S_IRUGO,
227 NULL, (void *) NULL, &dsp_debug_fops);
228
229 return misc_register(&dsp_misc);
230#else
231 return 0;
232#endif /* CONFIG_DEBUG_FS */
233}
234
235device_initcall(dsp_init);