dmxdev.c 26.3 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3
/*
 * dmxdev.c - DVB demultiplexer device
 *
4 5
 * Copyright (C) 2000 Ralph Metzler & Marcus Metzler
 *		      for convergence integrated media GmbH
L
Linus Torvalds 已提交
6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public License
 * as published by the Free Software Foundation; either version 2.1
 * of the License, or (at your option) any later version.
 *
 * 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 Lesser 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.
 *
 */

#include <linux/spinlock.h>
#include <linux/slab.h>
#include <linux/vmalloc.h>
#include <linux/module.h>
#include <linux/moduleparam.h>
#include <linux/sched.h>
#include <linux/poll.h>
#include <linux/ioctl.h>
#include <linux/wait.h>
#include <asm/uaccess.h>
#include <asm/system.h>
#include "dmxdev.h"

static int debug;

module_param(debug, int, 0644);
MODULE_PARM_DESC(debug, "Turn on/off debugging (default:off).");

#define dprintk	if (debug) printk

43 44
static int dvb_dmxdev_buffer_write(struct dvb_ringbuffer *buf,
				   const u8 *src, size_t len)
L
Linus Torvalds 已提交
45
{
46
	ssize_t free;
L
Linus Torvalds 已提交
47 48 49 50 51 52

	if (!len)
		return 0;
	if (!buf->data)
		return 0;

53 54
	free = dvb_ringbuffer_free(buf);
	if (len > free) {
L
Linus Torvalds 已提交
55
		dprintk("dmxdev: buffer overflow\n");
56
		return -EOVERFLOW;
L
Linus Torvalds 已提交
57
	}
58 59

	return dvb_ringbuffer_write(buf, src, len);
L
Linus Torvalds 已提交
60 61
}

62
static ssize_t dvb_dmxdev_buffer_read(struct dvb_ringbuffer *src,
63 64
				      int non_blocking, char __user *buf,
				      size_t count, loff_t *ppos)
L
Linus Torvalds 已提交
65
{
66 67 68
	size_t todo;
	ssize_t avail;
	ssize_t ret = 0;
L
Linus Torvalds 已提交
69 70 71 72

	if (!src->data)
		return 0;

73 74 75 76
	if (src->error) {
		ret = src->error;
		dvb_ringbuffer_flush(src);
		return ret;
L
Linus Torvalds 已提交
77 78
	}

79 80 81 82 83
	for (todo = count; todo > 0; todo -= ret) {
		if (non_blocking && dvb_ringbuffer_empty(src)) {
			ret = -EWOULDBLOCK;
			break;
		}
L
Linus Torvalds 已提交
84

85 86 87 88 89
		ret = wait_event_interruptible(src->queue,
					       !dvb_ringbuffer_empty(src) ||
					       (src->error != 0));
		if (ret < 0)
			break;
L
Linus Torvalds 已提交
90

91 92 93 94
		if (src->error) {
			ret = src->error;
			dvb_ringbuffer_flush(src);
			break;
L
Linus Torvalds 已提交
95 96
		}

97
		avail = dvb_ringbuffer_avail(src);
98 99
		if (avail > todo)
			avail = todo;
100 101 102 103 104 105

		ret = dvb_ringbuffer_read(src, buf, avail, 1);
		if (ret < 0)
			break;

		buf += ret;
L
Linus Torvalds 已提交
106
	}
107 108

	return (count - todo) ? (count - todo) : ret;
L
Linus Torvalds 已提交
109 110
}

111
static struct dmx_frontend *get_fe(struct dmx_demux *demux, int type)
L
Linus Torvalds 已提交
112 113 114
{
	struct list_head *head, *pos;

115
	head = demux->get_frontends(demux);
L
Linus Torvalds 已提交
116 117 118
	if (!head)
		return NULL;
	list_for_each(pos, head)
119
		if (DMX_FE_ENTRY(pos)->source == type)
L
Linus Torvalds 已提交
120 121 122 123 124 125 126
			return DMX_FE_ENTRY(pos);

	return NULL;
}

static int dvb_dvr_open(struct inode *inode, struct file *file)
{
127 128
	struct dvb_device *dvbdev = file->private_data;
	struct dmxdev *dmxdev = dvbdev->priv;
L
Linus Torvalds 已提交
129 130
	struct dmx_frontend *front;

131
	dprintk("function : %s\n", __FUNCTION__);
L
Linus Torvalds 已提交
132

133
	if (mutex_lock_interruptible(&dmxdev->mutex))
L
Linus Torvalds 已提交
134 135
		return -ERESTARTSYS;

136 137
	if ((file->f_flags & O_ACCMODE) == O_RDWR) {
		if (!(dmxdev->capabilities & DMXDEV_CAP_DUPLEX)) {
138
			mutex_unlock(&dmxdev->mutex);
L
Linus Torvalds 已提交
139 140 141 142
			return -EOPNOTSUPP;
		}
	}

143
	if ((file->f_flags & O_ACCMODE) == O_RDONLY) {
144 145
		void *mem = vmalloc(DVR_BUFFER_SIZE);
		if (!mem) {
146 147 148
			mutex_unlock(&dmxdev->mutex);
			return -ENOMEM;
		}
149
		dvb_ringbuffer_init(&dmxdev->dvr_buffer, mem, DVR_BUFFER_SIZE);
L
Linus Torvalds 已提交
150 151
	}

152 153
	if ((file->f_flags & O_ACCMODE) == O_WRONLY) {
		dmxdev->dvr_orig_fe = dmxdev->demux->frontend;
L
Linus Torvalds 已提交
154 155

		if (!dmxdev->demux->write) {
156
			mutex_unlock(&dmxdev->mutex);
L
Linus Torvalds 已提交
157 158 159
			return -EOPNOTSUPP;
		}

160
		front = get_fe(dmxdev->demux, DMX_MEMORY_FE);
L
Linus Torvalds 已提交
161 162

		if (!front) {
163
			mutex_unlock(&dmxdev->mutex);
L
Linus Torvalds 已提交
164 165 166 167 168
			return -EINVAL;
		}
		dmxdev->demux->disconnect_frontend(dmxdev->demux);
		dmxdev->demux->connect_frontend(dmxdev->demux, front);
	}
169
	mutex_unlock(&dmxdev->mutex);
L
Linus Torvalds 已提交
170 171 172 173 174
	return 0;
}

