📄 ueagle-atm.c
字号:
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 + -