int sme_send(void *dev, char *buf, unsigned len) { struct local *l = dev; volatile struct desc *txd; unsigned txstat, loop; /* send a single frame with no T1_TER|T1_TCH designation */ wbinv(buf, len); txd = &l->txd[l->tx]; txd->xd2 = htole32(VTOPHYS(buf)); txd->xd1 = htole32(T1_FS | T1_LS | (len & T1_FL)); txd->xd0 = htole32(T0_OWN | (len & T0_FL) << 16); wbinv(txd, sizeof(struct desc)); CSR_WRITE(l, TXPOLLD, 01); /* start transmission */ loop = 100; do { txstat = le32toh(txd->xd0); if (txstat & T0_ES) break; if ((txstat & T0_OWN) == 0) goto done; DELAY(10); inv(txd, sizeof(struct desc)); } while (--loop != 0); printf("xmit failed\n"); return -1; done: l->tx ^= 1; return len; }
int kse_send(void *dev, char *buf, unsigned len) { struct local *l = dev; volatile struct desc *txd; unsigned txstat, loop; wbinv(buf, len); txd = &l->txd; txd->xd2 = htole32(VTOPHYS(buf)); txd->xd1 = htole32(T1_FS | T1_LS | (len & T1_TBS_MASK)); txd->xd0 = htole32(T0_OWN); wbinv(txd, sizeof(struct desc)); CSR_WRITE(l, MDTSC, 01); /* start transmission */ loop = 100; do { txstat = le32toh(txd->xd0); if ((txstat & T0_OWN) == 0) goto done; DELAY(10); inv(txd, sizeof(struct desc)); } while (--loop != 0); printf("xmit failed\n"); return -1; done: return len; }
int sme_recv(void *dev, char *buf, unsigned maxlen, unsigned timo) { struct local *l = dev; volatile struct desc *rxd; unsigned bound, rxstat, len; uint8_t *ptr; bound = 1000 * timo; printf("recving with %u sec. timeout\n", timo); again: rxd = &l->rxd[l->rx]; do { inv(rxd, sizeof(struct desc)); rxstat = le32toh(rxd->xd0); if ((rxstat & R0_OWN) == 0) goto gotone; DELAY(1000); /* 1 milli second */ } while (--bound > 0); errno = 0; return -1; gotone: if (rxstat & R0_ES) { rxd->xd0 = htole32(R0_OWN); wbinv(rxd, sizeof(struct desc)); l->rx ^= 1; CSR_WRITE(l, RXPOLLD, 01); /* restart receiving */ goto again; } /* good frame */ len = (rxstat & R0_FL) >> 16 /* no FCS included */; if (len > maxlen) len = maxlen; ptr = l->rxstore[l->rx]; inv(ptr, len); memcpy(buf, ptr, len); rxd->xd0 = htole32(R0_OWN); wbinv(rxd, sizeof(struct desc)); l->rx ^= 1; CSR_WRITE(l, RXPOLLD, 01); /* necessary? */ return len; }
void * sme_init(unsigned tag, void *data) { struct local *l; struct desc *txd, *rxd; unsigned mac32, mac16, val, fdx; uint8_t *en; l = ALLOC(struct local, 32); /* desc alignment */ memset(l, 0, sizeof(struct local)); l->csr = DEVTOV(pcicfgread(tag, 0x1c)); /* BAR3 mem space, LE */ l->phy = 1; /* 9420 internal PHY */ en = data; mac32 = CSR_READ(l, ADDRL); mac16 = CSR_READ(l, ADDRH); en[0] = mac32; en[1] = mac32 >> 8; en[2] = mac32 >> 16; en[3] = mac32 >> 24; en[4] = mac16; en[5] = mac16 >> 8; printf("MAC address %02x:%02x:%02x:%02x:%02x:%02x\n", en[0], en[1], en[2], en[3], en[4], en[5]); DPRINTF(("PHY %d (%04x.%04x)\n", l->phy, mii_read(l, l->phy, 2), mii_read(l, l->phy, 3))); mii_dealan(l, 5); /* speed and duplexity can be seen in MII 31 */ val = mii_read(l, l->phy, 31); fdx = !!(val & (1U << 4)); printf("%s", (val & (1U << 3)) ? "100Mbps" : "10Mbps"); if (fdx) printf("-FDX"); printf("\n"); txd = &l->txd[0]; rxd = &l->rxd[0]; rxd[0].xd0 = htole32(R0_OWN); rxd[0].xd1 = htole32(R1_RCH | FRAMESIZE); rxd[0].xd2 = htole32(VTOPHYS(l->rxstore[0])); rxd[0].xd3 = htole32(VTOPHYS(&rxd[1])); rxd[1].xd0 = htole32(R0_OWN); rxd[1].xd1 = htole32(R1_RER | FRAMESIZE); rxd[1].xd2 = htole32(VTOPHYS(l->rxstore[1])); /* R1_RER neglects xd3 */ l->tx = l->rx = 0; wbinv(l, sizeof(struct local)); CSR_WRITE(l, TXDBASE, VTOPHYS(txd)); CSR_WRITE(l, RXDBASE, VTOPHYS(rxd)); val = MACCR_TXEN | MACCR_RXEN; if (fdx) val |= MACCR_FDPX; CSR_WRITE(l, BUSMODE, 0); CSR_WRITE(l, DMACCTL, DMACCTL_ST | DMACCTL_SR); CSR_WRITE(l, MAC_CR, val); /* (FDX), Tx/Rx enable */ CSR_WRITE(l, RXPOLLD, 01); /* start receiving */ return l; }