Skip site navigation (1)Skip section navigation (2)
Date:      Fri, 11 Jan 2008 00:45:55 GMT
From:      Hans Petter Selasky <hselasky@FreeBSD.org>
To:        Perforce Change Reviews <perforce@FreeBSD.org>
Subject:   PERFORCE change 133006 for review
Message-ID:  <200801110045.m0B0jt5Y049295@repoman.freebsd.org>

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

Change 133006 by hselasky@hselasky_laptop001 on 2008/01/11 00:45:51

	
	This commit only affects code that compiles on NetBSD platforms.
	
	Update BUS-DMA code for NetBSD.

Affected files ...

.. //depot/projects/usb/src/sys/dev/usb/usb_subr.c#85 edit

Differences ...

==== //depot/projects/usb/src/sys/dev/usb/usb_subr.c#85 (text+ko) ====

@@ -2522,97 +2522,260 @@
 
 #ifdef __NetBSD__
 
-bus_dma_tag_t
-usbd_dma_tag_alloc(bus_dma_tag_t parent, uint32_t seg_size,
-    uint32_t alignment, uint32_t max_size, uint8_t single_seg)
+/*------------------------------------------------------------------------*
+ *	usbd_dma_tag_create - allocate a DMA tag
+ *
+ * NOTE: If the "align" parameter has a value of 1 the DMA-tag will
+ * allow multi-segment mappings. Else all mappings are single-segment.
+ *------------------------------------------------------------------------*/
+void
+usbd_dma_tag_create(bus_dma_tag_t tag_parent, struct usbd_dma_tag *udt,
+    uint32_t size, uint32_t align)
 {
-	/* FreeBSD specific */
-	return (parent);
+	uint32_t nseg;
+
+	if (align == 1) {
+		nseg = (2 + (size / USB_PAGE_SIZE));
+	} else {
+		nseg = 1;
+	}
+
+	udt->p_seg = malloc(nseg * sizeof(*(udt->p_seg)),
+	    M_USB, M_WAITOK | M_ZERO);
+
+	if (udt->p_seg == NULL) {
+		return;
+	}
+	udt->tag = tag_parent;
+	udt->n_seg = nseg;
+	return;
 }
 
+/*------------------------------------------------------------------------*
+ *	usbd_dma_tag_free - free a DMA tag
+ *------------------------------------------------------------------------*/
 void
-usbd_dma_tag_free(bus_dma_tag_t tag)
+usbd_dma_tag_destroy(struct usbd_dma_tag *udt)
+{
+	free(udt->p_seg, M_USB);
+	return;
+}
+
+/*------------------------------------------------------------------------*
+ *	usbd_pc_alloc_mem_cb
+ *------------------------------------------------------------------------*/
+static void
+usbd_pc_alloc_mem_cb(struct usbd_page_cache *pc, bus_dma_segment_t *segs,
+    int nseg, int error)
 {
-	/* FreeBSD specific */
+	struct usbd_xfer *xfer;
+	struct usbd_page *pg;
+	uint32_t rem;
+	uint8_t owned;
+
+	xfer = pc->xfer;
+
+	/*
+	 * XXX There is sometimes recursive locking here.
+	 * XXX We should try to find a better solution.
+	 * XXX Until further the "owned" variable does
+	 * XXX the trick.
+	 */
+
+	if (error) {
+		if (xfer) {
+			owned = mtx_owned(xfer->priv_mtx);
+			if (!owned)
+				mtx_lock(xfer->priv_mtx);
+			xfer->usb_root->dma_error = 1;
+			usbd_bdma_done_event(xfer->usb_root);
+			if (!owned)
+				mtx_unlock(xfer->priv_mtx);
+		}
+		return;
+	}
+	pg = pc->page_start;
+	pg->physaddr = segs->ds_addr & ~(USB_PAGE_SIZE - 1);
+	rem = segs->ds_addr & (USB_PAGE_SIZE - 1);
+	pc->page_offset_buf = rem;
+	pc->page_offset_end += rem;
+	nseg--;
+
+	while (nseg > 0) {
+		nseg--;
+		segs++;
+		pg++;
+		pg->physaddr = segs->ds_addr & ~(USB_PAGE_SIZE - 1);
+	}
+
+	if (xfer) {
+		owned = mtx_owned(xfer->priv_mtx);
+		if (!owned)
+			mtx_lock(xfer->priv_mtx);
+		usbd_bdma_done_event(xfer->usb_root);
+		if (!owned)
+			mtx_unlock(xfer->priv_mtx);
+	}
 	return;
 }
 