static int dvb_dvr_release(struct inode *inode, struct file *file)
{
175 176
	struct dvb_device *dvbdev = file->private_data;
	struct dmxdev *dmxdev = dvbdev->priv;
L
Linus Torvalds 已提交
177

178
	if (mutex_lock_interruptible(&dmxdev->mutex))
L
Linus Torvalds 已提交
179 180
		return -ERESTARTSYS;

181
	if ((file->f_flags & O_ACCMODE) == O_WRONLY) {
L
Linus Torvalds 已提交
182 183 184 185
		dmxdev->demux->disconnect_frontend(dmxdev->demux);
		dmxdev->demux->connect_frontend(dmxdev->demux,
						dmxdev->dvr_orig_fe);
	}
186
	if ((file->f_flags & O_ACCMODE) == O_RDONLY) {
L
Linus Torvalds 已提交
187
		if (dmxdev->dvr_buffer.data) {
188
			void *mem = dmxdev->dvr_buffer.data;
L
Linus Torvalds 已提交
189 190
			mb();
			spin_lock_irq(&dmxdev->lock);
191
			dmxdev->dvr_buffer.data = NULL;
L
Linus Torvalds 已提交
192 193 194 195
			spin_unlock_irq(&dmxdev->lock);
			vfree(mem);
		}
	}
196
	mutex_unlock(&dmxdev->mutex);
L
Linus Torvalds 已提交
197 198 199 200
	return 0;
}

static ssize_t dvb_dvr_write(struct file *file, const char __user *buf,
201
			     size_t count, loff_t *ppos)
L
Linus Torvalds 已提交
202
{
203 204
	struct dvb_device *dvbdev = file->private_data;
	struct dmxdev *dmxdev = dvbdev->priv;
L
Linus Torvalds 已提交
205 206 207 208
	int ret;

	if (!dmxdev->demux->write)
		return -EOPNOTSUPP;
209
	if ((file->f_flags & O_ACCMODE) != O_WRONLY)
L
Linus Torvalds 已提交
210
		return -EINVAL;
211
	if (mutex_lock_interruptible(&dmxdev->mutex))
L
Linus Torvalds 已提交
212
		return -ERESTARTSYS;
213
	ret = dmxdev->demux->write(dmxdev->demux, buf, count);
214
	mutex_unlock(&dmxdev->mutex);
L
Linus Torvalds 已提交
215 216 217 218
	return ret;
}

static ssize_t dvb_dvr_read(struct file *file, char __user *buf, size_t count,
219
			    loff_t *ppos)
L
Linus Torvalds 已提交
220
{
221 222
	struct dvb_device *dvbdev = file->private_data;
	struct dmxdev *dmxdev = dvbdev->priv;
L
Linus Torvalds 已提交
223 224
	int ret;

225
	//mutex_lock(&dmxdev->mutex);
226 227 228
	ret = dvb_dmxdev_buffer_read(&dmxdev->dvr_buffer,
				     file->f_flags & O_NONBLOCK,
				     buf, count, ppos);
229
	//mutex_unlock(&dmxdev->mutex);
L
Linus Torvalds 已提交
230 231 232
	return ret;
}

233 234
static inline void dvb_dmxdev_filter_state_set(struct dmxdev_filter
					       *dmxdevfilter, int state)
L
Linus Torvalds 已提交
235 236
{
	spin_lock_irq(&dmxdevfilter->dev->lock);
237
	dmxdevfilter->state = state;
L
Linus Torvalds 已提交
238 239 240
	spin_unlock_irq(&dmxdevfilter->dev->lock);
}

241 242
static int dvb_dmxdev_set_buffer_size(struct dmxdev_filter *dmxdevfilter,
				      unsigned long size)
L
Linus Torvalds 已提交
243
{
244
	struct dvb_ringbuffer *buf = &dmxdevfilter->buffer;
L
Linus Torvalds 已提交
245 246
	void *mem;

247
	if (buf->size == size)
L
Linus Torvalds 已提交
248
		return 0;
249
	if (dmxdevfilter->state >= DMXDEV_STATE_GO)
L
Linus Torvalds 已提交
250 251
		return -EBUSY;
	spin_lock_irq(&dmxdevfilter->dev->lock);
252 253 254
	mem = buf->data;
	buf->data = NULL;
	buf->size = size;
255
	dvb_ringbuffer_flush(buf);
L
Linus Torvalds 已提交
256 257 258 259
	spin_unlock_irq(&dmxdevfilter->dev->lock);
	vfree(mem);

	if (buf->size) {
260
		mem = vmalloc(dmxdevfilter->buffer.size);
L
Linus Torvalds 已提交
261 262 263
		if (!mem)
			return -ENOMEM;
		spin_lock_irq(&dmxdevfilter->dev->lock);
264
		buf->data = mem;
L
Linus Torvalds 已提交
265 266 267 268 269 270 271
		spin_unlock_irq(&dmxdevfilter->dev->lock);
	}
	return 0;
}

static void dvb_dmxdev_filter_timeout(unsigned long data)
{
272
	struct dmxdev_filter *dmxdevfilter = (struct dmxdev_filter *)data;
L
Linus Torvalds 已提交
273

274
	dmxdevfilter->buffer.error = -ETIMEDOUT;
L
Linus Torvalds 已提交
275
	spin_lock_irq(&dmxdevfilter->dev->lock);
276
	dmxdevfilter->state = DMXDEV_STATE_TIMEDOUT;
L
Linus Torvalds 已提交
277 278 279 280 281 282
	spin_unlock_irq(&dmxdevfilter->dev->lock);
	wake_up(&dmxdevfilter->buffer.queue);
}

