Skip site navigation (1)Skip section navigation (2)
Date:      Wed, 15 Dec 2004 14:59:10 -0800
From:      Julian Elischer <julian@elischer.org>
To:        Scott Long <scottl@freebsd.org>
Cc:        FreeBSD Current <freebsd-current@freebsd.org>
Subject:   Re: bus_dma question
Message-ID:  <41C0C1BE.8070801@elischer.org>
In-Reply-To: <41BF96E8.2080300@freebsd.org>
References:  <41BF8C1A.2090801@elischer.org> <41BF96E8.2080300@freebsd.org>

next in thread | previous in thread | raw e-mail | index | archive | help
Scott this is just what I was looking for.

This should be added to the busdma man page I think..


Scott Long wrote:

> Julian Elischer wrote:
>
>>
>> The bus_dma man page is good. but it there documentation of how to 
>> extract the physical page addresses from a map? In other words, if I 
>> defien a tag for a device and then create a map using 
>> bus_dmamap_create, and then load it using bus_dmamap_load, what is 
>> the recommended method of extracting the list of addresses to feed 
>> into a DMA scatter_gather list? Looking at various drivers I'm 
>> getting more and more confused.. There is  a lot of home-grown s/g list
>> generation code out there.. is there a "right" way to do this?
>>
>
> The function pointer that you provide to bus_dmamap_load() is called
> when the S/G map is ready (it might be deferred due to needing bounce
> pages, so you cannot assume that it will be called before 
> bus_dmamap_load() returns).  The S/G list and length is provided in
> the arguments to the callback function.  So:
>
> static void
> my_start(my_softc, my_data_ptr, my_data_len)
> {
>     /* Prepare resources for sending data to the hardware */
>     [...]
>
>     /*
>      * Hand off to busdma to generate the S/G list.
>      */
>     error = bus_dmamap_load(my_softc->dmat, my_softc->dmamap,
>         my_data_ptr, my_data_len, my_callback_func, my_softc, 0);
>     if (error == EINPROGRESS) {
>         /*
>          * The callback will be called later.  Don't queue up
>          * any more commands until it has so that we don't spin
>          * on busdma.
>          */
>         my_softc->flags |= FROZEN;
>     }
>
>     return;
> }
>
> static void
> my_callack_func(void *arg, bus_dma_segment_t *segs, int nseg, int error)
> {
>     my_softc_t *my_softc = arg;
>
>     if (error) {
>         /* Handle error */
>         [...]
>         return;
>
>     for (i = 0; i < nsegs; i++) {
>         my_sg[i].phys = segs[i].ds_addr;
>         mg_sg[i].len = segs[i].ds_len;
>     }
>
>     if (dma is from host memory to the hardware)
>     bus_dmamap_sync(my_dmat, my_dmamap, BUS_DMASYNC_PREWRITE);
>
>     if (dma is from the hardware to host memory)
>     bus_dmamap_sync(my_dmat, my_dmamap, BUS_DMASYNC_PREREAD);
>
>     /*
>      * Send S/G list to the hardware and tell the hardware to start
>      * the transaction.
>      */
>     [...]
>
>     return;
> }
>
> Note that you cannot pass an error value or anything else from the 
> callback function to the caller since you have no control over when
> the callback will actually be called.  Well, if you use the 
> BUS_DMA_NOWAIT flag for bus_dmamap_load(), you'll be guaranteed to get
> an error back instead of a deferral, but that should only be used with
> great care.
>
> Also, don't forget to call bus_dmamap_sync() with the POSTREAD and 
> POSTWRITE ops once the transaction is complete.  This is very important
> now that >4GB of memory is easy to aquire; devices like UHCI and ATA can
> still only do 32bit S/G lists, so bouncing is required on i386 and
> amd64.  You also need to do a bus_dmamap_unload() call to free whatever
> busdma resources were used for the transaction, so the sequence should
> be something like:
>
> static void
> my_intr(void *arg)
> {
>     /* Check and handle interrupt */
>     [...]
>
>     /* return busdma resources */
>     if (dma is from host memory to the hardware)
>     bus_dmamap_sync(my_dmat, my_dmamap, BUS_DMASYNC_POSTWRITE);
>
>     if (dma is from the hardware to host memory)
>     bus_dmamap_sync(my_dmat, my_dmamap, BUS_DMASYNC_POSTREAD);
>
>     bus_dmamap_unload(my_dmat, my_dmamap);
>
>     /* Process the data that was DMA'd */
>     [...]
>
>     return;
> }
>
> There will likely be changes to the busdma API for 6.0 that will
> simplify this a bit, make it faster, and make it resemble NetBSD a
> little more, but I'm still working on them.
>
> Scott




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