summaryrefslogtreecommitdiff
path: root/sys/dev/pci/noct.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/pci/noct.c')
-rw-r--r--sys/dev/pci/noct.c211
1 files changed, 200 insertions, 11 deletions
diff --git a/sys/dev/pci/noct.c b/sys/dev/pci/noct.c
index f7ea3245d83..eb976f5df4a 100644
--- a/sys/dev/pci/noct.c
+++ b/sys/dev/pci/noct.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: noct.c,v 1.3 2002/06/21 03:26:40 jason Exp $ */
+/* $OpenBSD: noct.c,v 1.4 2002/06/21 17:45:29 jason Exp $ */
/*
* Copyright (c) 2002 Jason L. Wright (jason@thought.net)
@@ -80,6 +80,12 @@ void noct_pkh_init(struct noct_softc *);
void noct_pkh_intr(struct noct_softc *);
void noct_pkh_tick(void *);
+void noct_ea_enable(struct noct_softc *);
+void noct_ea_disable(struct noct_softc *);
+void noct_ea_init(struct noct_softc *);
+void noct_ea_intr(struct noct_softc *);
+void noct_ea_tick(void *);
+
u_int64_t noct_read_8(struct noct_softc *, u_int32_t);
void noct_write_8(struct noct_softc *, u_int32_t, u_int64_t);
@@ -163,6 +169,7 @@ noct_attach(parent, self, aux)
noct_rng_init(sc);
noct_pkh_init(sc);
+ noct_ea_init(sc);
return;
@@ -191,6 +198,11 @@ noct_intr(vsc)
noct_pkh_intr(sc);
}
+ if (reg & BRDGSTS_CCH_INT) {
+ r = 1;
+ noct_ea_intr(sc);
+ }
+
return (r);
}
@@ -227,7 +239,7 @@ noct_ram_write(sc, adr, dat)
/* wait for pending writes to finish */
for (;;) {
reg = NOCT_READ_4(sc, NOCT_EA_CTX_ADDR);
- if ((reg & CTXADDR_WRITEPEND) == 0)
+ if ((reg & EACTXADDR_WRITEPEND) == 0)
break;
}
@@ -237,7 +249,7 @@ noct_ram_write(sc, adr, dat)
for (;;) {
reg = NOCT_READ_4(sc, NOCT_EA_CTX_ADDR);
- if ((reg & CTXADDR_WRITEPEND) == 0)
+ if ((reg & EACTXADDR_WRITEPEND) == 0)
break;
}
}
@@ -253,15 +265,15 @@ noct_ram_read(sc, adr)
/* wait for pending reads to finish */
for (;;) {
reg = NOCT_READ_4(sc, NOCT_EA_CTX_ADDR);
- if ((reg & CTXADDR_READPEND) == 0)
+ if ((reg & EACTXADDR_READPEND) == 0)
break;
}
- NOCT_WRITE_4(sc, NOCT_EA_CTX_ADDR, adr | CTXADDR_READPEND);
+ NOCT_WRITE_4(sc, NOCT_EA_CTX_ADDR, adr | EACTXADDR_READPEND);
for (;;) {
reg = NOCT_READ_4(sc, NOCT_EA_CTX_ADDR);
- if ((reg & CTXADDR_READPEND) == 0)
+ if ((reg & EACTXADDR_READPEND) == 0)
break;
}
@@ -281,8 +293,8 @@ noct_pkh_disable(sc)
NOCT_WRITE_4(sc, NOCT_BRDG_CTL,
NOCT_READ_4(sc, NOCT_BRDG_CTL) & ~(BRDGCTL_PKIRQ_ENA));
- /* Turn of PK interrupts */
- NOCT_READ_4(sc, NOCT_PKH_IER);
+ /* Turn off PK interrupts */
+ r = NOCT_READ_4(sc, NOCT_PKH_IER);
r &= ~(PKHIER_CMDSI | PKHIER_SKSWR | PKHIER_SKSOFF | PKHIER_PKHLEN |
PKHIER_PKHOPCODE | PKHIER_BADQBASE | PKHIER_LOADERR |
PKHIER_STOREERR | PKHIER_CMDERR | PKHIER_ILL | PKHIER_PKERESV |
@@ -457,7 +469,7 @@ noct_pkh_tick(vsc)
if (sc->sc_pkhbusy)
goto out;
nop = &sc->sc_pkhcmd[sc->sc_pkhwp].nop;
- nop->op = PKH_OP_SI | PKH_OP_CODE_NOP;
+ nop->op = htole32(PKH_OP_SI | PKH_OP_CODE_NOP);
nop->unused[0] = nop->unused[1] = nop->unused[2] = nop->unused[3] = 0;
nop->unused[4] = nop->unused[5] = nop->unused[6] = 0;
sc->sc_pkhbusy = 1;
@@ -512,9 +524,9 @@ noct_rng_enable(sc)
u_int32_t r;
adr = sc->sc_rngmap->dm_segs[0].ds_addr;
- NOCT_WRITE_4(sc, NOCT_RNG_BAR1, (adr >> 32) & 0xffffffff);
+ NOCT_WRITE_4(sc, NOCT_RNG_Q_BASE_HI, (adr >> 32) & 0xffffffff);
NOCT_WRITE_4(sc, NOCT_RNG_Q_LEN, NOCT_RNG_QLEN);
- NOCT_WRITE_4(sc, NOCT_RNG_BAR0, (adr >> 0 ) & 0xffffffff);
+ NOCT_WRITE_4(sc, NOCT_RNG_Q_BASE_LO, (adr >> 0 ) & 0xffffffff);
NOCT_WRITE_8(sc, NOCT_RNG_CTL,
RNGCTL_RNG_ENA |
@@ -655,6 +667,183 @@ noct_rng_tick(vsc)
}
void
+noct_ea_disable(sc)
+ struct noct_softc *sc;
+{
+ u_int32_t r;
+
+ /* Turn off EA irq */
+ NOCT_WRITE_4(sc, NOCT_BRDG_CTL,
+ NOCT_READ_4(sc, NOCT_BRDG_CTL) & ~(BRDGCTL_EAIRQ_ENA));
+
+ /* Turn off EA interrupts */
+ r = NOCT_READ_4(sc, NOCT_EA_IER);
+ r &= ~(EAIER_QALIGN | EAIER_CMDCMPL | EAIER_OPERR | EAIER_CMDREAD |
+ EAIER_CMDWRITE | EAIER_DATAREAD | EAIER_DATAWRITE |
+ EAIER_INTRNLLEN | EAIER_EXTRNLLEN | EAIER_DESBLOCK |
+ EAIER_DESKEY | EAIER_ILL);
+ NOCT_WRITE_4(sc, NOCT_EA_IER, r);
+
+ /* Disable EA unit */
+ r = NOCT_READ_4(sc, NOCT_EA_CSR);
+ r &= ~EACSR_ENABLE;
+ NOCT_WRITE_4(sc, NOCT_EA_CSR, r);
+ for (;;) {
+ r = NOCT_READ_4(sc, NOCT_EA_CSR);
+ if ((r & EACSR_BUSY) == 0)
+ break;
+ }
+
+ /* Clear status bits */
+ r = NOCT_READ_4(sc, NOCT_EA_CSR);
+ r |= EACSR_QALIGN | EACSR_CMDCMPL | EACSR_OPERR | EACSR_CMDREAD |
+ EACSR_CMDWRITE | EACSR_DATAREAD | EACSR_DATAWRITE |
+ EACSR_INTRNLLEN | EACSR_EXTRNLLEN | EACSR_DESBLOCK |
+ EACSR_DESKEY | EACSR_ILL;
+ NOCT_WRITE_4(sc, NOCT_EA_CSR, r);
+}
+
+void
+noct_ea_enable(sc)
+ struct noct_softc *sc;
+{
+ u_int64_t adr;
+
+ sc->sc_eawp = 0;
+
+ adr = sc->sc_eamap->dm_segs[0].ds_addr;
+ NOCT_WRITE_4(sc, NOCT_EA_Q_BASE_HI, (adr >> 32) & 0xffffffff);
+ NOCT_WRITE_4(sc, NOCT_EA_Q_LEN, NOCT_EA_QLEN);
+ NOCT_WRITE_4(sc, NOCT_EA_Q_BASE_LO, (adr >> 0) & 0xffffffff);
+
+ NOCT_WRITE_4(sc, NOCT_EA_IER,
+ EAIER_QALIGN | EAIER_CMDCMPL | EAIER_OPERR | EAIER_CMDREAD |
+ EAIER_CMDWRITE | EAIER_DATAREAD | EAIER_DATAWRITE |
+ EAIER_INTRNLLEN | EAIER_EXTRNLLEN | EAIER_DESBLOCK |
+ EAIER_DESKEY | EAIER_ILL);
+
+ NOCT_WRITE_4(sc, NOCT_EA_CSR,
+ NOCT_READ_4(sc, NOCT_EA_CSR) | EACSR_ENABLE);
+
+ NOCT_WRITE_4(sc, NOCT_BRDG_CTL,
+ NOCT_READ_4(sc, NOCT_BRDG_CTL) | BRDGCTL_EAIRQ_ENA);
+}
+
+void
+noct_ea_init(sc)
+ struct noct_softc *sc;
+{
+ bus_dma_segment_t seg;
+ int rseg;
+
+ if (bus_dmamem_alloc(sc->sc_dmat, NOCT_EA_BUFSIZE,
+ PAGE_SIZE, 0, &seg, 1, &rseg, BUS_DMA_NOWAIT)) {
+ printf("%s: failed ea buf alloc\n", sc->sc_dv.dv_xname);
+ goto fail;
+ }
+ if (bus_dmamem_map(sc->sc_dmat, &seg, rseg, NOCT_EA_BUFSIZE,
+ (caddr_t *)&sc->sc_eacmd, BUS_DMA_NOWAIT)) {
+ printf("%s: failed ea buf map\n", sc->sc_dv.dv_xname);
+ goto fail_1;
+ }
+ if (bus_dmamap_create(sc->sc_dmat, NOCT_EA_BUFSIZE, rseg,
+ NOCT_EA_BUFSIZE, 0, BUS_DMA_NOWAIT, &sc->sc_eamap)) {
+ printf("%s: failed ea map create\n", sc->sc_dv.dv_xname);
+ goto fail_2;
+ }
+ if (bus_dmamap_load_raw(sc->sc_dmat, sc->sc_eamap,
+ &seg, rseg, NOCT_EA_BUFSIZE, BUS_DMA_NOWAIT)) {
+ printf("%s: failed ea buf load\n", sc->sc_dv.dv_xname);
+ goto fail_3;
+ }
+
+ noct_ea_disable(sc);
+ noct_ea_enable(sc);
+
+ if (hz > 100)
+ sc->sc_eatick = hz/100;
+ else
+ sc->sc_eatick = 1;
+ timeout_set(&sc->sc_eato, noct_ea_tick, sc);
+ timeout_add(&sc->sc_eato, sc->sc_eatick);
+
+ return;
+
+fail_3:
+ bus_dmamap_destroy(sc->sc_dmat, sc->sc_eamap);
+fail_2:
+ bus_dmamem_unmap(sc->sc_dmat,
+ (caddr_t)sc->sc_eacmd, NOCT_EA_BUFSIZE);
+fail_1:
+ bus_dmamem_free(sc->sc_dmat, &seg, rseg);
+fail:
+ sc->sc_eacmd = NULL;
+ sc->sc_eamap = NULL;
+}
+
+void
+noct_ea_tick(vsc)
+ void *vsc;
+{
+ struct noct_softc *sc = vsc;
+ struct noct_ea_cmd *nop;
+ int s, i;
+
+ s = splnet();
+ nop = &sc->sc_eacmd[sc->sc_eawp];
+ for (i = 0; i < EA_CMD_WORDS; i++)
+ nop->buf[i] = 0;
+ nop->buf[0] = htole32(EA_0_SI);
+ nop->buf[1] = htole32(EA_OP_NOP);
+ if (++sc->sc_eawp == NOCT_EA_ENTRIES)
+ sc->sc_eawp = 0;
+ NOCT_WRITE_4(sc, NOCT_EA_Q_PTR, sc->sc_eawp);
+ splx(s);
+ timeout_add(&sc->sc_eato, sc->sc_eatick);
+}
+
+void
+noct_ea_intr(sc)
+ struct noct_softc *sc;
+{
+ u_int32_t csr;
+
+ csr = NOCT_READ_4(sc, NOCT_EA_CSR);
+ NOCT_WRITE_4(sc, NOCT_EA_CSR, csr |
+ EACSR_QALIGN | EACSR_CMDCMPL | EACSR_OPERR | EACSR_CMDREAD |
+ EACSR_CMDWRITE | EACSR_DATAREAD | EACSR_DATAWRITE |
+ EACSR_INTRNLLEN | EACSR_EXTRNLLEN | EACSR_DESBLOCK |
+ EACSR_DESKEY | EACSR_ILL);
+
+ if (csr & EACSR_CMDCMPL) {
+ /* command completed... */
+ }
+
+ if (csr & EACSR_QALIGN)
+ printf("%s: ea bad queue alignment\n", sc->sc_dv.dv_xname);
+ if (csr & EACSR_OPERR)
+ printf("%s: ea bad opcode\n", sc->sc_dv.dv_xname);
+ if (csr & EACSR_CMDREAD)
+ printf("%s: ea command read error\n", sc->sc_dv.dv_xname);
+ if (csr & EACSR_CMDWRITE)
+ printf("%s: ea command write error\n", sc->sc_dv.dv_xname);
+ if (csr & EACSR_DATAREAD)
+ printf("%s: ea data read error\n", sc->sc_dv.dv_xname);
+ if (csr & EACSR_DATAWRITE)
+ printf("%s: ea data write error\n", sc->sc_dv.dv_xname);
+ if (csr & EACSR_INTRNLLEN)
+ printf("%s: ea bad internal len\n", sc->sc_dv.dv_xname);
+ if (csr & EACSR_EXTRNLLEN)
+ printf("%s: ea bad external len\n", sc->sc_dv.dv_xname);
+ if (csr & EACSR_DESBLOCK)
+ printf("%s: ea bad des block\n", sc->sc_dv.dv_xname);
+ if (csr & EACSR_DESKEY)
+ printf("%s: ea bad des key\n", sc->sc_dv.dv_xname);
+ if (csr & EACSR_ILL)
+ printf("%s: ea illegal access\n", sc->sc_dv.dv_xname);
+}
+
+void
noct_write_8(sc, reg, val)
struct noct_softc *sc;
u_int32_t reg;