Skip site navigation (1)Skip section navigation (2)
Date:      Wed, 29 Mar 2006 07:24:29 GMT
From:      Warner Losh <imp@FreeBSD.org>
To:        Perforce Change Reviews <perforce@freebsd.org>
Subject:   PERFORCE change 94231 for review
Message-ID:  <200603290724.k2T7OTro011018@repoman.freebsd.org>

next in thread | raw e-mail | index | archive | help
http://perforce.freebsd.org/chv.cgi?CH=94231

Change 94231 by imp@imp_Speedy on 2006/03/29 07:24:23

	Use the PDC to transmit.  The tx side is easy, since we don't
	try to double buffer.  We get a bunch of bytes, and try to transmit
	them.  Once it is done, we'll get more from the main uart driver.
	I think this works, but I've not tested it.
	
	The Rx side is harder.  We don't know, a priori, the number of
	bytes we'll get.  We have to deal with both ENDTRX interrupts,
	as well as TIMEOUT interrupts.  We necessarily have to have
	double buffering.  The uart infrastructure doesn't really
	have this concept, so more study is needed.  We can fake it with
	really really large buffers, but that's just a statistical
	solution...

Affected files ...

.. //depot/projects/arm/src/sys/arm/at91/uart_dev_at91usart.c#19 edit

Differences ...

==== //depot/projects/arm/src/sys/arm/at91/uart_dev_at91usart.c#19 (text+ko) ====

@@ -52,9 +52,11 @@
 	struct uart_softc base;
 	bus_dma_tag_t dmatag;		/* bus dma tag for mbufs */
 	bus_dmamap_t tx_map;
+	bus_dmamap_t rx_map;
 };
 
-#define      DEFAULT_RCLK    AT91C_MASTER_CLOCK
+#define DEFAULT_RCLK		AT91C_MASTER_CLOCK
+#define	USART_BUFFER_SIZE	128
 
 #define	RD4(bas, reg)		\
 	bus_space_read_4((bas)->bst, (bas)->bsh, uart_regofs(bas, reg))
@@ -297,41 +299,59 @@
 
 	atsc = (struct at91_usart_softc *)sc;
 
-	sc->sc_txfifosz = 128;	/* Really 64k, but 128 seems a good number */
-	sc->sc_rxfifosz = 128;
+	sc->sc_txfifosz = USART_BUFFER_SIZE;
+	sc->sc_rxfifosz = USART_BUFFER_SIZE;
 	sc->sc_hwiflow = 0;
 
 	/*
 	 * Allocate DMA tags and maps
 	 */
 	err = bus_dma_tag_create(NULL, 1, 0, BUS_SPACE_MAXADDR_32BIT,
-	    BUS_SPACE_MAXADDR, NULL, NULL, MCLBYTES, 1, MCLBYTES,
-	    BUS_DMA_ALLOCNOW, NULL, NULL, &atsc->dmatag);
+	    BUS_SPACE_MAXADDR, NULL, NULL, USART_BUFFER_SIZE, 1,
+	    USART_BUFFER_SIZE, BUS_DMA_ALLOCNOW, NULL, NULL, &atsc->dmatag);
 	if (err != 0)
 		goto errout;
 	err = bus_dmamap_create(atsc->dmatag, 0, &atsc->tx_map);
 	if (err != 0)
 	    goto errout;
+	err = bus_dmamap_create(atsc->dmatag, 0, &atsc->rx_map);
+	if (err != 0)
+	    goto errout;
 errout:;
+	// XXX bad
 	return (err);
 }
+
+static void
+at91_getaddr(void *arg, bus_dma_segment_t *segs, int nsegs, int error)
+{
+	if (error != 0)
+		return;
+	*(bus_addr_t *)arg = segs[0].ds_addr;
+}
+
+
 static int
 at91_usart_bus_transmit(struct uart_softc *sc)
 {
-	int i;
+	bus_addr_t addr;
+	struct at91_usart_softc *atsc;
+
+	atsc = (struct at91_usart_softc *)sc;
+	if (bus_dmamap_load(atsc->dmatag, atsc->tx_map, sc->sc_txbuf,
+	    sc->sc_txdatasz, at91_getaddr, &addr, 0) != 0)
+		return (EAGAIN);
+	bus_dmamap_sync(atsc->dmatag, atsc->tx_map, BUS_DMASYNC_PREWRITE);
 
-	/* XXX VERY sub-optimial */
 	mtx_lock_spin(&sc->sc_hwmtx);
 	sc->sc_txbusy = 1;
-#if 0
-	/* XXX
-	 * We can setup the PDC to transfer the whole buffer
-	 * here.
+	/*
+	 * Setup the PDC to transfer the data and interrupt us when it
+	 * is done.  We've already requested the interrupt.
 	 */
-#else
-	for (i = 0; i < sc->sc_txdatasz; i++)
-		at91_usart_putc(&sc->sc_bas, sc->sc_txbuf[i]);
-#endif
+	WR4(&sc->sc_bas, PDC_TPR, addr);
+	WR4(&sc->sc_bas, PDC_TCR, sc->sc_txdatasz);
+	WR4(&sc->sc_bas, PDC_PTCR, PDC_PTCR_TXTEN);
 	mtx_unlock_spin(&sc->sc_hwmtx);
 #ifdef USART0_CONSOLE
 	/*
@@ -394,7 +414,9 @@
 {
 	int csr = RD4(&sc->sc_bas, USART_CSR);
 	int ipend = 0;
-	
+	struct at91_usart_softc *atsc;
+
+	atsc = (struct at91_usart_softc *)sc;
 #ifdef USART0_CONSOLE
 	/* 
 	 * XXX: We have to cheat for skyeye, as it will return 0xff for all
@@ -404,6 +426,11 @@
 		return (0);
 #endif
 	   
+	if (csr & USART_CSR_ENDTX) {
+		bus_dmamap_sync(atsc->dmatag, atsc->tx_map,
+		    BUS_DMASYNC_POSTWRITE);
+		bus_dmamap_unload(atsc->dmatag, atsc->tx_map);
+	}
 	mtx_lock_spin(&sc->sc_hwmtx);
 	if (csr & USART_CSR_TXRDY && sc->sc_txbusy)
 		ipend |= SER_INT_TXIDLE;



Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?200603290724.k2T7OTro011018>