static void dvb_dmxdev_filter_timer(struct dmxdev_filter *dmxdevfilter)
{
283
	struct dmx_sct_filter_params *para = &dmxdevfilter->params.sec;
L
Linus Torvalds 已提交
284 285 286

	del_timer(&dmxdevfilter->timer);
	if (para->timeout) {
287 288 289 290
		dmxdevfilter->timer.function = dvb_dmxdev_filter_timeout;
		dmxdevfilter->timer.data = (unsigned long)dmxdevfilter;
		dmxdevfilter->timer.expires =
		    jiffies + 1 + (HZ / 2 + HZ * para->timeout) / 1000;
L
Linus Torvalds 已提交
291 292 293 294 295
		add_timer(&dmxdevfilter->timer);
	}
}

static int dvb_dmxdev_section_callback(const u8 *buffer1, size_t buffer1_len,
296 297 298
				       const u8 *buffer2, size_t buffer2_len,
				       struct dmx_section_filter *filter,
				       enum dmx_success success)
L
Linus Torvalds 已提交
299
{
300
	struct dmxdev_filter *dmxdevfilter = filter->priv;
L
Linus Torvalds 已提交
301 302 303 304 305 306 307
	int ret;

	if (dmxdevfilter->buffer.error) {
		wake_up(&dmxdevfilter->buffer.queue);
		return 0;
	}
	spin_lock(&dmxdevfilter->dev->lock);
308
	if (dmxdevfilter->state != DMXDEV_STATE_GO) {
L
Linus Torvalds 已提交
309 310 311 312 313 314
		spin_unlock(&dmxdevfilter->dev->lock);
		return 0;
	}
	del_timer(&dmxdevfilter->timer);
	dprintk("dmxdev: section callback %02x %02x %02x %02x %02x %02x\n",
		buffer1[0], buffer1[1],
315 316 317 318 319 320
		buffer1[2], buffer1[3], buffer1[4], buffer1[5]);
	ret = dvb_dmxdev_buffer_write(&dmxdevfilter->buffer, buffer1,
				      buffer1_len);
	if (ret == buffer1_len) {
		ret = dvb_dmxdev_buffer_write(&dmxdevfilter->buffer, buffer2,
					      buffer2_len);
L
Linus Torvalds 已提交
321
	}
322
	if (ret < 0) {
323 324
		dvb_ringbuffer_flush(&dmxdevfilter->buffer);
		dmxdevfilter->buffer.error = ret;
L
Linus Torvalds 已提交
325
	}
326 327
	if (dmxdevfilter->params.sec.flags & DMX_ONESHOT)
		dmxdevfilter->state = DMXDEV_STATE_DONE;
L
Linus Torvalds 已提交
328 329 330 331 332 333
	spin_unlock(&dmxdevfilter->dev->lock);
	wake_up(&dmxdevfilter->buffer.queue);
	return 0;
}

static int dvb_dmxdev_ts_callback(const u8 *buffer1, size_t buffer1_len,
334 335 336
				  const u8 *buffer2, size_t buffer2_len,
				  struct dmx_ts_feed *feed,
				  enum dmx_success success)
L
Linus Torvalds 已提交
337
{
338
	struct dmxdev_filter *dmxdevfilter = feed->priv;
339
	struct dvb_ringbuffer *buffer;
L
Linus Torvalds 已提交
340 341 342
	int ret;

	spin_lock(&dmxdevfilter->dev->lock);
343
	if (dmxdevfilter->params.pes.output == DMX_OUT_DECODER) {
L
Linus Torvalds 已提交
344 345 346 347
		spin_unlock(&dmxdevfilter->dev->lock);
		return 0;
	}

348 349
	if (dmxdevfilter->params.pes.output == DMX_OUT_TAP)
		buffer = &dmxdevfilter->buffer;
L
Linus Torvalds 已提交
350
	else
351
		buffer = &dmxdevfilter->dev->dvr_buffer;
L
Linus Torvalds 已提交
352 353 354 355 356
	if (buffer->error) {
		spin_unlock(&dmxdevfilter->dev->lock);
		wake_up(&buffer->queue);
		return 0;
	}
357 358 359 360
	ret = dvb_dmxdev_buffer_write(buffer, buffer1, buffer1_len);
	if (ret == buffer1_len)
		ret = dvb_dmxdev_buffer_write(buffer, buffer2, buffer2_len);
	if (ret < 0) {
361 362
		dvb_ringbuffer_flush(buffer);
		buffer->error = ret;
L
Linus Torvalds 已提交
363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390
	}
	spin_unlock(&dmxdevfilter->dev->lock);
	wake_up(&buffer->queue);
	return 0;
}

/* stop feed but only mark the specified filter as stopped (state set) */
static int dvb_dmxdev_feed_stop(struct dmxdev_filter *dmxdevfilter)
{
	dvb_dmxdev_filter_state_set(dmxdevfilter, DMXDEV_STATE_SET);

	switch (dmxdevfilter->type) {
	case DMXDEV_TYPE_SEC:
		del_timer(&dmxdevfilter->timer);
		dmxdevfilter->feed.sec->stop_filtering(dmxdevfilter->feed.sec);
		break;
	case DMXDEV_TYPE_PES:
		dmxdevfilter->feed.ts->stop_filtering(dmxdevfilter->feed.ts);
		break;
	default:
		return -EINVAL;
	}
	return 0;
}

/* start feed associated with the specified filter */
static int dvb_dmxdev_feed_start(struct dmxdev_filter *filter)
{
391
	dvb_dmxdev_filter_state_set(filter, DMXDEV_STATE_GO);
L
Linus Torvalds 已提交
392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412

	switch (filter->type) {
	case DMXDEV_TYPE_SEC:
		return filter->feed.sec->start_filtering(filter->feed.sec);
	case DMXDEV_TYPE_PES:
		return filter->feed.ts->start_filtering(filter->feed.ts);
	default:
		return -EINVAL;
	}

	return 0;
}

/* restart section feed if it has filters left associated with it,
   otherwise release the feed */
static int dvb_dmxdev_feed_restart(struct dmxdev_filter *filter)
{
	int i;
	struct dmxdev *dmxdev = filter->dev;
	u16 pid = filter->params.sec.pid;

413 414 415 416
	for (i = 0; i < dmxdev->filternum; i++)
		if (dmxdev->filter[i].state >= DMXDEV_STATE_GO &&
		    dmxdev->filter[i].type == DMXDEV_TYPE_SEC &&
		    dmxdev->filter[i].params.sec.pid == pid) {
L
Linus Torvalds 已提交
417 418 419 420
			dvb_dmxdev_feed_start(&dmxdev->filter[i]);
			return 0;
		}

421 422
	filter->dev->demux->release_section_feed(dmxdev->demux,
						 filter->feed.sec);
L
Linus Torvalds 已提交
423 424 425 426 427 428

	return 0;
}