-void   *
-usbd_mem_alloc_sub(bus_dma_tag_t tag, struct usbd_page *page,
-    uint32_t size, uint32_t alignment)
+/*------------------------------------------------------------------------*
+ *	usbd_pc_alloc_mem - allocate DMA'able memory
+ *
+ * Returns:
+ *    0: Success
+ * Else: Failure
+ *------------------------------------------------------------------------*/
+uint8_t
+usbd_pc_alloc_mem(bus_dma_tag_t parent_tag, struct usbd_dma_tag *utag,
+    struct usbd_page_cache *pc, struct usbd_page *pg, uint32_t size,
+    uint32_t align, uint8_t utag_max)
 {
 	caddr_t ptr = NULL;
+	bus_dma_tag_t tag;
+	bus_dmamap_t map;
+	int seg_count;
 
-	page->tag = tag;
-	page->seg_count = 1;
+	if (align != 1) {
+		/*
+	         * The alignment must be greater or equal to the
+	         * "size" else the object can be split between two
+	         * memory pages and we get a problem!
+	         */
+		while (align < size) {
+			align *= 2;
+			if (align == 0) {
+				goto done_5;
+			}
+		}
+	}
+	/* get the correct DMA tag */
+	utag = usbd_dma_tag_setup(parent_tag, utag, size, align, utag_max);
+	if (utag == NULL) {
+		goto done_5;
+	}
+	/* get the DMA tag */
+	tag = utag->tag;
 
-	if (bus_dmamem_alloc(page->tag, size, alignment, 0,
-	    &page->seg, 1,
-	    &page->seg_count, BUS_DMA_WAITOK)) {
+	if (bus_dmamem_alloc(tag, size, align, 0, utag->p_seg,
+	    utag->n_seg, &seg_count, BUS_DMA_WAITOK)) {
 		goto done_4;
 	}
-	if (bus_dmamem_map(page->tag, &page->seg, page->seg_count, size,
+	if (bus_dmamem_map(tag, utag->p_seg, seg_count, size,
 	    &ptr, BUS_DMA_WAITOK | BUS_DMA_COHERENT)) {
 		goto done_3;
 	}
-	if (bus_dmamap_create(page->tag, size, 1, size,
-	    0, BUS_DMA_WAITOK, &page->map)) {
+	if (bus_dmamap_create(tag, size, utag->n_seg, USB_PAGE_SIZE,
+	    0, BUS_DMA_WAITOK, &map)) {
 		goto done_2;
 	}
-	if (bus_dmamap_load(page->tag, page->map, ptr, size, NULL,
+	if (bus_dmamap_load(tag, map, ptr, size, NULL,
 	    BUS_DMA_WAITOK)) {
 		goto done_1;
 	}
-	page->physaddr = page->map->dm_segs[0].ds_addr;
-	page->buffer = ptr;
-	page->length = size;
+	pc->p_seg = malloc(seg_count * sizeof(*(pc->p_seg)),
+	    M_USB, M_WAITOK | M_ZERO);
+	if (pc->p_seg == NULL) {
+		goto done_0;
+	}
+	/* store number if actual segments used */
+	pc->n_seg = seg_count;
+
+	/* make a copy of the segments */
+	bcopy(utag->p_seg, pc->p_seg,
+	    seg_count * sizeof(*(pc->p_seg)));
+
+	/* setup page cache */
+	pc->buffer = ptr;
+	pc->page_start = pg;
+	pc->page_offset_buf = 0;
+	pc->page_offset_end = size;
+	pc->map = map;
+	pc->tag = tag;
+
+	usbd_pc_alloc_mem_cb(pc, utag->p_seg, seg_count, 0);
 
-	usbd_page_cpu_invalidate(page);
 	bzero(ptr, size);
-	usbd_page_cpu_flush(page);
+
+	usbd_pc_cpu_flush(pc);
 
-#ifdef USB_DEBUG
-	if (usbdebug > 14) {
-		printf("%s: %p, %d bytes, phys=%p\n",
-		    __FUNCTION__, ptr, size,
-		    ((char *)0) + page->physaddr);
-	}
-#endif
-	return (ptr);
+	return (0);
 
+done_0:
+	bus_dmamap_unload(tag, map);
 done_1:
-	bus_dmamap_destroy(page->tag, page->map);
-
+	bus_dmamap_destroy(tag, map);
 done_2:
-	bus_dmamem_unmap(page->tag, ptr, size);
-
+	bus_dmamem_unmap(tag, ptr, size);
 done_3:
-	bus_dmamem_free(page->tag, &page->seg, page->seg_count);
+	bus_dmamem_free(tag, utag->p_seg, seg_count);
+done_4:
+	/* utag is destroyed later */
+done_5:
+	/* reset most of the page cache */
+	pc->buffer = NULL;
+	pc->page_start = NULL;
+	pc->page_offset_buf = 0;
+	pc->page_offset_end = 0;
+	pc->map = NULL;
+	pc->tag = NULL;
+	pc->n_seg = 0;
+	pc->p_seg = NULL;
+	return (1);
+}
 
-done_4:
-	return (NULL);
+/*------------------------------------------------------------------------*
+ *	usbd_pc_free_mem - free DMA memory
+ *
+ * This function is NULL safe.
+ *------------------------------------------------------------------------*/
+void
+usbd_pc_free_mem(struct usbd_page_cache *pc)
+{
+	if (pc && pc->buffer) {
+		bus_dmamap_unload(pc->tag, pc->map);
+		bus_dmamap_destroy(pc->tag, pc->map);
+		bus_dmamem_unmap(pc->tag, pc->buffer,
+		    pc->page_offset_end - pc->page_offset_buf);
+		bus_dmamem_free(pc->tag, pc->p_seg, pc->n_seg);
+		free(pc->p_seg, M_USB);
+		pc->buffer = NULL;
+	}
+	return;
 }
 
+/*------------------------------------------------------------------------*
+ *	usbd_pc_load_mem - load virtual memory into DMA
+ *------------------------------------------------------------------------*/
 void
-usbd_mem_free_sub(struct usbd_page *page)
+usbd_pc_load_mem(struct usbd_page_cache *pc, uint32_t size)
 {
-	/*
-	 * NOTE: make a copy of "tag", "map", and "buffer" in case "page" is
-	 * part of the allocated memory:
-	 */
-	struct usbd_page temp = *page;
+	int error;
+
+	/* sanity check */
+	if (pc->xfer == NULL) {
+		panic("This page cache is not loadable!\n");
+		return;
+	}
+	/* setup page cache */
+	pc->page_offset_buf = 0;
+	pc->page_offset_end = size;
+
+	if (size > 0) {
+
+		pc->xfer->usb_root->dma_refcount++;
 
-	bus_dmamap_unload(temp.tag, temp.map);
-	bus_dmamap_destroy(temp.tag, temp.map);
-	bus_dmamem_unmap(temp.tag, temp.buffer, temp.length);
-	bus_dmamem_free(temp.tag, &temp.seg, temp.seg_count);
+		/* try to load memory into DMA */
+		if (bus_dmamap_load(pc->tag, pc->map, pc->buffer,
+		    size, NULL, BUS_DMA_NOWAIT)) {
+			error = ENOMEM;
+		} else {
+			error = 0;
+		}
 
-#ifdef USB_DEBUG
-	if (usbdebug > 14) {
-		printf("%s: %p\n",
-		    __FUNCTION__, temp.buffer);
+		usbd_pc_alloc_mem_cb(pc, pc->map->dm_segs,
+		    pc->map->dm_nsegs, error);
 	}
-#endif
 	return;
 }
 
@@ -2641,11 +2804,70 @@
 
 	len = pc->page_offset_end - pc->page_offset_buf;
 
-	bus_dmamap_sync(page->tag, page->map, 0, len,
+	bus_dmamap_sync(pc->tag, pc->map, 0, len,
 	    BUS_DMASYNC_PREWRITE | BUS_DMASYNC_PREREAD);
 	return;
 }
 
+/*------------------------------------------------------------------------*
+ *	usbd_pc_dmamap_create
+ *
+ * Returns:
+ *    0: Success
+ * Else: Failure
+ *------------------------------------------------------------------------*/
+uint8_t
+usbd_pc_dmamap_create(struct usbd_page_cache *pc, uint32_t size)
+{
+	struct usbd_memory_info *info;
+	struct usbd_dma_tag *utag;
+	bus_dma_tag_t tag;
+
+	/* sanity check */
+	if (pc->xfer == NULL) {
+		goto error;
+	}
+	info = pc->xfer->usb_root;
+	tag = pc->xfer->udev->bus->dma_tag_parent;
+
+	utag = usbd_dma_tag_setup(tag, info->dma_tag_p,
+	    size, 1, info->dma_tag_max);
+	if (utag == NULL) {
+		goto error;
+	}
+	if (bus_dmamap_create(utag->tag, size, utag->n_seg,
+	    USB_PAGE_SIZE, 0, BUS_DMA_WAITOK, &(pc->map))) {
+		goto error;
+	}
+	pc->tag = utag->tag;
+	pc->p_seg = utag->p_seg;
+	pc->n_seg = utag->n_seg;
+	return 0;			/* success */
+
+error:
+	pc->map = NULL;
+	pc->tag = NULL;
+	pc->p_seg = NULL;
+	pc->n_seg = 0;
+	return 1;			/* failure */
+}
+
+/*------------------------------------------------------------------------*
+ *	usbd_pc_dmamap_destroy
+ *
+ * This function is NULL safe.
+ *------------------------------------------------------------------------*/
+void
+usbd_pc_dmamap_destroy(struct usbd_page_cache *pc)
+{
+	if (pc && pc->tag) {
+		bus_dmamap_destroy(pc->tag, pc->map);
+		pc->tag = NULL;
+		pc->map = NULL;
+	}
+	return;
+}
+
 #endif
 
 /*------------------------------------------------------------------------*



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