diff options
-rw-r--r-- | sys/dev/pci/ubsec.c | 147 | ||||
-rw-r--r-- | sys/dev/pci/ubsecreg.h | 11 | ||||
-rw-r--r-- | sys/dev/pci/ubsecvar.h | 4 |
3 files changed, 130 insertions, 32 deletions
diff --git a/sys/dev/pci/ubsec.c b/sys/dev/pci/ubsec.c index 9cd2eec4c28..e88b62aa67f 100644 --- a/sys/dev/pci/ubsec.c +++ b/sys/dev/pci/ubsec.c @@ -1,4 +1,4 @@ -/* $OpenBSD: ubsec.c,v 1.19 2000/07/31 21:57:49 deraadt Exp $ */ +/* $OpenBSD: ubsec.c,v 1.20 2000/08/11 19:38:15 deraadt Exp $ */ /* * Copyright (c) 2000 Jason L. Wright (jason@thought.net) @@ -200,7 +200,8 @@ ubsec_intr(arg) struct ubsec_softc *sc = arg; volatile u_int32_t stat, a; struct ubsec_q *q; - int npkts = 0; + struct ubsec_mcr *mcr; + int npkts = 0, i; stat = READ_REG(sc, BS_STAT); @@ -217,18 +218,39 @@ ubsec_intr(arg) while (!SIMPLEQ_EMPTY(&sc->sc_qchip)) { q = SIMPLEQ_FIRST(&sc->sc_qchip); #ifdef UBSEC_DEBUG - printf("mcr_flags %x %x %x\n", &q->q_mcr, q->q_mcr.mcr_flags, - READ_REG(sc, BS_ERR)); + printf("mcr_flags %x %x %x\n", q->q_mcr, + q->q_mcr->mcr_flags, READ_REG(sc, BS_ERR)); #endif - if ((q->q_mcr.mcr_flags & UBS_MCR_DONE) == 0) + if ((q->q_mcr->mcr_flags & UBS_MCR_DONE) == 0) break; npkts++; SIMPLEQ_REMOVE_HEAD(&sc->sc_qchip, q, q_next); #ifdef UBSEC_DEBUG printf("intr: callback q %08x flags %04x\n", q, - q->q_mcr.mcr_flags); + q->q_mcr->mcr_flags); #endif + mcr = q->q_mcr; ubsec_callback(q); + + /* + * search for further sc_qchip ubsec_q's that share + * the same MCR, and complete them too, they must be + * at the top. + */ + for (i = 1; i < mcr->mcr_pkts; i++) { + q = SIMPLEQ_FIRST(&sc->sc_qchip); + if (q && q->q_mcr == mcr) { + printf("found a share\n"); + SIMPLEQ_REMOVE_HEAD(&sc->sc_qchip, + q, q_next); + ubsec_callback(q); + } else { + printf("HUH!\n"); + break; + } + } + + free(mcr, M_DEVBUF); } #ifdef UBSEC_DEBUG if (npkts > 1) @@ -241,6 +263,7 @@ ubsec_intr(arg) printf("%s: dmaerr %s@%08x\n", sc->sc_dv.dv_xname, (a & BS_ERR_READ) ? "read" : "write", a & ~BS_ERR_READ); + panic("to let theo see things"); } ubsec_feed(sc); @@ -252,14 +275,65 @@ ubsec_feed(sc) struct ubsec_softc *sc; { struct ubsec_q *q; + struct ubsec_mcr *mcr, *mcr2; + int npkts, i; + + npkts = sc->sc_nqueue; + if (npkts > 20) + npkts = 20; +#ifdef not_working_yet + if (npkts < 2) + goto feed1; +#else + goto feed1; +#endif + + if (READ_REG(sc, BS_STAT) & BS_STAT_MCR1_FULL) + return (0); + + mcr = (struct ubsec_mcr *)malloc(sizeof(struct ubsec_mcr) + + (npkts-1) * sizeof(struct ubsec_mcr_add), M_DEVBUF, M_NOWAIT); + if (mcr == NULL) + goto feed1; + + printf("merging %d records\n", npkts); + + for (mcr2 = mcr, i = 0; i < npkts; i++) { + q = SIMPLEQ_FIRST(&sc->sc_queue); + SIMPLEQ_REMOVE_HEAD(&sc->sc_queue, q, q_next); + --sc->sc_nqueue; + SIMPLEQ_INSERT_TAIL(&sc->sc_qchip, q, q_next); + + /* + * first packet contains a full mcr, others contain + * a shortened one + */ + if (i == 0) { + bcopy(q->q_mcr, mcr2, sizeof(struct ubsec_mcr)); + mcr2 += sizeof(struct ubsec_mcr); + } else { + void *v; + + v = ((void *)q->q_mcr) + sizeof(struct ubsec_mcr) - + sizeof(struct ubsec_mcr_add); + bcopy(v, mcr2, sizeof(struct ubsec_mcr_add)); + mcr2 += sizeof(struct ubsec_mcr_add); + } + free(q->q_mcr, M_DEVBUF); + q->q_mcr = mcr; + } + mcr->mcr_pkts = npkts; + WRITE_REG(sc, BS_MCR1, (u_int32_t)vtophys(mcr)); + return (0); +feed1: while (!SIMPLEQ_EMPTY(&sc->sc_queue)) { if (READ_REG(sc, BS_STAT) & BS_STAT_MCR1_FULL) break; q = SIMPLEQ_FIRST(&sc->sc_queue); - WRITE_REG(sc, BS_MCR1, (u_int32_t)vtophys(&q->q_mcr)); + WRITE_REG(sc, BS_MCR1, (u_int32_t)vtophys(q->q_mcr)); #ifdef UBSEC_DEBUG - printf("feed: q->chip %08x %08x\n", q, (u_int32_t)vtophys(&q->q_mcr)); + printf("feed: q->chip %08x %08x\n", q, (u_int32_t)vtophys(q->q_mcr)); #endif SIMPLEQ_REMOVE_HEAD(&sc->sc_queue, q, q_next); --sc->sc_nqueue; @@ -368,9 +442,18 @@ ubsec_process(crp) } bzero(q, sizeof(struct ubsec_q)); - q->q_mcr.mcr_pkts = 1; - q->q_mcr.mcr_flags = 0; - q->q_mcr.mcr_cmdctxp = vtophys(&q->q_ctx); + q->q_mcr = (struct ubsec_mcr *)malloc(sizeof(struct ubsec_mcr), + M_DEVBUF, M_NOWAIT); + if (q->q_mcr == NULL) { + free(q, M_DEVBUF); + err = ENOMEM; + goto errout; + } + bzero(q->q_mcr, sizeof(struct ubsec_mcr)); + + q->q_mcr->mcr_pkts = 1; + q->q_mcr->mcr_flags = 0; + q->q_mcr->mcr_cmdctxp = vtophys(&q->q_ctx); q->q_sc = sc; q->q_crp = crp; @@ -395,7 +478,7 @@ ubsec_process(crp) maccrd = crd1; enccrd = NULL; } else if (crd1->crd_alg == CRYPTO_DES_CBC || - crd1->crd_alg == CRYPTO_3DES_CBC) { + crd1->crd_alg == CRYPTO_3DES_CBC) { maccrd = NULL; enccrd = crd1; } else { @@ -544,7 +627,7 @@ ubsec_process(crp) err = ENOMEM; goto errout; } - q->q_mcr.mcr_pktlen = stheend; + q->q_mcr->mcr_pktlen = stheend; #ifdef UBSEC_DEBUG printf("src skip: %d\n", sskip); @@ -567,7 +650,7 @@ ubsec_process(crp) } if (j == 0) - pb = &q->q_mcr.mcr_ipktbuf; + pb = &q->q_mcr->mcr_ipktbuf; else pb = &q->q_srcpkt[j - 1]; @@ -593,10 +676,10 @@ ubsec_process(crp) j++; } #ifdef UBSEC_DEBUG - printf(" buf[%x]: %d@%x -> %x\n", vtophys(&q->q_mcr), - q->q_mcr.mcr_ipktbuf.pb_len, - q->q_mcr.mcr_ipktbuf.pb_addr, - q->q_mcr.mcr_ipktbuf.pb_next); + printf(" buf[%x]: %d@%x -> %x\n", vtophys(q->q_mcr), + q->q_mcr->mcr_ipktbuf.pb_len, + q->q_mcr->mcr_ipktbuf.pb_addr, + q->q_mcr->mcr_ipktbuf.pb_next); for (i = 0; i < j - 1; i++) { printf(" buf[%x]: %d@%x -> %x\n", vtophys(&q->q_srcpkt[i]), q->q_srcpkt[i].pb_len, @@ -606,14 +689,14 @@ ubsec_process(crp) #endif if (enccrd == NULL && maccrd != NULL) { - q->q_mcr.mcr_opktbuf.pb_addr = 0; - q->q_mcr.mcr_opktbuf.pb_len = 0; - q->q_mcr.mcr_opktbuf.pb_next = + q->q_mcr->mcr_opktbuf.pb_addr = 0; + q->q_mcr->mcr_opktbuf.pb_len = 0; + q->q_mcr->mcr_opktbuf.pb_next = (u_int32_t)vtophys(&q->q_macbuf[0]); printf("opkt: %x %x %x\n", - q->q_mcr.mcr_opktbuf.pb_addr, - q->q_mcr.mcr_opktbuf.pb_len, - q->q_mcr.mcr_opktbuf.pb_next); + q->q_mcr->mcr_opktbuf.pb_addr, + q->q_mcr->mcr_opktbuf.pb_len, + q->q_mcr->mcr_opktbuf.pb_next); } else { if (!nicealign) { int totlen, len; @@ -689,7 +772,7 @@ ubsec_process(crp) } if (j == 0) - pb = &q->q_mcr.mcr_opktbuf; + pb = &q->q_mcr->mcr_opktbuf; else pb = &q->q_dstpkt[j - 1]; @@ -711,10 +794,10 @@ ubsec_process(crp) } #ifdef UBSEC_DEBUG printf(" buf[%d, %x]: %d@%x -> %x\n", 0, - vtophys(&q->q_mcr), - q->q_mcr.mcr_opktbuf.pb_len, - q->q_mcr.mcr_opktbuf.pb_addr, - q->q_mcr.mcr_opktbuf.pb_next); + vtophys(q->q_mcr), + q->q_mcr->mcr_opktbuf.pb_len, + q->q_mcr->mcr_opktbuf.pb_addr, + q->q_mcr->mcr_opktbuf.pb_next); for (i = 0; i < j - 1; i++) { printf(" buf[%d, %x]: %d@%x -> %x\n", i+1, vtophys(&q->q_dstpkt[i]), @@ -734,6 +817,8 @@ ubsec_process(crp) errout: if (q != NULL) { + if (q->q_mcr) + free(q->q_mcr, M_DEVBUF); if (q->q_src_m != q->q_dst_m) m_freem(q->q_dst_m); free(q, M_DEVBUF); @@ -764,6 +849,10 @@ ubsec_callback(q) break; } + /* + * note that q->q_mcr is not freed, because ubsec_intr() has to + * deal with possible sharing + */ free(q, M_DEVBUF); crypto_done(crp); } diff --git a/sys/dev/pci/ubsecreg.h b/sys/dev/pci/ubsecreg.h index e3fb558af2b..304c95c4c46 100644 --- a/sys/dev/pci/ubsecreg.h +++ b/sys/dev/pci/ubsecreg.h @@ -1,4 +1,4 @@ -/* $OpenBSD: ubsecreg.h,v 1.4 2000/06/18 03:37:22 jason Exp $ */ +/* $OpenBSD: ubsecreg.h,v 1.5 2000/08/11 19:38:15 deraadt Exp $ */ /* * Copyright (c) 2000 Theo de Raadt @@ -107,6 +107,15 @@ struct ubsec_mcr { volatile u_int16_t mcr_pktlen; struct ubsec_pktbuf mcr_opktbuf; /* output chain header */ }; + +struct ubsec_mcr_add { + volatile u_int32_t mcr_cmdctxp; /* command ctx pointer */ + struct ubsec_pktbuf mcr_ipktbuf; /* input chain header */ + volatile u_int16_t mcr_reserved; + volatile u_int16_t mcr_pktlen; + struct ubsec_pktbuf mcr_opktbuf; /* output chain header */ +}; + #define UBS_MCR_DONE 0x0001 /* mcr has been processed */ #define UBS_MCR_ERROR 0x0002 /* error in processing */ #define UBS_MCR_ERRORCODE 0xff00 /* error type */ diff --git a/sys/dev/pci/ubsecvar.h b/sys/dev/pci/ubsecvar.h index 6bc5ff44fac..7960ba26b0a 100644 --- a/sys/dev/pci/ubsecvar.h +++ b/sys/dev/pci/ubsecvar.h @@ -1,4 +1,4 @@ -/* $OpenBSD: ubsecvar.h,v 1.6 2000/07/29 23:42:00 jason Exp $ */ +/* $OpenBSD: ubsecvar.h,v 1.7 2000/08/11 19:38:16 deraadt Exp $ */ /* * Copyright (c) 2000 Theo de Raadt @@ -43,7 +43,7 @@ struct ubsec_softc { struct ubsec_q { SIMPLEQ_ENTRY(ubsec_q) q_next; struct cryptop *q_crp; - struct ubsec_mcr q_mcr; + struct ubsec_mcr *q_mcr; struct ubsec_pktbuf q_srcpkt[MAX_SCATTER-1]; struct ubsec_pktbuf q_dstpkt[MAX_SCATTER-1]; struct ubsec_pktctx q_ctx; |