static int dvb_dmxdev_filter_stop(struct dmxdev_filter *dmxdevfilter)
{
429
	if (dmxdevfilter->state < DMXDEV_STATE_GO)
L
Linus Torvalds 已提交
430 431 432 433 434 435 436 437 438
		return 0;

	switch (dmxdevfilter->type) {
	case DMXDEV_TYPE_SEC:
		if (!dmxdevfilter->feed.sec)
			break;
		dvb_dmxdev_feed_stop(dmxdevfilter);
		if (dmxdevfilter->filter.sec)
			dmxdevfilter->feed.sec->
439 440
			    release_filter(dmxdevfilter->feed.sec,
					   dmxdevfilter->filter.sec);
L
Linus Torvalds 已提交
441
		dvb_dmxdev_feed_restart(dmxdevfilter);
442
		dmxdevfilter->feed.sec = NULL;
L
Linus Torvalds 已提交
443 444 445 446 447 448
		break;
	case DMXDEV_TYPE_PES:
		if (!dmxdevfilter->feed.ts)
			break;
		dvb_dmxdev_feed_stop(dmxdevfilter);
		dmxdevfilter->dev->demux->
449 450 451
		    release_ts_feed(dmxdevfilter->dev->demux,
				    dmxdevfilter->feed.ts);
		dmxdevfilter->feed.ts = NULL;
L
Linus Torvalds 已提交
452 453
		break;
	default:
454
		if (dmxdevfilter->state == DMXDEV_STATE_ALLOCATED)
L
Linus Torvalds 已提交
455 456 457
			return 0;
		return -EINVAL;
	}
458 459

	dvb_ringbuffer_flush(&dmxdevfilter->buffer);
L
Linus Torvalds 已提交
460 461 462 463 464
	return 0;
}

static inline int dvb_dmxdev_filter_reset(struct dmxdev_filter *dmxdevfilter)
{
465
	if (dmxdevfilter->state < DMXDEV_STATE_SET)
L
Linus Torvalds 已提交
466 467
		return 0;

468
	dmxdevfilter->type = DMXDEV_TYPE_NONE;
L
Linus Torvalds 已提交
469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484
	dvb_dmxdev_filter_state_set(dmxdevfilter, DMXDEV_STATE_ALLOCATED);
	return 0;
}

static int dvb_dmxdev_filter_start(struct dmxdev_filter *filter)
{
	struct dmxdev *dmxdev = filter->dev;
	void *mem;
	int ret, i;

	if (filter->state < DMXDEV_STATE_SET)
		return -EINVAL;

	if (filter->state >= DMXDEV_STATE_GO)
		dvb_dmxdev_filter_stop(filter);

485
	if (!filter->buffer.data) {
L
Linus Torvalds 已提交
486
		mem = vmalloc(filter->buffer.size);
487 488
		if (!mem)
			return -ENOMEM;
L
Linus Torvalds 已提交
489
		spin_lock_irq(&filter->dev->lock);
490
		filter->buffer.data = mem;
L
Linus Torvalds 已提交
491 492 493
		spin_unlock_irq(&filter->dev->lock);
	}

494
	dvb_ringbuffer_flush(&filter->buffer);
L
Linus Torvalds 已提交
495 496 497 498

	switch (filter->type) {
	case DMXDEV_TYPE_SEC:
	{
499 500 501 502 503 504
		struct dmx_sct_filter_params *para = &filter->params.sec;
		struct dmx_section_filter **secfilter = &filter->filter.sec;
		struct dmx_section_feed **secfeed = &filter->feed.sec;

		*secfilter = NULL;
		*secfeed = NULL;
L
Linus Torvalds 已提交
505 506 507


		/* find active filter/feed with same PID */
508
		for (i = 0; i < dmxdev->filternum; i++) {
L
Linus Torvalds 已提交
509
			if (dmxdev->filter[i].state >= DMXDEV_STATE_GO &&
510 511
			    dmxdev->filter[i].type == DMXDEV_TYPE_SEC &&
			    dmxdev->filter[i].params.sec.pid == para->pid) {
L
Linus Torvalds 已提交
512 513 514 515 516 517 518
				*secfeed = dmxdev->filter[i].feed.sec;
				break;
			}
		}

		/* if no feed found, try to allocate new one */
		if (!*secfeed) {
519 520 521 522 523 524
			ret = dmxdev->demux->allocate_section_feed(dmxdev->demux,
								   secfeed,
								   dvb_dmxdev_section_callback);
			if (ret < 0) {
				printk("DVB (%s): could not alloc feed\n",
				       __FUNCTION__);
L
Linus Torvalds 已提交
525 526 527
				return ret;
			}

528 529 530 531 532
			ret = (*secfeed)->set(*secfeed, para->pid, 32768,
					      (para->flags & DMX_CHECK_CRC) ? 1 : 0);
			if (ret < 0) {
				printk("DVB (%s): could not set feed\n",
				       __FUNCTION__);
L
Linus Torvalds 已提交
533 534 535 536 537 538 539
				dvb_dmxdev_feed_restart(filter);
				return ret;
			}
		} else {
			dvb_dmxdev_feed_stop(filter);
		}

540
		ret = (*secfeed)->allocate_filter(*secfeed, secfilter);
L
Linus Torvalds 已提交
541 542 543
		if (ret < 0) {
			dvb_dmxdev_feed_restart(filter);
			filter->feed.sec->start_filtering(*secfeed);
544
			dprintk("could not get filter\n");
L
Linus Torvalds 已提交
545 546 547 548 549 550
			return ret;
		}

		(*secfilter)->priv = filter;

		memcpy(&((*secfilter)->filter_value[3]),
551
		       &(para->filter.filter[1]), DMX_FILTER_SIZE - 1);
L
Linus Torvalds 已提交
552
		memcpy(&(*secfilter)->filter_mask[3],
553
		       &para->filter.mask[1], DMX_FILTER_SIZE - 1);
L
Linus Torvalds 已提交
554
		memcpy(&(*secfilter)->filter_mode[3],
555
		       &para->filter.mode[1], DMX_FILTER_SIZE - 1);
L
Linus Torvalds 已提交
556

557 558 559 560 561
		(*secfilter)->filter_value[0] = para->filter.filter[0];
		(*secfilter)->filter_mask[0] = para->filter.mask[0];
		(*secfilter)->filter_mode[0] = para->filter.mode[0];
		(*secfilter)->filter_mask[1] = 0;
		(*secfilter)->filter_mask[2] = 0;
L
Linus Torvalds 已提交
562 563 564

		filter->todo = 0;

565
		ret = filter->feed.sec->start_filtering(filter->feed.sec);
L
Linus Torvalds 已提交
566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582
		if (ret < 0)
			return ret;

		dvb_dmxdev_filter_timer(filter);
		break;
	}
	case DMXDEV_TYPE_PES:
	{
		struct timespec timeout = { 0 };
		struct dmx_pes_filter_params *para = &filter->params.pes;
		dmx_output_t otype;
		int ret;
		int ts_type;
		enum dmx_ts_pes ts_pes;
		struct dmx_ts_feed **tsfeed = &filter->feed.ts;

		filter->feed.ts = NULL;
583
		otype = para->output;
L
Linus Torvalds 已提交
584

585
		ts_pes = (enum dmx_ts_pes)para->pes_type;
L
Linus Torvalds 已提交
586

587 588
		if (ts_pes < DMX_PES_OTHER)
			ts_type = TS_DECODER;
L
Linus Torvalds 已提交
589
		else
590
			ts_type = 0;
L
Linus Torvalds 已提交
591 592 593 594 595

		if (otype == DMX_OUT_TS_TAP)
			ts_type |= TS_PACKET;

		if (otype == DMX_OUT_TAP)
596
			ts_type |= TS_PAYLOAD_ONLY | TS_PACKET;
L
Linus Torvalds 已提交
597

598 599 600 601
		ret = dmxdev->demux->allocate_ts_feed(dmxdev->demux,
						      tsfeed,
						      dvb_dmxdev_ts_callback);
		if (ret < 0)
L
Linus Torvalds 已提交
602 603
			return ret;

604
		(*tsfeed)->priv = filter;
L
Linus Torvalds 已提交
605 606

		ret = (*tsfeed)->set(*tsfeed, para->pid, ts_type, ts_pes,
607
				     32768, timeout);
L
Linus Torvalds 已提交
608
		if (ret < 0) {
609 610
			dmxdev->demux->release_ts_feed(dmxdev->demux,
						       *tsfeed);
L
Linus Torvalds 已提交
611 612 613 614
			return ret;
		}

		ret = filter->feed.ts->start_filtering(filter->feed.ts);
615
		if (ret < 0) {
616 617
			dmxdev->demux->release_ts_feed(dmxdev->demux,
						       *tsfeed);
L
Linus Torvalds 已提交
618
			return ret;
619
		}
L
Linus Torvalds 已提交
620 621 622 623 624 625 626 627 628 629 630 631 632

		break;
	}
	default:
		return -EINVAL;
	}

	dvb_dmxdev_filter_state_set(filter, DMXDEV_STATE_GO);
	return 0;
}

