ref: 2d101158e48e424a318b4027be2b3db72d057443
parent: efe1c9087a1deaadc8c541d699ff5f3f47ac1e5c
author: cinap_lenrek <[email protected]>
date: Tue Jul 9 14:52:01 EDT 2013
add erik quanstros yukon2 ethernet driver
--- /dev/null
+++ b/sys/src/9/pc/etheryuk.c
@@ -1,0 +1,2362 @@
+/*
+ * marvell 88e8057 yukon2
+ * copyright © 2009-10 erik quanstrom
+ */
+#include "u.h"
+#include "../port/lib.h"
+#include "mem.h"
+#include "dat.h"
+#include "fns.h"
+#include "io.h"
+#include "../port/error.h"
+#include "../port/netif.h"
+#include "etherif.h"
+
+#define Pciwaddrh(x) 0
+#define Pciwaddrl(x) PCIWADDR(x)
+#define is64() (sizeof(uintptr) == 8)
+#define dprint(...) if(debug) print(__VA_ARGS__); else {}
+
+extern void sfence(void);
+
+enum {
+ Nctlr = 4,
+ Nrb = 1024,
+ Rbalign = 64,
+ Fprobe = 1<<0,
+ Sringcnt = 2048,
+ Tringcnt = 512,
+// Rringcnt = Nrb,
+ Rringcnt = 512,
+ Rringl = Rringcnt - 8,
+};
+
+enum {
+ /* pci registers */
+ Pciphy = 0x40,
+ Pciclk = 0x80,
+ Pciasp = 0x84,
+ Pcistate = 0x88,
+ Pcicf0 = 0x90,
+ Pcicf1 = 0x94,
+
+ /* “csr” registers */
+ Ctst = 0x0004/2, /* control and status */
+ Pwrctl = 0x0007, /* power control */
+ Isr = 0x0008/4, /* interrupt src */
+ Ism = 0x000c/4, /* interrupt mask */
+ Hwe = 0x0010/4, /* hw error */
+ Hwem = 0x0014/4, /* hw error mask*/
+ Isrc2 = 0x001c/4,
+ Eisr = 0x0024/4,
+ Lisr = 0x0028/4, /* leave isr */
+ Macadr = 0x0100, /* mac address 2ports*3 */
+ Pmd = 0x0119,
+ Maccfg = 0x011a,
+ Chip = 0x011b,
+ Ramcnt = 0x011c, /* # of 4k blocks */
+ Hres = 0x011e,
+ Clkgate = 0x011d,
+ Clkctl = 0x0120/4,
+ Tstctl1 = 0x0158,
+ Tstctl2 = 0x0159,
+ Gpio = 0x015c/4,
+
+ Rictl = 0x01a0, /* ri ram buffer ctl */
+ Rib = 0x0190, /* ri buffer0 */
+
+ /* other unoffset registers */
+ Asfcs = 0x0e68, /* asf command and status */
+ Asfhost = 0x0e6c/4,
+
+ Statctl = 0x0e80/4, /* status */
+ Stattl = 0x0e84/2, /* tail (previous) status addr */
+ Stataddr = 0x0e88/4, /* status address low */
+ Statth = 0x0e98/2,
+ Stathd = 0x0e9c/2,
+ Statwm = 0x0eac, /* stat watermark */
+ Statiwm = 0x0ead, /* isr stat watermark */
+
+ Dpolltm = 0x0e08/4, /* descriptor pool timer */
+
+ /* timers */
+ Tgv = 0x0e14/4, /* gmac timer current value */
+ Tgc = 0x0e18, /* gmac timer ctl */
+ Tgt = 0x0e1a, /* gmac timer test */
+
+ Tsti = 0x0ec0/4, /* stat tx timer ini */
+ Tlti = 0x0eb0/4, /* level */
+ Titi = 0x0ed0/4, /* isr */
+
+ Tstc = 0x0ec8, /* stat tx timer ctl */
+ Tltc = 0x0eb8, /* level timer ctl */
+ Titc = 0x0ed8, /* isr timer ctl */
+
+ /* “gmac” registers */
+ Stat = 0x000/2,
+ Ctl = 0x004/2,
+ Txctl = 0x008/2,
+ Rxctl = 0x00c/2,
+ Txflow = 0x010/2,
+ Txparm = 0x014/2,
+ Serctl = 0x018/2, /* serial mode */
+ Mchash = 0x034/2, /* 4 registers; 4 bytes apart */
+
+ /* interrupt sources and masks */
+ Txirq = 0x044/2,
+ Rxirq = 0x048/2,
+ Trirq = 0x04c/2, /* tx/rx overflow irq source */
+ Txmask = 0x050/2,
+ Rxmask = 0x054/2,
+ Trmask = 0x058/2,
+
+ Smictl = 0x080/2, /* serial mode control */
+ Smidata = 0x084/2,
+ Phyaddr = 0x088/2,
+
+ Ea0 = 0x01c/2, /* 3 16 bit gmac registers */
+ Ea1 = 0x028/2,
+
+ Stats = 0x0100/4,
+
+ /* mac registers */
+ Txactl = 0x210, /* transmit arbiter ctl */
+
+ Grxea = 0x0c40/4, /* rx fifo end address */
+ Gfrxctl = 0x0c48/4, /* gmac rxfifo ctl */
+ Grxfm = 0x0c4c/4, /* fifo flush mask */
+ Grxft = 0x0c50/4, /* fifo flush threshold */
+ Grxtt = 0x0c54/4, /* rx truncation threshold */
+ Gmfea = 0x0d40/4, /* end address */
+ Gmfae = 0x0d44/4, /* almost empty thresh */
+ Gmfctl = 0x0d48/4, /* tx gmac fifo ctl */
+
+ Rxphi = 0x0c58, /* pause high watermark */
+ Rxplo = 0x0c5c, /* pause low watermark */
+
+ Rxwp = 0x0c60/4,
+ Rxwlev = 0x0c68/4,
+ Rxrp = 0x0c70/4,
+ Rxrlev = 0x0c78/4,
+
+ Mac = 0x0f00/4, /* global mac control */
+ Phy = 0x0f04/4, /* phy control register */
+
+ Irq = 0x0f08, /* irq source */
+ Irqm = 0x0f0c, /* irq mask */
+ Linkctl = 0x0f10,
+
+ /* queue registers; all offsets from Qbase*/
+ Qbase = 0x0400,
+ Qportsz = 0x0080, /* BOTCH; tx diff is 2x rx diff */
+
+ Qr = 0x000,
+ Qtxs = 0x200,
+ Qtx = 0x280,
+
+ /* queue offsets */
+ Qd = 0x00,
+ Qvlan = 0x20,
+ Qdone = 0x24,
+ Qaddrl = 0x28,
+ Qaddrh = 0x2c,
+ Qbc = 0x30,
+ Qcsr = 0x34, /* 32bit */
+ Qtest = 0x38,
+ Qwm = 0x40,
+
+ /* buffer registers; all offsets from Rbase */
+ Rbase = 0x0800,
+
+ Rstart = 0x00,
+ Rend = 0x04,
+ Rwp = 0x08,
+ Rrp = 0x0c,
+ Rpon = 0x10, /* pause frames on */
+ Rpoff = 0x14, /* pause frames off */
+ Rhon = 0x18, /* high-priority frames on */
+ Rhoff = 0x1c, /* high-priority frames off */
+ Rctl = 0x28,
+
+ /* prefetch */
+ Pbase = 0x450,
+ Pctl = 0x00,
+ Plidx = 0x04, /* last addr; 16 bit */
+ Paddrl = 0x08,
+ Paddrh = 0x0c,
+ Pgetidx = 0x10, /* 16 bit */
+ Pputidx = 0x14, /* 16 bit */
+ Pfifow = 0x20, /* 8 bit */
+ Pfifor = 0x24, /* 8 bit */
+ Pfifowm = 0x20, /* 8 bit */
+
+ /* indirect phy registers */
+ Phyctl = 0x000,
+ Phystat = 0x001,
+ Phyid0 = 0x002,
+ Phyid1 = 0x003,
+ Phyana = 0x004, /* auto neg advertisement */
+ Phylpa = 0x005, /* link partner ability */
+ Phyanee = 0x006, /* auto neg adv expansion */
+ Phynp = 0x007, /* next page */
+ Phylnp = 0x008, /* link partner next page */
+ Gbectl = 0x009,
+ Gbestat = 0x00a,
+ Phyphy = 0x010, /* phy specific ctl */
+ Phylstat = 0x011,
+ Phyintm = 0x012, /* phy interrupt mask */
+ Phyint = 0x013,
+ Phyextctl = 0x014,
+ Phyrxe = 0x015, /* rx error counter */
+ Phypage = 0x016, /* external address */
+ Phypadr = 0x01d, /* phy page address */
+};
+
+enum {
+ /* Pciasp */
+ Aspforce = 1<<15,
+ Aspglinkdn = 1<<14, /* gphy link down */
+ Aspfempty = 1<<13,
+ Aspclkrun = 1<<12,
+ Aspmsk = Aspforce | Aspglinkdn | Aspfempty | Aspclkrun,
+
+ /* Pcistate */
+ Vmain = 3<<27,
+
+ /* Stat */
+ Sfast = 1<<15, /* 100mbit */
+ Duplex = 1<<14,
+ Txnofc = 1<<13, /* tx flow control disabled */
+ Link = 1<<12, /* link up */
+ Pausest = 1<<11, /* pause state */
+ Txactive = 1<<10,
+ Excesscol = 1<<9,
+ Latecol = 1<<8,
+ Physc = 1<<5, /* phy status change */
+ Sgbe = 1<<4, /* gbe speed */
+ Rxnofc = 1<<2, /* rx flow control disabled */
+ Promisc = 1<<1, /* promiscuous mode enabled */
+
+ /* Ctl */
+ Promiscen = 1<<14,
+ Txfcdis = 1<<13,
+ Txen = 1<<12,
+ Rxen = 1<<11,
+ Bursten = 1<<10,
+ Loopen = 1<<9,
+ Gbeen = 1<<7,
+ Fpass = 1<<6, /* "force link pass" ? */
+ Duplexen = 1<<5,
+ Rxfcdis = 1<<4,
+ Fasten = 1<<3, /* enable 100mbit */
+ Adudis = 1<<2, /* disable auto upd duplex */
+ Afcdis = 1<<1, /* disable auto upd flow ctl */
+ Aspddis = 1<<0, /* disable auto upd speed */
+
+ /* Rxctl */
+ Ufilter = 1<<15, /* unicast filter */
+ Mfilter = 1<<14, /* multicast filter */
+ Rmcrc = 1<<13, /* remove frame crc */
+
+ /* Serctl */
+ Vlanen = 1<<9,
+ Jumboen = 1<<8,
+
+ /* Txactl */
+ Txaclr = 1<<1,
+ Txarst = 1<<0,
+
+ /* Asfcs: yukex only */
+ Asfbrrst = 1<<9, /* bridge reset */
+ Asfcpurst = 1<<8, /* cpu reset */
+ Asfucrst = 3<<0, /* µctlr reset */
+
+ /* Asfcs */
+ Asfhvos = 1<<4, /* os present */
+ Asfrst = 1<<3,
+ Asfrun = 1<<2,
+ Asfcirq = 1<<1,
+ Afsirq = 1<<0,
+
+ /* Statctl */
+ Statirqclr = 1<<4,
+ Staton = 1<<3,
+ Statoff = 1<<2,
+ Statclr = 1<<1,
+ Statrst = 1<<0,
+
+ /* Mac */
+ Nomacsec = 1<<13 | 1<<11,
+ Nortx = 1<<9,
+ Macpause = 1<<3,
+ Macpauseoff = 1<<2,
+ Macrstclr = 1<<1,
+ Macrst = 1<<0,
+
+ /* Phy */
+ Gphyrstclr = 1<<1,
+ Gphyrst = 1<<0,
+
+ /* Irqm */
+ Txovfl = 1<<5, /* tx counter overflow */
+ Rxovfl = 1<<4, /* rx counter overflow */
+ Txurun = 1<<3, /* transmit fifo underrun */
+ Txdone = 1<<2, /* frame tx done */
+ Rxorun = 1<<1, /* rx fifo overrun */
+ Rxdone = 1<<0, /* frame rx done */
+
+ /* Linkctl */
+ Linkclr = 1<<1,
+ Linkrst = 1<<0,
+
+ /* Smictl */
+ Smiread = 1<<5,
+ Smiwrite = 0<<5,
+ Smirdone = 1<<4,
+ Smibusy = 1<<3,
+
+ /* Phyaddr */
+ Mibclear = 1<<5,
+
+ /* Ctst */
+ Asfdis = 1<<12, /* asf disable */
+ Clken = 1<<11, /* enable clock */
+
+ Swirq = 1<<7,
+ Swirqclr = 1<<6,
+ Mstopped = 1<<5, /* master is stopped */
+ Mstop = 1<<4, /* stop master */
+ Mstrclr = 1<<3, /* master reset clear */
+ Mstrrset = 1<<2, /* master reset */
+ Swclr = 1<<1,
+ Swrst = 1<<0,
+
+ /* Pwrctl */
+ Vauxen = 1<<7,
+ Vauxdis = 1<<6,
+ Vccen = 1<<5,
+ Vccdis = 1<<4,
+ Vauxon = 1<<3,
+ Vauxoff = 1<<2,
+ Vccon = 1<<1,
+ Vccoff = 1<<0,
+
+ /* timers */
+ Tstart = 1<<2,
+ Tstop = 1<<1,
+ Tclrirq = 1<<0,
+
+ /* Dpolltm */
+ Pollstart = 1<<1,
+ Pollstop = 1<<0,
+
+ /* csr interrupts: Isrc2, Eisr, etc. */
+ Ihwerr = 1<<31,
+ Ibmu = 1<<30, /* sring irq */
+ Isoftware = 1<<25,
+
+ Iphy = 1<<4,
+ Imac = 1<<3,
+ Irx = 1<<2,
+ Itxs = 1<<1, /* descriptor error */
+ Itx = 1<<0, /* descriptor error */
+
+ Iport = 0x1f,
+ Iphy2base = 8,
+ Ierror = (Imac | Itx | Irx)*(1 | 1<<Iphy2base),
+
+ /* hwe interrupts: Hwe Hwem */
+ Htsof = 1<<29, /* timer stamp overflow */
+ Hsensor = 1<<28,
+ Hmerr = 1<<27, /* master error */
+ Hstatus = 1<<26, /* status exception */
+ Hpcie = 1<<25, /* pcie error */
+ Hpcie2 = 1<<24, /* " */
+
+ Hrparity = 1<<5, /* ram read parity error */
+ Hwparity = 1<<4, /* ram write parity error */
+ Hmfault = 1<<3, /* mac fault */
+ Hrxparity = 1<<2, /* rx parity */
+ Htcptxs = 1<<1, /* tcp length mismatch */
+ Htcptxa = 1<<0, /* tcp length mismatch */
+
+ H1base = 1<<0,
+ H2base = 1<<8,
+ Hmask = 0x3f,
+ Hdflt = Htsof | Hmerr | Hstatus | Hmask*(H1base | H2base),
+
+ /* Clkctl */
+ Clkdiven = 1<<1,
+ Clkdivdis = 1<<0,
+
+ /* Clkgate */
+ Link2inactive = 1<<7,
+
+ /* Phyctl */
+ Phyrst = 1<<15,
+ Phy100 = 1<<14, /* manual enable 100mbit */
+ Aneen = 1<<12, /* auto negotiation enable */
+ Phyoff = 1<<11, /* turn phy off */
+ Anerst = 1<<9, /* auto neg reset */
+ Phydpx = 1<<8,
+ Phy1000 = 1<<5, /* manual enable gbe */
+
+ /* Phyana */
+ Annp = 1<<15, /* request next page */
+ Anack = 1<<14, /* ack rx (read only) */
+ Anrf = 1<<13, /* remote fault */
+ Anpa = 1<<11, /* try asymmetric pause */
+ Anp = 1<<10, /* try pause */
+ An100f = 1<<8,
+ An100h = 1<<7,
+ An10f = 1<<6,
+ An10h = 1<<5,
+ Anonly = 1<<0,
+ Anall = An100f | An100h | An10f | An10h | Anonly,
+
+ /* Gbectl */
+ Gbef = 1<<9, /* auto neg gbe full */
+ Gbeh = 1<<8, /* auto neg gbe half */
+ Gbexf = 1<<6, /* auto neg gbe full fiber */
+ Gbexh = 1<<5, /* auto neg gbe full fiber */
+
+ /* Phyphy */
+ Pptf = 3<<14, /* tx fifo depth */
+ Pprf = 3<<12, /* rx fifo depth */
+ Pped = 3<<8, /* energy detect */
+ Ppmdix = 3<<5, /* mdix conf */
+ Ppmdixa = 3<<5, /* automdix */
+
+ Ppengy = 1<<14, /* fe+ enable energy detect */
+ Ppscrdis = 1<<9, /* fe+ scrambler disable */
+ Ppnpe = 1<<12, /* fe+ enable next page */
+
+ /* Phylstat */
+ Physpd = 3<<14,
+ Phydupx = 1<<13,
+ Phypr = 1<<12, /* page rx */
+ Phydone = 1<<11, /* speed and duplex neg. done */
+ Plink = 1<<10,
+ Pwirelen = 7<<7,
+ Pmdi = 1<<6,
+ Pdwnsh = 1<<5, /* downshift */
+ Penergy = 1<<4, /* energy detect */
+ Ptxpause = 1<<3, /* tx pause enabled */
+ Prxpause = 1<<2, /* rx pause enabled */
+ Ppol = 1<<2, /* polarity */
+ Pjarjar = 1<<1, /* mesa no understasa */
+
+ /* Phyintm */
+ Anerr = 1<<15, /* an error */
+ Lsp = 1<<14, /* link speed change */
+ Andc = 1<<13, /* an duplex change */
+ Anok = 1<<11,
+ Lsc = 1<<10, /* link status change */
+ Symerr = 1<<9, /* symbol error */
+ Fcarr = 1<<8, /* false carrier */
+ Fifoerr = 1<<7,
+ Mdich = 1<<6,
+ Downsh = 1<<5,
+ Engych = 1<<4, /* energy change */
+ Dtech = 1<<2, /* dte power det status */
+ Polch = 1<<1, /* polarity change */
+ Jabber = 1<<0,
+
+ /* Phyextctl */
+ Dnmstr = 1<<9, /* master downshift; 0: 1x; 1: 2x; 2: 3x */
+ Dnslv = 1<<8,
+
+ /* Tgc */
+ Tgstart = 1<<2,
+ Tgstop = 1<<1,
+ Tgclr = 1<<0, /* clear irq */
+
+ /* Tstctl1 */
+ Tstwen = 1<<1, /* enable config reg r/w */
+ Tstwdis = 1<<0, /* disable config reg r/w */
+
+ /* Gpio */
+ Norace = 1<<13,
+
+ /* Rictl */
+ Rirpclr = 1<<9,
+ Riwpclr = 1<<8,
+ Riclr = 1<<1,
+ Rirst = 1<<0,
+
+ /* Rbase opcodes */
+ Rsfon = 1<<5, /* enable store/fwd */
+ Rsfoff = 1<<4,
+ Renable = 1<<3,
+ Rdisable = 1<<2,
+ Rrstclr = 1<<1,
+ Rrst = 1<<0,
+
+ /* Qbase opcodes */
+ Qidle = 1<<31,
+ Qtcprx = 1<<30,
+ Qiprx = 1<<29,
+ Qrssen = 1<<15,
+ Qrssdis = 1<<14,
+ Qsumen = 1<<13, /* tcp/ip cksum */
+ Qsumdis = 1<<12,
+ Qcirqpar = 1<<11, /* clear irq on parity errors */
+ Qcirqck = 1<<10,
+ Qstop = 1<<9,
+ Qstart = 1<<8,
+ Qfifoon = 1<<7,
+ Qfifooff = 1<<6,
+ Qfifoen = 1<<5,
+ Qfiforst = 1<<4,
+ Qenable = 1<<3,
+ Qdisable = 1<<2,
+ Qrstclr = 1<<1,
+ Qrst = 1<<0,
+
+ Qallclr = Qfiforst | Qfifooff | Qrstclr,
+ Qgo = Qcirqpar | Qcirqck | Qstart | Qfifoen | Qenable,
+
+ /* Qtest bits */
+ Qckoff = 1<<31, /* tx: auto checksum off */
+ Qckon = 1<<30,
+ Qramdis = 1<<24, /* rx: ram disable */
+
+ /* Pbase opcodes */
+ Prefon = 1<<3, /* prefetch on */
+ Prefoff = 1<<2,
+ Prefrstclr = 1<<1,
+ Prefrst = 1<<0,
+
+ /* ring opcodes */
+ Hw = 0x80, /* bitmask */
+ Ock = 0x12, /* tcp checksum start */
+ Oaddr64 = 0x21,
+ Obuf = 0x40,
+ Opkt = 0x41,
+ Orxstat = 0x60,
+ Orxts = 0x61, /* rx timestamp */
+ Orxvlan = 0x62,
+ Orxchks = 0x64,
+ Otxidx = 0x68,
+ Omacs = 0x6c, /* macsec */
+ Oputidx = 0x70,
+
+ /* ring status */
+ Eop = 0x80,
+
+ /* Gfrxctl */
+ Gftrunc = 1<<27,
+ Gftroff = 1<<26,
+
+ Gfroon = 1<<19, /* flush on rx overrun */
+ Gfrooff = 1<<18,
+ Gffon = 1<<7, /* rx fifo flush mode on */
+ Gffoff = 1<<6,
+ Gfon = 1<<3,
+ Gfoff = 1<<2,
+ Gfrstclr = 1<<1,
+ Gfrst = 1<<0,
+
+ /* Gmfctl */
+ Gmfsfoff = 1<<31, /* disable store-forward (ec ultra) */
+ Gmfsfon = 1<<30, /* able store-forward (ec ultra) */
+ Gmfvon = 1<<25, /* vlan tag on */
+ Gmfvoff = 1<<24, /* vlan off */
+ Gmfjon = 1<<23, /* jumbo on (ec ultra) */
+ Gmfjoff = 1<<22, /* jumbo off */
+ Gmfcfu = 1<<6, /* clear fifio underrun irq */
+ Gmfcfc = 1<<5, /* clear frame complete irq */
+ Gmfcpe = 1<<4, /* clear parity error irq */
+ Gmfon = 1<<3,
+ Gmfoff = 1<<2,
+ Gmfclr = 1<<1,
+ Gmfrst = 1<<0,
+
+ /* rx frame */
+ Flen = 0x7fff<<17,
+ Fvlan = 1<<13,
+ Fjabbr = 1<<12,
+ Ftoosm = 1<<11,
+ Fmc = 1<<10, /* multicast */
+ Fbc = 1<<9,
+ Fok = 1<<8, /* good frame */
+ Fokfc = 1<<7,
+ Fbadfc = 1<<6,
+ Fmiierr = 1<<5,
+ Ftoobg = 1<<4, /* oversized */
+ Ffrag = 1<<3, /* fragment */
+ Fcrcerr = 1<<1,
+ Ffifoof = 1<<0, /* fifo overflow */
+ Ferror = Ffifoof | Fcrcerr | Ffrag | Ftoobg
+ | Fmiierr | Fbadfc | Ftoosm | Fjabbr,
+
+ /* rx checksum bits in Status.ctl */
+ Badck = 5, /* arbitrary bad checksum */
+
+ Ctcpok = 1<<7, /* tcp or udp cksum ok */
+ Cisip6 = 1<<3,
+ Cisip4 = 1<<1,
+
+ /* more status ring rx bits */
+ Rxvlan = 1<<13,
+ Rxjab = 1<<12, /* jabber */
+ Rxsmall = 1<<11, /* too small */
+ Rxmc = 1<<10, /* multicast */
+ Rxbc = 1<<9, /* bcast */
+ Rxok = 1<<8,
+ Rxfcok = 1<<7, /* flow control pkt */
+ Rxfcbad = 1<<6,
+ Rxmiierr = 1<<5,
+ Rxbig = 1<<4, /* too big */
+ Rxfrag = 1<<3,
+ Rxcrcerr = 1<<1,
+ Rxfov = 1<<0, /* fifo overflow */
+ Rxerror = Rxfov | Rxcrcerr | Rxfrag | Rxbig | Rxmiierr
+ | Rxfcbad | Rxsmall | Rxjab,
+};
+
+enum {
+ Ffiber = 1<<0,
+ Fgbe = 1<<1,
+ Fnewphy = 1<<2,
+ Fapwr = 1<<3,
+ Fnewle = 1<<4,
+ Fram = 1<<5,
+ Fancy =Fgbe | Fnewphy | Fapwr,
+
+ Yukxl = 0,
+ Yukecu,
+ Yukex,
+ Yukec,
+ Yukfe,
+ Yukfep,
+ Yuksup,
+ Yukul2,
+ Yukba, /* doesn't exist */
+ Yukopt,
+ Nyuk,
+};
+
+typedef struct Chipid Chipid;
+typedef struct Ctlr Ctlr;
+typedef void (*Freefn)(Block*);
+typedef struct Kproc Kproc;
+typedef struct Mc Mc;
+typedef struct Stattab Stattab;
+typedef struct Status Status;
+typedef struct Sring Sring;
+typedef struct Vtab Vtab;
+
+struct Chipid {
+ uchar feat;
+ uchar okrev;
+ uchar mhz;
+ char *name;
+};
+
+struct Kproc {
+ Rendez;
+ uint event;
+};
+
+struct Sring {
+ uint wp;
+ uint rp;
+ uint cnt;
+ uint m;
+ Status *r;
+};
+
+struct Ctlr {
+ Pcidev *p;
+ Ctlr *oport; /* port 2 */
+ uchar qno;
+ uchar attach;
+ uchar rxinit;
+ uchar txinit;
+ uchar flag;
+ uchar feat;
+ uchar type;
+ uchar rev;
+ uchar nports;
+ uchar portno;
+ uintptr io;
+ uchar *reg8;
+ ushort *reg16;
+ uint *reg;
+ uint rbsz;
+ uchar ra[Eaddrlen];
+ uint mca;
+ uint nmc;
+ Mc *mc;
+ void *alloc;
+ Sring status;
+ Sring tx;
+ Block *tbring[Tringcnt];
+ Sring rx;
+ Block *rbring[Rringcnt];
+ Kproc txmit;
+ Kproc rxmit;
+ Kproc iproc;
+};
+
+struct Mc {
+ Mc *next;
+ uchar ea[Eaddrlen];
+};
+
+struct Stattab {
+ uint offset;
+ char *name;
+};
+
+struct Status {
+ uchar status[4];
+ uchar l[2];
+ uchar ctl;
+ uchar op;
+};
+
+struct Vtab {
+ int vid;
+ int did;
+ int mtu;
+ char *name;
+};
+
+static Chipid idtab[] = {
+[Yukxl] Fgbe | Fnewphy, 0xff, 156, "yukon-2 xl",
+[Yukecu] Fancy, 0xff, 125, "yukon-2 ec ultra",
+[Yukex] Fancy | Fnewle, 0xff, 125, "yukon-2 extreme",
+[Yukec] Fgbe, 2, 125, "yukon-2 ec",
+[Yukfe] 0, 0xff, 100, "yukon-2 fe",
+[Yukfep] Fnewphy|Fapwr | Fnewle, 0xff, 50, "yukon-2 fe+",
+[Yuksup] Fgbe | Fnewphy | Fnewle, 0xff, 125, "yukon-2 supreme",
+[Yukul2] Fgbe |Fapwr, 0xff, 125, "yukon-2 ultra2",
+[Yukba] 0, 0, 0, "??",
+[Yukopt] Fancy, 0xff, 125, "yukon-2 optima",
+};
+
+static Vtab vtab[] = {
+ 0x11ab, 0x4354, 1514, "88e8040", /* unsure on mtu */
+ 0x11ab, 0x4362, 1514, "88e8053",
+ 0x11ab, 0x4364, 1514, "88e8056",
+ 0x11ab, 0x4380, 1514, "88e8057",
+ 0x11ab, 0x436b, 1514, "88e8071", /* unsure on mtu */
+ 0x1186, 0x4b00, 9000, "dge-560t",
+ 0x1186, 0x4b02, 1514, "dge-550sx",
+ 0x1186, 0x4b03, 1514, "dge-550t",
+};
+
+static Stattab stattab[] = {
+ 0, "rx ucast",
+ 8, "rx bcast",
+ 16, "rx pause",
+ 24, "rx mcast",
+ 32, "rx chk seq",
+
+ 48, "rx ok low",
+ 56, "rx ok high",
+ 64, "rx bad low",
+ 72, "rx bad high",
+
+ 80, "rx frames < 64",
+ 88, "rx frames < 64 fcs",
+ 96, "rx frames 64",
+ 104, "rx frames 65-127",
+ 112, "rx frames 128-255",
+ 120, "rx frames 256-511",
+ 128, "rx frames 512-1023",
+ 136, "rx frames 1024-1518",
+ 144, "rx frames 1519-mtu",
+ 152, "rx frames too long",
+ 160, "rx jabber",
+ 176, "rx fifo oflow",
+
+ 192, "tx ucast",
+ 200, "tx bcast",
+ 208, "tx pause",
+ 216, "tx mcast",
+
+ 224, "tx ok low",
+ 232, "tx ok hi",
+
+ 240, "tx frames 64",
+ 248, "tx frames 65-127",
+ 256, "tx frames 128-255",
+ 264, "tx frames 256-511",
+ 272, "tx frames 512-1023",
+ 280, "tx frames 1024-1518",
+ 288, "tx frames 1519-mtu",
+
+ 304, "tx coll",
+ 312, "tx late coll",
+ 320, "tx excess coll",
+ 328, "tx mul col",
+ 336, "tx single col",
+ 344, "tx underrun",
+};
+
+static uint phypwr[] = {1<<26, 1<<27};
+static uint coma[] = {1<<28, 1<<29};
+static uchar nilea[Eaddrlen];
+static int debug;
+static Ctlr *ctlrtab[Nctlr];
+static int nctlr;
+static struct {
+ union {
+ struct {
+ Lock;
+ Block *b;
+ uint starve;
+ };
+ uchar pad[128]; /* cacheline */
+ };
+ Kproc *k;
+ Block *x;
+ uint nfast;
+ uint nslow;
+} rbtab[Nctlr];
+
+static int
+icansleep(void *v)
+{
+ Kproc *k;
+
+ k = v;
+ return k->event != 0;
+}
+
+static void
+unstarve(Kproc *k)
+{
+ k->event = 1;
+ wakeup(k);
+}
+
+static void
+starve(Kproc *k)
+{
+ sleep(k, icansleep, k);
+ k->event = 0;
+}
+
+static Status*
+getslot(Sring *r, Kproc *k)
+{
+ if(r->rp + r->m - r->wp & ~r->m)
+ starve(k);
+ return r->r + (r->wp++ & r->m);
+}
+
+static int
+getnslot(Sring *r, uint *wp, Status **t, uint n)
+{
+ int i;
+
+ if(r->rp + r->m - (n - 1) - wp[0] & ~r->m)
+ return -1;
+ for(i = 0; i < n; i++)
+ t[i] = r->r + (wp[0]++ & r->m);
+ return 0;
+}
+
+/* assume allocs come from a single thread; 30*0.999x speedup */
+static Block*
+rballoc(int t)
+{
+ Block *b;
+
+ if((b = rbtab[t].x) != nil){
+ rbtab[t].nfast++;
+ rbtab[t].x = b->next;
+ b->next = nil;
+b->ref = 1;
+ return b;
+ }
+
+ ilock(&rbtab[t]);
+ b = rbtab[t].x = rbtab[t].b;
+ rbtab[t].b = nil;
+ if(b == nil)
+ rbtab[t].starve = 1;
+ iunlock(&rbtab[t]);
+
+ rbtab[t].nslow++;
+ if(b != nil){
+ rbtab[t].x = b->next;
+ b->next = nil;
+b->ref = 1;
+ }
+ return b;
+}
+
+static void
+rbfree(Block *b, int t)
+{
+ b->rp = b->wp = (uchar*)ROUND((uintptr)b->base, Rbalign);
+ b->flag &= ~(Bipck | Budpck | Btcpck | Bpktck);
+ ilock(&rbtab[t]);
+ b->next = rbtab[t].b;
+ if(b->next == nil && rbtab[t].starve){
+ rbtab[t].starve = 0;
+ unstarve(rbtab[t].k);
+ }
+ rbtab[t].b = b;
+ iunlock(&rbtab[t]);
+}
+
+static void
+rbfree0(Block *b)
+{
+ rbfree(b, 0);
+}
+
+static void
+rbfree1(Block *b)
+{
+ rbfree(b, 1);
+}
+
+static void
+rbfree2(Block *b)
+{
+ rbfree(b, 2);
+}
+
+static void
+rbfree3(Block *b)
+{
+ rbfree(b, 3);
+}
+
+static Freefn freetab[Nctlr] = {
+ rbfree0,
+ rbfree1,
+ rbfree2,
+ rbfree3,
+};
+
+static uint
+macread32(Ctlr *c, uint r)
+{
+ return c->reg[c->portno*0x20 + r];
+}
+
+static void
+macwrite32(Ctlr *c, uint r, uint v)
+{
+ c->reg[c->portno*0x20 + r] = v;
+}
+
+static uint
+macread16(Ctlr *c, uint r)
+{
+ return c->reg16[c->portno*0x40 + r];
+}
+
+static void
+macwrite16(Ctlr *c, uint r, uint v)
+{
+ c->reg16[c->portno*0x40 + r] = v;
+}
+
+static uint
+macread8(Ctlr *c, uint r)
+{
+ return c->reg8[c->portno*0x80 + r];
+}
+
+static void
+macwrite8(Ctlr *c, uint r, uint v)
+{
+ c->reg8[c->portno*0x80 + r] = v;
+}
+
+static uint gmac32[2] = {
+ 0x2800/4,
+ 0x3800/4,
+};
+
+static ushort
+gmacread32(Ctlr *c, uint r)
+{
+ return c->reg[gmac32[c->portno] + r];
+}
+
+static void
+gmacwrite32(Ctlr *c, uint r, uint v)
+{
+ c->reg[gmac32[c->portno] + r] = v;
+}
+
+static uint gmac[2] = {
+ 0x2800/2,
+ 0x3800/2,
+};
+
+static ushort
+gmacread(Ctlr *c, uint r)
+{
+ return c->reg16[gmac[c->portno] + r];
+}
+
+static void
+gmacwrite(Ctlr *c, uint r, ushort v)
+{
+ c->reg16[gmac[c->portno] + r] = v;
+}
+
+static uint
+qrread(Ctlr *c, uint r)
+{
+ return c->reg[Qbase + c->portno*Qportsz + r>>2];
+}
+
+static void
+qrwrite(Ctlr *c, uint r, uint v)
+{
+ c->reg[Qbase + c->portno*Qportsz + r>>2] = v;
+}
+
+static uint
+qrread16(Ctlr *c, uint r)
+{
+ return c->reg16[Qbase + c->portno*Qportsz + r>>1];
+}
+
+static void
+qrwrite16(Ctlr *c, uint r, uint v)
+{
+ c->reg16[Qbase + c->portno*Qportsz + r>>1] = v;
+}
+
+static uint
+qrread8(Ctlr *c, uint r)
+{
+ return c->reg8[Qbase + c->portno*Qportsz + r>>0];
+}
+
+static void
+qrwrite8(Ctlr *c, uint r, uint v)
+{
+ c->reg8[Qbase + c->portno*Qportsz + r>>0] = v;
+}
+
+static uint
+rrread32(Ctlr *c, uint r)
+{
+ return c->reg[Rbase + c->portno*Qportsz + r>>2];
+}
+
+static void
+rrwrite32(Ctlr *c, uint r, uint v)
+{
+ c->reg[Rbase + c->portno*Qportsz + r>>2] = v;
+}
+
+static void
+rrwrite8(Ctlr *c, uint r, uint v)
+{
+ c->reg8[Rbase + c->portno*Qportsz + r] = v;
+}
+
+static uint
+rrread8(Ctlr *c, uint r)
+{
+ return c->reg8[Rbase + c->portno*Qportsz + r];
+}
+
+static uint
+prread32(Ctlr *c, uint r)
+{
+ return c->reg[Pbase + c->portno*Qportsz + r>>2];
+}
+
+static void
+prwrite32(Ctlr *c, uint r, uint v)
+{
+ c->reg[Pbase + c->portno*Qportsz + r>>2] = v;
+}
+
+static uint
+prread16(Ctlr *c, uint r)
+{
+ return c->reg16[Pbase + c->portno*Qportsz + r>>1];
+}
+
+static void
+prwrite16(Ctlr *c, uint r, uint v)
+{
+ c->reg16[Pbase + c->portno*Qportsz + r>>1] = v;
+}
+
+static ushort
+phyread(Ctlr *c, uint r)
+{
+ ushort v;
+
+ gmacwrite(c, Smictl, Smiread | r<<6);
+ for(;;){
+ v = gmacread(c, Smictl);
+ if(v == 0xffff)
+ error("phy read");
+ if(v & Smirdone)
+ return gmacread(c, Smidata);
+ microdelay(10);
+ }
+}
+
+static ushort
+phywrite(Ctlr *c, uint r, ushort v)
+{
+ gmacwrite(c, Smidata, v);
+ gmacwrite(c, Smictl, Smiwrite | r<<6);
+ for(;;){
+ v = gmacread(c, Smictl);
+ if(v == 0xffff)
+ error("phy write");
+ if((v & Smibusy) == 0)
+ return gmacread(c, Smidata);
+ microdelay(10);
+ }
+}
+
+static uvlong lorder = 0x0706050403020100ull;
+
+static uvlong
+getle(uchar *t, int w)
+{
+ uint i;
+ uvlong r;
+
+ r = 0;
+ for(i = w; i != 0; )
+ r = r<<8 | t[--i];
+ return r;
+}
+
+static void
+putle(uchar *t, uvlong r, int w)
+{
+ uchar *o, *f;
+ uint i;
+
+ f = (uchar*)&r;
+ o = (uchar*)&lorder;
+ for(i = 0; i < w; i++)
+ t[o[i]] = f[i];
+}
+
+static void
+bufinit(Ctlr *c, uint q, uint start, uint end)
+{
+ uint t;
+
+ rrwrite8(c, q + Rctl, Rrstclr);
+ rrwrite32(c, q + Rstart, start);
+ rrwrite32(c, q + Rend, end-1);
+ rrwrite32(c, q + Rwp, start);
+ rrwrite32(c, q + Rrp, start);
+
+ if(q == Qr || q == Qr + Qportsz){
+ t = start-end;
+ rrwrite32(c, q + Rpon, t - 8192/8);
+ rrwrite32(c, q + Rpoff, t - 16384/8);
+ } else
+ rrwrite8(c, q + Rctl, Rsfon);
+ rrwrite8(c, q + Rctl, Renable);
+ rrread8(c, q + Rctl);
+}
+
+static void
+qinit(Ctlr *c, uint queue)
+{
+ qrwrite(c, queue + Qcsr, Qallclr);
+ qrwrite(c, queue + Qcsr, Qgo);
+ qrwrite(c, queue + Qcsr, Qfifoon);
+ qrwrite16(c, queue + Qwm, 0x600); /* magic */
+// qrwrite16(c, queue + Qwm, 0x80); /* pcie magic; assume pcie; no help */
+}
+
+/* initialized prefetching */
+static void
+pinit(Ctlr *c, uint queue, Sring *r)
+{
+ union {
+ uchar u[4];
+ uint l;
+ } u;
+
+ prwrite32(c, queue + Pctl, Prefrst);
+ prwrite32(c, queue + Pctl, Prefrstclr);
+ putle(u.u, Pciwaddrh(r->r), 4);
+ prwrite32(c, queue + Paddrh, u.l);
+ putle(u.u, Pciwaddrl(r->r), 4);
+ prwrite32(c, queue + Paddrl, u.l);
+ prwrite16(c, queue + Plidx, r->m);
+ prwrite32(c, queue + Pctl, Prefon);
+ prread32(c, queue + Pctl);
+}
+
+static void
+txinit(Ether *e)
+{
+ Ctlr *c;
+ Sring *r;
+
+ c = e->ctlr;
+ r = &c->tx;
+ if(c->txinit == 1)
+ return;
+ c->txinit = 1;
+ r->wp = 0;
+ r->rp = 0;
+ qinit(c, Qtx);
+ pinit(c, Qtx, &c->tx);
+}
+
+static void
+linkup(Ctlr *c, uint w)
+{
+ static Lock l;
+
+ lock(&l);
+ gmacwrite(c, Ctl, w|gmacread(c, Ctl));
+ unlock(&l);
+}
+
+static void
+tproc(void *v)
+{
+ Block *b;
+ Ctlr *c;
+ Ether *e;
+ Kproc *k;
+ Sring *r;
+ Status *t;
+
+ e = v;
+ c = e->ctlr;
+ k = &c->txmit;
+ r = &c->tx;
+
+ txinit(e);
+ linkup(c, Txen);
+ for(;;){
+ if((b = qbread(e->oq, 100000)) == nil)
+ break;
+ if(Pciwaddrh(b->rp) != 0){
+ t = getslot(r, k);
+ t->ctl = 0;
+ t->op = Oaddr64 | Hw;
+ putle(t->status, Pciwaddrh(b->rp), 4);
+ }
+ t = getslot(r, k);
+ c->tbring[t - r->r] = b;
+ putle(t->status, Pciwaddrl(b->rp), 4);
+ putle(t->l, BLEN(b), 2);
+ t->op = Opkt | Hw;
+ t->ctl = Eop;
+ sfence();
+ prwrite16(c, Qtx + Pputidx, r->wp & r->m);
+ }
+ print("#l%d: tproc: queue closed\n", e->ctlrno);
+ pexit("queue closed", 1);
+}
+
+static void
+rxinit(Ether *e)
+{
+ int i;
+ Ctlr *c;
+ Block *b;
+ Sring *r;
+ Status *t;
+
+ c = e->ctlr;
+ r = &c->rx;
+ if(c->rxinit == 1)
+ return;
+ c->rxinit = 1;
+ for(i = 0; i < Nrb; i++){
+ b = allocb(c->rbsz + Rbalign);
+ b->free = freetab[c->qno];
+ freeb(b);
+ }
+
+ qinit(c, Qr);
+ if(c->type == Yukecu && (c->rev == 2 || c->rev == 3))
+ qrwrite(c, Qr + Qtest, Qramdis);
+ pinit(c, Qr, &c->rx);
+
+ if((c->flag & Fnewle) == 0){
+ t = getslot(r, &c->rxmit);
+ putle(t->status, 14<<16 | 14, 4);
+ t->ctl = 0;
+ t->op = Ock | Hw;
+ qrwrite(c, Qr + Qcsr, Qsumen);
+ }
+ macwrite32(c, Gfrxctl, Gftroff);
+}
+
+/* debug; remove */
+#include "yukdump.h"
+static int
+rxscrew(Ether *e, Sring *r, Status *t, uint wp)
+{
+ Ctlr *c;
+
+ c = e->ctlr;
+ if(wp - r->rp > r->cnt){
+ print("rxscrew1 wp %ud(%ud) rp %ud %lud\n", wp, r->wp, r->rp, t-r->r);
+ return -1;
+ }
+ if(c->rbring[t - r->r]){
+ print("rxscrew2 wp %ud rp %ud %lud\n", wp, r->rp, t-r->r);
+ descriptorfu(e, Qr);
+ return -1;
+ }
+ return 0;
+}
+
+static int
+replenish(Ether *e, Ctlr *c)
+{
+ int req, n, lim;
+ uint wp;
+ Block *b;
+ Sring *r;
+ Status *tab[2], *t;
+
+ r = &c->rx;
+ wp = r->wp;
+ req = 1 + is64();
+
+ lim = r->cnt/2;
+ if(lim > 128)
+ lim = 128; /* hw limit? */
+ for(n = 0; n < lim; n++){
+ b = rballoc(c->qno);
+ if(b == nil || getnslot(r, &wp, tab, req) == -1){
+ freeb(b);
+ break;
+ }
+ t = tab[0];
+ if(is64()){
+ putle(t->status, Pciwaddrh(b->wp), 4);
+ t->ctl = 0;
+ t->op = Oaddr64 | Hw;
+ t = tab[1];
+ }
+ if(rxscrew(e, r, t, wp) == -1)
+ break;
+ assert(c->rbring[t - r->r] == nil);
+ c->rbring[t - r->r] = b;
+
+ putle(t->status, Pciwaddrl(b->wp), 4);
+ putle(t->l, c->rbsz, 2);
+ t->ctl = 0;
+ t->op = Opkt | Hw;
+ }
+ if(n>0){
+ sfence();
+ prwrite16(c, Qr + Pputidx, wp & r->m);
+ r->wp = wp;
+ dprint("yuk: replenish %d %ud-%ud [%d-%d]\n", n, r->rp, wp, r->rp&r->m, wp&r->m);
+ }
+ return lim - n == 0;
+}
+
+static void
+rproc(void *v)
+{
+ Ctlr *c;
+ Ether *e;
+ Kproc *k;
+
+ e = v;
+ c = e->ctlr;
+ k = &c->rxmit;
+
+ rxinit(e);
+ linkup(c, Rxen);
+ for(;;)
+ if(replenish(e, c) == 0){
+ starve(k);
+ print("yuk: rx unstarve?\n");
+ }
+}
+
+static void
+promiscuous(void *a, int on)
+{
+ uint r;
+ Ether *e;
+ Ctlr *c;
+
+ e = a;
+ c = e->ctlr;
+ r = gmacread(c, Rxctl);
+ if(on)
+ r &= ~(Ufilter|Mfilter);
+ else
+ r |= Ufilter|Mfilter;
+ gmacwrite(c, Rxctl, r);
+}
+
+static uchar pauseea[] = {1, 0x80, 0xc2, 0, 0, 1};
+
+static void
+multicast(void *a, uchar *ea, int on)
+{
+ uchar f[8];
+ uint i, r, b;
+ Ctlr *c;
+ Ether *e;
+ Mc **ll, *l, *p;
+
+ e = a;
+ c = e->ctlr;
+ r = gmacread(c, Rxctl);
+ if(on){
+ for(ll = &c->mc; *ll != nil; ll = &(*ll)->next)
+ if(memcmp((*ll)->ea, ea, Eaddrlen) == 0)
+ return;
+ *ll = malloc(sizeof **ll);
+ memmove((*ll)->ea, ea, Eaddrlen);
+ }else{
+ for(p = nil, l = c->mc; l != nil; p = l, l = l->next)
+ if(memcmp(l->ea, ea, Eaddrlen) == 0)
+ break;
+ if(l == nil)
+ return;
+ if(p != nil)
+ p->next = l->next;
+ else
+ c->mc = l->next;
+ free(l);
+ }
+ memset(f, 0, sizeof f);
+ if(0 /* flow control */){
+ b = ethercrc(pauseea, Eaddrlen) & 0x3f;
+ f[b>>3] |= 1 << (b & 7);
+ }
+ for(l = c->mc; l != nil; l = l->next){
+ b = ethercrc(l->ea, Eaddrlen) & 0x3f;
+ f[b>>3] |= 1 << (b & 7);
+ }
+ for(i = 0; i < sizeof f / 2; i++)
+ gmacwrite(c, Mchash + 2*i, f[i] | f[i+1]<<8);
+ gmacwrite(c, Rxctl, r | Mfilter);
+}
+
+static int spdtab[4] = {
+ 10, 100, 1000, 0,
+};
+
+static void
+link(Ether *e)
+{
+ uint i, s, spd;
+ Ctlr *c;
+
+ c = e->ctlr;
+ i = phyread(c, Phyint);
+ s = phyread(c, Phylstat);
+ dprint("#l%d: yuk: link %.8ux %.8ux\n", e->ctlrno, i, s);
+ spd = 0;
+ e->link = (s & Plink) != 0;
+ if(e->link && c->feat&Ffiber)
+ spd = 1000;
+ else if(e->link){
+ spd = s & Physpd;
+ spd >>= 14;
+ spd = spdtab[spd];
+ }
+ e->mbps = spd;
+ dprint("#l%d: yuk: link %d spd %d\n", e->ctlrno, e->link, e->mbps);
+}
+
+static void
+txcleanup(Ctlr *c, uint end)
+{
+ uint rp0, rp;
+ Block *b;
+ Sring *r;
+ Status *t;
+
+ r = &c->tx;
+ rp0 = r->rp & r->m;
+ for(rp = rp0; rp != end; rp = r->rp & r->m){
+ t = r->r + rp;
+ r->rp++;
+ if((t->ctl & Eop) == 0)
+ continue;
+ b = c->tbring[rp];
+ c->tbring[rp] = nil;
+ if(b != nil)
+ freeb(b);
+ }
+ if(r->wp - r->rp > 16){ /* BOTCH */
+ print("TX unstarve %ud - %ud \n", r->wp, r->rp);
+ unstarve(&c->txmit);
+ }
+}
+
+static void
+rx(Ether *e, uint l, uint x, uint flag)
+{
+ uint cnt, i, rp;
+ Block *b;
+ Ctlr *c;
+ Sring *r;
+
+ c = e->ctlr;
+ r = &c->rx;
+ for(rp = r->rp;;){
+ i = rp++&r->m;
+ b = c->rbring[i];
+ c->rbring[i] = nil;
+ if(b != nil)
+ break;
+ }
+ cnt = x>>16 & 0x7fff;
+ if(cnt != l || x&Rxerror &&
+ !(c->type == Yukfep && c->rev == 0)){
+ print("#l%d: yuk rx error %.4ux\n", e->ctlrno, x&0xffff);
+ freeb(b);
+ }else{
+ b->wp += l;
+ b->lim = b->wp; /* lie like a dog */
+ b->flag |= flag;
+ etheriq(e, b, 1);
+ }
+ r->rp = rp;
+}
+
+static uint
+cksum(Ctlr *c, uint ck, uint css)
+{
+ if(c->flag & Fnewle && css&(Cisip4|Cisip6) && css&Ctcpok)
+ return Bipck | Btcpck | Budpck;
+ else if(ck == 0xffff || ck == 0)
+ return Bipck;
+ return 0;
+}
+
+static void
+sring(Ether *e)
+{
+ uint i, p, lim, op, l, x;
+ Ctlr *c;
+ Sring *r;
+ Status *s;
+ static uint ck = Badck;
+
+ c = e->ctlr;
+ r = &c->status;
+ lim = r->rp & r->m;
+ p = 0;
+ for(;;){
+ if((r->rp & r->m) == lim){
+ lim = c->reg16[Stathd];
+ if((r->rp & r->m) == lim)
+ break;
+ }
+ i = r->rp & r->m;
+ s = r->r + i;
+ op = s->op;
+ if((op & Hw) == 0)
+ break;
+ op &= ~Hw;
+ switch(op){
+ case Orxchks:
+ ck = getle(s->status, 4) & 0xffff;
+ break;
+ case Orxstat:
+ l = getle(s->l, 2);
+ x = getle(s->status, 4);
+ rx(e, l, x, cksum(c, ck, s->ctl));
+ ck = Badck;
+ p++;
+ break;
+ case Otxidx:
+ l = getle(s->l, 2);
+ x = getle(s->status, 4);
+ txcleanup(c, x & 0xfff);
+
+ x = l>>24 & 0xff | l<< 8;
+ x &= 0xfff;
+ if(x != 0 && c->oport)
+ txcleanup(c->oport, x);
+ break;
+ default:
+ print("#l%d: yuk: funny opcode %.2ux\n", e->ctlrno, op);
+ break;
+ }
+ s->op = 0;
+ r->rp++;
+ }
+ while(p && replenish(e, c) != 0)
+ ;
+ c->reg[Statctl] = Statirqclr;
+}
+
+enum {
+ Pciaer = 0x1d00,
+ Pciunc = 0x0004,
+};
+
+static void
+hwerror(Ether *e, uint cause)
+{
+ uint u;
+ Ctlr *c;
+
+ c = e->ctlr;
+ cause = c->reg[Hwe];
+ if(cause == 0)
+ print("hwe: no cause\n");
+ if(cause & Htsof){
+ c->reg8[Tgc] = Tgclr;
+ cause &= ~Htsof;
+ }
+ if(cause & (Hmerr | Hstatus)){
+ c->reg8[Tstctl1] = Tstwen;
+ u = pcicfgr16(c->p, PciPSR) | 0x7800;
+ pcicfgw16(c->p, PciPSR, u);
+ c->reg8[Tstctl1] = Tstwdis;
+ cause &= ~(Hmerr | Hstatus);
+ }
+ if(cause & Hpcie){
+ c->reg8[Tstctl1] = Tstwen;
+ c->reg[Pciaer + Pciunc>>2] = ~0;
+ u = c->reg[Pciaer + Pciunc>>2];
+ USED(u);
+ print("#l%d: pcierror %.8ux\n", e->ctlrno, u);
+ c->reg8[Tstctl1] = Tstwdis;
+ cause &= ~Hpcie;
+ }
+ if(cause & Hrxparity){
+ print("#l%d: ram parity read error. bug? ca %.8ux\n", e->ctlrno, cause);
+ qrwrite(c, Qtx + Qcsr, Qcirqpar);
+ cause &= ~Hrxparity;
+ }
+ if(cause & Hrparity){
+ print("#l%d: ram parity read error. bug? ca %.8ux\n", e->ctlrno, cause);
+ descriptorfu(e, Qr);
+ descriptorfu(e, Qtx);
+ c->reg16[Rictl + c->portno*0x40>>1] = Rirpclr;
+ cause &= ~Hrparity;
+ }
+ if(cause & Hwparity){
+ print("#l%d: ram parity write error. bug? ca %.8ux\n", e->ctlrno, cause);
+ descriptorfu(e, Qr);
+ descriptorfu(e, Qtx);
+ c->reg16[Rictl + c->portno*0x40>>1] = Riwpclr;
+ cause &= ~Hwparity;
+ }
+ if(cause & Hmfault){
+ print("#l%d: mac parity error\n", e->ctlrno);
+ macwrite32(c, Gmfctl, Gmfcpe);
+ cause &= ~Hmfault;
+ }
+ if(cause)
+ print("#l%d: leftover hwe %.8ux\n", e->ctlrno, cause);
+}
+
+static void
+macintr(Ether *e)
+{
+ uint cause;
+ Ctlr *c;
+
+ c = e->ctlr;
+ cause = macread8(c, Irq);
+ cause &= ~(Rxdone | Txdone);
+ if(cause == 0)
+ return;
+ print("#l%d: mac error %.8ux\n", e->ctlrno, cause);
+ if(cause & Txovfl){
+ gmacread32(c, Txirq);
+ cause &= ~Txovfl;
+ }
+ if(cause & Rxovfl){
+ gmacread32(c, Rxirq);
+ cause &= ~Rxovfl;
+ }
+ if(cause & Rxorun){
+ macwrite32(c, Gfrxctl, Gmfcfu);
+ cause &= ~Rxorun;
+ }
+ if(cause & Txurun){
+ macwrite32(c, Gmfctl, Gmfcfu);
+ cause &= ~Txurun;
+ }
+ if(cause)
+ print("#l%d: leftover mac error %.8ux\n", e->ctlrno, cause);
+}
+
+static struct {
+ uint i;
+ uint q;
+ char *s;
+} emap[] = {
+ Irx, Qr, "qr",
+ Itxs, Qtxs, "qtxs",
+ Itx, Qtx, "qtx",
+ Irx<<Iphy2base, Qr + 0x80, "qr1",
+ Itxs<<Iphy2base, Qtxs + 0x100, "qtxs1",
+ Itx<<Iphy2base, Qtx + 0x100, "qtx1",
+};
+
+static void
+eerror(Ether *e, uint cause)
+{
+ uint i, o, q;
+ Ctlr *c;
+
+ c = e->ctlr;
+
+ if(cause & Imac){
+ macintr(e);
+ cause &= ~Imac;
+ }
+ if(cause & (Irx | Itxs | Itx)*(1 | 1<<Iphy2base))
+ for(i = 0; i < nelem(emap); i++){
+ if((cause & emap[i].i) == 0)
+ continue;
+ q = emap[i].q;
+ o = prread16(c, q + Pgetidx);
+ print("#l%d: yuk: bug: %s: @%d ca=%.8ux\n",
+ e->ctlrno, emap[i].s, o, cause);
+ descriptorfu(e, q);
+ qrwrite(c, emap[i].q + Qcsr, Qcirqck);
+ cause &= ~emap[i].i;
+ }
+ if(cause)
+ print("#l%d: leftover error %.8ux\n", e->ctlrno, cause);
+}
+
+static void
+iproc(void *v)
+{
+ uint cause, d;
+ Ether *e;
+ Ctlr *c;
+ Kproc *k;
+
+ e = v;
+ c = e->ctlr;
+ k = &c->iproc;
+
+ for(;;){
+ starve(k);
+ cause = c->reg[Eisr];
+ if(cause & Iphy)
+ link(e);
+ if(cause & Ihwerr)
+ hwerror(e, cause);
+ if(cause & Ierror)
+ eerror(e, cause & Ierror);
+ if(cause & Ibmu)
+ sring(e);
+ d = c->reg[Lisr];
+ USED(d);
+ }
+}
+
+static void
+interrupt(Ureg*, void *v)
+{
+ uint cause;
+ Ctlr *c;
+ Ether *e;
+
+ e = v;
+ c = e->ctlr;
+
+ cause = c->reg[Isrc2];
+ if(cause != 0 && cause != ~0)
+ unstarve(&c->iproc);
+}
+
+static void
+storefw(Ctlr *c)
+{
+ if(c->type == Yukex && c->rev != 1
+ || c->type == Yukfep
+ || c->type == Yuksup)
+ macwrite32(c, Gmfctl, Gmfjon | Gmfsfon);
+ else{
+ macwrite32(c, Gmfae, 0x8000 | 0x70); /* tx gmac fifo */
+ macwrite32(c, Gmfctl, Gmfsfoff);
+ }
+}
+
+static void
+raminit(Ctlr *c)
+{
+ uint ram, rx;
+
+ if(ram = c->reg8[Ramcnt] * 4096/8){ /* in qwords */
+ c->flag |= Fram;
+ rx = ROUNDUP((2*ram)/3, 1024/8);
+ bufinit(c, Qr, 0, rx);
+ bufinit(c, Qtx, rx, ram);
+ rrwrite8(c, Qtxs + Rctl, Rrst); /* sync tx off */
+ }else{
+ macwrite8(c, Rxplo, 768/8);
+ macwrite8(c, Rxphi, 1024/8);
+ storefw(c);
+ }
+}
+
+static void
+attach(Ether *e)
+{
+ char buf[KNAMELEN];
+ Ctlr *c;
+ static Lock l;
+
+ c = e->ctlr;
+ if(c->attach == 1)
+ return;
+ lock(&l);
+ if(c->attach == 1){
+ unlock(&l);
+ return;
+ }
+ c->attach = 1;
+ unlock(&l);
+
+ snprint(buf, sizeof buf, "#l%dtproc", e->ctlrno);
+ kproc(buf, tproc, e);
+ snprint(buf, sizeof buf, "#l%drproc", e->ctlrno);
+ kproc(buf, rproc, e);
+ snprint(buf, sizeof buf, "#l%diproc", e->ctlrno);
+ kproc(buf, iproc, e);
+
+ c->reg[Ism] |= Ibmu | Iport<<Iphy2base*c->portno;
+}
+
+static long
+ifstat(Ether *e0, void *a, long n, ulong offset)
+{
+ char *s, *e, *p;
+ int i;
+ uint u;
+ Ctlr *c;
+
+ c = e0->ctlr;
+ p = s = malloc(READSTR);
+ e = p + READSTR;
+ for(i = 0; i < nelem(stattab); i++){
+ u = gmacread32(c, Stats + stattab[i].offset/4);
+ if(u > 0)
+ p = seprint(p, e, "%s\t%ud\n", stattab[i].name, u);
+ }
+ p = seprint(p, e, "stat %.4ux ctl %.3ux\n", gmacread(c, Stat), gmacread(c, Ctl));
+ p = seprint(p, e, "irq %.8ux\n", c->reg[Isrc2]);
+ p = seprint(p, e, "pref %.8ux %.4ux\n", prread32(c, Qr + Pctl), prread16(c, Qr + Pgetidx));
+ p = seprint(p, e, "nfast %ud nslow %ud\n", rbtab[c->qno].nfast, rbtab[c->qno].nslow);
+ if(debug){
+ p = dumppci(c, p, e);
+ p = dumpgmac(c, p, e);
+ p = dumpmac(c, p, e);
+ p = dumpreg(c, p, e);
+ }
+ seprint(p, e, "%s rev %d phy %s\n", idtab[c->type].name,
+ c->rev, c->feat&Ffiber? "fiber": "copper");
+ n = readstr(offset, a, n, s);
+ free(s);
+ return n;
+}
+
+static Cmdtab ctltab[] = {
+ 1, "debug", 1,
+ 2, "descriptorfu", 1,
+};
+
+static long
+ctl(Ether *e, void *buf, long n)
+{
+ Cmdbuf *cb;
+ Cmdtab *t;
+
+ cb = parsecmd(buf, n);
+ if(waserror()){
+ free(cb);
+ nexterror();
+ }
+ t = lookupcmd(cb, ctltab, nelem(ctltab));
+ switch(t->index){
+ case 0:
+ debug ^= 1;
+ break;
+ case 1:
+ descriptorfu(e, Qr);
+ break;
+ }
+ free(cb);
+ poperror();
+ return n;
+}
+
+static uint
+yukpcicfgr32(Ctlr *c, uint r)
+{
+ return c->reg[r + 0x1c00>>2];
+}
+
+static void
+yukpcicfgw32(Ctlr *c, uint r, uint v)
+{
+ c->reg[r + 0x1c00>>2] = v;
+}
+
+static void
+phypower(Ctlr *c)
+{
+ uint u, u0;
+
+ u = u0 = yukpcicfgr32(c, Pciphy);
+ u &= ~phypwr[c->portno];
+ if(c->type == Yukxl && c->rev > 1)
+ u |= coma[c->portno];
+ if(u != u0 || 1){
+ c->reg8[Tstctl1] = Tstwen;
+ yukpcicfgw32(c, Pciphy, u);
+ c->reg8[Tstctl1] = Tstwdis;
+ }
+ if(c->type == Yukfe)
+ c->reg8[Phyctl] = Aneen;
+ else if(c->flag & Fapwr)
+ macwrite32(c, Phy, Gphyrstclr);
+}
+
+static void
+phyinit(Ctlr *c)
+{
+ uint u;
+
+ if((c->feat & Fnewphy) == 0){
+ u = phyread(c, Phyextctl);
+ u &= ~0xf70; /* clear downshift counters */
+ u |= 0x7<<4; /* mac tx clock = 25mhz */
+ if(c->type == Yukec)
+ u |= 2*Dnmstr | Dnslv;
+ else
+ u |= Dnslv;
+ phywrite(c, Phyextctl, u);
+ }
+ u = phyread(c, Phyphy);
+
+ /* questionable value */
+ if(c->feat & Ffiber)
+ u &= ~Ppmdix;
+ else if(c->feat & Fgbe){
+ u &= ~Pped;
+ u |= Ppmdixa;
+ if(c->flag & Fnewphy){
+ // u &= ~(7<<12);
+ // u |= 2*(1<<12) | 1<<11; /* like 2*Dnmstr | Dnslv */
+ u |= 2*(1<<9) | 1<<11;
+ }
+ }else{
+ u |= Ppmdixa >> 1; /* why the shift? */
+ if(c->type == Yukfep && c->rev == 0){
+ }
+ }
+
+ phywrite(c, Phyphy, u);
+ /* copper/fiber specific stuff gmacwrite(c, Ctl, 0); */
+ gmacwrite(c, Ctl, 0);
+ if(c->feat & Fgbe)
+ if(c->feat & Ffiber)
+ phywrite(c, Gbectl, Gbexf | Gbexh);
+ else
+ phywrite(c, Gbectl, Gbef | Gbeh);
+ phywrite(c, Phyana, Anall);
+ phywrite(c, Phyctl, Phyrst | Anerst | Aneen);
+ /* chip specific stuff? */
+ if (c->type == Yukfep){
+ u = phyread(c, Phyphy) | Ppnpe;
+ u &= ~(Ppengy | Ppscrdis);
+ phywrite(c, Phyphy, u);
+// phywrite(c, 0x16, 0x0b54); /* write to fe_led_par */
+
+ /* yukfep and rev 0: apply workaround for integrated resistor calibration */
+ phywrite(c, Phypadr, 17);
+ phywrite(c, 0x1e, 0x3f60);
+ }
+ phywrite(c, Phyintm, Anok | Anerr | Lsc);
+ dprint("phyid %.4ux step %.4ux\n", phyread(c, 2), phyread(c, 3));
+}
+
+static int
+identify(Ctlr *c)
+{
+ char t;
+
+ pcicfgw32(c->p, Pciclk, 0);
+ c->reg16[Ctst] = Swclr;
+
+ c->type = c->reg8[Chip] - 0xb3;
+ c->rev = c->reg8[Maccfg]>>4 & 0xf;
+ if(c->type >= Nyuk)
+ return -1;
+ if(idtab[c->type].okrev != 0xff)
+ if(c->rev != idtab[c->type].okrev)
+ return -1;
+ c->feat |= idtab[c->type].feat;
+
+ t = c->reg8[Pmd];
+ if(t == 'L' || t == 'S' || t == 'P')
+ c->feat |= Ffiber;
+ c->portno = 0;
+ /* check second port ... whatever */
+ return 0;
+}
+
+static uint
+µ2clk(Ctlr *c, int µs)
+{
+ return idtab[c->type].mhz * µs;
+}
+
+static void
+gmacsetea(Ctlr *c, uint r)
+{
+ uchar *ra;
+ int i;
+
+ ra = c->ra;
+ for(i = 0; i < Eaddrlen; i += 2)
+ gmacwrite(c, r + i, ra[i + 0] | ra[i + 1]<<8);
+}
+
+static int
+reset(Ctlr *c)
+{
+ uint i, j;
+ Block *b;
+
+ identify(c);
+
+ if(c->type == Yukex)
+ c->reg16[Asfcs/2] &= ~(Asfbrrst | Asfcpurst | Asfucrst);
+ else
+ c->reg8[Asfcs] = Asfrst;
+ c->reg16[Ctst] = Asfdis;
+
+ c->reg16[Ctst] = Swrst;
+ c->reg16[Ctst] = Swclr;
+
+ c->reg8[Tstctl1] = Tstwen;
+ pcicfgw16(c->p, PciPSR, pcicfgr16(c->p, PciPSR) | 0xf100);
+ c->reg16[Ctst] = Mstrclr;
+ /* fixup pcie extended error goes here */
+
+ c->reg8[Pwrctl] = Vauxen | Vccen | Vauxoff | Vccon;
+ c->reg[Clkctl] = Clkdivdis;
+ if(c->type == Yukxl && c->rev > 1)
+ c->reg8[Clkgate] = ~Link2inactive;
+ else
+ c->reg8[Clkgate] = 0;
+ if(c->flag & Fapwr){
+ pcicfgw32(c->p, Pciclk, 0);
+ pcicfgw32(c->p, Pciasp, pcicfgr32(c->p, Pciasp) & Aspmsk);
+ pcicfgw32(c->p, Pcistate, pcicfgr32(c->p, Pcistate) & Vmain);
+ pcicfgw32(c->p, Pcicf1, 0);
+ c->reg[Gpio] |= Norace;
+ print("yuk2: advanced power %.8ux\n", c->reg[Gpio]);
+ }
+ c->reg8[Tstctl1] = Tstwdis;
+
+ for(i = 0; i < c->nports; i++){
+ macwrite8(c, Linkctl, Linkrst);
+ macwrite8(c, Linkctl, Linkclr);
+ if(c->type == Yukex || c->type == Yuksup)
+ macwrite16(c, Mac, Nomacsec | Nortx);
+ }
+
+ c->reg[Dpolltm] = Pollstop;
+
+ for(i = 0; i < c->nports; i++)
+ macwrite8(c, Txactl, Txaclr);
+ for(i = 0; i < c->nports; i++){
+ c->reg8[i*64 + Rictl] = Riclr;
+ for(j = 0; j < 12; j++)
+ c->reg8[i*64 + Rib + j] = 36; /* qword times */
+ }
+
+ c->reg[Hwem] = Hdflt;
+ macwrite8(c, Irqm, 0);
+ for(i = 0; i < 4; i++)
+ gmacwrite(c, Mchash + 2*i, 0);
+ gmacwrite(c, Rxctl, Ufilter | Mfilter | Rmcrc);
+
+ for(i = 0; i < nelem(c->tbring); i++)
+ if(b = c->tbring[i]){
+ c->tbring[i] = nil;
+ freeb(b);
+ }
+ for(i = 0; i < nelem(c->rbring); i++)
+ if(b = c->rbring[i]){
+ c->rbring[i] = nil;
+ freeb(b);
+ }
+
+ memset(c->tbring, 0, sizeof c->tbring[0] * nelem(c->tbring));
+ memset(c->rbring, 0, sizeof c->rbring[0] * nelem(c->rbring));
+ memset(c->tx.r, 0, sizeof c->tx.r[0] * c->tx.cnt);
+ memset(c->rx.r, 0, sizeof c->rx.r[0] * c->rx.cnt);
+ memset(c->status.r, 0, sizeof c->status.r[0] * c->status.cnt);
+ c->reg[Statctl] = Statrst;
+ c->reg[Statctl] = Statclr;
+ c->reg[Stataddr + 0] = Pciwaddrl(c->status.r);
+ c->reg[Stataddr + 4] = Pciwaddrh(c->status.r);
+ c->reg16[Stattl] = c->status.m;
+ c->reg16[Statth] = 10;
+ c->reg8[Statwm] = 16;
+ if(c->type == Yukxl && c->rev == 0)
+ c->reg8[Statiwm] = 4;
+ else
+ c->reg8[Statiwm] = 4; //16;
+
+ /* set transmit, isr, level timers */
+ c->reg[Tsti] = µ2clk(c, 1000);
+ c->reg[Titi] = µ2clk(c, 20);
+ c->reg[Tlti] = µ2clk(c, 100);
+
+ c->reg[Statctl] = Staton;
+
+ c->reg8[Tstc] = Tstart;
+ c->reg8[Tltc] = Tstart;
+ c->reg8[Titc] = Tstart;
+
+ return 0;
+}
+
+static void
+macinit(Ctlr *c)
+{
+ uint r, i;
+
+ r = macread32(c, Phy) & ~(Gphyrst | Gphyrstclr);
+ macwrite32(c, Phy, r | Gphyrst);
+ macwrite32(c, Phy, r | Gphyrstclr);
+ /* macwrite32(c, Mac, Macrst); ? */
+ macwrite32(c, Mac, Macrstclr);
+
+ if(c->type == Yukxl && c->rev == 0 && c->portno == 1){
+ }
+
+ macread8(c, Irq);
+ macwrite8(c, Irqm, Txurun);
+
+ phypower(c);
+ phyinit(c);
+
+ gmacwrite(c, Phyaddr, (r = gmacread(c, Phyaddr)) | Mibclear);
+ for(i = 0; i < nelem(stattab); i++)
+ gmacread32(c, Stats + stattab[i].offset/4);
+ gmacwrite(c, Phyaddr, r);
+
+ gmacwrite(c, Txctl, 4<<10); /* collision distance */
+ gmacwrite(c, Txflow, 0xffff); /* flow control */
+ gmacwrite(c, Txparm, 3<<14 | 0xb<<9 | 0x1c<<4 | 4);
+ gmacwrite(c, Rxctl, Ufilter | Mfilter | Rmcrc);
+ gmacwrite(c, Serctl, 0x04<<11 /* blind */ | Jumboen | 0x1e /* ipig */);
+
+ gmacsetea(c, Ea0);
+ gmacsetea(c, Ea1);
+
+ gmacwrite(c, Txmask, 0);
+ gmacwrite(c, Rxmask, 0);
+ gmacwrite(c, Trmask, 0);
+
+ macwrite32(c, Gfrxctl, Gfrstclr);
+ r = Gfon | Gffon;
+ if(c->type == Yukex || c->type == Yukfep)
+ r |= Gfroon;
+ macwrite32(c, Gfrxctl, r);
+ if(c->type == Yukxl)
+ macwrite32(c, Grxfm, 0);
+ else
+ macwrite32(c, Grxfm, Ferror);
+ if(c->type == Yukfep && c->rev == 0)
+ macwrite32(c, Grxft, 0x178);
+ else
+ macwrite32(c, Grxft, 0xb);
+
+ macwrite32(c, Gmfctl, Gmfclr); /* clear reset */
+ macwrite32(c, Gmfctl, Gmfon); /* on */
+
+ raminit(c);
+ if(c->type == Yukfep && c->rev == 0)
+ c->reg[Gmfea] = c->reg[Gmfea] & ~3;
+
+ c->rxinit = 0;
+ c->txinit = 0;
+}
+
+static void*
+slice(void **v, uint r, uint sz)
+{
+ uintptr a;
+
+ a = (uintptr)*v;
+ a = ROUNDUP(a, r);
+ *v = (void*)(a + sz);
+ return (void*)a;
+}
+
+static void
+setupr(Sring *r, uint cnt)
+{
+ r->rp = 0;
+ r->wp = 0;
+ r->cnt = cnt;
+ r->m = cnt - 1;
+}
+
+static int
+setup(Ctlr *c)
+{
+ uint n;
+ void *v, *mem;
+ Pcidev *p;
+
+ p = c->p;
+ c->io = p->mem[0].bar&~0xf;
+ mem = vmap(c->io, p->mem[0].size);
+ if(mem == nil){
+ print("yuk: cant map %#p\n", c->io);
+ return -1;
+ }
+ c->p = p;
+ c->reg = (uint*)mem;
+ c->reg8 = (uchar*)mem;
+ c->reg16 = (ushort*)mem;
+ if(memcmp(c->ra, nilea, sizeof c->ra) == 0)
+ memmove(c->ra, c->reg8 + Macadr + 8*c->portno, Eaddrlen);
+
+ setupr(&c->status, Sringcnt);
+ setupr(&c->tx, Tringcnt);
+ setupr(&c->rx, Rringcnt);
+
+ n = sizeof c->status.r[0] * (c->status.cnt + c->tx.cnt + c->rx.cnt);
+ n += 16*4096*2; /* rounding slop */
+ c->alloc = xspanalloc(n, 16*4096, 0); /* unknown alignment constraints */
+ memset(c->alloc, 0, n);
+
+ v = c->alloc;
+ c->status.r = slice(&v, 16*4096, sizeof c->status.r[0] * c->status.cnt);
+ c->tx.r = slice(&v, 16*4096, sizeof c->tx.r[0] * c->tx.cnt);
+ c->rx.r = slice(&v, 16*4096, sizeof c->rx.r[0] * c->rx.cnt);
+
+ c->nports = 1; /* BOTCH */
+ pcisetbme(p);
+ if(reset(c)){
+ print("yuk: cant reset\n");
+ pciclrbme(p);
+ free(c->alloc);
+ vunmap(mem, p->mem[0].size);
+ return -1;
+ }
+ macinit(c);
+ return 0;
+}
+
+static void
+shutdown(Ether *e)
+{
+ Ctlr *c;
+ Pcidev *p;
+
+ c = e->ctlr;
+
+ reset(c);
+ if(0){
+ p = c->p;
+ vunmap(c->reg, p->mem[0].size);
+ free(c->alloc);
+ }
+}
+
+static void
+scan(void)
+{
+ int i;
+ Pcidev *p;
+ Ctlr *c;
+
+ for(p = nil; p = pcimatch(p, 0, 0); ){
+ for(i = 0; i < nelem(vtab); i++)
+ if(vtab[i].vid == p->vid)
+ if(vtab[i].did == p->did)
+ break;
+ if(i == nelem(vtab))
+ continue;
+ if(nctlr == nelem(ctlrtab)){
+ print("yuk: too many controllers\n");
+ return;
+ }
+ c = malloc(sizeof *c);
+ c->p = p;
+ c->qno = nctlr;
+ rbtab[c->qno].k = &c->rxmit;
+ c->rbsz = vtab[i].mtu;
+ ctlrtab[nctlr++] = c;
+ }
+}
+
+static int
+pnp(Ether *e)
+{
+ int i;
+ Ctlr *c;
+
+ if(nctlr == 0)
+ scan();
+ for(i = 0;; i++){
+ if(i == nctlr)
+ return -1;
+ c = ctlrtab[i];
+ if(c == nil || c->flag&Fprobe)
+ continue;
+ if(e->port != 0 && e->port != (ulong)c->reg)
+ continue;
+ c->flag |= Fprobe;
+ if(setup(c) != 0)
+ continue;
+ break;
+ }
+ e->ctlr = c;
+ e->port = c->io;
+ e->irq = c->p->intl;
+ e->tbdf = c->p->tbdf;
+ e->mbps = 1000;
+ e->maxmtu = c->rbsz;
+ memmove(e->ea, c->ra, Eaddrlen);
+ e->arg = e;
+ e->attach = attach;
+ e->ctl = ctl;
+ e->ifstat = ifstat;
+ e->interrupt = interrupt;
+ e->multicast = multicast;
+ e->promiscuous = promiscuous;
+ e->shutdown = shutdown;
+ e->transmit = nil;
+
+ return 0;
+}
+
+void
+etheryuklink(void)
+{
+ addethercard("yuk", pnp);
+}
--- a/sys/src/9/pc/mkfile
+++ b/sys/src/9/pc/mkfile
@@ -105,6 +105,7 @@
$ETHER: etherif.h ../port/netif.h
$AUDIO: ../port/audioif.h
ether8003.$O ether8390.$O: ether8390.h
+etheryuk.$O: yukdump.h
$VGA mouse.$O: screen.h /sys/include/memdraw.h
devfloppy.$O: floppy.h
archmp.$O mp.$O: apbootstrap.h
--- a/sys/src/9/pc/pccpuf
+++ b/sys/src/9/pc/pccpuf
@@ -66,6 +66,7 @@
# etherm10g pci
ethersink
ethersmc devi82365 cis
+ etheryuk pci
etherwavelan wavelan devi82365 cis pci
etheriwl pci wifi
etherrt2860 pci wifi
--- a/sys/src/9/pc/pcf
+++ b/sys/src/9/pc/pcf
@@ -67,6 +67,7 @@
ethervt6105m pci ethermii
ethersink
ethersmc devi82365 cis
+ etheryuk pci
etherwavelan wavelan devi82365 cis pci
etheriwl pci wifi
etherrt2860 pci wifi
--- /dev/null
+++ b/sys/src/9/pc/yukdump.h
@@ -1,0 +1,296 @@
+typedef struct Regdump Regdump;
+struct Regdump {
+ uint offset;
+ uint size;
+ char *name;
+};
+
+Regdump pcireg[] = {
+ Pciphy, 32, "Pciphy",
+ Pciclk, 32, "Pciclk",
+ Pcistate, 32, "Pcistate",
+};
+
+static Regdump gmacreg[] = {
+ Stat, 16, "Stat",
+ Ctl, 16, "Ctl",
+ Txctl, 16, "Txctl",
+ Rxctl, 16, "Rxctl",
+ Txflow, 16, "Txflow",
+ Txparm, 16, "Txparm",
+ Serctl, 16, "Serctl",
+ Txirq, 16, "Txirq",
+ Rxirq, 16, "Rxirq",
+ Trirq, 16, "Trirq",
+ Txmask, 16, "Txmask",
+ Rxmask, 16, "Rxmask",
+ Trmask, 16, "Trmask",
+ Smictl, 16, "Smictl",
+ Smidata, 16, "Smidata",
+ Phyaddr, 16, "Phyaddr",
+ Mchash+0, 16, "Mchash",
+ Mchash+2, 16, "Mchash",
+ Mchash+4, 16, "Mchash",
+ Mchash+6, 16, "Mchash",
+ Ea0, 16, "Ea0",
+ Ea0+2, 16, "Ea0",
+ Ea0+4, 16, "Ea0",
+ Ea1, 16, "Ea1",
+ Ea1+2, 16, "Ea1",
+ Ea1+4, 16, "Ea1",
+};
+
+static Regdump macreg[] = {
+ Txactl, 8, "Txactl",
+ Gfrxctl, 32, "Gfrxctl",
+ Grxfm, 32, "Grxfm",
+ Grxft, 32, "Grxft",
+ Grxtt, 32, "Grxtt",
+ Gmfctl, 32, "Gmfctl",
+ Mac, 32, "Mac",
+ Phy, 32, "Phy",
+ Irqm, 8, "Irqm",
+ Linkctl, 8, "Linkctl",
+
+ Rxwp, 32, "Rxwp",
+ Rxrp, 32, "Rxrp",
+ Rxrlev, 32, "Rxrlev",
+
+};
+
+static Regdump reg[] = {
+ Ctst, 16, "Ctst",
+ Pwrctl, 8, "Pwrctl",
+ Isr, 32, "Isr",
+ Ism, 32, "Ism",
+ Hwe, 32, "Hwe",
+ Hwem, 32, "Hwem",
+ Isrc2, 32, "Isrc2",
+
+ Macadr/2, 16, "Macadr",
+ Macadr/2+1, 16, "Macadr",
+ Macadr/2+2, 16, "Macadr",
+
+ Pmd, 8, "Pmd",
+ Maccfg, 8, "Maccfg",
+ Chip, 8, "Chip",
+ Ramcnt, 8, "Ramcnt",
+ Clkgate, 8, "Clkgate",
+ Hres, 8, "Hres",
+ Clkctl, 32, "Clkctl",
+ Tstctl1, 8, "Tstctl1",
+
+ Asfcs, 8, "Asfcs",
+ Asfhost, 32, "Asfhost",
+ Statctl, 32, "Statctl",
+ Stattl, 16, "Stattl",
+ Stataddr, 32, "Stataddr",
+ Statth, 16, "Statth",
+ Stathd, 16, "Stathd",
+ Statwm, 8, "Statwm",
+ Statiwm, 8, "Statiwm",
+};
+
+static char*
+dumppci(Ctlr *c, char *p, char *e)
+{
+ int i;
+ Regdump *r;
+
+ r = pcireg;
+ p = seprint(p, e, "/* pci reg */\n");
+ for(i = 0; i < nelem(pcireg); i++)
+ switch(r[i].size){
+ default:
+ p = seprint(p, e, "%s: bug size %d\n", r[i].name, r[i].size);
+ break;
+ case 32:
+ p = seprint(p, e, "%s: %.8ux\n", r[i].name, pcicfgr32(c->p, r[i].offset));
+ break;
+ }
+ return p;
+}
+
+static char*
+dumpgmac(Ctlr *c, char *p, char *e)
+{
+ int i;
+ Regdump *r;
+
+ r = gmacreg;
+ p = seprint(p, e, "/* gmac reg */\n");
+ for(i = 0; i < nelem(gmacreg); i++)
+ switch(r[i].size){
+ default:
+ p = seprint(p, e, "%s: bug size %d\n", r[i].name, r[i].size);
+ break;
+ case 16:
+ p = seprint(p, e, "%s: %.4ux\n", r[i].name, gmacread(c, r[i].offset));
+ break;
+ case 32:
+ p = seprint(p, e, "%s: %.8ux\n", r[i].name, gmacread32(c, r[i].offset));
+ break;
+ }
+ return p;
+}
+
+static char*
+dumpmac(Ctlr *c, char *p, char *e)
+{
+ int i;
+ Regdump *r;
+
+ r = macreg;
+ p = seprint(p, e, "/* mac reg */\n");
+ for(i = 0; i < nelem(macreg); i++)
+ switch(r[i].size){
+ default:
+ p = seprint(p, e, "%s: bug size %d\n", r[i].name, r[i].size);
+ break;
+ case 8:
+ p = seprint(p, e, "%s: %.2ux\n", r[i].name, macread8(c, r[i].offset));
+ break;
+ case 32:
+ p = seprint(p, e, "%s: %.8ux\n", r[i].name, macread32(c, r[i].offset));
+ break;
+ }
+ return p;
+}
+
+static char*
+dumpreg(Ctlr *c, char *p, char *e)
+{
+ int i;
+ Regdump *r;
+
+ r = reg;
+ p = seprint(p, e, "/* reg */\n");
+ for(i = 0; i < nelem(reg); i++)
+ switch(r[i].size){
+ default:
+ p = seprint(p, e, "%s: bug size %d\n", r[i].name, r[i].size);
+ break;
+ case 8:
+ p = seprint(p, e, "%s: %.2ux\n", r[i].name, c->reg8[r[i].offset]);
+ break;
+ case 16:
+ p = seprint(p, e, "%s: %.4ux\n", r[i].name, c->reg16[r[i].offset]);
+ break;
+ case 32:
+ p = seprint(p, e, "%s: %.8ux\n", r[i].name, c->reg[r[i].offset]);
+ break;
+ }
+ return p;
+}
+
+static char *optab[256] = {
+[Orxchks] "rxsum",
+[Orxstat] "rxstat",
+[Otxidx] "txidx",
+};
+
+static char*
+rs(uint r)
+{
+ char *s;
+
+ s = optab[r & 0xff];
+ if(s == nil)
+ s = "";
+ return s;
+}
+
+static int
+nblocktab(int qno)
+{
+ uint i;
+ Block *b;
+
+ ilock(&rbtab[qno]);
+ b = rbtab[qno].b;
+ for(i = 0;; i++){
+ if(b == nil)
+ break;
+ b = b->next;
+ }
+ iunlock(&rbtab[qno]);
+ return i;
+}
+
+static char*
+dumpring(Sring *r, Block **t, int qno, char *p, char *e)
+{
+ int m, n;
+ uint i;
+
+ p = seprint(p, e,
+ "bring: rp %ud wp %ud cnt %ud m %ud : ",
+ r->rp, r->wp, r->cnt, r->m);
+ m = -1;
+ n = 0;
+ for(i = 0; i < r->cnt; i++){
+ n += t[i] != nil;
+ if(m>=0 && t[i] == nil){
+ p = seprint(p, e, "%ud", m);
+ if(i != m + 1)
+ p = seprint(p, e, "-%ud", i-1);
+ p = seprint(p, e, " ");
+ m = -1;
+ }else if(m<0 && t[i] != nil)
+ m = i;
+ }
+ if(m>=0){
+ p = seprint(p, e, "%ud", m);
+ if(i != m + 1)
+ p = seprint(p, e, "-%ud ", i-1);
+ }
+ p = seprint(p, e, "n=%d/%d", n, r->cnt);
+ return seprint(p, e, " list %d\n", nblocktab(qno));
+}
+
+/* debug; remove */
+static void
+descriptorfu(Ether *e, uint qoff)
+{
+ char buf[PRINTSIZE], *p, *f;
+ uint q, qm1;
+ Block *b, *a, **bring;
+ Ctlr *c;
+ Status *t, *v;
+ Sring *r;
+
+ c = e->ctlr;
+ f = buf + sizeof buf;
+ if(qoff == Qtx){
+ bring = c->tbring;
+ r = &c->tx;
+ p = seprint(buf, f, "tx ");
+ }else{
+ bring = c->rbring;
+ r = &c->rx;
+ p = seprint(buf, f, "rx ");
+ }
+
+ q = prread16(c, qoff + Pgetidx);
+ print(" getidx %ud\n", q);
+ if(q >= r->cnt){
+ q &= r->m;
+ print(" try getidx %ud\n", q);
+ }
+ qm1 = q-1 & r->m;
+ t = r->r + q;
+ v = r->r + qm1;
+ b = bring[q];
+ a = bring[qm1];
+ print(" %0.4d %.2ux l=%d p=%#p @%#p\n", q, t->op,
+ (uint)getle(t->l, 2), KZERO+(ulong)getle(t->status, 4), Pciwaddrl(t));
+ print(" %0.4d %.2ux l=%d p=%#p @%#p\n", qm1, t->op,
+ (uint)getle(v->l, 2), KZERO+(ulong)getle(v->status, 4), Pciwaddrl(v));
+ if(r == &c->rx)
+ print(" %#p %#p (wp)\n", b? b->wp: 0, a? a->wp: 0);
+ else
+ print(" %#p %#p (rp)\n", b? b->rp: 0, a? a->rp: 0);
+
+ dumpring(r, bring, c->qno, p, f);
+ print(" %s", buf);
+}