mixart_hwdep.c

来自「优龙2410linux2.6.8内核源代码」· C语言 代码 · 共 573 行 · 第 1/2 页

C
573
字号
		snd_printk(KERN_DEBUG "error MSG_CONSOLE_GET_CLOCK_UID : err=%x\n", console_mgr.error_code);		return -EINVAL;	}	/* used later for clock issues ! */	mgr->uid_console_manager = console_mgr.uid;	request.message_id = MSG_SYSTEM_ENUM_PHYSICAL_IO;	request.uid = (mixart_uid_t){0,0};	request.data = &console_mgr.uid;	request.size = sizeof(console_mgr.uid);	err = snd_mixart_send_msg(mgr, &request, sizeof(phys_io), &phys_io);	if( (err < 0) || ( phys_io.error_code != 0 ) ) {		snd_printk(KERN_ERR "error MSG_SYSTEM_ENUM_PHYSICAL_IO err(%x) error_code(%x)\n", err, phys_io.error_code );		return -EINVAL;	}	snd_assert(phys_io.nb_uid >= (MIXART_MAX_CARDS * 2),  return -EINVAL); /* min 2 phys io per card (analog in + analog out) */	for(k=0; k<mgr->num_cards; k++) {		mgr->chip[k]->uid_in_analog_physio = phys_io.uid[k];		mgr->chip[k]->uid_out_analog_physio = phys_io.uid[phys_io.nb_uid/2 + k]; 	}	return 0;}static int mixart_first_init(mixart_mgr_t *mgr){	u32 k;	int err;	mixart_msg_t request;	if((err = mixart_enum_connectors(mgr)) < 0) return err;	if((err = mixart_enum_physio(mgr)) < 0) return err;	/* send a synchro command to card (necessary to do this before first MSG_STREAM_START_STREAM_GRP_PACKET) */	/* though why not here */	request.message_id = MSG_SYSTEM_SEND_SYNCHRO_CMD;	request.uid = (mixart_uid_t){0,0};	request.data = NULL;	request.size = 0;	/* this command has no data. response is a 32 bit status */	err = snd_mixart_send_msg(mgr, &request, sizeof(k), &k);	if( (err < 0) || (k != 0) ) {		snd_printk(KERN_ERR "error MSG_SYSTEM_SEND_SYNCHRO_CMD\n");		return err == 0 ? -EINVAL : err;	}	return 0;}/* firmware base addresses (when hard coded) */#define MIXART_MOTHERBOARD_XLX_BASE_ADDRESS   0x00600000static int mixart_hwdep_dsp_load(snd_hwdep_t *hw, snd_hwdep_dsp_image_t *dsp){	mixart_mgr_t* mgr = snd_magic_cast(mixart_mgr_t, hw->private_data, return -ENXIO);	int           err, card_index;	u32           status_xilinx, status_elf, status_daught;	u32           val;	/* read motherboard xilinx status */	status_xilinx = readl_be( MIXART_MEM( mgr,MIXART_PSEUDOREG_MXLX_STATUS_OFFSET ));	/* read elf status */	status_elf = readl_be( MIXART_MEM( mgr,MIXART_PSEUDOREG_ELF_STATUS_OFFSET ));	/* read daughterboard xilinx status */	status_daught = readl_be( MIXART_MEM( mgr,MIXART_PSEUDOREG_DXLX_STATUS_OFFSET ));	/* motherboard xilinx status 5 will say that the board is performing a reset */	if( status_xilinx == 5 ) {		snd_printk( KERN_ERR "miXart is resetting !\n");		return -EAGAIN; /* try again later */	}	switch (dsp->index)   {	case MIXART_MOTHERBOARD_XLX_INDEX:		/* xilinx already loaded ? */ 		if( status_xilinx == 4 ) {			snd_printk( KERN_DEBUG "xilinx is already loaded !\n");			return 0;		}		/* the status should be 0 == "idle" */		if( status_xilinx != 0 ) {			snd_printk( KERN_ERR "xilinx load error ! status = %d\n", status_xilinx);			return -EIO; /* modprob -r may help ? */		}		/* check xilinx validity */		snd_assert(((u32*)(dsp->image))[0]==0xFFFFFFFF, return -EINVAL);		snd_assert(dsp->length % 4 == 0, return -EINVAL);		/* set xilinx status to copying */		writel_be( 1, MIXART_MEM( mgr, MIXART_PSEUDOREG_MXLX_STATUS_OFFSET ));		/* setup xilinx base address */		writel_be( MIXART_MOTHERBOARD_XLX_BASE_ADDRESS, MIXART_MEM( mgr,MIXART_PSEUDOREG_MXLX_BASE_ADDR_OFFSET ));		/* setup code size for xilinx file */		writel_be( dsp->length, MIXART_MEM( mgr, MIXART_PSEUDOREG_MXLX_SIZE_OFFSET ));		/* copy xilinx code */		if (copy_from_user_toio(  MIXART_MEM( mgr, MIXART_MOTHERBOARD_XLX_BASE_ADDRESS),  dsp->image,  dsp->length))			return -EFAULT;    		/* set xilinx status to copy finished */		writel_be( 2, MIXART_MEM( mgr, MIXART_PSEUDOREG_MXLX_STATUS_OFFSET ));		/* return, because no further processing needed */		return 0;	case MIXART_MOTHERBOARD_ELF_INDEX:		if( status_elf == 4 ) {			snd_printk( KERN_DEBUG "elf file already loaded !\n");			return 0;		}		/* the status should be 0 == "idle" */		if( status_elf != 0 ) {			snd_printk( KERN_ERR "elf load error ! status = %d\n", status_elf);			return -EIO; /* modprob -r may help ? */		}		/* wait for xilinx status == 4 */		err = mixart_wait_nice_for_register_value( mgr, MIXART_PSEUDOREG_MXLX_STATUS_OFFSET, 1, 4, 500); /* 5sec */		if (err < 0) {			snd_printk( KERN_ERR "xilinx was not loaded or could not be started\n");			return err;		}		/* init some data on the card */		writel_be( 0, MIXART_MEM( mgr, MIXART_PSEUDOREG_BOARDNUMBER ) ); /* set miXart boardnumber to 0 */		writel_be( 0, MIXART_MEM( mgr, MIXART_FLOWTABLE_PTR ) );         /* reset pointer to flow table on miXart */		/* set elf status to copying */		writel_be( 1, MIXART_MEM( mgr, MIXART_PSEUDOREG_ELF_STATUS_OFFSET ));		/* process the copying of the elf packets */		err = mixart_load_elf( mgr, dsp);		if (err < 0) return err;		/* set elf status to copy finished */		writel_be( 2, MIXART_MEM( mgr, MIXART_PSEUDOREG_ELF_STATUS_OFFSET ));		/* wait for elf status == 4 */		err = mixart_wait_nice_for_register_value( mgr, MIXART_PSEUDOREG_ELF_STATUS_OFFSET, 1, 4, 300); /* 3sec */		if (err < 0) {			snd_printk( KERN_ERR "elf could not be started\n");			return err;		}		/* miXart waits at this point on the pointer to the flow table */		writel_be( (u32)mgr->flowinfo.addr, MIXART_MEM( mgr, MIXART_FLOWTABLE_PTR ) ); /* give pointer of flow table to miXart */		return 0;  /* return, another xilinx file has to be loaded before */	case MIXART_AESEBUBOARD_XLX_INDEX:	default:		/* elf and xilinx should be loaded */		if( (status_elf != 4) || (status_xilinx != 4) ) {			printk( KERN_ERR "xilinx or elf not successfully loaded\n");			return -EIO; /* modprob -r may help ? */		}		/* wait for daughter detection != 0 */		err = mixart_wait_nice_for_register_value( mgr, MIXART_PSEUDOREG_DBRD_PRESENCE_OFFSET, 0, 0, 30); /* 300msec */		if (err < 0) {			snd_printk( KERN_ERR "error starting elf file\n");			return err;		}		/* the board type can now be retrieved */		mgr->board_type = (DAUGHTER_TYPE_MASK & readl_be( MIXART_MEM( mgr, MIXART_PSEUDOREG_DBRD_TYPE_OFFSET)));		if (mgr->board_type == MIXART_DAUGHTER_TYPE_NONE)			break;  /* no daughter board; the file does not have to be loaded, continue after the switch */		/* only if aesebu daughter board presence (elf code must run)  */ 		if (mgr->board_type != MIXART_DAUGHTER_TYPE_AES )			return -EINVAL;		/* daughter should be idle */		if( status_daught != 0 ) {			printk( KERN_ERR "daughter load error ! status = %d\n", status_daught);			return -EIO; /* modprob -r may help ? */		} 		/* check daughterboard xilinx validity */		snd_assert(((u32*)(dsp->image))[0]==0xFFFFFFFF, return -EINVAL);		snd_assert(dsp->length % 4 == 0, return -EINVAL);		/* inform mixart about the size of the file */		writel_be( dsp->length, MIXART_MEM( mgr, MIXART_PSEUDOREG_DXLX_SIZE_OFFSET ));		/* set daughterboard status to 1 */		writel_be( 1, MIXART_MEM( mgr, MIXART_PSEUDOREG_DXLX_STATUS_OFFSET ));		/* wait for status == 2 */		err = mixart_wait_nice_for_register_value( mgr, MIXART_PSEUDOREG_DXLX_STATUS_OFFSET, 1, 2, 30); /* 300msec */		if (err < 0) {			snd_printk( KERN_ERR "daughter board load error\n");			return err;		}		/* get the address where to write the file */		val = readl_be( MIXART_MEM( mgr, MIXART_PSEUDOREG_DXLX_BASE_ADDR_OFFSET ));		snd_assert(val != 0, return -EINVAL);		/* copy daughterboard xilinx code */		if (copy_from_user_toio(  MIXART_MEM( mgr, val),  dsp->image,  dsp->length))			return -EFAULT;		/* set daughterboard status to 4 */		writel_be( 4, MIXART_MEM( mgr, MIXART_PSEUDOREG_DXLX_STATUS_OFFSET ));		/* continue with init */		break;	} /* end of switch file index*/        /* wait for daughter status == 3 */        err = mixart_wait_nice_for_register_value( mgr, MIXART_PSEUDOREG_DXLX_STATUS_OFFSET, 1, 3, 300); /* 3sec */        if (err < 0) {		snd_printk( KERN_ERR "daughter board could not be initialised\n");		return err;	}	/* init mailbox (communication with embedded) */	snd_mixart_init_mailbox(mgr);	/* first communication with embedded */	err = mixart_first_init(mgr);        if (err < 0) {		snd_printk( KERN_ERR "miXart could not be set up\n");		return err;	}       	/* create devices and mixer in accordance with HW options*/        for (card_index = 0; card_index < mgr->num_cards; card_index++) {		mixart_t *chip = mgr->chip[card_index];		if ((err = snd_mixart_create_pcm(chip)) < 0)			return err;		if (card_index == 0) {			if ((err = snd_mixart_create_mixer(chip->mgr)) < 0)	        		return err;		}		if ((err = snd_card_register(chip->card)) < 0)			return err;	};	snd_printdd("miXart firmware downloaded and successfully set up\n");	return 0;}int snd_mixart_hwdep_new(mixart_mgr_t *mgr){	int err;	snd_hwdep_t *hw;	/* only create hwdep interface for first cardX (see "index" module parameter)*/	if ((err = snd_hwdep_new(mgr->chip[0]->card, SND_MIXART_HWDEP_ID, 0, &hw)) < 0)		return err;	hw->iface = SNDRV_HWDEP_IFACE_MIXART;	hw->private_data = mgr;	hw->ops.open = mixart_hwdep_open;	hw->ops.release = mixart_hwdep_release;	hw->ops.dsp_status = mixart_hwdep_dsp_status;	hw->ops.dsp_load = mixart_hwdep_dsp_load;	hw->exclusive = 1;	sprintf(hw->name,  SND_MIXART_HWDEP_ID);	mgr->hwdep = hw;	mgr->hwdep->dsp_loaded = 0;	return 0;}

⌨️ 快捷键说明

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