static int dvb_demux_open(struct inode *inode, struct file *file)
{
633 634
	struct dvb_device *dvbdev = file->private_data;
	struct dmxdev *dmxdev = dvbdev->priv;
L
Linus Torvalds 已提交
635 636 637 638 639 640
	int i;
	struct dmxdev_filter *dmxdevfilter;

	if (!dmxdev->filter)
		return -EINVAL;

641
	if (mutex_lock_interruptible(&dmxdev->mutex))
L
Linus Torvalds 已提交
642 643
		return -ERESTARTSYS;

644 645
	for (i = 0; i < dmxdev->filternum; i++)
		if (dmxdev->filter[i].state == DMXDEV_STATE_FREE)
L
Linus Torvalds 已提交
646 647
			break;

648
	if (i == dmxdev->filternum) {
649
		mutex_unlock(&dmxdev->mutex);
L
Linus Torvalds 已提交
650 651 652
		return -EMFILE;
	}

653
	dmxdevfilter = &dmxdev->filter[i];
654
	mutex_init(&dmxdevfilter->mutex);
655
	file->private_data = dmxdevfilter;
L
Linus Torvalds 已提交
656

657
	dvb_ringbuffer_init(&dmxdevfilter->buffer, NULL, 8192);
658
	dmxdevfilter->type = DMXDEV_TYPE_NONE;
L
Linus Torvalds 已提交
659
	dvb_dmxdev_filter_state_set(dmxdevfilter, DMXDEV_STATE_ALLOCATED);
660
	dmxdevfilter->feed.ts = NULL;
L
Linus Torvalds 已提交
661 662
	init_timer(&dmxdevfilter->timer);

663
	mutex_unlock(&dmxdev->mutex);
L
Linus Torvalds 已提交
664 665 666
	return 0;
}

667 668
static int dvb_dmxdev_filter_free(struct dmxdev *dmxdev,
				  struct dmxdev_filter *dmxdevfilter)
L
Linus Torvalds 已提交
669
{
670
	if (mutex_lock_interruptible(&dmxdev->mutex))
L
Linus Torvalds 已提交
671 672
		return -ERESTARTSYS;

673 674
	if (mutex_lock_interruptible(&dmxdevfilter->mutex)) {
		mutex_unlock(&dmxdev->mutex);
L
Linus Torvalds 已提交
675 676 677 678 679 680 681
		return -ERESTARTSYS;
	}

	dvb_dmxdev_filter_stop(dmxdevfilter);
	dvb_dmxdev_filter_reset(dmxdevfilter);

	if (dmxdevfilter->buffer.data) {
682
		void *mem = dmxdevfilter->buffer.data;
L
Linus Torvalds 已提交
683 684

		spin_lock_irq(&dmxdev->lock);
685
		dmxdevfilter->buffer.data = NULL;
L
Linus Torvalds 已提交
686 687 688 689 690 691
		spin_unlock_irq(&dmxdev->lock);
		vfree(mem);
	}

	dvb_dmxdev_filter_state_set(dmxdevfilter, DMXDEV_STATE_FREE);
	wake_up(&dmxdevfilter->buffer.queue);
692 693
	mutex_unlock(&dmxdevfilter->mutex);
	mutex_unlock(&dmxdev->mutex);
L
Linus Torvalds 已提交
694 695 696 697 698 699 700
	return 0;
}

