av7110.c

来自「Linux Kernel 2.6.9 for OMAP1710」· C语言 代码 · 共 1,699 行 · 第 1/4 页

C
1,699
字号
        txbuf=irdebi(av7110, DEBINOSWAP, TX_BUFF, 0, 2);	len = (av7110->debilen + 3) & ~3;//        DEB_D(("GPIO0 irq %d %d\n", av7110->debitype, av7110->debilen));        print_time("gpio");//       DEB_D(("GPIO0 irq %02x\n", av7110->debitype&0xff));                switch (av7110->debitype&0xff) {        case DATA_TS_PLAY:        case DATA_PES_PLAY:                break;	case DATA_MPEG_VIDEO_EVENT:	{		u32 h_ar;		struct video_event event;                av7110->video_size.w = irdebi(av7110, DEBINOSWAP, STATUS_MPEG_WIDTH, 0, 2);                h_ar = irdebi(av7110, DEBINOSWAP, STATUS_MPEG_HEIGHT_AR, 0, 2);                iwdebi(av7110, DEBINOSWAP, IRQ_STATE_EXT, 0, 2);                iwdebi(av7110, DEBINOSWAP, RX_BUFF, 0, 2);		av7110->video_size.h = h_ar & 0xfff;		DEB_D(("GPIO0 irq: DATA_MPEG_VIDEO_EVENT: w/h/ar = %u/%u/%u\n",				av7110->video_size.w,				av7110->video_size.h,				av7110->video_size.aspect_ratio));		event.type = VIDEO_EVENT_SIZE_CHANGED;		event.u.size.w = av7110->video_size.w;		event.u.size.h = av7110->video_size.h;		switch ((h_ar >> 12) & 0xf)		{		case 3:			av7110->video_size.aspect_ratio = VIDEO_FORMAT_16_9;			event.u.size.aspect_ratio = VIDEO_FORMAT_16_9;			av7110->videostate.video_format = VIDEO_FORMAT_16_9;			break;		case 4:			av7110->video_size.aspect_ratio = VIDEO_FORMAT_221_1;			event.u.size.aspect_ratio = VIDEO_FORMAT_221_1;			av7110->videostate.video_format = VIDEO_FORMAT_221_1;			break;		default:			av7110->video_size.aspect_ratio = VIDEO_FORMAT_4_3;			event.u.size.aspect_ratio = VIDEO_FORMAT_4_3;			av7110->videostate.video_format = VIDEO_FORMAT_4_3;		}		dvb_video_add_event(av7110, &event);		break;	}        case DATA_CI_PUT:        {                int avail;                struct dvb_ringbuffer *cibuf=&av7110->ci_wbuffer;                avail=dvb_ringbuffer_avail(cibuf);                if (avail<=2) {                        iwdebi(av7110, DEBINOSWAP, IRQ_STATE_EXT, 0, 2);                        iwdebi(av7110, DEBINOSWAP, TX_LEN, 0, 2);                        iwdebi(av7110, DEBINOSWAP, TX_BUFF, 0, 2);                        break;                }                 len= DVB_RINGBUFFER_PEEK(cibuf,0)<<8;                len|=DVB_RINGBUFFER_PEEK(cibuf,1);                if (avail<len+2) {                        iwdebi(av7110, DEBINOSWAP, IRQ_STATE_EXT, 0, 2);                        iwdebi(av7110, DEBINOSWAP, TX_LEN, 0, 2);                        iwdebi(av7110, DEBINOSWAP, TX_BUFF, 0, 2);                        break;                }                 DVB_RINGBUFFER_SKIP(cibuf,2);                 dvb_ringbuffer_read(cibuf,av7110->debi_virt,len,0);                wake_up(&cibuf->queue);                iwdebi(av7110, DEBINOSWAP, TX_LEN, len, 2);                iwdebi(av7110, DEBINOSWAP, IRQ_STATE_EXT, len, 2);		saa7146_wait_for_debi_done(av7110->dev);                saa7146_write(av7110->dev, IER,                               saa7146_read(av7110->dev, IER) | MASK_19 );		if (len < 5)			len = 5; /* we want a real DEBI DMA */                iwdebi(av7110, DEBISWAB, DPRAM_BASE+txbuf, 0, (len+3)&~3);                spin_unlock(&av7110->debilock);                return;        }        case DATA_MPEG_PLAY:                if (!av7110->playing) {                        iwdebi(av7110, DEBINOSWAP, IRQ_STATE_EXT, 0, 2);                        iwdebi(av7110, DEBINOSWAP, TX_LEN, 0, 2);                        iwdebi(av7110, DEBINOSWAP, TX_BUFF, 0, 2);                        break;                }                len=0;                if (av7110->debitype&0x100) {                        spin_lock(&av7110->aout.lock);			len=av7110_pes_play(av7110->debi_virt, &av7110->aout, 2048);                        spin_unlock(&av7110->aout.lock);                }                if (len<=0 && (av7110->debitype&0x200)                        &&av7110->videostate.play_state!=VIDEO_FREEZED) {                        spin_lock(&av7110->avout.lock);			len=av7110_pes_play(av7110->debi_virt, &av7110->avout, 2048);                        spin_unlock(&av7110->avout.lock);                }                if (len<=0) {                        iwdebi(av7110, DEBINOSWAP, IRQ_STATE_EXT, 0, 2);                        iwdebi(av7110, DEBINOSWAP, TX_LEN, 0, 2);                        iwdebi(av7110, DEBINOSWAP, TX_BUFF, 0, 2);                        break;                }                 DEB_D(("GPIO0 PES_PLAY len=%04x\n", len));                        iwdebi(av7110, DEBINOSWAP, TX_LEN, len, 2);                iwdebi(av7110, DEBINOSWAP, IRQ_STATE_EXT, len, 2);		saa7146_wait_for_debi_done(av7110->dev);                saa7146_write(av7110->dev, IER,                               saa7146_read(av7110->dev, IER) | MASK_19 );                iwdebi(av7110, DEBISWAB, DPRAM_BASE+txbuf, 0, (len+3)&~3);                spin_unlock(&av7110->debilock);                return;        case DATA_BMP_LOAD:                len=av7110->debilen;                if (!len) {                        av7110->bmp_state=BMP_LOADED;                        iwdebi(av7110, DEBINOSWAP, IRQ_STATE_EXT, 0, 2);                        iwdebi(av7110, DEBINOSWAP, TX_LEN, 0, 2);                        iwdebi(av7110, DEBINOSWAP, TX_BUFF, 0, 2);                        wake_up(&av7110->bmpq);                        break;                }                if (len>av7110->bmplen)                        len=av7110->bmplen;                if (len>2*1024)                        len=2*1024;                iwdebi(av7110, DEBINOSWAP, TX_LEN, len, 2);                iwdebi(av7110, DEBINOSWAP, IRQ_STATE_EXT, len, 2);                memcpy(av7110->debi_virt, av7110->bmpbuf+av7110->bmpp, len);                av7110->bmpp+=len;                av7110->bmplen-=len;		saa7146_wait_for_debi_done(av7110->dev);                saa7146_write(av7110->dev, IER,                               saa7146_read(av7110->dev, IER) | MASK_19 );		if (len < 5)			len = 5; /* we want a real DEBI DMA */                iwdebi(av7110, DEBISWAB, DPRAM_BASE+txbuf, 0, (len+3)&~3);                spin_unlock(&av7110->debilock);                return;        case DATA_CI_GET:        case DATA_COMMON_INTERFACE:        case DATA_FSECTION:        case DATA_IPMPE:        case DATA_PIPING:                if (!len || len>4*1024) {                        iwdebi(av7110, DEBINOSWAP, RX_BUFF, 0, 2);                        break;		}		/* fall through */        case DATA_TS_RECORD:        case DATA_PES_RECORD:		saa7146_wait_for_debi_done(av7110->dev);                saa7146_write(av7110->dev, IER,                               saa7146_read(av7110->dev, IER) | MASK_19);                irdebi(av7110, DEBISWAB, DPRAM_BASE+rxbuf, 0, len);                spin_unlock(&av7110->debilock);                return;        case DATA_DEBUG_MESSAGE:		saa7146_wait_for_debi_done(av7110->dev);                if (!len || len>0xff) {                        iwdebi(av7110, DEBINOSWAP, RX_BUFF, 0, 2);                        break;                }                saa7146_write(av7110->dev, IER,                               saa7146_read(av7110->dev, IER) | MASK_19);                irdebi(av7110, DEBISWAB, Reserved, 0, len);                spin_unlock(&av7110->debilock);                return;        case DATA_IRCOMMAND:                 IR_handle(av7110,                           swahw32(irdebi(av7110, DEBINOSWAP, Reserved, 0, 4)));                iwdebi(av7110, DEBINOSWAP, RX_BUFF, 0, 2);                break;        default:                printk("gpioirq unknown type=%d len=%d\n",                        av7110->debitype, av7110->debilen);                break;        }              ARM_ClearMailBox(av7110);        av7110->debitype=-1;        spin_unlock(&av7110->debilock);}#ifdef CONFIG_DVB_AV7110_OSDstatic int dvb_osd_ioctl(struct inode *inode, struct file *file,			 unsigned int cmd, void *parg){	struct dvb_device *dvbdev = (struct dvb_device *) file->private_data;	struct av7110 *av7110 = (struct av7110 *) dvbdev->priv;	DEB_EE(("av7110: %p\n", av7110));	if (cmd == OSD_SEND_CMD)		return av7110_osd_cmd(av7110, (osd_cmd_t *) parg);	return -EINVAL;        }static struct file_operations dvb_osd_fops = {	.owner		= THIS_MODULE,	.ioctl		= dvb_generic_ioctl,	.open		= dvb_generic_open,	.release	= dvb_generic_release,};static struct dvb_device dvbdev_osd = {	.priv		= NULL,	.users		= 1,	.writers	= 1,	.fops		= &dvb_osd_fops,	.kernel_ioctl	= dvb_osd_ioctl,};#endif /* CONFIG_DVB_AV7110_OSD */static inline int SetPIDs(struct av7110 *av7110, u16 vpid, u16 apid, u16 ttpid,			  u16 subpid, u16 pcrpid)        {	DEB_EE(("av7110: %p\n", av7110));	if (vpid == 0x1fff || apid == 0x1fff ||	    ttpid == 0x1fff || subpid == 0x1fff || pcrpid == 0x1fff) {		vpid = apid = ttpid = subpid = pcrpid = 0;		av7110->pids[DMX_PES_VIDEO] = 0;		av7110->pids[DMX_PES_AUDIO] = 0;		av7110->pids[DMX_PES_TELETEXT] = 0;		av7110->pids[DMX_PES_PCR] = 0;	}	return av7110_fw_cmd(av7110, COMTYPE_PIDFILTER, MultiPID, 5,			     pcrpid, vpid, apid, ttpid, subpid);}void ChangePIDs(struct av7110 *av7110, u16 vpid, u16 apid, u16 ttpid,		u16 subpid, u16 pcrpid){	DEB_EE(("av7110: %p\n", av7110));        	if (down_interruptible(&av7110->pid_mutex))		return;	if (!(vpid & 0x8000))		av7110->pids[DMX_PES_VIDEO] = vpid;	if (!(apid & 0x8000))		av7110->pids[DMX_PES_AUDIO] = apid;	if (!(ttpid & 0x8000))		av7110->pids[DMX_PES_TELETEXT] = ttpid;	if (!(pcrpid & 0x8000))		av7110->pids[DMX_PES_PCR] = pcrpid;		av7110->pids[DMX_PES_SUBTITLE] = 0;	if (av7110->fe_synced) {		pcrpid = av7110->pids[DMX_PES_PCR];		SetPIDs(av7110, vpid, apid, ttpid, subpid, pcrpid);        }	up(&av7110->pid_mutex);}/****************************************************************************** * hardware filter functions ******************************************************************************/static int StartHWFilter(struct dvb_demux_filter *dvbdmxfilter){	struct dvb_demux_feed *dvbdmxfeed = dvbdmxfilter->feed;	struct av7110 *av7110 = (struct av7110 *) dvbdmxfeed->demux->priv;	u16 buf[20];	int ret, i;	u16 handle;//	u16 mode=0x0320;	u16 mode=0xb96a;	DEB_EE(("av7110: %p\n",av7110));	if (dvbdmxfilter->type == DMX_TYPE_SEC) {		if (hw_sections) {			buf[4] = (dvbdmxfilter->filter.filter_value[0] << 8) |				dvbdmxfilter->maskandmode[0];			for (i = 3; i < 18; i++)				buf[i + 4 - 2] =					(dvbdmxfilter->filter.filter_value[i] << 8) |					dvbdmxfilter->maskandmode[i];			mode = 4;                }	} else if ((dvbdmxfeed->ts_type & TS_PACKET) &&		   !(dvbdmxfeed->ts_type & TS_PAYLOAD_ONLY)) {		av7110_p2t_init(&av7110->p2t_filter[dvbdmxfilter->index], dvbdmxfeed);        }	buf[0] = (COMTYPE_PID_FILTER << 8) + AddPIDFilter;	buf[1] = 16;	buf[2] = dvbdmxfeed->pid;	buf[3] = mode;	ret = av7110_fw_request(av7110, buf, 20, &handle, 1);	if (ret < 0) {		printk("StartHWFilter error\n");		return ret;        }	av7110->handle2filter[handle] = dvbdmxfilter;	dvbdmxfilter->hw_handle = handle;	return ret;                }static int StopHWFilter(struct dvb_demux_filter *dvbdmxfilter){	struct av7110 *av7110 = (struct av7110 *) dvbdmxfilter->feed->demux->priv;	u16 buf[3];	u16 answ[2];	int ret;	u16 handle;        	DEB_EE(("av7110: %p\n", av7110));	handle = dvbdmxfilter->hw_handle;	if (handle > 32) {		DEB_S(("dvb: StopHWFilter tried to stop invalid filter %d.\n",		       handle));		DEB_S(("dvb: filter type = %d\n", dvbdmxfilter->type));		return 0;                }	av7110->handle2filter[handle] = NULL;	buf[0] = (COMTYPE_PID_FILTER << 8) + DelPIDFilter;	buf[1] = 1;	buf[2] = handle;	ret = av7110_fw_request(av7110, buf, 3, answ, 2);	if (ret)		printk("StopHWFilter error\n");	if (answ[1] != handle) {		DEB_S(("dvb: filter %d shutdown error :%d\n", handle, answ[1]));		ret = -1;        }        return ret;}static void dvb_feed_start_pid(struct dvb_demux_feed *dvbdmxfeed){	struct dvb_demux *dvbdmx = dvbdmxfeed->demux;	struct av7110 *av7110 = (struct av7110 *) dvbdmx->priv;	u16 *pid = dvbdmx->pids, npids[5];	int i;	DEB_EE(("av7110: %p\n",av7110));	npids[0] = npids[1] = npids[2] = npids[3] = npids[4] = 0xffff;	i = dvbdmxfeed->pes_type;	npids[i] = (pid[i]&0x8000) ? 0 : pid[i];	if ((i == 2) && npids[i] && (dvbdmxfeed->ts_type & TS_PACKET)) {		npids[i] = 0;		ChangePIDs(av7110, npids[1], npids[0], npids[2], npids[3], npids[4]);		StartHWFilter(dvbdmxfeed->filter);		return;	}	if (dvbdmxfeed->pes_type <= 2 || dvbdmxfeed->pes_type == 4)		ChangePIDs(av7110, npids[1], npids[0], npids[2], npids[3], npids[4]);        	if (dvbdmxfeed->pes_type < 2 && npids[0])		if (av7110->fe_synced)			av7110_fw_cmd(av7110, COMTYPE_PIDFILTER, Scan, 0);	if ((dvbdmxfeed->ts_type & TS_PACKET)) {		if (dvbdmxfeed->pes_type == 0 && !(dvbdmx->pids[0] & 0x8000))			av7110_av_start_record(av7110, RP_AUDIO, dvbdmxfeed);		if (dvbdmxfeed->pes_type == 1 && !(dvbdmx->pids[1] & 0x8000))			av7110_av_start_record(av7110, RP_VIDEO, dvbdmxfeed);	}}                static void dvb_feed_stop_pid(struct dvb_demux_feed *dvbdmxfeed){	struct dvb_demux *dvbdmx = dvbdmxfeed->demux;	struct av7110 *av7110 = (struct av7110 *) dvbdmx->priv;	u16 *pid = dvbdmx->pids, npids[5];	int i;                	DEB_EE(("av7110: %p\n", av7110));                	if (dvbdmxfeed->pes_type <= 1) {		av7110_av_stop(av7110, dvbdmxfeed->pes_type ?  RP_VIDEO : RP_AUDIO);		if (!av7110->rec_mode)			dvbdmx->recording = 0;		if (!av7110->playing)			dvbdmx->playing = 0;	}	npids[0] = npids[1] = npids[2] = npids[3] = npids[4] = 0xffff;	i = dvbdmxfeed->pes_type;	switch (i) {	case 2: //teletext		if (dvbdmxfeed->ts_type & TS_PACKET)			StopHWFilter(dvbdmxfeed->filter);		npids[2] = 0;                break;	case 0:	case 1:

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?