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

📄 ueagle-atm.c

📁 LINUX 2.6.17.4的源码
💻 C
📖 第 1 页 / 共 4 页
字号:
	sc->stats.phy.mflags |= sc->stats.phy.flags;	/* in case of a flags ( for example delineation LOSS (& 0x10)),	 * we check the status again in order to detect the failure earlier	 */	if (sc->stats.phy.flags) {		uea_dbg(INS_TO_USBDEV(sc), "Stat flag = 0x%x\n",		       sc->stats.phy.flags);		return 0;	}	ret = uea_read_cmv(sc, SA_RATE, 0, &data);	if (ret < 0)		return ret;	/* in bulk mode the modem have problem with high rate	 * changing internal timing could improve things, but the	 * value is misterious.	 * ADI930 don't support it (-EPIPE error).	 */	if (UEA_CHIP_VERSION(sc) != ADI930		    && !use_iso[sc->modem_index]		    && sc->stats.phy.dsrate != (data >> 16) * 32) {		/* Original timming from ADI(used in windows driver)		 * 0x20ffff>>16 * 32 = 32 * 32 = 1Mbits		 */		u16 timeout = (data <= 0x20ffff) ? 0 : 1;		ret = uea_request(sc, UEA_SET_TIMEOUT, timeout, 0, NULL);		uea_info(INS_TO_USBDEV(sc),				"setting new timeout %d%s\n", timeout,				ret < 0?" failed":"");	}	sc->stats.phy.dsrate = (data >> 16) * 32;	sc->stats.phy.usrate = (data & 0xffff) * 32;	UPDATE_ATM_STAT(link_rate, sc->stats.phy.dsrate * 1000 / 424);	ret = uea_read_cmv(sc, SA_DIAG, 23, &data);	if (ret < 0)		return ret;	sc->stats.phy.dsattenuation = (data & 0xff) / 2;	ret = uea_read_cmv(sc, SA_DIAG, 47, &data);	if (ret < 0)		return ret;	sc->stats.phy.usattenuation = (data & 0xff) / 2;	ret = uea_read_cmv(sc, SA_DIAG, 25, &sc->stats.phy.dsmargin);	if (ret < 0)		return ret;	ret = uea_read_cmv(sc, SA_DIAG, 49, &sc->stats.phy.usmargin);	if (ret < 0)		return ret;	ret = uea_read_cmv(sc, SA_DIAG, 51, &sc->stats.phy.rxflow);	if (ret < 0)		return ret;	ret = uea_read_cmv(sc, SA_DIAG, 52, &sc->stats.phy.txflow);	if (ret < 0)		return ret;	ret = uea_read_cmv(sc, SA_DIAG, 54, &sc->stats.phy.dsunc);	if (ret < 0)		return ret;	/* only for atu-c */	ret = uea_read_cmv(sc, SA_DIAG, 58, &sc->stats.phy.usunc);	if (ret < 0)		return ret;	ret = uea_read_cmv(sc, SA_DIAG, 53, &sc->stats.phy.dscorr);	if (ret < 0)		return ret;	/* only for atu-c */	ret = uea_read_cmv(sc, SA_DIAG, 57, &sc->stats.phy.uscorr);	if (ret < 0)		return ret;	ret = uea_read_cmv(sc, SA_INFO, 8, &sc->stats.phy.vidco);	if (ret < 0)		return ret;	ret = uea_read_cmv(sc, SA_INFO, 13, &sc->stats.phy.vidcpe);	if (ret < 0)		return ret;	return 0;}static int request_cmvs(struct uea_softc *sc,		 struct uea_cmvs **cmvs, const struct firmware **fw){	int ret, size;	u8 *data;	char *file;	char cmv_name[FIRMWARE_NAME_MAX]; /* 30 bytes stack variable */	if (cmv_file[sc->modem_index] == NULL) {		if (UEA_CHIP_VERSION(sc) == ADI930)			file = (IS_ISDN(sc)) ? "CMV9i.bin" : "CMV9p.bin";		else			file = (IS_ISDN(sc)) ? "CMVei.bin" : "CMVep.bin";	} else		file = cmv_file[sc->modem_index];	strcpy(cmv_name, FW_DIR);	strlcat(cmv_name, file, sizeof(cmv_name));	ret = request_firmware(fw, cmv_name, &sc->usb_dev->dev);	if (ret < 0) {		uea_err(INS_TO_USBDEV(sc),		       "requesting firmware %s failed with error %d\n",		       cmv_name, ret);		return ret;	}	data = (u8 *) (*fw)->data;	size = *data * sizeof(struct uea_cmvs) + 1;	if (size != (*fw)->size) {		uea_err(INS_TO_USBDEV(sc), "firmware %s is corrupted\n",		       cmv_name);		release_firmware(*fw);		return -EILSEQ;	}	*cmvs = (struct uea_cmvs *)(data + 1);	return *data;}/* Start boot post firmware modem: * - send reset commands through usb control pipe * - start workqueue for DSP loading * - send CMV options to modem */static int uea_start_reset(struct uea_softc *sc){	u16 zero = 0;	/* ;-) */	int i, len, ret;	struct uea_cmvs *cmvs;	const struct firmware *cmvs_fw;	uea_enters(INS_TO_USBDEV(sc));	uea_info(INS_TO_USBDEV(sc), "(re)booting started\n");	/* mask interrupt */	sc->booting = 1;	/* We need to set this here because, a ack timeout could have occured,	 * but before we start the reboot, the ack occurs and set this to 1.	 * So we will failed to wait Ready CMV.	 */	sc->cmv_ack = 0;	UPDATE_ATM_STAT(signal, ATM_PHY_SIG_LOST);	/* reset statistics */	memset(&sc->stats, 0, sizeof(struct uea_stats));	/* tell the modem that we want to boot in IDMA mode */	uea_request(sc, UEA_SET_MODE, UEA_LOOPBACK_ON, 0, NULL);	uea_request(sc, UEA_SET_MODE, UEA_BOOT_IDMA, 0, NULL); 	/* enter reset mode */	uea_request(sc, UEA_SET_MODE, UEA_START_RESET, 0, NULL);	/* original driver use 200ms, but windows driver use 100ms */	msleep(100);	/* leave reset mode */	uea_request(sc, UEA_SET_MODE, UEA_END_RESET, 0, NULL); 	/* clear tx and rx mailboxes */	uea_request(sc, UEA_SET_2183_DATA, UEA_MPTX_MAILBOX, 2, &zero);	uea_request(sc, UEA_SET_2183_DATA, UEA_MPRX_MAILBOX, 2, &zero);	uea_request(sc, UEA_SET_2183_DATA, UEA_SWAP_MAILBOX, 2, &zero);	msleep(1000);	sc->cmv_function = MAKEFUNCTION(ADSLDIRECTIVE, MODEMREADY);	/* demask interrupt */	sc->booting = 0;	/* start loading DSP */	sc->pageno = 0;	sc->ovl = 0;	schedule_work(&sc->task);	/* wait for modem ready CMV */	ret = wait_cmv_ack(sc);	if (ret < 0)		return ret;	uea_vdbg(INS_TO_USBDEV(sc), "Ready CMV received\n");	/* Enter in R-IDLE (cmv) until instructed otherwise */	ret = uea_write_cmv(sc, SA_CNTL, 0, 1);	if (ret < 0)		return ret;	/* get options */ 	ret = len = request_cmvs(sc, &cmvs, &cmvs_fw);	if (ret < 0)		return ret;	/* send options */	for (i = 0; i < len; i++) {		ret = uea_write_cmv(sc, FW_GET_LONG(&cmvs[i].address),					FW_GET_WORD(&cmvs[i].offset),					FW_GET_LONG(&cmvs[i].data));		if (ret < 0)			goto out;	}	/* Enter in R-ACT-REQ */	ret = uea_write_cmv(sc, SA_CNTL, 0, 2);	uea_vdbg(INS_TO_USBDEV(sc), "Entering in R-ACT-REQ state\n");out:	release_firmware(cmvs_fw);	sc->reset = 0;	uea_leaves(INS_TO_USBDEV(sc));	return ret;}/* * In case of an error wait 1s before rebooting the modem * if the modem don't request reboot (-EAGAIN). * Monitor the modem every 1s. */static int uea_kthread(void *data){	struct uea_softc *sc = data;	int ret = -EAGAIN;	uea_enters(INS_TO_USBDEV(sc));	while (!kthread_should_stop()) {		if (ret < 0 || sc->reset)			ret = uea_start_reset(sc);		if (!ret)			ret = uea_stat(sc);		if (ret != -EAGAIN)			msleep(1000);	}	uea_leaves(INS_TO_USBDEV(sc));	return ret;}/* Load second usb firmware for ADI930 chip */static int load_XILINX_firmware(struct uea_softc *sc){	const struct firmware *fw_entry;	int ret, size, u, ln;	u8 *pfw, value;	char *fw_name = FW_DIR "930-fpga.bin";	uea_enters(INS_TO_USBDEV(sc));	ret = request_firmware(&fw_entry, fw_name, &sc->usb_dev->dev);	if (ret) {		uea_err(INS_TO_USBDEV(sc), "firmware %s is not available\n",		       fw_name);		goto err0;	}	pfw = fw_entry->data;	size = fw_entry->size;	if (size != 0x577B) {		uea_err(INS_TO_USBDEV(sc), "firmware %s is corrupted\n",		       fw_name);		ret = -EILSEQ;		goto err1;	}	for (u = 0; u < size; u += ln) {		ln = min(size - u, 64);		ret = uea_request(sc, 0xe, 0, ln, pfw + u);		if (ret < 0) {			uea_err(INS_TO_USBDEV(sc),			       "elsa download data failed (%d)\n", ret);			goto err1;		}	}	/* finish to send the fpga */	ret = uea_request(sc, 0xe, 1, 0, NULL);	if (ret < 0) {		uea_err(INS_TO_USBDEV(sc),				"elsa download data failed (%d)\n", ret);		goto err1;	}	/* Tell the modem we finish : de-assert reset */	value = 0;	ret = uea_send_modem_cmd(sc->usb_dev, 0xe, 1, &value);	if (ret < 0)		uea_err(sc->usb_dev, "elsa de-assert failed with error %d\n", ret);err1:	release_firmware(fw_entry);err0:	uea_leaves(INS_TO_USBDEV(sc));	return ret;}/* The modem send us an ack. First with check if it right */static void uea_dispatch_cmv(struct uea_softc *sc, struct cmv* cmv){	uea_enters(INS_TO_USBDEV(sc));	if (le16_to_cpu(cmv->wPreamble) != PREAMBLE)		goto bad1;	if (cmv->bDirection != MODEMTOHOST)		goto bad1;	/* FIXME : ADI930 reply wrong preambule (func = 2, sub = 2) to	 * the first MEMACESS cmv. Ignore it...	 */	if (cmv->bFunction != sc->cmv_function) {		if (UEA_CHIP_VERSION(sc) == ADI930				&& cmv->bFunction ==  MAKEFUNCTION(2, 2)) {			cmv->wIndex = cpu_to_le16(sc->cmv_idx);			put_unaligned(cpu_to_le32(sc->cmv_address), &cmv->dwSymbolicAddress);			cmv->wOffsetAddress = cpu_to_le16(sc->cmv_offset);		}		else			goto bad2;	}	if (cmv->bFunction == MAKEFUNCTION(ADSLDIRECTIVE, MODEMREADY)) {		wake_up_cmv_ack(sc);		uea_leaves(INS_TO_USBDEV(sc));		return;	}	/* in case of MEMACCESS */	if (le16_to_cpu(cmv->wIndex) != sc->cmv_idx ||	    le32_to_cpu(get_unaligned(&cmv->dwSymbolicAddress)) !=	    sc->cmv_address	    || le16_to_cpu(cmv->wOffsetAddress) != sc->cmv_offset)		goto bad2;	sc->data = le32_to_cpu(get_unaligned(&cmv->dwData));	sc->data = sc->data << 16 | sc->data >> 16;	wake_up_cmv_ack(sc);	uea_leaves(INS_TO_USBDEV(sc));	return;bad2:	uea_err(INS_TO_USBDEV(sc), "unexpected cmv received,"			"Function : %d, Subfunction : %d\n",			FUNCTION_TYPE(cmv->bFunction),			FUNCTION_SUBTYPE(cmv->bFunction));	uea_leaves(INS_TO_USBDEV(sc));	return;bad1:	uea_err(INS_TO_USBDEV(sc), "invalid cmv received, "			"wPreamble %d, bDirection %d\n",			le16_to_cpu(cmv->wPreamble), cmv->bDirection);	uea_leaves(INS_TO_USBDEV(sc));}/* * interrupt handler */static void uea_intr(struct urb *urb, struct pt_regs *regs){	struct uea_softc *sc = urb->context;	struct intr_pkt *intr = urb->transfer_buffer;	uea_enters(INS_TO_USBDEV(sc));	if (unlikely(urb->status < 0)) {		uea_err(INS_TO_USBDEV(sc), "uea_intr() failed with %d\n",		       urb->status);		return;	}	/* device-to-host interrupt */	if (intr->bType != 0x08 || sc->booting) {		uea_err(INS_TO_USBDEV(sc), "wrong interrupt\n");		goto resubmit;	}	switch (le16_to_cpu(intr->wInterrupt)) {	case INT_LOADSWAPPAGE:		sc->pageno = intr->bSwapPageNo;		sc->ovl = intr->bOvl >> 4 | intr->bOvl << 4;		schedule_work(&sc->task);		break;	case INT_INCOMINGCMV:		uea_dispatch_cmv(sc, &intr->u.s2.cmv);		break;	default:		uea_err(INS_TO_USBDEV(sc), "unknown interrupt %u\n",		       le16_to_cpu(intr->wInterrupt));	}resubmit:	usb_submit_urb(sc->urb_int, GFP_ATOMIC);}/* * Start the modem : init the data and start kernel thread */static int uea_boot(struct uea_softc *sc){	int ret;	struct intr_pkt *intr;	uea_enters(INS_TO_USBDEV(sc));	INIT_WORK(&sc->task, uea_load_page, sc);	init_waitqueue_head(&sc->sync_q);	init_waitqueue_head(&sc->cmv_ack_wait);	if (UEA_CHIP_VERSION(sc) == ADI930)		load_XILINX_firmware(sc);	intr = kmalloc(INTR_PKT_SIZE, GFP_KERNEL);	if (!intr) {		uea_err(INS_TO_USBDEV(sc),		       "cannot allocate interrupt package\n");		uea_leaves(INS_TO_USBDEV(sc));		return -ENOMEM;	}	sc->urb_int = usb_alloc_urb(0, GFP_KERNEL);	if (!sc->urb_int) {		uea_err(INS_TO_USBDEV(sc), "cannot allocate interrupt URB\n");		goto err;	}	usb_fill_int_urb(sc->urb_int, sc->usb_dev,			 usb_rcvintpipe(sc->usb_dev, UEA_INTR_PIPE),			 intr, INTR_PKT_SIZE, uea_intr, sc,			 sc->usb_dev->actconfig->interface[0]->altsetting[0].			 endpoint[0].desc.bInterval);	ret = usb_submit_urb(sc->urb_int, GFP_KERNEL);	if (ret < 0) {		uea_err(INS_TO_USBDEV(sc),		       "urb submition failed with error %d\n", ret);		goto err;	}	sc->kthread = kthread_run(uea_kthread, sc, "ueagle-atm");	if (sc->kthread == ERR_PTR(-ENOMEM)) {		uea_err(INS_TO_USBDEV(sc), "failed to create thread\n");		goto err2;	}	uea_leaves(INS_TO_USBDEV(sc));	return 0;err2:	usb_kill_urb(sc->urb_int);err:	usb_free_urb(sc->urb_int);	sc->urb_int = NULL;	kfree(intr);	uea_leaves(INS_TO_USBDEV(sc));	return -ENOMEM;}/*

⌨️ 快捷键说明

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