static inline void invert_mode(dmx_filter_t *filter)
{
	int i;

701 702
	for (i = 0; i < DMX_FILTER_SIZE; i++)
		filter->mode[i] ^= 0xff;
L
Linus Torvalds 已提交
703 704 705
}

static int dvb_dmxdev_filter_set(struct dmxdev *dmxdev,
706 707
				 struct dmxdev_filter *dmxdevfilter,
				 struct dmx_sct_filter_params *params)
L
Linus Torvalds 已提交
708
{
709
	dprintk("function : %s\n", __FUNCTION__);
L
Linus Torvalds 已提交
710 711 712

	dvb_dmxdev_filter_stop(dmxdevfilter);

713
	dmxdevfilter->type = DMXDEV_TYPE_SEC;
L
Linus Torvalds 已提交
714 715 716 717 718
	memcpy(&dmxdevfilter->params.sec,
	       params, sizeof(struct dmx_sct_filter_params));
	invert_mode(&dmxdevfilter->params.sec.filter);
	dvb_dmxdev_filter_state_set(dmxdevfilter, DMXDEV_STATE_SET);

719
	if (params->flags & DMX_IMMEDIATE_START)
L
Linus Torvalds 已提交
720 721 722 723 724 725
		return dvb_dmxdev_filter_start(dmxdevfilter);

	return 0;
}

static int dvb_dmxdev_pes_filter_set(struct dmxdev *dmxdev,
726 727
				     struct dmxdev_filter *dmxdevfilter,
				     struct dmx_pes_filter_params *params)
L
Linus Torvalds 已提交
728 729 730
{
	dvb_dmxdev_filter_stop(dmxdevfilter);

731
	if (params->pes_type > DMX_PES_OTHER || params->pes_type < 0)
L
Linus Torvalds 已提交
732 733
		return -EINVAL;

734 735 736
	dmxdevfilter->type = DMXDEV_TYPE_PES;
	memcpy(&dmxdevfilter->params, params,
	       sizeof(struct dmx_pes_filter_params));
L
Linus Torvalds 已提交
737 738 739

	dvb_dmxdev_filter_state_set(dmxdevfilter, DMXDEV_STATE_SET);

740
	if (params->flags & DMX_IMMEDIATE_START)
L
Linus Torvalds 已提交
741 742 743 744 745 746
		return dvb_dmxdev_filter_start(dmxdevfilter);

	return 0;
}

static ssize_t dvb_dmxdev_read_sec(struct dmxdev_filter *dfil,
747 748
				   struct file *file, char __user *buf,
				   size_t count, loff_t *ppos)
L
Linus Torvalds 已提交
749 750
{
	int result, hcount;
751 752 753 754 755 756 757 758 759 760 761
	int done = 0;

	if (dfil->todo <= 0) {
		hcount = 3 + dfil->todo;
		if (hcount > count)
			hcount = count;
		result = dvb_dmxdev_buffer_read(&dfil->buffer,
						file->f_flags & O_NONBLOCK,
						buf, hcount, ppos);
		if (result < 0) {
			dfil->todo = 0;
L
Linus Torvalds 已提交
762 763
			return result;
		}
764
		if (copy_from_user(dfil->secheader - dfil->todo, buf, result))
L
Linus Torvalds 已提交
765
			return -EFAULT;
766 767 768 769 770
		buf += result;
		done = result;
		count -= result;
		dfil->todo -= result;
		if (dfil->todo > -3)
L
Linus Torvalds 已提交
771
			return done;
772
		dfil->todo = ((dfil->secheader[1] << 8) | dfil->secheader[2]) & 0xfff;
L
Linus Torvalds 已提交
773 774 775
		if (!count)
			return done;
	}
776 777 778 779 780 781
	if (count > dfil->todo)
		count = dfil->todo;
	result = dvb_dmxdev_buffer_read(&dfil->buffer,
					file->f_flags & O_NONBLOCK,
					buf, count, ppos);
	if (result < 0)
L
Linus Torvalds 已提交
782
		return result;
783 784
	dfil->todo -= result;
	return (result + done);
L
Linus Torvalds 已提交
785 786 787
}

static ssize_t
788 789
dvb_demux_read(struct file *file, char __user *buf, size_t count,
	       loff_t *ppos)
L
Linus Torvalds 已提交
790
{
791 792
	struct dmxdev_filter *dmxdevfilter = file->private_data;
	int ret;
L
Linus Torvalds 已提交
793

794
	if (mutex_lock_interruptible(&dmxdevfilter->mutex))
L
Linus Torvalds 已提交
795 796
		return -ERESTARTSYS;

797 798
	if (dmxdevfilter->type == DMXDEV_TYPE_SEC)
		ret = dvb_dmxdev_read_sec(dmxdevfilter, file, buf, count, ppos);
L
Linus Torvalds 已提交
799
	else
800 801 802
		ret = dvb_dmxdev_buffer_read(&dmxdevfilter->buffer,
					     file->f_flags & O_NONBLOCK,
					     buf, count, ppos);
L
Linus Torvalds 已提交
803

804
	mutex_unlock(&dmxdevfilter->mutex);
L
Linus Torvalds 已提交
805 806 807 808 809 810
	return ret;
}

