⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 av7110_hw.c

📁 linux-2.6.15.6
💻 C
📖 第 1 页 / 共 3 页
字号:
/* * av7110_hw.c: av7110 low level hardware access and firmware interface * * Copyright (C) 1999-2002 Ralph  Metzler *                       & Marcus Metzler for convergence integrated media GmbH * * originally based on code by: * Copyright (C) 1998,1999 Christian Theiss <mistert@rz.fh-augsburg.de> * * 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; either version 2 * 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 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. * Or, point your browser to http://www.gnu.org/copyleft/gpl.html * * the project's page is at http://www.linuxtv.org/dvb/ *//* for debugging ARM communication: *///#define COM_DEBUG#include <stdarg.h>#include <linux/types.h>#include <linux/kernel.h>#include <linux/string.h>#include <linux/sched.h>#include <linux/delay.h>#include <linux/byteorder/swabb.h>#include <linux/smp_lock.h>#include <linux/fs.h>#include "av7110.h"#include "av7110_hw.h"#define _NOHANDSHAKE/**************************************************************************** * DEBI functions ****************************************************************************//* This DEBI code is based on the Stradis driver   by Nathan Laredo <laredo@gnu.org> */int av7110_debiwrite(struct av7110 *av7110, u32 config,		     int addr, u32 val, int count){	struct saa7146_dev *dev = av7110->dev;	if (count <= 0 || count > 32764) {		printk("%s: invalid count %d\n", __FUNCTION__, count);		return -1;	}	if (saa7146_wait_for_debi_done(av7110->dev, 0) < 0) {		printk("%s: wait_for_debi_done failed\n", __FUNCTION__);		return -1;	}	saa7146_write(dev, DEBI_CONFIG, config);	if (count <= 4)		/* immediate transfer */		saa7146_write(dev, DEBI_AD, val);	else			/* block transfer */		saa7146_write(dev, DEBI_AD, av7110->debi_bus);	saa7146_write(dev, DEBI_COMMAND, (count << 17) | (addr & 0xffff));	saa7146_write(dev, MC2, (2 << 16) | 2);	return 0;}u32 av7110_debiread(struct av7110 *av7110, u32 config, int addr, int count){	struct saa7146_dev *dev = av7110->dev;	u32 result = 0;	if (count > 32764 || count <= 0) {		printk("%s: invalid count %d\n", __FUNCTION__, count);		return 0;	}	if (saa7146_wait_for_debi_done(av7110->dev, 0) < 0) {		printk("%s: wait_for_debi_done #1 failed\n", __FUNCTION__);		return 0;	}	saa7146_write(dev, DEBI_AD, av7110->debi_bus);	saa7146_write(dev, DEBI_COMMAND, (count << 17) | 0x10000 | (addr & 0xffff));	saa7146_write(dev, DEBI_CONFIG, config);	saa7146_write(dev, MC2, (2 << 16) | 2);	if (count > 4)		return count;	if (saa7146_wait_for_debi_done(av7110->dev, 0) < 0) {		printk("%s: wait_for_debi_done #2 failed\n", __FUNCTION__);		return 0;	}	result = saa7146_read(dev, DEBI_AD);	result &= (0xffffffffUL >> ((4 - count) * 8));	return result;}/* av7110 ARM core boot stuff */#if 0void av7110_reset_arm(struct av7110 *av7110){	saa7146_setgpio(av7110->dev, RESET_LINE, SAA7146_GPIO_OUTLO);	/* Disable DEBI and GPIO irq */	SAA7146_IER_DISABLE(av7110->dev, MASK_19 | MASK_03);	SAA7146_ISR_CLEAR(av7110->dev, MASK_19 | MASK_03);	saa7146_setgpio(av7110->dev, RESET_LINE, SAA7146_GPIO_OUTHI);	msleep(30);	/* the firmware needs some time to initialize */	ARM_ResetMailBox(av7110);	SAA7146_ISR_CLEAR(av7110->dev, MASK_19 | MASK_03);	SAA7146_IER_ENABLE(av7110->dev, MASK_03);	av7110->arm_ready = 1;	dprintk(1, "reset ARM\n");}#endif  /*  0  */static int waitdebi(struct av7110 *av7110, int adr, int state){	int k;	dprintk(4, "%p\n", av7110);	for (k = 0; k < 100; k++) {		if (irdebi(av7110, DEBINOSWAP, adr, 0, 2) == state)			return 0;		udelay(5);	}	return -ETIMEDOUT;}static int load_dram(struct av7110 *av7110, u32 *data, int len){	int i;	int blocks, rest;	u32 base, bootblock = BOOT_BLOCK;	dprintk(4, "%p\n", av7110);	blocks = len / BOOT_MAX_SIZE;	rest = len % BOOT_MAX_SIZE;	base = DRAM_START_CODE;	for (i = 0; i < blocks; i++) {		if (waitdebi(av7110, BOOT_STATE, BOOTSTATE_BUFFER_EMPTY) < 0) {			printk(KERN_ERR "dvb-ttpci: load_dram(): timeout at block %d\n", i);			return -ETIMEDOUT;		}		dprintk(4, "writing DRAM block %d\n", i);		mwdebi(av7110, DEBISWAB, bootblock,		       ((char*)data) + i * BOOT_MAX_SIZE, BOOT_MAX_SIZE);		bootblock ^= 0x1400;		iwdebi(av7110, DEBISWAB, BOOT_BASE, swab32(base), 4);		iwdebi(av7110, DEBINOSWAP, BOOT_SIZE, BOOT_MAX_SIZE, 2);		iwdebi(av7110, DEBINOSWAP, BOOT_STATE, BOOTSTATE_BUFFER_FULL, 2);		base += BOOT_MAX_SIZE;	}	if (rest > 0) {		if (waitdebi(av7110, BOOT_STATE, BOOTSTATE_BUFFER_EMPTY) < 0) {			printk(KERN_ERR "dvb-ttpci: load_dram(): timeout at last block\n");			return -ETIMEDOUT;		}		if (rest > 4)			mwdebi(av7110, DEBISWAB, bootblock,			       ((char*)data) + i * BOOT_MAX_SIZE, rest);		else			mwdebi(av7110, DEBISWAB, bootblock,			       ((char*)data) + i * BOOT_MAX_SIZE - 4, rest + 4);		iwdebi(av7110, DEBISWAB, BOOT_BASE, swab32(base), 4);		iwdebi(av7110, DEBINOSWAP, BOOT_SIZE, rest, 2);		iwdebi(av7110, DEBINOSWAP, BOOT_STATE, BOOTSTATE_BUFFER_FULL, 2);	}	if (waitdebi(av7110, BOOT_STATE, BOOTSTATE_BUFFER_EMPTY) < 0) {		printk(KERN_ERR "dvb-ttpci: load_dram(): timeout after last block\n");		return -ETIMEDOUT;	}	iwdebi(av7110, DEBINOSWAP, BOOT_SIZE, 0, 2);	iwdebi(av7110, DEBINOSWAP, BOOT_STATE, BOOTSTATE_BUFFER_FULL, 2);	if (waitdebi(av7110, BOOT_STATE, BOOTSTATE_BOOT_COMPLETE) < 0) {		printk(KERN_ERR "dvb-ttpci: load_dram(): final handshake timeout\n");		return -ETIMEDOUT;	}	return 0;}/* we cannot write av7110 DRAM directly, so load a bootloader into * the DPRAM which implements a simple boot protocol */static u8 bootcode[] = {  0xea, 0x00, 0x00, 0x0e, 0xe1, 0xb0, 0xf0, 0x0e, 0xe2, 0x5e, 0xf0, 0x04,  0xe2, 0x5e, 0xf0, 0x04, 0xe2, 0x5e, 0xf0, 0x08, 0xe2, 0x5e, 0xf0, 0x04,  0xe2, 0x5e, 0xf0, 0x04, 0xe2, 0x5e, 0xf0, 0x04, 0x2c, 0x00, 0x00, 0x24,  0x00, 0x00, 0x00, 0x0c, 0x00, 0x00, 0x00, 0x00, 0x2c, 0x00, 0x00, 0x34,  0x00, 0x00, 0x00, 0x00, 0xa5, 0xa5, 0x5a, 0x5a, 0x00, 0x1f, 0x15, 0x55,  0x00, 0x00, 0x00, 0x09, 0xe5, 0x9f, 0xd0, 0x7c, 0xe5, 0x9f, 0x40, 0x74,  0xe3, 0xa0, 0x00, 0x00, 0xe5, 0x84, 0x00, 0x00, 0xe5, 0x84, 0x00, 0x04,  0xe5, 0x9f, 0x10, 0x70, 0xe5, 0x9f, 0x20, 0x70, 0xe5, 0x9f, 0x30, 0x64,  0xe8, 0xb1, 0x1f, 0xe0, 0xe8, 0xa3, 0x1f, 0xe0, 0xe1, 0x51, 0x00, 0x02,  0xda, 0xff, 0xff, 0xfb, 0xe5, 0x9f, 0xf0, 0x50, 0xe1, 0xd4, 0x10, 0xb0,  0xe3, 0x51, 0x00, 0x00, 0x0a, 0xff, 0xff, 0xfc, 0xe1, 0xa0, 0x10, 0x0d,  0xe5, 0x94, 0x30, 0x04, 0xe1, 0xd4, 0x20, 0xb2, 0xe2, 0x82, 0x20, 0x3f,  0xe1, 0xb0, 0x23, 0x22, 0x03, 0xa0, 0x00, 0x02, 0xe1, 0xc4, 0x00, 0xb0,  0x0a, 0xff, 0xff, 0xf4, 0xe8, 0xb1, 0x1f, 0xe0, 0xe8, 0xa3, 0x1f, 0xe0,  0xe8, 0xb1, 0x1f, 0xe0, 0xe8, 0xa3, 0x1f, 0xe0, 0xe2, 0x52, 0x20, 0x01,  0x1a, 0xff, 0xff, 0xf9, 0xe2, 0x2d, 0xdb, 0x05, 0xea, 0xff, 0xff, 0xec,  0x2c, 0x00, 0x03, 0xf8, 0x2c, 0x00, 0x04, 0x00, 0x9e, 0x00, 0x08, 0x00,  0x2c, 0x00, 0x00, 0x74, 0x2c, 0x00, 0x00, 0xc0};int av7110_bootarm(struct av7110 *av7110){	struct saa7146_dev *dev = av7110->dev;	u32 ret;	int i;	dprintk(4, "%p\n", av7110);	saa7146_setgpio(dev, RESET_LINE, SAA7146_GPIO_OUTLO);	/* Disable DEBI and GPIO irq */	SAA7146_IER_DISABLE(av7110->dev, MASK_03 | MASK_19);	SAA7146_ISR_CLEAR(av7110->dev, MASK_19 | MASK_03);	/* enable DEBI */	saa7146_write(av7110->dev, MC1, 0x08800880);	saa7146_write(av7110->dev, DD1_STREAM_B, 0x00000000);	saa7146_write(av7110->dev, MC2, (MASK_09 | MASK_25 | MASK_10 | MASK_26));	/* test DEBI */	iwdebi(av7110, DEBISWAP, DPRAM_BASE, 0x76543210, 4);	if ((ret=irdebi(av7110, DEBINOSWAP, DPRAM_BASE, 0, 4)) != 0x10325476) {		printk(KERN_ERR "dvb-ttpci: debi test in av7110_bootarm() failed: "		       "%08x != %08x (check your BIOS 'Plug&Play OS' settings)\n",		       ret, 0x10325476);		return -1;	}	for (i = 0; i < 8192; i += 4)		iwdebi(av7110, DEBISWAP, DPRAM_BASE + i, 0x00, 4);	dprintk(2, "debi test OK\n");	/* boot */	dprintk(1, "load boot code\n");	saa7146_setgpio(dev, ARM_IRQ_LINE, SAA7146_GPIO_IRQLO);	//saa7146_setgpio(dev, DEBI_DONE_LINE, SAA7146_GPIO_INPUT);	//saa7146_setgpio(dev, 3, SAA7146_GPIO_INPUT);	mwdebi(av7110, DEBISWAB, DPRAM_BASE, bootcode, sizeof(bootcode));	iwdebi(av7110, DEBINOSWAP, BOOT_STATE, BOOTSTATE_BUFFER_FULL, 2);	if (saa7146_wait_for_debi_done(av7110->dev, 1)) {		printk(KERN_ERR "dvb-ttpci: av7110_bootarm(): "		       "saa7146_wait_for_debi_done() timed out\n");		return -ETIMEDOUT;	}	saa7146_setgpio(dev, RESET_LINE, SAA7146_GPIO_OUTHI);	mdelay(1);	dprintk(1, "load dram code\n");	if (load_dram(av7110, (u32 *)av7110->bin_root, av7110->size_root) < 0) {		printk(KERN_ERR "dvb-ttpci: av7110_bootarm(): "		       "load_dram() failed\n");		return -1;	}	saa7146_setgpio(dev, RESET_LINE, SAA7146_GPIO_OUTLO);	mdelay(1);	dprintk(1, "load dpram code\n");	mwdebi(av7110, DEBISWAB, DPRAM_BASE, av7110->bin_dpram, av7110->size_dpram);	if (saa7146_wait_for_debi_done(av7110->dev, 1)) {		printk(KERN_ERR "dvb-ttpci: av7110_bootarm(): "		       "saa7146_wait_for_debi_done() timed out after loading DRAM\n");		return -ETIMEDOUT;	}	saa7146_setgpio(dev, RESET_LINE, SAA7146_GPIO_OUTHI);	msleep(30);	/* the firmware needs some time to initialize */	//ARM_ClearIrq(av7110);	ARM_ResetMailBox(av7110);	SAA7146_ISR_CLEAR(av7110->dev, MASK_19 | MASK_03);	SAA7146_IER_ENABLE(av7110->dev, MASK_03);	av7110->arm_errors = 0;	av7110->arm_ready = 1;	return 0;}/**************************************************************************** * DEBI command polling ****************************************************************************/int av7110_wait_msgstate(struct av7110 *av7110, u16 flags){	unsigned long start;	u32 stat;	int err;	if (FW_VERSION(av7110->arm_app) <= 0x261c) {		/* not supported by old firmware */		msleep(50);		return 0;	}	/* new firmware */	start = jiffies;	for (;;) {		err = time_after(jiffies, start + ARM_WAIT_FREE);		if (down_interruptible(&av7110->dcomlock))			return -ERESTARTSYS;		stat = rdebi(av7110, DEBINOSWAP, MSGSTATE, 0, 2);		up(&av7110->dcomlock);		if ((stat & flags) == 0)			break;		if (err) {			printk(KERN_ERR "%s: timeout waiting for MSGSTATE %04x\n",				__FUNCTION__, stat & flags);			return -ETIMEDOUT;		}		msleep(1);	}	return 0;}static int __av7110_send_fw_cmd(struct av7110 *av7110, u16* buf, int length){	int i;	unsigned long start;	char *type = NULL;	u16 flags[2] = {0, 0};	u32 stat;	int err;//	dprintk(4, "%p\n", av7110);	if (!av7110->arm_ready) {		dprintk(1, "arm not ready.\n");		return -ENXIO;	}	start = jiffies;	while (1) {		err = time_after(jiffies, start + ARM_WAIT_FREE);		if (rdebi(av7110, DEBINOSWAP, COMMAND, 0, 2) == 0)			break;		if (err) {			printk(KERN_ERR "dvb-ttpci: %s(): timeout waiting for COMMAND idle\n", __FUNCTION__);			return -ETIMEDOUT;		}		msleep(1);	}	if (FW_VERSION(av7110->arm_app) <= 0x261f)		wdebi(av7110, DEBINOSWAP, COM_IF_LOCK, 0xffff, 2);#ifndef _NOHANDSHAKE	start = jiffies;	while (1) {		err = time_after(jiffies, start + ARM_WAIT_SHAKE);		if (rdebi(av7110, DEBINOSWAP, HANDSHAKE_REG, 0, 2) == 0)			break;		if (err) {			printk(KERN_ERR "dvb-ttpci: %s(): timeout waiting for HANDSHAKE_REG\n", __FUNCTION__);			return -ETIMEDOUT;		}		msleep(1);	}#endif	switch ((buf[0] >> 8) & 0xff) {	case COMTYPE_PIDFILTER:	case COMTYPE_ENCODER:	case COMTYPE_REC_PLAY:	case COMTYPE_MPEGDECODER:		type = "MSG";		flags[0] = GPMQOver;		flags[1] = GPMQFull;		break;	case COMTYPE_OSD:		type = "OSD";		flags[0] = OSDQOver;		flags[1] = OSDQFull;		break;	case COMTYPE_MISC:		if (FW_VERSION(av7110->arm_app) >= 0x261d) {			type = "MSG";			flags[0] = GPMQOver;			flags[1] = GPMQBusy;		}		break;

⌨️ 快捷键说明

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