• Home
  • History
  • Annotate
  • Line#
  • Navigate
  • Raw
  • Download
  • only in /asuswrt-rt-n18u-9.0.0.4.380.2695/release/src-rt-6.x.4708/linux/linux-2.6.36/drivers/staging/dream/qdsp5/
1/* arch/arm/mach-msm/qdsp5/snd.c
2 *
3 * interface to "snd" service on the baseband cpu
4 *
5 * Copyright (C) 2008 HTC Corporation
6 *
7 * This software is licensed under the terms of the GNU General Public
8 * License version 2, as published by the Free Software Foundation, and
9 * may be copied, distributed, and modified under those terms.
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
18#include <linux/module.h>
19#include <linux/fs.h>
20#include <linux/miscdevice.h>
21#include <linux/uaccess.h>
22#include <linux/kthread.h>
23#include <linux/delay.h>
24#include <linux/msm_audio.h>
25
26#include <asm/atomic.h>
27#include <asm/ioctls.h>
28#include <mach/board.h>
29#include <mach/msm_rpcrouter.h>
30
31struct snd_ctxt {
32	struct mutex lock;
33	int opened;
34	struct msm_rpc_endpoint *ept;
35	struct msm_snd_endpoints *snd_epts;
36};
37
38static struct snd_ctxt the_snd;
39
40#define RPC_SND_PROG    0x30000002
41#define RPC_SND_CB_PROG 0x31000002
42#if CONFIG_MSM_AMSS_VERSION == 6210
43#define RPC_SND_VERS	0x94756085 /* 2490720389 */
44#elif (CONFIG_MSM_AMSS_VERSION == 6220) || \
45      (CONFIG_MSM_AMSS_VERSION == 6225)
46#define RPC_SND_VERS	0xaa2b1a44 /* 2854951492 */
47#elif CONFIG_MSM_AMSS_VERSION == 6350
48#define RPC_SND_VERS 	MSM_RPC_VERS(1,0)
49#endif
50
51#define SND_SET_DEVICE_PROC 2
52#define SND_SET_VOLUME_PROC 3
53
54struct rpc_snd_set_device_args {
55	uint32_t device;
56	uint32_t ear_mute;
57	uint32_t mic_mute;
58
59	uint32_t cb_func;
60	uint32_t client_data;
61};
62
63struct rpc_snd_set_volume_args {
64	uint32_t device;
65	uint32_t method;
66	uint32_t volume;
67
68	uint32_t cb_func;
69	uint32_t client_data;
70};
71
72struct snd_set_device_msg {
73	struct rpc_request_hdr hdr;
74	struct rpc_snd_set_device_args args;
75};
76
77struct snd_set_volume_msg {
78	struct rpc_request_hdr hdr;
79	struct rpc_snd_set_volume_args args;
80};
81
82struct snd_endpoint *get_snd_endpoints(int *size);
83
84static inline int check_mute(int mute)
85{
86	return (mute == SND_MUTE_MUTED ||
87		mute == SND_MUTE_UNMUTED) ? 0 : -EINVAL;
88}
89
90static int get_endpoint(struct snd_ctxt *snd, unsigned long arg)
91{
92	int rc = 0, index;
93	struct msm_snd_endpoint ept;
94
95	if (copy_from_user(&ept, (void __user *)arg, sizeof(ept))) {
96		pr_err("snd_ioctl get endpoint: invalid read pointer.\n");
97		return -EFAULT;
98	}
99
100	index = ept.id;
101	if (index < 0 || index >= snd->snd_epts->num) {
102		pr_err("snd_ioctl get endpoint: invalid index!\n");
103		return -EINVAL;
104	}
105
106	ept.id = snd->snd_epts->endpoints[index].id;
107	strncpy(ept.name,
108		snd->snd_epts->endpoints[index].name,
109		sizeof(ept.name));
110
111	if (copy_to_user((void __user *)arg, &ept, sizeof(ept))) {
112		pr_err("snd_ioctl get endpoint: invalid write pointer.\n");
113		rc = -EFAULT;
114	}
115
116	return rc;
117}
118
119static long snd_ioctl(struct file *file, unsigned int cmd, unsigned long arg)
120{
121	struct snd_set_device_msg dmsg;
122	struct snd_set_volume_msg vmsg;
123	struct msm_snd_device_config dev;
124	struct msm_snd_volume_config vol;
125	struct snd_ctxt *snd = file->private_data;
126	int rc = 0;
127
128	mutex_lock(&snd->lock);
129	switch (cmd) {
130	case SND_SET_DEVICE:
131		if (copy_from_user(&dev, (void __user *) arg, sizeof(dev))) {
132			pr_err("snd_ioctl set device: invalid pointer.\n");
133			rc = -EFAULT;
134			break;
135		}
136
137		dmsg.args.device = cpu_to_be32(dev.device);
138		dmsg.args.ear_mute = cpu_to_be32(dev.ear_mute);
139		dmsg.args.mic_mute = cpu_to_be32(dev.mic_mute);
140		if (check_mute(dev.ear_mute) < 0 ||
141				check_mute(dev.mic_mute) < 0) {
142			pr_err("snd_ioctl set device: invalid mute status.\n");
143			rc = -EINVAL;
144			break;
145		}
146		dmsg.args.cb_func = -1;
147		dmsg.args.client_data = 0;
148
149		pr_info("snd_set_device %d %d %d\n", dev.device,
150						 dev.ear_mute, dev.mic_mute);
151
152		rc = msm_rpc_call(snd->ept,
153			SND_SET_DEVICE_PROC,
154			&dmsg, sizeof(dmsg), 5 * HZ);
155		break;
156
157	case SND_SET_VOLUME:
158		if (copy_from_user(&vol, (void __user *) arg, sizeof(vol))) {
159			pr_err("snd_ioctl set volume: invalid pointer.\n");
160			rc = -EFAULT;
161			break;
162		}
163
164		vmsg.args.device = cpu_to_be32(vol.device);
165		vmsg.args.method = cpu_to_be32(vol.method);
166		if (vol.method != SND_METHOD_VOICE) {
167			pr_err("snd_ioctl set volume: invalid method.\n");
168			rc = -EINVAL;
169			break;
170		}
171
172		vmsg.args.volume = cpu_to_be32(vol.volume);
173		vmsg.args.cb_func = -1;
174		vmsg.args.client_data = 0;
175
176		pr_info("snd_set_volume %d %d %d\n", vol.device,
177						vol.method, vol.volume);
178
179		rc = msm_rpc_call(snd->ept,
180			SND_SET_VOLUME_PROC,
181			&vmsg, sizeof(vmsg), 5 * HZ);
182		break;
183
184	case SND_GET_NUM_ENDPOINTS:
185		if (copy_to_user((void __user *)arg,
186				&snd->snd_epts->num, sizeof(unsigned))) {
187			pr_err("snd_ioctl get endpoint: invalid pointer.\n");
188			rc = -EFAULT;
189		}
190		break;
191
192	case SND_GET_ENDPOINT:
193		rc = get_endpoint(snd, arg);
194		break;
195
196	default:
197		pr_err("snd_ioctl unknown command.\n");
198		rc = -EINVAL;
199		break;
200	}
201	mutex_unlock(&snd->lock);
202
203	return rc;
204}
205
206static int snd_release(struct inode *inode, struct file *file)
207{
208	struct snd_ctxt *snd = file->private_data;
209
210	mutex_lock(&snd->lock);
211	snd->opened = 0;
212	mutex_unlock(&snd->lock);
213	return 0;
214}
215
216static int snd_open(struct inode *inode, struct file *file)
217{
218	struct snd_ctxt *snd = &the_snd;
219	int rc = 0;
220
221	mutex_lock(&snd->lock);
222	if (snd->opened == 0) {
223		if (snd->ept == NULL) {
224			snd->ept = msm_rpc_connect(RPC_SND_PROG, RPC_SND_VERS,
225					MSM_RPC_UNINTERRUPTIBLE);
226			if (IS_ERR(snd->ept)) {
227				rc = PTR_ERR(snd->ept);
228				snd->ept = NULL;
229				pr_err("snd: failed to connect snd svc\n");
230				goto err;
231			}
232		}
233		file->private_data = snd;
234		snd->opened = 1;
235	} else {
236		pr_err("snd already opened.\n");
237		rc = -EBUSY;
238	}
239
240err:
241	mutex_unlock(&snd->lock);
242	return rc;
243}
244
245static struct file_operations snd_fops = {
246	.owner		= THIS_MODULE,
247	.open		= snd_open,
248	.release	= snd_release,
249	.unlocked_ioctl	= snd_ioctl,
250};
251
252struct miscdevice snd_misc = {
253	.minor	= MISC_DYNAMIC_MINOR,
254	.name	= "msm_snd",
255	.fops	= &snd_fops,
256};
257
258static int snd_probe(struct platform_device *pdev)
259{
260	struct snd_ctxt *snd = &the_snd;
261	mutex_init(&snd->lock);
262	snd->snd_epts = (struct msm_snd_endpoints *)pdev->dev.platform_data;
263	return misc_register(&snd_misc);
264}
265
266static struct platform_driver snd_plat_driver = {
267	.probe = snd_probe,
268	.driver = {
269		.name = "msm_snd",
270		.owner = THIS_MODULE,
271	},
272};
273
274static int __init snd_init(void)
275{
276	return platform_driver_register(&snd_plat_driver);
277}
278
279module_init(snd_init);
280