static int dvb_demux_do_ioctl(struct inode *inode, struct file *file,
			      unsigned int cmd, void *parg)
{
811
	struct dmxdev_filter *dmxdevfilter = file->private_data;
812 813 814
	struct dmxdev *dmxdev = dmxdevfilter->dev;
	unsigned long arg = (unsigned long)parg;
	int ret = 0;
L
Linus Torvalds 已提交
815

816
	if (mutex_lock_interruptible(&dmxdev->mutex))
L
Linus Torvalds 已提交
817 818 819 820
		return -ERESTARTSYS;

	switch (cmd) {
	case DMX_START:
821 822
		if (mutex_lock_interruptible(&dmxdevfilter->mutex)) {
			mutex_unlock(&dmxdev->mutex);
L
Linus Torvalds 已提交
823 824
			return -ERESTARTSYS;
		}
825
		if (dmxdevfilter->state < DMXDEV_STATE_SET)
L
Linus Torvalds 已提交
826 827 828
			ret = -EINVAL;
		else
			ret = dvb_dmxdev_filter_start(dmxdevfilter);
829
		mutex_unlock(&dmxdevfilter->mutex);
L
Linus Torvalds 已提交
830 831 832
		break;

	case DMX_STOP:
833 834
		if (mutex_lock_interruptible(&dmxdevfilter->mutex)) {
			mutex_unlock(&dmxdev->mutex);
L
Linus Torvalds 已提交
835 836
			return -ERESTARTSYS;
		}
837
		ret = dvb_dmxdev_filter_stop(dmxdevfilter);
838
		mutex_unlock(&dmxdevfilter->mutex);
L
Linus Torvalds 已提交
839 840 841
		break;

	case DMX_SET_FILTER:
842 843
		if (mutex_lock_interruptible(&dmxdevfilter->mutex)) {
			mutex_unlock(&dmxdev->mutex);
L
Linus Torvalds 已提交
844 845
			return -ERESTARTSYS;
		}
846
		ret = dvb_dmxdev_filter_set(dmxdev, dmxdevfilter, parg);
847
		mutex_unlock(&dmxdevfilter->mutex);
L
Linus Torvalds 已提交
848 849 850
		break;

	case DMX_SET_PES_FILTER:
851 852
		if (mutex_lock_interruptible(&dmxdevfilter->mutex)) {
			mutex_unlock(&dmxdev->mutex);
L
Linus Torvalds 已提交
853 854
			return -ERESTARTSYS;
		}
855
		ret = dvb_dmxdev_pes_filter_set(dmxdev, dmxdevfilter, parg);
856
		mutex_unlock(&dmxdevfilter->mutex);
L
Linus Torvalds 已提交
857 858 859
		break;

	case DMX_SET_BUFFER_SIZE:
860 861
		if (mutex_lock_interruptible(&dmxdevfilter->mutex)) {
			mutex_unlock(&dmxdev->mutex);
L
Linus Torvalds 已提交
862 863
			return -ERESTARTSYS;
		}
864
		ret = dvb_dmxdev_set_buffer_size(dmxdevfilter, arg);
865
		mutex_unlock(&dmxdevfilter->mutex);
L
Linus Torvalds 已提交
866 867 868 869 870 871 872
		break;

	case DMX_GET_EVENT:
		break;

	case DMX_GET_PES_PIDS:
		if (!dmxdev->demux->get_pes_pids) {
873
			ret = -EINVAL;
L
Linus Torvalds 已提交
874 875
			break;
		}
876
		dmxdev->demux->get_pes_pids(dmxdev->demux, parg);
L
Linus Torvalds 已提交
877 878
		break;

879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894
	case DMX_GET_CAPS:
		if (!dmxdev->demux->get_caps) {
			ret = -EINVAL;
			break;
		}
		ret = dmxdev->demux->get_caps(dmxdev->demux, parg);
		break;

	case DMX_SET_SOURCE:
		if (!dmxdev->demux->set_source) {
			ret = -EINVAL;
			break;
		}
		ret = dmxdev->demux->set_source(dmxdev->demux, parg);
		break;

L
Linus Torvalds 已提交
895 896
	case DMX_GET_STC:
		if (!dmxdev->demux->get_stc) {
897
			ret = -EINVAL;
L
Linus Torvalds 已提交
898 899 900
			break;
		}
		ret = dmxdev->demux->get_stc(dmxdev->demux,
901 902 903
					     ((struct dmx_stc *)parg)->num,
					     &((struct dmx_stc *)parg)->stc,
					     &((struct dmx_stc *)parg)->base);
L
Linus Torvalds 已提交
904 905 906
		break;

	default:
907 908
		ret = -EINVAL;
		break;
L
Linus Torvalds 已提交
909
	}
910
	mutex_unlock(&dmxdev->mutex);
L
Linus Torvalds 已提交
911 912 913 914 915 916 917 918 919
	return ret;
}

static int dvb_demux_ioctl(struct inode *inode, struct file *file,
			   unsigned int cmd, unsigned long arg)
{
	return dvb_usercopy(inode, file, cmd, arg, dvb_demux_do_ioctl);
}

920
static unsigned int dvb_demux_poll(struct file *file, poll_table *wait)
L
Linus Torvalds 已提交
921
{
922
	struct dmxdev_filter *dmxdevfilter = file->private_data;
L
Linus Torvalds 已提交
923 924 925 926 927 928 929 930 931 932 933 934 935 936 937
	unsigned int mask = 0;

	if (!dmxdevfilter)
		return -EINVAL;

	poll_wait(file, &dmxdevfilter->buffer.queue, wait);

	if (dmxdevfilter->state != DMXDEV_STATE_GO &&
	    dmxdevfilter->state != DMXDEV_STATE_DONE &&
	    dmxdevfilter->state != DMXDEV_STATE_TIMEDOUT)
		return 0;

	if (dmxdevfilter->buffer.error)
		mask |= (POLLIN | POLLRDNORM | POLLPRI | POLLERR);

938
	if (!dvb_ringbuffer_empty(&dmxdevfilter->buffer))
L
Linus Torvalds 已提交
939 940 941 942 943 944 945
		mask |= (POLLIN | POLLRDNORM | POLLPRI);

	return mask;
}

static int dvb_demux_release(struct inode *inode, struct file *file)
{
946
	struct dmxdev_filter *dmxdevfilter = file->private_data;
L
Linus Torvalds 已提交
947 948 949 950 951 952
	struct dmxdev *dmxdev = dmxdevfilter->dev;

	return dvb_dmxdev_filter_free(dmxdev, dmxdevfilter);
}

