📄 soc.c
字号:
request = sw_cq->pool + sw_cq->in; fch = &request->fchdr; switch (fcmd->proto) { case TYPE_SCSI_FCP: request->shdr.token = TOKEN(TYPE_SCSI_FCP, port->mask, fcmd->token); request->data[0].base = fc->dma_scsi_cmd + fcmd->token * sizeof(fcp_cmd); request->data[0].count = sizeof(fcp_cmd); request->data[1].base = fc->dma_scsi_rsp + fcmd->token * fc->rsp_size; request->data[1].count = fc->rsp_size; if (fcmd->data) { request->shdr.segcnt = 3; i = fc->scsi_cmd_pool[fcmd->token].fcp_data_len; request->shdr.bytecnt = i; request->data[2].base = fcmd->data; request->data[2].count = i; request->type = (fc->scsi_cmd_pool[fcmd->token].fcp_cntl & FCP_CNTL_WRITE) ? SOC_CQTYPE_IO_WRITE : SOC_CQTYPE_IO_READ; } else { request->shdr.segcnt = 2; request->shdr.bytecnt = 0; request->data[2].base = 0; request->data[2].count = 0; request->type = SOC_CQTYPE_SIMPLE; } FILL_FCHDR_RCTL_DID(fch, R_CTL_COMMAND, fc->did); FILL_FCHDR_SID(fch, fc->sid); FILL_FCHDR_TYPE_FCTL(fch, TYPE_SCSI_FCP, F_CTL_FIRST_SEQ | F_CTL_SEQ_INITIATIVE); FILL_FCHDR_SEQ_DF_SEQ(fch, 0, 0, 0); FILL_FCHDR_OXRX(fch, 0xffff, 0xffff); fch->param = 0; request->shdr.flags = port->flags; request->shdr.class = 2; break; case PROTO_OFFLINE: memset (request, 0, sizeof(*request)); request->shdr.token = TOKEN(PROTO_OFFLINE, port->mask, fcmd->token); request->type = SOC_CQTYPE_OFFLINE; FILL_FCHDR_RCTL_DID(fch, R_CTL_COMMAND, fc->did); FILL_FCHDR_SID(fch, fc->sid); FILL_FCHDR_TYPE_FCTL(fch, TYPE_SCSI_FCP, F_CTL_FIRST_SEQ | F_CTL_SEQ_INITIATIVE); FILL_FCHDR_SEQ_DF_SEQ(fch, 0, 0, 0); FILL_FCHDR_OXRX(fch, 0xffff, 0xffff); request->shdr.flags = port->flags; break; case PROTO_REPORT_AL_MAP: /* SOC only supports Point-to-Point topology, no FC-AL, sorry... */ return -ENOSYS; default: request->shdr.token = TOKEN(fcmd->proto, port->mask, fcmd->token); request->shdr.class = 2; request->shdr.flags = port->flags; memcpy (fch, &fcmd->fch, sizeof(fc_hdr)); request->data[0].count = fcmd->cmdlen; request->data[1].count = fcmd->rsplen; request->type = fcmd->class; switch (fcmd->class) { case FC_CLASS_OUTBOUND: request->data[0].base = fcmd->cmd; request->data[0].count = fcmd->cmdlen; request->type = SOC_CQTYPE_OUTBOUND; request->shdr.bytecnt = fcmd->cmdlen; request->shdr.segcnt = 1; break; case FC_CLASS_INBOUND: request->data[0].base = fcmd->rsp; request->data[0].count = fcmd->rsplen; request->type = SOC_CQTYPE_INBOUND; request->shdr.bytecnt = 0; request->shdr.segcnt = 1; break; case FC_CLASS_SIMPLE: request->data[0].base = fcmd->cmd; request->data[1].base = fcmd->rsp; request->data[0].count = fcmd->cmdlen; request->data[1].count = fcmd->rsplen; request->type = SOC_CQTYPE_SIMPLE; request->shdr.bytecnt = fcmd->cmdlen; request->shdr.segcnt = 2; break; case FC_CLASS_IO_READ: case FC_CLASS_IO_WRITE: request->data[0].base = fcmd->cmd; request->data[1].base = fcmd->rsp; request->data[0].count = fcmd->cmdlen; request->data[1].count = fcmd->rsplen; request->type = (fcmd->class == FC_CLASS_IO_READ) ? SOC_CQTYPE_IO_READ : SOC_CQTYPE_IO_WRITE; if (fcmd->data) { request->data[2].base = fcmd->data; request->data[2].count = fcmd->datalen; request->shdr.bytecnt = fcmd->datalen; request->shdr.segcnt = 3; } else { request->shdr.bytecnt = 0; request->shdr.segcnt = 2; } break; }; break; }; request->count = 1; request->flags = 0; request->seqno = sw_cq->seqno; /* And now tell the SOC about it */ if (++sw_cq->in > sw_cq->last) { sw_cq->in = 0; sw_cq->seqno++; } SOD(("Putting %08x into cmd\n", SOC_CMD_RSP_QALL | (sw_cq->in << 24) | (SOC_CMD_REQ_Q0 << qno))) sbus_writel(SOC_CMD_RSP_QALL | (sw_cq->in << 24) | (SOC_CMD_REQ_Q0 << qno), s->regs + CMD); /* Read so that command is completed. */ sbus_readl(s->regs + CMD); return 0;}static inline void soc_download_fw(struct soc *s){#ifdef HAVE_SOC_UCODE xram_copy_to (s->xram, soc_ucode, sizeof(soc_ucode)); xram_bzero (s->xram + sizeof(soc_ucode), 32768 - sizeof(soc_ucode));#endif}/* Check for what the best SBUS burst we can use happens * to be on this machine. */static inline void soc_init_bursts(struct soc *s, struct sbus_dev *sdev){ int bsizes, bsizes_more; bsizes = (prom_getintdefault(sdev->prom_node,"burst-sizes",0xff) & 0xff); bsizes_more = (prom_getintdefault(sdev->bus->prom_node, "burst-sizes", 0xff) & 0xff); bsizes &= bsizes_more; if ((bsizes & 0x7f) == 0x7f) s->cfg = SOC_CFG_BURST_64; else if ((bsizes & 0x3f) == 0x3f) s->cfg = SOC_CFG_BURST_32; else if ((bsizes & 0x1f) == 0x1f) s->cfg = SOC_CFG_BURST_16; else s->cfg = SOC_CFG_BURST_4;}static inline void soc_init(struct sbus_dev *sdev, int no){ unsigned char tmp[60]; int propl; struct soc *s; static int version_printed = 0; soc_hw_cq cq[8]; int size, i; int irq; s = kmalloc (sizeof (struct soc), GFP_KERNEL); if (s == NULL) return; memset (s, 0, sizeof(struct soc)); s->soc_no = no; SOD(("socs %08lx soc_intr %08lx soc_hw_enque %08x\n", (long)socs, (long)soc_intr, (long)soc_hw_enque)) if (version_printed++ == 0) printk (version);#ifdef MODULE s->port[0].fc.module = &__this_module; s->port[1].fc.module = &__this_module;#else s->port[0].fc.module = NULL; s->port[1].fc.module = NULL;#endif s->next = socs; socs = s; s->port[0].fc.dev = sdev; s->port[1].fc.dev = sdev; s->port[0].s = s; s->port[1].s = s; s->port[0].fc.next = &s->port[1].fc; /* World Wide Name of SOC */ propl = prom_getproperty (sdev->prom_node, "soc-wwn", tmp, sizeof(tmp)); if (propl != sizeof (fc_wwn)) { s->wwn.naaid = NAAID_IEEE; s->wwn.lo = 0x12345678; } else memcpy (&s->wwn, tmp, sizeof (fc_wwn)); propl = prom_getproperty (sdev->prom_node, "port-wwns", tmp, sizeof(tmp)); if (propl != 2 * sizeof (fc_wwn)) { s->port[0].fc.wwn_nport.naaid = NAAID_IEEE_EXT; s->port[0].fc.wwn_nport.hi = s->wwn.hi; s->port[0].fc.wwn_nport.lo = s->wwn.lo; s->port[1].fc.wwn_nport.naaid = NAAID_IEEE_EXT; s->port[1].fc.wwn_nport.nportid = 1; s->port[1].fc.wwn_nport.hi = s->wwn.hi; s->port[1].fc.wwn_nport.lo = s->wwn.lo; } else { memcpy (&s->port[0].fc.wwn_nport, tmp, sizeof (fc_wwn)); memcpy (&s->port[1].fc.wwn_nport, tmp + sizeof (fc_wwn), sizeof (fc_wwn)); } memcpy (&s->port[0].fc.wwn_node, &s->wwn, sizeof (fc_wwn)); memcpy (&s->port[1].fc.wwn_node, &s->wwn, sizeof (fc_wwn)); SOD(("Got wwns %08x%08x ports %08x%08x and %08x%08x\n", *(u32 *)&s->port[0].fc.wwn_nport, s->port[0].fc.wwn_nport.lo, *(u32 *)&s->port[0].fc.wwn_nport, s->port[0].fc.wwn_nport.lo, *(u32 *)&s->port[1].fc.wwn_nport, s->port[1].fc.wwn_nport.lo)) s->port[0].fc.sid = 1; s->port[1].fc.sid = 17; s->port[0].fc.did = 2; s->port[1].fc.did = 18; s->port[0].fc.reset = soc_reset; s->port[1].fc.reset = soc_reset; if (sdev->num_registers == 1) { /* Probably SunFire onboard SOC */ s->xram = sbus_ioremap(&sdev->resource[0], 0, 0x10000UL, "soc xram"); s->regs = sbus_ioremap(&sdev->resource[0], 0x10000UL, 0x10UL, "soc regs"); } else { /* Probably SOC sbus card */ s->xram = sbus_ioremap(&sdev->resource[1], 0, sdev->reg_addrs[1].reg_size, "soc xram"); s->regs = sbus_ioremap(&sdev->resource[2], 0, sdev->reg_addrs[2].reg_size, "soc regs"); } soc_init_bursts(s, sdev); SOD(("Disabling SOC\n")) soc_disable (s); irq = sdev->irqs[0]; if (request_irq (irq, soc_intr, SA_SHIRQ, "SOC", (void *)s)) { soc_printk ("Cannot order irq %d to go\n", irq); socs = s->next; return; } SOD(("SOC uses IRQ%s\n", __irq_itoa(irq))) s->port[0].fc.irq = irq; s->port[1].fc.irq = irq; sprintf (s->port[0].fc.name, "soc%d port A", no); sprintf (s->port[1].fc.name, "soc%d port B", no); s->port[0].flags = SOC_FC_HDR | SOC_PORT_A; s->port[1].flags = SOC_FC_HDR | SOC_PORT_B; s->port[1].mask = (1 << 11); s->port[0].fc.hw_enque = soc_hw_enque; s->port[1].fc.hw_enque = soc_hw_enque; soc_download_fw (s); SOD(("Downloaded firmware\n")) /* Now setup xram circular queues */ memset (cq, 0, sizeof(cq)); size = (SOC_CQ_REQ0_SIZE + SOC_CQ_REQ1_SIZE) * sizeof(soc_req); s->req_cpu = sbus_alloc_consistent(sdev, size, &s->req_dvma); s->req[0].pool = s->req_cpu; cq[0].address = s->req_dvma; s->req[1].pool = s->req[0].pool + SOC_CQ_REQ0_SIZE; s->req[0].hw_cq = (soc_hw_cq *)(s->xram + SOC_CQ_REQ_OFFSET); s->req[1].hw_cq = (soc_hw_cq *)(s->xram + SOC_CQ_REQ_OFFSET + sizeof(soc_hw_cq)); s->rsp[0].hw_cq = (soc_hw_cq *)(s->xram + SOC_CQ_RSP_OFFSET); s->rsp[1].hw_cq = (soc_hw_cq *)(s->xram + SOC_CQ_RSP_OFFSET + sizeof(soc_hw_cq)); cq[1].address = cq[0].address + (SOC_CQ_REQ0_SIZE * sizeof(soc_req)); cq[4].address = 1; cq[5].address = 1; cq[0].last = SOC_CQ_REQ0_SIZE - 1; cq[1].last = SOC_CQ_REQ1_SIZE - 1; cq[4].last = SOC_CQ_RSP0_SIZE - 1; cq[5].last = SOC_CQ_RSP1_SIZE - 1; for (i = 0; i < 8; i++) cq[i].seqno = 1; s->req[0].last = SOC_CQ_REQ0_SIZE - 1; s->req[1].last = SOC_CQ_REQ1_SIZE - 1; s->rsp[0].last = SOC_CQ_RSP0_SIZE - 1; s->rsp[1].last = SOC_CQ_RSP1_SIZE - 1; s->req[0].seqno = 1; s->req[1].seqno = 1; s->rsp[0].seqno = 1; s->rsp[1].seqno = 1; xram_copy_to (s->xram + SOC_CQ_REQ_OFFSET, cq, sizeof(cq)); /* Make our sw copy of SOC service parameters */ xram_copy_from (s->serv_params, s->xram + 0x140, sizeof (s->serv_params)); s->port[0].fc.common_svc = (common_svc_parm *)s->serv_params; s->port[0].fc.class_svcs = (svc_parm *)(s->serv_params + 0x20); s->port[1].fc.common_svc = (common_svc_parm *)&s->serv_params; s->port[1].fc.class_svcs = (svc_parm *)(s->serv_params + 0x20); soc_enable (s); SOD(("Enabled SOC\n"))}#ifndef MODULEint __init soc_probe(void)#elseint init_module(void)#endif{ struct sbus_bus *sbus; struct sbus_dev *sdev = 0; struct soc *s; int cards = 0; for_each_sbus(sbus) { for_each_sbusdev(sdev, sbus) { if(!strcmp(sdev->prom_name, "SUNW,soc")) { soc_init(sdev, cards); cards++; } } } if (!cards) return -EIO; for_each_soc(s) if (s->next) s->port[1].fc.next = &s->next->port[0].fc; fcp_init (&socs->port[0].fc); return 0;}EXPORT_NO_SYMBOLS;#ifdef MODULEvoid cleanup_module(void){ struct soc *s; int irq; struct sbus_dev *sdev; for_each_soc(s) { irq = s->port[0].fc.irq; disable_irq (irq); free_irq (irq, s); fcp_release(&(s->port[0].fc), 2); sdev = s->port[0].fc.dev; if (sdev->num_registers == 1) { sbus_iounmap(s->xram, 0x10000UL); sbus_iounmap(s->regs, 0x10UL); } else { sbus_iounmap(s->xram, sdev->reg_addrs[1].reg_size); sbus_iounmap(s->regs, sdev->reg_addrs[2].reg_size); } sbus_free_consistent(sdev, (SOC_CQ_REQ0_SIZE+SOC_CQ_REQ1_SIZE)*sizeof(soc_req), s->req_cpu, s->req_dvma); }}#endif
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -