| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | /*
 | 
					
						
							|  |  |  |  *  compress_core.c - compress offload core | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  Copyright (C) 2011 Intel Corporation | 
					
						
							|  |  |  |  *  Authors:	Vinod Koul <vinod.koul@linux.intel.com> | 
					
						
							|  |  |  |  *		Pierre-Louis Bossart <pierre-louis.bossart@linux.intel.com> | 
					
						
							|  |  |  |  *  ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  This program is free software; you can redistribute it and/or modify | 
					
						
							|  |  |  |  *  it under the terms of the GNU General Public License as published by | 
					
						
							|  |  |  |  *  the Free Software Foundation; version 2 of the License. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  This program is distributed in the hope that it will be useful, but | 
					
						
							|  |  |  |  *  WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
					
						
							|  |  |  |  *  General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  You should have received a copy of the GNU General Public License along | 
					
						
							|  |  |  |  *  with this program; if not, write to the Free Software Foundation, Inc., | 
					
						
							|  |  |  |  *  59 Temple Place, Suite 330, Boston, MA 02111-1307 USA. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #define FORMAT(fmt) "%s: %d: " fmt, __func__, __LINE__
 | 
					
						
							|  |  |  | #define pr_fmt(fmt) KBUILD_MODNAME ": " FORMAT(fmt)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <linux/file.h>
 | 
					
						
							|  |  |  | #include <linux/fs.h>
 | 
					
						
							|  |  |  | #include <linux/list.h>
 | 
					
						
							| 
									
										
										
										
											2013-04-18 11:03:46 +01:00
										 |  |  | #include <linux/math64.h>
 | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | #include <linux/mm.h>
 | 
					
						
							|  |  |  | #include <linux/mutex.h>
 | 
					
						
							|  |  |  | #include <linux/poll.h>
 | 
					
						
							|  |  |  | #include <linux/slab.h>
 | 
					
						
							|  |  |  | #include <linux/sched.h>
 | 
					
						
							| 
									
										
										
										
											2013-04-18 11:03:46 +01:00
										 |  |  | #include <linux/types.h>
 | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | #include <linux/uio.h>
 | 
					
						
							|  |  |  | #include <linux/uaccess.h>
 | 
					
						
							|  |  |  | #include <linux/module.h>
 | 
					
						
							|  |  |  | #include <sound/core.h>
 | 
					
						
							|  |  |  | #include <sound/initval.h>
 | 
					
						
							|  |  |  | #include <sound/compress_params.h>
 | 
					
						
							|  |  |  | #include <sound/compress_offload.h>
 | 
					
						
							|  |  |  | #include <sound/compress_driver.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* TODO:
 | 
					
						
							|  |  |  |  * - add substream support for multiple devices in case of | 
					
						
							|  |  |  |  *	SND_DYNAMIC_MINORS is not used | 
					
						
							|  |  |  |  * - Multiple node representation | 
					
						
							|  |  |  |  *	driver should be able to register multiple nodes | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static DEFINE_MUTEX(device_mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct snd_compr_file { | 
					
						
							|  |  |  | 	unsigned long caps; | 
					
						
							|  |  |  | 	struct snd_compr_stream stream; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * a note on stream states used: | 
					
						
							|  |  |  |  * we use follwing states in the compressed core | 
					
						
							|  |  |  |  * SNDRV_PCM_STATE_OPEN: When stream has been opened. | 
					
						
							|  |  |  |  * SNDRV_PCM_STATE_SETUP: When stream has been initialized. This is done by | 
					
						
							|  |  |  |  *	calling SNDRV_COMPRESS_SET_PARAMS. running streams will come to this | 
					
						
							|  |  |  |  *	state at stop by calling SNDRV_COMPRESS_STOP, or at end of drain. | 
					
						
							|  |  |  |  * SNDRV_PCM_STATE_RUNNING: When stream has been started and is | 
					
						
							|  |  |  |  *	decoding/encoding and rendering/capturing data. | 
					
						
							|  |  |  |  * SNDRV_PCM_STATE_DRAINING: When stream is draining current data. This is done | 
					
						
							|  |  |  |  *	by calling SNDRV_COMPRESS_DRAIN. | 
					
						
							|  |  |  |  * SNDRV_PCM_STATE_PAUSED: When stream is paused. This is done by calling | 
					
						
							|  |  |  |  *	SNDRV_COMPRESS_PAUSE. It can be stopped or resumed by calling | 
					
						
							|  |  |  |  *	SNDRV_COMPRESS_STOP or SNDRV_COMPRESS_RESUME respectively. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static int snd_compr_open(struct inode *inode, struct file *f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct snd_compr *compr; | 
					
						
							|  |  |  | 	struct snd_compr_file *data; | 
					
						
							|  |  |  | 	struct snd_compr_runtime *runtime; | 
					
						
							|  |  |  | 	enum snd_compr_direction dirn; | 
					
						
							|  |  |  | 	int maj = imajor(inode); | 
					
						
							|  |  |  | 	int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-11 14:12:43 +03:00
										 |  |  | 	if ((f->f_flags & O_ACCMODE) == O_WRONLY) | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 		dirn = SND_COMPRESS_PLAYBACK; | 
					
						
							| 
									
										
										
										
											2012-09-11 14:12:43 +03:00
										 |  |  | 	else if ((f->f_flags & O_ACCMODE) == O_RDONLY) | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 		dirn = SND_COMPRESS_CAPTURE; | 
					
						
							| 
									
										
										
										
											2012-09-11 14:12:43 +03:00
										 |  |  | 	else | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (maj == snd_major) | 
					
						
							|  |  |  | 		compr = snd_lookup_minor_data(iminor(inode), | 
					
						
							|  |  |  | 					SNDRV_DEVICE_TYPE_COMPRESS); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		return -EBADFD; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (compr == NULL) { | 
					
						
							|  |  |  | 		pr_err("no device data!!!\n"); | 
					
						
							|  |  |  | 		return -ENODEV; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (dirn != compr->direction) { | 
					
						
							|  |  |  | 		pr_err("this device doesn't support this direction\n"); | 
					
						
							| 
									
										
										
										
											2012-10-16 13:05:59 +02:00
										 |  |  | 		snd_card_unref(compr->card); | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	data = kzalloc(sizeof(*data), GFP_KERNEL); | 
					
						
							| 
									
										
										
										
											2012-10-16 13:05:59 +02:00
										 |  |  | 	if (!data) { | 
					
						
							|  |  |  | 		snd_card_unref(compr->card); | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 		return -ENOMEM; | 
					
						
							| 
									
										
										
										
											2012-10-16 13:05:59 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 	data->stream.ops = compr->ops; | 
					
						
							|  |  |  | 	data->stream.direction = dirn; | 
					
						
							|  |  |  | 	data->stream.private_data = compr->private_data; | 
					
						
							|  |  |  | 	data->stream.device = compr; | 
					
						
							|  |  |  | 	runtime = kzalloc(sizeof(*runtime), GFP_KERNEL); | 
					
						
							|  |  |  | 	if (!runtime) { | 
					
						
							|  |  |  | 		kfree(data); | 
					
						
							| 
									
										
										
										
											2012-10-16 13:05:59 +02:00
										 |  |  | 		snd_card_unref(compr->card); | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 		return -ENOMEM; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	runtime->state = SNDRV_PCM_STATE_OPEN; | 
					
						
							|  |  |  | 	init_waitqueue_head(&runtime->sleep); | 
					
						
							|  |  |  | 	data->stream.runtime = runtime; | 
					
						
							|  |  |  | 	f->private_data = (void *)data; | 
					
						
							|  |  |  | 	mutex_lock(&compr->lock); | 
					
						
							|  |  |  | 	ret = compr->ops->open(&data->stream); | 
					
						
							|  |  |  | 	mutex_unlock(&compr->lock); | 
					
						
							|  |  |  | 	if (ret) { | 
					
						
							|  |  |  | 		kfree(runtime); | 
					
						
							|  |  |  | 		kfree(data); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-10-16 13:05:59 +02:00
										 |  |  | 	snd_card_unref(compr->card); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int snd_compr_free(struct inode *inode, struct file *f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct snd_compr_file *data = f->private_data; | 
					
						
							| 
									
										
										
										
											2013-09-13 17:43:16 +01:00
										 |  |  | 	struct snd_compr_runtime *runtime = data->stream.runtime; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch (runtime->state) { | 
					
						
							|  |  |  | 	case SNDRV_PCM_STATE_RUNNING: | 
					
						
							|  |  |  | 	case SNDRV_PCM_STATE_DRAINING: | 
					
						
							|  |  |  | 	case SNDRV_PCM_STATE_PAUSED: | 
					
						
							|  |  |  | 		data->stream.ops->trigger(&data->stream, SNDRV_PCM_TRIGGER_STOP); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 	data->stream.ops->free(&data->stream); | 
					
						
							|  |  |  | 	kfree(data->stream.runtime->buffer); | 
					
						
							|  |  |  | 	kfree(data->stream.runtime); | 
					
						
							|  |  |  | 	kfree(data); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-11 13:44:53 +00:00
										 |  |  | static int snd_compr_update_tstamp(struct snd_compr_stream *stream, | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 		struct snd_compr_tstamp *tstamp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (!stream->ops->pointer) | 
					
						
							| 
									
										
										
										
											2013-02-11 13:44:53 +00:00
										 |  |  | 		return -ENOTSUPP; | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 	stream->ops->pointer(stream, tstamp); | 
					
						
							|  |  |  | 	pr_debug("dsp consumed till %d total %d bytes\n", | 
					
						
							|  |  |  | 		tstamp->byte_offset, tstamp->copied_total); | 
					
						
							| 
									
										
										
										
											2013-04-18 11:01:03 +01:00
										 |  |  | 	if (stream->direction == SND_COMPRESS_PLAYBACK) | 
					
						
							|  |  |  | 		stream->runtime->total_bytes_transferred = tstamp->copied_total; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		stream->runtime->total_bytes_available = tstamp->copied_total; | 
					
						
							| 
									
										
										
										
											2013-02-11 13:44:53 +00:00
										 |  |  | 	return 0; | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static size_t snd_compr_calc_avail(struct snd_compr_stream *stream, | 
					
						
							|  |  |  | 		struct snd_compr_avail *avail) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-02-11 13:44:53 +00:00
										 |  |  | 	memset(avail, 0, sizeof(*avail)); | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 	snd_compr_update_tstamp(stream, &avail->tstamp); | 
					
						
							| 
									
										
										
										
											2013-02-11 13:44:53 +00:00
										 |  |  | 	/* Still need to return avail even if tstamp can't be filled in */ | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (stream->runtime->total_bytes_available == 0 && | 
					
						
							| 
									
										
										
										
											2013-04-18 11:01:03 +01:00
										 |  |  | 			stream->runtime->state == SNDRV_PCM_STATE_SETUP && | 
					
						
							|  |  |  | 			stream->direction == SND_COMPRESS_PLAYBACK) { | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 		pr_debug("detected init and someone forgot to do a write\n"); | 
					
						
							|  |  |  | 		return stream->runtime->buffer_size; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	pr_debug("app wrote %lld, DSP consumed %lld\n", | 
					
						
							|  |  |  | 			stream->runtime->total_bytes_available, | 
					
						
							|  |  |  | 			stream->runtime->total_bytes_transferred); | 
					
						
							|  |  |  | 	if (stream->runtime->total_bytes_available == | 
					
						
							|  |  |  | 				stream->runtime->total_bytes_transferred) { | 
					
						
							| 
									
										
										
										
											2013-04-18 11:01:03 +01:00
										 |  |  | 		if (stream->direction == SND_COMPRESS_PLAYBACK) { | 
					
						
							|  |  |  | 			pr_debug("both pointers are same, returning full avail\n"); | 
					
						
							|  |  |  | 			return stream->runtime->buffer_size; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			pr_debug("both pointers are same, returning no avail\n"); | 
					
						
							|  |  |  | 			return 0; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-18 11:01:03 +01:00
										 |  |  | 	avail->avail = stream->runtime->total_bytes_available - | 
					
						
							|  |  |  | 			stream->runtime->total_bytes_transferred; | 
					
						
							|  |  |  | 	if (stream->direction == SND_COMPRESS_PLAYBACK) | 
					
						
							|  |  |  | 		avail->avail = stream->runtime->buffer_size - avail->avail; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-18 10:59:23 +01:00
										 |  |  | 	pr_debug("ret avail as %lld\n", avail->avail); | 
					
						
							|  |  |  | 	return avail->avail; | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline size_t snd_compr_get_avail(struct snd_compr_stream *stream) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct snd_compr_avail avail; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return snd_compr_calc_avail(stream, &avail); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | snd_compr_ioctl_avail(struct snd_compr_stream *stream, unsigned long arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct snd_compr_avail ioctl_avail; | 
					
						
							|  |  |  | 	size_t avail; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	avail = snd_compr_calc_avail(stream, &ioctl_avail); | 
					
						
							|  |  |  | 	ioctl_avail.avail = avail; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (copy_to_user((__u64 __user *)arg, | 
					
						
							|  |  |  | 				&ioctl_avail, sizeof(ioctl_avail))) | 
					
						
							|  |  |  | 		return -EFAULT; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int snd_compr_write_data(struct snd_compr_stream *stream, | 
					
						
							|  |  |  | 	       const char __user *buf, size_t count) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	void *dstn; | 
					
						
							|  |  |  | 	size_t copy; | 
					
						
							|  |  |  | 	struct snd_compr_runtime *runtime = stream->runtime; | 
					
						
							| 
									
										
										
										
											2013-04-18 11:03:46 +01:00
										 |  |  | 	/* 64-bit Modulus */ | 
					
						
							|  |  |  | 	u64 app_pointer = div64_u64(runtime->total_bytes_available, | 
					
						
							|  |  |  | 				    runtime->buffer_size); | 
					
						
							|  |  |  | 	app_pointer = runtime->total_bytes_available - | 
					
						
							|  |  |  | 		      (app_pointer * runtime->buffer_size); | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-18 11:03:46 +01:00
										 |  |  | 	dstn = runtime->buffer + app_pointer; | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 	pr_debug("copying %ld at %lld\n", | 
					
						
							| 
									
										
										
										
											2013-04-18 11:03:46 +01:00
										 |  |  | 			(unsigned long)count, app_pointer); | 
					
						
							|  |  |  | 	if (count < runtime->buffer_size - app_pointer) { | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 		if (copy_from_user(dstn, buf, count)) | 
					
						
							|  |  |  | 			return -EFAULT; | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2013-04-18 11:03:46 +01:00
										 |  |  | 		copy = runtime->buffer_size - app_pointer; | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 		if (copy_from_user(dstn, buf, copy)) | 
					
						
							|  |  |  | 			return -EFAULT; | 
					
						
							|  |  |  | 		if (copy_from_user(runtime->buffer, buf + copy, count - copy)) | 
					
						
							|  |  |  | 			return -EFAULT; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* if DSP cares, let it know data has been written */ | 
					
						
							|  |  |  | 	if (stream->ops->ack) | 
					
						
							|  |  |  | 		stream->ops->ack(stream, count); | 
					
						
							|  |  |  | 	return count; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static ssize_t snd_compr_write(struct file *f, const char __user *buf, | 
					
						
							|  |  |  | 		size_t count, loff_t *offset) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct snd_compr_file *data = f->private_data; | 
					
						
							|  |  |  | 	struct snd_compr_stream *stream; | 
					
						
							|  |  |  | 	size_t avail; | 
					
						
							|  |  |  | 	int retval; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (snd_BUG_ON(!data)) | 
					
						
							|  |  |  | 		return -EFAULT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	stream = &data->stream; | 
					
						
							|  |  |  | 	mutex_lock(&stream->device->lock); | 
					
						
							|  |  |  | 	/* write is allowed when stream is running or has been steup */ | 
					
						
							|  |  |  | 	if (stream->runtime->state != SNDRV_PCM_STATE_SETUP && | 
					
						
							|  |  |  | 			stream->runtime->state != SNDRV_PCM_STATE_RUNNING) { | 
					
						
							|  |  |  | 		mutex_unlock(&stream->device->lock); | 
					
						
							|  |  |  | 		return -EBADFD; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	avail = snd_compr_get_avail(stream); | 
					
						
							|  |  |  | 	pr_debug("avail returned %ld\n", (unsigned long)avail); | 
					
						
							|  |  |  | 	/* calculate how much we can write to buffer */ | 
					
						
							|  |  |  | 	if (avail > count) | 
					
						
							|  |  |  | 		avail = count; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-18 11:01:38 +01:00
										 |  |  | 	if (stream->ops->copy) { | 
					
						
							|  |  |  | 		char __user* cbuf = (char __user*)buf; | 
					
						
							|  |  |  | 		retval = stream->ops->copy(stream, cbuf, avail); | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 		retval = snd_compr_write_data(stream, buf, avail); | 
					
						
							| 
									
										
										
										
											2013-04-18 11:01:38 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 	if (retval > 0) | 
					
						
							|  |  |  | 		stream->runtime->total_bytes_available += retval; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* while initiating the stream, write should be called before START
 | 
					
						
							|  |  |  | 	 * call, so in setup move state */ | 
					
						
							|  |  |  | 	if (stream->runtime->state == SNDRV_PCM_STATE_SETUP) { | 
					
						
							|  |  |  | 		stream->runtime->state = SNDRV_PCM_STATE_PREPARED; | 
					
						
							|  |  |  | 		pr_debug("stream prepared, Houston we are good to go\n"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mutex_unlock(&stream->device->lock); | 
					
						
							|  |  |  | 	return retval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static ssize_t snd_compr_read(struct file *f, char __user *buf, | 
					
						
							|  |  |  | 		size_t count, loff_t *offset) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-04-18 11:02:08 +01:00
										 |  |  | 	struct snd_compr_file *data = f->private_data; | 
					
						
							|  |  |  | 	struct snd_compr_stream *stream; | 
					
						
							|  |  |  | 	size_t avail; | 
					
						
							|  |  |  | 	int retval; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (snd_BUG_ON(!data)) | 
					
						
							|  |  |  | 		return -EFAULT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	stream = &data->stream; | 
					
						
							|  |  |  | 	mutex_lock(&stream->device->lock); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-29 14:25:23 +05:30
										 |  |  | 	/* read is allowed when stream is running, paused, draining and setup
 | 
					
						
							|  |  |  | 	 * (yes setup is state which we transition to after stop, so if user | 
					
						
							|  |  |  | 	 * wants to read data after stop we allow that) | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	switch (stream->runtime->state) { | 
					
						
							|  |  |  | 	case SNDRV_PCM_STATE_OPEN: | 
					
						
							|  |  |  | 	case SNDRV_PCM_STATE_PREPARED: | 
					
						
							|  |  |  | 	case SNDRV_PCM_STATE_XRUN: | 
					
						
							|  |  |  | 	case SNDRV_PCM_STATE_SUSPENDED: | 
					
						
							|  |  |  | 	case SNDRV_PCM_STATE_DISCONNECTED: | 
					
						
							| 
									
										
										
										
											2013-04-18 11:02:08 +01:00
										 |  |  | 		retval = -EBADFD; | 
					
						
							|  |  |  | 		goto out; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	avail = snd_compr_get_avail(stream); | 
					
						
							|  |  |  | 	pr_debug("avail returned %ld\n", (unsigned long)avail); | 
					
						
							|  |  |  | 	/* calculate how much we can read from buffer */ | 
					
						
							|  |  |  | 	if (avail > count) | 
					
						
							|  |  |  | 		avail = count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (stream->ops->copy) { | 
					
						
							|  |  |  | 		retval = stream->ops->copy(stream, buf, avail); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		retval = -ENXIO; | 
					
						
							|  |  |  | 		goto out; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (retval > 0) | 
					
						
							|  |  |  | 		stream->runtime->total_bytes_transferred += retval; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  | 	mutex_unlock(&stream->device->lock); | 
					
						
							|  |  |  | 	return retval; | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int snd_compr_mmap(struct file *f, struct vm_area_struct *vma) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return -ENXIO; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline int snd_compr_get_poll(struct snd_compr_stream *stream) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (stream->direction == SND_COMPRESS_PLAYBACK) | 
					
						
							|  |  |  | 		return POLLOUT | POLLWRNORM; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		return POLLIN | POLLRDNORM; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static unsigned int snd_compr_poll(struct file *f, poll_table *wait) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct snd_compr_file *data = f->private_data; | 
					
						
							|  |  |  | 	struct snd_compr_stream *stream; | 
					
						
							|  |  |  | 	size_t avail; | 
					
						
							|  |  |  | 	int retval = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (snd_BUG_ON(!data)) | 
					
						
							|  |  |  | 		return -EFAULT; | 
					
						
							|  |  |  | 	stream = &data->stream; | 
					
						
							|  |  |  | 	if (snd_BUG_ON(!stream)) | 
					
						
							|  |  |  | 		return -EFAULT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mutex_lock(&stream->device->lock); | 
					
						
							| 
									
										
										
										
											2013-10-22 11:26:48 +01:00
										 |  |  | 	if (stream->runtime->state == SNDRV_PCM_STATE_OPEN) { | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 		retval = -EBADFD; | 
					
						
							|  |  |  | 		goto out; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	poll_wait(f, &stream->runtime->sleep, wait); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	avail = snd_compr_get_avail(stream); | 
					
						
							|  |  |  | 	pr_debug("avail is %ld\n", (unsigned long)avail); | 
					
						
							|  |  |  | 	/* check if we have at least one fragment to fill */ | 
					
						
							|  |  |  | 	switch (stream->runtime->state) { | 
					
						
							|  |  |  | 	case SNDRV_PCM_STATE_DRAINING: | 
					
						
							|  |  |  | 		/* stream has been woken up after drain is complete
 | 
					
						
							|  |  |  | 		 * draining done so set stream state to stopped | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		retval = snd_compr_get_poll(stream); | 
					
						
							|  |  |  | 		stream->runtime->state = SNDRV_PCM_STATE_SETUP; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case SNDRV_PCM_STATE_RUNNING: | 
					
						
							|  |  |  | 	case SNDRV_PCM_STATE_PREPARED: | 
					
						
							|  |  |  | 	case SNDRV_PCM_STATE_PAUSED: | 
					
						
							|  |  |  | 		if (avail >= stream->runtime->fragment_size) | 
					
						
							|  |  |  | 			retval = snd_compr_get_poll(stream); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		if (stream->direction == SND_COMPRESS_PLAYBACK) | 
					
						
							|  |  |  | 			retval = POLLOUT | POLLWRNORM | POLLERR; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			retval = POLLIN | POLLRDNORM | POLLERR; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  | 	mutex_unlock(&stream->device->lock); | 
					
						
							|  |  |  | 	return retval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | snd_compr_get_caps(struct snd_compr_stream *stream, unsigned long arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int retval; | 
					
						
							|  |  |  | 	struct snd_compr_caps caps; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!stream->ops->get_caps) | 
					
						
							|  |  |  | 		return -ENXIO; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-21 14:07:29 +03:00
										 |  |  | 	memset(&caps, 0, sizeof(caps)); | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 	retval = stream->ops->get_caps(stream, &caps); | 
					
						
							|  |  |  | 	if (retval) | 
					
						
							|  |  |  | 		goto out; | 
					
						
							|  |  |  | 	if (copy_to_user((void __user *)arg, &caps, sizeof(caps))) | 
					
						
							|  |  |  | 		retval = -EFAULT; | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  | 	return retval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | snd_compr_get_codec_caps(struct snd_compr_stream *stream, unsigned long arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int retval; | 
					
						
							|  |  |  | 	struct snd_compr_codec_caps *caps; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!stream->ops->get_codec_caps) | 
					
						
							|  |  |  | 		return -ENXIO; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-22 10:38:26 +02:00
										 |  |  | 	caps = kzalloc(sizeof(*caps), GFP_KERNEL); | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 	if (!caps) | 
					
						
							|  |  |  | 		return -ENOMEM; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	retval = stream->ops->get_codec_caps(stream, caps); | 
					
						
							|  |  |  | 	if (retval) | 
					
						
							|  |  |  | 		goto out; | 
					
						
							|  |  |  | 	if (copy_to_user((void __user *)arg, caps, sizeof(*caps))) | 
					
						
							|  |  |  | 		retval = -EFAULT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  | 	kfree(caps); | 
					
						
							|  |  |  | 	return retval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* revisit this with snd_pcm_preallocate_xxx */ | 
					
						
							|  |  |  | static int snd_compr_allocate_buffer(struct snd_compr_stream *stream, | 
					
						
							|  |  |  | 		struct snd_compr_params *params) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned int buffer_size; | 
					
						
							|  |  |  | 	void *buffer; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	buffer_size = params->buffer.fragment_size * params->buffer.fragments; | 
					
						
							|  |  |  | 	if (stream->ops->copy) { | 
					
						
							|  |  |  | 		buffer = NULL; | 
					
						
							|  |  |  | 		/* if copy is defined the driver will be required to copy
 | 
					
						
							|  |  |  | 		 * the data from core | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		buffer = kmalloc(buffer_size, GFP_KERNEL); | 
					
						
							|  |  |  | 		if (!buffer) | 
					
						
							|  |  |  | 			return -ENOMEM; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	stream->runtime->fragment_size = params->buffer.fragment_size; | 
					
						
							|  |  |  | 	stream->runtime->fragments = params->buffer.fragments; | 
					
						
							|  |  |  | 	stream->runtime->buffer = buffer; | 
					
						
							|  |  |  | 	stream->runtime->buffer_size = buffer_size; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-17 11:51:25 +05:30
										 |  |  | static int snd_compress_check_input(struct snd_compr_params *params) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* first let's check the buffer parameter's */ | 
					
						
							|  |  |  | 	if (params->buffer.fragment_size == 0 || | 
					
						
							|  |  |  | 			params->buffer.fragments > SIZE_MAX / params->buffer.fragment_size) | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-17 11:51:26 +05:30
										 |  |  | 	/* now codec parameters */ | 
					
						
							|  |  |  | 	if (params->codec.id == 0 || params->codec.id > SND_AUDIOCODEC_MAX) | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (params->codec.ch_in == 0 || params->codec.ch_out == 0) | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-17 11:51:25 +05:30
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | static int | 
					
						
							|  |  |  | snd_compr_set_params(struct snd_compr_stream *stream, unsigned long arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct snd_compr_params *params; | 
					
						
							|  |  |  | 	int retval; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (stream->runtime->state == SNDRV_PCM_STATE_OPEN) { | 
					
						
							|  |  |  | 		/*
 | 
					
						
							|  |  |  | 		 * we should allow parameter change only when stream has been | 
					
						
							|  |  |  | 		 * opened not in other cases | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		params = kmalloc(sizeof(*params), GFP_KERNEL); | 
					
						
							|  |  |  | 		if (!params) | 
					
						
							|  |  |  | 			return -ENOMEM; | 
					
						
							| 
									
										
										
										
											2012-01-23 21:02:57 +01:00
										 |  |  | 		if (copy_from_user(params, (void __user *)arg, sizeof(*params))) { | 
					
						
							|  |  |  | 			retval = -EFAULT; | 
					
						
							|  |  |  | 			goto out; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-09-17 11:51:25 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  | 		retval = snd_compress_check_input(params); | 
					
						
							|  |  |  | 		if (retval) | 
					
						
							|  |  |  | 			goto out; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 		retval = snd_compr_allocate_buffer(stream, params); | 
					
						
							|  |  |  | 		if (retval) { | 
					
						
							| 
									
										
										
										
											2012-01-23 21:02:57 +01:00
										 |  |  | 			retval = -ENOMEM; | 
					
						
							|  |  |  | 			goto out; | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-09-17 11:51:25 +05:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 		retval = stream->ops->set_params(stream, params); | 
					
						
							|  |  |  | 		if (retval) | 
					
						
							|  |  |  | 			goto out; | 
					
						
							| 
									
										
										
										
											2013-04-18 11:02:08 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-14 16:52:51 +05:30
										 |  |  | 		stream->metadata_set = false; | 
					
						
							|  |  |  | 		stream->next_track = false; | 
					
						
							| 
									
										
										
										
											2013-04-18 11:02:08 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if (stream->direction == SND_COMPRESS_PLAYBACK) | 
					
						
							|  |  |  | 			stream->runtime->state = SNDRV_PCM_STATE_SETUP; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			stream->runtime->state = SNDRV_PCM_STATE_PREPARED; | 
					
						
							| 
									
										
										
										
											2012-01-23 21:02:57 +01:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 		return -EPERM; | 
					
						
							| 
									
										
										
										
											2012-01-23 21:02:57 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | out: | 
					
						
							|  |  |  | 	kfree(params); | 
					
						
							|  |  |  | 	return retval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | snd_compr_get_params(struct snd_compr_stream *stream, unsigned long arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct snd_codec *params; | 
					
						
							|  |  |  | 	int retval; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!stream->ops->get_params) | 
					
						
							|  |  |  | 		return -EBADFD; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-22 10:38:26 +02:00
										 |  |  | 	params = kzalloc(sizeof(*params), GFP_KERNEL); | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 	if (!params) | 
					
						
							|  |  |  | 		return -ENOMEM; | 
					
						
							|  |  |  | 	retval = stream->ops->get_params(stream, params); | 
					
						
							|  |  |  | 	if (retval) | 
					
						
							|  |  |  | 		goto out; | 
					
						
							|  |  |  | 	if (copy_to_user((char __user *)arg, params, sizeof(*params))) | 
					
						
							|  |  |  | 		retval = -EFAULT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  | 	kfree(params); | 
					
						
							|  |  |  | 	return retval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-14 16:52:51 +05:30
										 |  |  | static int | 
					
						
							|  |  |  | snd_compr_get_metadata(struct snd_compr_stream *stream, unsigned long arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct snd_compr_metadata metadata; | 
					
						
							|  |  |  | 	int retval; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!stream->ops->get_metadata) | 
					
						
							|  |  |  | 		return -ENXIO; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (copy_from_user(&metadata, (void __user *)arg, sizeof(metadata))) | 
					
						
							|  |  |  | 		return -EFAULT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	retval = stream->ops->get_metadata(stream, &metadata); | 
					
						
							|  |  |  | 	if (retval != 0) | 
					
						
							|  |  |  | 		return retval; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (copy_to_user((void __user *)arg, &metadata, sizeof(metadata))) | 
					
						
							|  |  |  | 		return -EFAULT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | snd_compr_set_metadata(struct snd_compr_stream *stream, unsigned long arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct snd_compr_metadata metadata; | 
					
						
							|  |  |  | 	int retval; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!stream->ops->set_metadata) | 
					
						
							|  |  |  | 		return -ENXIO; | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	* we should allow parameter change only when stream has been | 
					
						
							|  |  |  | 	* opened not in other cases | 
					
						
							|  |  |  | 	*/ | 
					
						
							|  |  |  | 	if (copy_from_user(&metadata, (void __user *)arg, sizeof(metadata))) | 
					
						
							|  |  |  | 		return -EFAULT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	retval = stream->ops->set_metadata(stream, &metadata); | 
					
						
							|  |  |  | 	stream->metadata_set = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return retval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | static inline int | 
					
						
							|  |  |  | snd_compr_tstamp(struct snd_compr_stream *stream, unsigned long arg) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-02-11 13:44:53 +00:00
										 |  |  | 	struct snd_compr_tstamp tstamp = {0}; | 
					
						
							|  |  |  | 	int ret; | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-11 13:44:53 +00:00
										 |  |  | 	ret = snd_compr_update_tstamp(stream, &tstamp); | 
					
						
							|  |  |  | 	if (ret == 0) | 
					
						
							|  |  |  | 		ret = copy_to_user((struct snd_compr_tstamp __user *)arg, | 
					
						
							|  |  |  | 			&tstamp, sizeof(tstamp)) ? -EFAULT : 0; | 
					
						
							|  |  |  | 	return ret; | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int snd_compr_pause(struct snd_compr_stream *stream) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int retval; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (stream->runtime->state != SNDRV_PCM_STATE_RUNNING) | 
					
						
							|  |  |  | 		return -EPERM; | 
					
						
							|  |  |  | 	retval = stream->ops->trigger(stream, SNDRV_PCM_TRIGGER_PAUSE_PUSH); | 
					
						
							| 
									
										
										
										
											2012-06-12 16:16:17 +05:30
										 |  |  | 	if (!retval) | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 		stream->runtime->state = SNDRV_PCM_STATE_PAUSED; | 
					
						
							|  |  |  | 	return retval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int snd_compr_resume(struct snd_compr_stream *stream) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int retval; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (stream->runtime->state != SNDRV_PCM_STATE_PAUSED) | 
					
						
							|  |  |  | 		return -EPERM; | 
					
						
							|  |  |  | 	retval = stream->ops->trigger(stream, SNDRV_PCM_TRIGGER_PAUSE_RELEASE); | 
					
						
							|  |  |  | 	if (!retval) | 
					
						
							|  |  |  | 		stream->runtime->state = SNDRV_PCM_STATE_RUNNING; | 
					
						
							|  |  |  | 	return retval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int snd_compr_start(struct snd_compr_stream *stream) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int retval; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (stream->runtime->state != SNDRV_PCM_STATE_PREPARED) | 
					
						
							|  |  |  | 		return -EPERM; | 
					
						
							|  |  |  | 	retval = stream->ops->trigger(stream, SNDRV_PCM_TRIGGER_START); | 
					
						
							|  |  |  | 	if (!retval) | 
					
						
							|  |  |  | 		stream->runtime->state = SNDRV_PCM_STATE_RUNNING; | 
					
						
							|  |  |  | 	return retval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int snd_compr_stop(struct snd_compr_stream *stream) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int retval; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (stream->runtime->state == SNDRV_PCM_STATE_PREPARED || | 
					
						
							|  |  |  | 			stream->runtime->state == SNDRV_PCM_STATE_SETUP) | 
					
						
							|  |  |  | 		return -EPERM; | 
					
						
							|  |  |  | 	retval = stream->ops->trigger(stream, SNDRV_PCM_TRIGGER_STOP); | 
					
						
							|  |  |  | 	if (!retval) { | 
					
						
							| 
									
										
										
										
											2013-10-24 16:37:31 +05:30
										 |  |  | 		snd_compr_drain_notify(stream); | 
					
						
							| 
									
										
										
										
											2012-06-12 16:16:18 +05:30
										 |  |  | 		stream->runtime->total_bytes_available = 0; | 
					
						
							|  |  |  | 		stream->runtime->total_bytes_transferred = 0; | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 	} | 
					
						
							|  |  |  | 	return retval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-24 16:37:31 +05:30
										 |  |  | static int snd_compress_wait_for_drain(struct snd_compr_stream *stream) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-11-07 10:08:22 +01:00
										 |  |  | 	int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-24 16:37:31 +05:30
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * We are called with lock held. So drop the lock while we wait for | 
					
						
							|  |  |  | 	 * drain complete notfication from the driver | 
					
						
							|  |  |  | 	 * | 
					
						
							|  |  |  | 	 * It is expected that driver will notify the drain completion and then | 
					
						
							|  |  |  | 	 * stream will be moved to SETUP state, even if draining resulted in an | 
					
						
							|  |  |  | 	 * error. We can trigger next track after this. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	stream->runtime->state = SNDRV_PCM_STATE_DRAINING; | 
					
						
							|  |  |  | 	mutex_unlock(&stream->device->lock); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-07 10:08:22 +01:00
										 |  |  | 	/* we wait for drain to complete here, drain can return when
 | 
					
						
							|  |  |  | 	 * interruption occurred, wait returned error or success. | 
					
						
							|  |  |  | 	 * For the first two cases we don't do anything different here and | 
					
						
							|  |  |  | 	 * return after waking up | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ret = wait_event_interruptible(stream->runtime->sleep, | 
					
						
							|  |  |  | 			(stream->runtime->state != SNDRV_PCM_STATE_DRAINING)); | 
					
						
							|  |  |  | 	if (ret == -ERESTARTSYS) | 
					
						
							|  |  |  | 		pr_debug("wait aborted by a signal"); | 
					
						
							|  |  |  | 	else if (ret) | 
					
						
							|  |  |  | 		pr_debug("wait for drain failed with %d\n", ret); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-24 16:37:31 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  | 	wake_up(&stream->runtime->sleep); | 
					
						
							|  |  |  | 	mutex_lock(&stream->device->lock); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-07 10:08:22 +01:00
										 |  |  | 	return ret; | 
					
						
							| 
									
										
										
										
											2013-10-24 16:37:31 +05:30
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | static int snd_compr_drain(struct snd_compr_stream *stream) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int retval; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (stream->runtime->state == SNDRV_PCM_STATE_PREPARED || | 
					
						
							|  |  |  | 			stream->runtime->state == SNDRV_PCM_STATE_SETUP) | 
					
						
							|  |  |  | 		return -EPERM; | 
					
						
							| 
									
										
										
										
											2013-10-24 16:37:31 +05:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 	retval = stream->ops->trigger(stream, SND_COMPR_TRIGGER_DRAIN); | 
					
						
							| 
									
										
										
										
											2013-10-24 16:37:31 +05:30
										 |  |  | 	if (retval) { | 
					
						
							| 
									
										
										
										
											2013-11-07 10:08:22 +01:00
										 |  |  | 		pr_debug("SND_COMPR_TRIGGER_DRAIN failed %d\n", retval); | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 		wake_up(&stream->runtime->sleep); | 
					
						
							| 
									
										
										
										
											2013-10-24 16:37:31 +05:30
										 |  |  | 		return retval; | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-10-24 16:37:31 +05:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-07 10:08:22 +01:00
										 |  |  | 	return snd_compress_wait_for_drain(stream); | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-14 16:52:51 +05:30
										 |  |  | static int snd_compr_next_track(struct snd_compr_stream *stream) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int retval; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* only a running stream can transition to next track */ | 
					
						
							|  |  |  | 	if (stream->runtime->state != SNDRV_PCM_STATE_RUNNING) | 
					
						
							|  |  |  | 		return -EPERM; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* you can signal next track isf this is intended to be a gapless stream
 | 
					
						
							|  |  |  | 	 * and current track metadata is set | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	if (stream->metadata_set == false) | 
					
						
							|  |  |  | 		return -EPERM; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	retval = stream->ops->trigger(stream, SND_COMPR_TRIGGER_NEXT_TRACK); | 
					
						
							|  |  |  | 	if (retval != 0) | 
					
						
							|  |  |  | 		return retval; | 
					
						
							|  |  |  | 	stream->metadata_set = false; | 
					
						
							|  |  |  | 	stream->next_track = true; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int snd_compr_partial_drain(struct snd_compr_stream *stream) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int retval; | 
					
						
							|  |  |  | 	if (stream->runtime->state == SNDRV_PCM_STATE_PREPARED || | 
					
						
							|  |  |  | 			stream->runtime->state == SNDRV_PCM_STATE_SETUP) | 
					
						
							|  |  |  | 		return -EPERM; | 
					
						
							|  |  |  | 	/* stream can be drained only when next track has been signalled */ | 
					
						
							|  |  |  | 	if (stream->next_track == false) | 
					
						
							|  |  |  | 		return -EPERM; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	retval = stream->ops->trigger(stream, SND_COMPR_TRIGGER_PARTIAL_DRAIN); | 
					
						
							| 
									
										
										
										
											2013-10-24 16:37:31 +05:30
										 |  |  | 	if (retval) { | 
					
						
							| 
									
										
										
										
											2013-11-07 10:08:22 +01:00
										 |  |  | 		pr_debug("Partial drain returned failure\n"); | 
					
						
							| 
									
										
										
										
											2013-10-24 16:37:31 +05:30
										 |  |  | 		wake_up(&stream->runtime->sleep); | 
					
						
							|  |  |  | 		return retval; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-02-14 16:52:51 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  | 	stream->next_track = false; | 
					
						
							| 
									
										
										
										
											2013-10-24 16:37:31 +05:30
										 |  |  | 	return snd_compress_wait_for_drain(stream); | 
					
						
							| 
									
										
										
										
											2013-02-14 16:52:51 +05:30
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | static long snd_compr_ioctl(struct file *f, unsigned int cmd, unsigned long arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct snd_compr_file *data = f->private_data; | 
					
						
							|  |  |  | 	struct snd_compr_stream *stream; | 
					
						
							|  |  |  | 	int retval = -ENOTTY; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (snd_BUG_ON(!data)) | 
					
						
							|  |  |  | 		return -EFAULT; | 
					
						
							|  |  |  | 	stream = &data->stream; | 
					
						
							|  |  |  | 	if (snd_BUG_ON(!stream)) | 
					
						
							|  |  |  | 		return -EFAULT; | 
					
						
							|  |  |  | 	mutex_lock(&stream->device->lock); | 
					
						
							|  |  |  | 	switch (_IOC_NR(cmd)) { | 
					
						
							|  |  |  | 	case _IOC_NR(SNDRV_COMPRESS_IOCTL_VERSION): | 
					
						
							| 
									
										
										
										
											2013-07-29 15:10:22 +05:30
										 |  |  | 		retval = put_user(SNDRV_COMPRESS_VERSION, | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 				(int __user *)arg) ? -EFAULT : 0; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case _IOC_NR(SNDRV_COMPRESS_GET_CAPS): | 
					
						
							|  |  |  | 		retval = snd_compr_get_caps(stream, arg); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case _IOC_NR(SNDRV_COMPRESS_GET_CODEC_CAPS): | 
					
						
							|  |  |  | 		retval = snd_compr_get_codec_caps(stream, arg); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case _IOC_NR(SNDRV_COMPRESS_SET_PARAMS): | 
					
						
							|  |  |  | 		retval = snd_compr_set_params(stream, arg); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case _IOC_NR(SNDRV_COMPRESS_GET_PARAMS): | 
					
						
							|  |  |  | 		retval = snd_compr_get_params(stream, arg); | 
					
						
							|  |  |  | 		break; | 
					
						
							| 
									
										
										
										
											2013-02-14 16:52:51 +05:30
										 |  |  | 	case _IOC_NR(SNDRV_COMPRESS_SET_METADATA): | 
					
						
							|  |  |  | 		retval = snd_compr_set_metadata(stream, arg); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case _IOC_NR(SNDRV_COMPRESS_GET_METADATA): | 
					
						
							|  |  |  | 		retval = snd_compr_get_metadata(stream, arg); | 
					
						
							|  |  |  | 		break; | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 	case _IOC_NR(SNDRV_COMPRESS_TSTAMP): | 
					
						
							|  |  |  | 		retval = snd_compr_tstamp(stream, arg); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case _IOC_NR(SNDRV_COMPRESS_AVAIL): | 
					
						
							|  |  |  | 		retval = snd_compr_ioctl_avail(stream, arg); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case _IOC_NR(SNDRV_COMPRESS_PAUSE): | 
					
						
							|  |  |  | 		retval = snd_compr_pause(stream); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case _IOC_NR(SNDRV_COMPRESS_RESUME): | 
					
						
							|  |  |  | 		retval = snd_compr_resume(stream); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case _IOC_NR(SNDRV_COMPRESS_START): | 
					
						
							|  |  |  | 		retval = snd_compr_start(stream); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case _IOC_NR(SNDRV_COMPRESS_STOP): | 
					
						
							|  |  |  | 		retval = snd_compr_stop(stream); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case _IOC_NR(SNDRV_COMPRESS_DRAIN): | 
					
						
							|  |  |  | 		retval = snd_compr_drain(stream); | 
					
						
							|  |  |  | 		break; | 
					
						
							| 
									
										
										
										
											2013-02-14 16:52:51 +05:30
										 |  |  | 	case _IOC_NR(SNDRV_COMPRESS_PARTIAL_DRAIN): | 
					
						
							|  |  |  | 		retval = snd_compr_partial_drain(stream); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case _IOC_NR(SNDRV_COMPRESS_NEXT_TRACK): | 
					
						
							|  |  |  | 		retval = snd_compr_next_track(stream); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 	} | 
					
						
							|  |  |  | 	mutex_unlock(&stream->device->lock); | 
					
						
							|  |  |  | 	return retval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct file_operations snd_compr_file_ops = { | 
					
						
							|  |  |  | 		.owner =	THIS_MODULE, | 
					
						
							|  |  |  | 		.open =		snd_compr_open, | 
					
						
							|  |  |  | 		.release =	snd_compr_free, | 
					
						
							|  |  |  | 		.write =	snd_compr_write, | 
					
						
							|  |  |  | 		.read =		snd_compr_read, | 
					
						
							|  |  |  | 		.unlocked_ioctl = snd_compr_ioctl, | 
					
						
							|  |  |  | 		.mmap =		snd_compr_mmap, | 
					
						
							|  |  |  | 		.poll =		snd_compr_poll, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int snd_compress_dev_register(struct snd_device *device) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int ret = -EINVAL; | 
					
						
							|  |  |  | 	char str[16]; | 
					
						
							|  |  |  | 	struct snd_compr *compr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (snd_BUG_ON(!device || !device->device_data)) | 
					
						
							|  |  |  | 		return -EBADFD; | 
					
						
							|  |  |  | 	compr = device->device_data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	sprintf(str, "comprC%iD%i", compr->card->number, compr->device); | 
					
						
							|  |  |  | 	pr_debug("reg %s for device %s, direction %d\n", str, compr->name, | 
					
						
							|  |  |  | 			compr->direction); | 
					
						
							|  |  |  | 	/* register compressed device */ | 
					
						
							|  |  |  | 	ret = snd_register_device(SNDRV_DEVICE_TYPE_COMPRESS, compr->card, | 
					
						
							|  |  |  | 			compr->device, &snd_compr_file_ops, compr, str); | 
					
						
							|  |  |  | 	if (ret < 0) { | 
					
						
							|  |  |  | 		pr_err("snd_register_device failed\n %d", ret); | 
					
						
							|  |  |  | 		return ret; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int snd_compress_dev_disconnect(struct snd_device *device) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct snd_compr *compr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	compr = device->device_data; | 
					
						
							| 
									
										
										
										
											2013-09-13 17:43:17 +01:00
										 |  |  | 	snd_unregister_device(SNDRV_DEVICE_TYPE_COMPRESS, compr->card, | 
					
						
							|  |  |  | 		compr->device); | 
					
						
							| 
									
										
										
										
											2011-12-23 10:36:39 +05:30
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * snd_compress_new: create new compress device | 
					
						
							|  |  |  |  * @card: sound card pointer | 
					
						
							|  |  |  |  * @device: device number | 
					
						
							|  |  |  |  * @dirn: device direction, should be of type enum snd_compr_direction | 
					
						
							|  |  |  |  * @compr: compress device pointer | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int snd_compress_new(struct snd_card *card, int device, | 
					
						
							|  |  |  | 			int dirn, struct snd_compr *compr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	static struct snd_device_ops ops = { | 
					
						
							|  |  |  | 		.dev_free = NULL, | 
					
						
							|  |  |  | 		.dev_register = snd_compress_dev_register, | 
					
						
							|  |  |  | 		.dev_disconnect = snd_compress_dev_disconnect, | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	compr->card = card; | 
					
						
							|  |  |  | 	compr->device = device; | 
					
						
							|  |  |  | 	compr->direction = dirn; | 
					
						
							|  |  |  | 	return snd_device_new(card, SNDRV_DEV_COMPRESS, compr, &ops); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | EXPORT_SYMBOL_GPL(snd_compress_new); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int snd_compress_add_device(struct snd_compr *device) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!device->card) | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* register the card */ | 
					
						
							|  |  |  | 	ret = snd_card_register(device->card); | 
					
						
							|  |  |  | 	if (ret) | 
					
						
							|  |  |  | 		goto out; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  | 	pr_err("failed with %d\n", ret); | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int snd_compress_remove_device(struct snd_compr *device) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return snd_card_free(device->card); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * snd_compress_register - register compressed device | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @device: compressed device to register | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int snd_compress_register(struct snd_compr *device) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int retval; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (device->name == NULL || device->dev == NULL || device->ops == NULL) | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pr_debug("Registering compressed device %s\n", device->name); | 
					
						
							|  |  |  | 	if (snd_BUG_ON(!device->ops->open)) | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 	if (snd_BUG_ON(!device->ops->free)) | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 	if (snd_BUG_ON(!device->ops->set_params)) | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 	if (snd_BUG_ON(!device->ops->trigger)) | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mutex_init(&device->lock); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* register a compressed card */ | 
					
						
							|  |  |  | 	mutex_lock(&device_mutex); | 
					
						
							|  |  |  | 	retval = snd_compress_add_device(device); | 
					
						
							|  |  |  | 	mutex_unlock(&device_mutex); | 
					
						
							|  |  |  | 	return retval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | EXPORT_SYMBOL_GPL(snd_compress_register); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int snd_compress_deregister(struct snd_compr *device) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	pr_debug("Removing compressed device %s\n", device->name); | 
					
						
							|  |  |  | 	mutex_lock(&device_mutex); | 
					
						
							|  |  |  | 	snd_compress_remove_device(device); | 
					
						
							|  |  |  | 	mutex_unlock(&device_mutex); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | EXPORT_SYMBOL_GPL(snd_compress_deregister); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int __init snd_compress_init(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void __exit snd_compress_exit(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module_init(snd_compress_init); | 
					
						
							|  |  |  | module_exit(snd_compress_exit); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | MODULE_DESCRIPTION("ALSA Compressed offload framework"); | 
					
						
							|  |  |  | MODULE_AUTHOR("Vinod Koul <vinod.koul@linux.intel.com>"); | 
					
						
							|  |  |  | MODULE_LICENSE("GPL v2"); |