static struct file_operations dvb_demux_fops = {
953 954 955 956 957 958
	.owner = THIS_MODULE,
	.read = dvb_demux_read,
	.ioctl = dvb_demux_ioctl,
	.open = dvb_demux_open,
	.release = dvb_demux_release,
	.poll = dvb_demux_poll,
L
Linus Torvalds 已提交
959 960 961
};

static struct dvb_device dvbdev_demux = {
962 963 964 965
	.priv = NULL,
	.users = 1,
	.writers = 1,
	.fops = &dvb_demux_fops
L
Linus Torvalds 已提交
966 967 968
};

static int dvb_dvr_do_ioctl(struct inode *inode, struct file *file,
969
			    unsigned int cmd, void *parg)
L
Linus Torvalds 已提交
970
{
971 972
	struct dvb_device *dvbdev = file->private_data;
	struct dmxdev *dmxdev = dvbdev->priv;
973
	int ret;
L
Linus Torvalds 已提交
974

975
	if (mutex_lock_interruptible(&dmxdev->mutex))
L
Linus Torvalds 已提交
976 977 978 979 980
		return -ERESTARTSYS;

	switch (cmd) {
	case DMX_SET_BUFFER_SIZE:
		// FIXME: implement
981
		ret = 0;
L
Linus Torvalds 已提交
982 983 984
		break;

	default:
985 986
		ret = -EINVAL;
		break;
L
Linus Torvalds 已提交
987
	}
988
	mutex_unlock(&dmxdev->mutex);
L
Linus Torvalds 已提交
989 990 991 992
	return ret;
}

static int dvb_dvr_ioctl(struct inode *inode, struct file *file,
993
			 unsigned int cmd, unsigned long arg)
L
Linus Torvalds 已提交
994 995 996 997
{
	return dvb_usercopy(inode, file, cmd, arg, dvb_dvr_do_ioctl);
}

998
static unsigned int dvb_dvr_poll(struct file *file, poll_table *wait)
L
Linus Torvalds 已提交
999
{
1000 1001
	struct dvb_device *dvbdev = file->private_data;
	struct dmxdev *dmxdev = dvbdev->priv;
L
Linus Torvalds 已提交
1002 1003
	unsigned int mask = 0;

1004
	dprintk("function : %s\n", __FUNCTION__);
L
Linus Torvalds 已提交
1005 1006 1007

	poll_wait(file, &dmxdev->dvr_buffer.queue, wait);

1008
	if ((file->f_flags & O_ACCMODE) == O_RDONLY) {
L
Linus Torvalds 已提交
1009 1010 1011
		if (dmxdev->dvr_buffer.error)
			mask |= (POLLIN | POLLRDNORM | POLLPRI | POLLERR);

1012
		if (!dvb_ringbuffer_empty(&dmxdev->dvr_buffer))
L
Linus Torvalds 已提交
1013 1014 1015 1016 1017 1018 1019 1020
			mask |= (POLLIN | POLLRDNORM | POLLPRI);
	} else
		mask |= (POLLOUT | POLLWRNORM | POLLPRI);

	return mask;
}

static struct file_operations dvb_dvr_fops = {
1021 1022 1023 1024 1025 1026 1027
	.owner = THIS_MODULE,
	.read = dvb_dvr_read,
	.write = dvb_dvr_write,
	.ioctl = dvb_dvr_ioctl,
	.open = dvb_dvr_open,
	.release = dvb_dvr_release,
	.poll = dvb_dvr_poll,
L
Linus Torvalds 已提交
1028 1029 1030
};

static struct dvb_device dvbdev_dvr = {
1031 1032 1033 1034
	.priv = NULL,
	.users = 1,
	.writers = 1,
	.fops = &dvb_dvr_fops
L
Linus Torvalds 已提交
1035 1036
};

1037
int dvb_dmxdev_init(struct dmxdev *dmxdev, struct dvb_adapter *dvb_adapter)
L
Linus Torvalds 已提交
1038 1039 1040 1041 1042 1043
{
	int i;

	if (dmxdev->demux->open(dmxdev->demux) < 0)
		return -EUSERS;

1044
	dmxdev->filter = vmalloc(dmxdev->filternum * sizeof(struct dmxdev_filter));
L
Linus Torvalds 已提交
1045 1046 1047
	if (!dmxdev->filter)
		return -ENOMEM;

1048
	mutex_init(&dmxdev->mutex);
L
Linus Torvalds 已提交
1049
	spin_lock_init(&dmxdev->lock);
1050 1051 1052 1053 1054
	for (i = 0; i < dmxdev->filternum; i++) {
		dmxdev->filter[i].dev = dmxdev;
		dmxdev->filter[i].buffer.data = NULL;
		dvb_dmxdev_filter_state_set(&dmxdev->filter[i],
					    DMXDEV_STATE_FREE);
L
Linus Torvalds 已提交
1055 1056
	}

1057 1058 1059 1060
	dvb_register_device(dvb_adapter, &dmxdev->dvbdev, &dvbdev_demux, dmxdev,
			    DVB_DEVICE_DEMUX);
	dvb_register_device(dvb_adapter, &dmxdev->dvr_dvbdev, &dvbdev_dvr,
			    dmxdev, DVB_DEVICE_DVR);
L
Linus Torvalds 已提交
1061

1062
	dvb_ringbuffer_init(&dmxdev->dvr_buffer, NULL, 8192);
L
Linus Torvalds 已提交
1063 1064 1065

	return 0;
}
1066

L
Linus Torvalds 已提交
1067 1068
EXPORT_SYMBOL(dvb_dmxdev_init);

1069
void dvb_dmxdev_release(struct dmxdev *dmxdev)
L
Linus Torvalds 已提交
1070 1071 1072 1073 1074
{
	dvb_unregister_device(dmxdev->dvbdev);
	dvb_unregister_device(dmxdev->dvr_dvbdev);

	vfree(dmxdev->filter);
1075
	dmxdev->filter = NULL;
L
Linus Torvalds 已提交
1076 1077
	dmxdev->demux->close(dmxdev->demux);
}
1078

L
Linus Torvalds 已提交
1079
EXPORT_SYMBOL(dvb_dmxdev_release);