Received: by 2002:a05:6a10:f347:0:0:0:0 with SMTP id d7csp1772177pxu; Tue, 24 Nov 2020 08:34:16 -0800 (PST) X-Google-Smtp-Source: ABdhPJw1Kb83ngVzWO00FWVTpeMIC3gbXk7XSmIEHpOv17Uc/3BD/pBaVG3Z7HVcLWSOs4RencBi X-Received: by 2002:a17:906:ceca:: with SMTP id si10mr4701850ejb.409.1606235656308; Tue, 24 Nov 2020 08:34:16 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1606235656; cv=none; d=google.com; s=arc-20160816; b=EyWGkubEGzmRErqdZNm8yVIb/E1A2pujfKaXKRwJ79p8eTXpCln/SFhPmtEiu+0QoB DkFw+iLybDktniFmrDLhEGnKxJYIuZ5OOhximKdKY7uSOjqqjTS65NBte2oV/5JZfiku nt46npVdcbyAYzlybW/5JXd9aFVVYEMUWMp9cTp91c8lhWQOJ/YqC5M3SMNZTTAc4zAx 7nYlelSrAyk18n+u/zZ7FaazuikWSUsiSex6ZcePAoZruc7IML7i34u7/5G1BeDZocS9 TQS7EILS4Cq/47pMcoa1E9Gn3Lng4QaIAb1SsLaTsXdl5CAHB3Am1aNh5iquYdXhhYlu EqOQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:content-language :in-reply-to:mime-version:user-agent:date:message-id:from:references :to:subject; bh=Xc/Fu6ku3odRoVWzmstP976Gs71rieavwGZfRKuXfZM=; b=UpE6z96fNqI6GuybFA01islF/0+wa3dGobiBwL+80El+cvP+2gAUigYL5cyYdhmPet cEiwor9rj+WULK5pMntHRwE9Q73o0CHtZ1jIszOAbOHtrKaQTGpUJQ1LfecaFORoy0IL AaliTu8ixSmwCWwGRjO2vNVHaYplyDTE3XI8FxGY8mgG0FGKwf9uK0JqDTius/44SFLh LqhRIFVdQgn/Q/HY+JxrXH43wGatHjjrNsOTXLHCD1ECk1vG4pfLpgDDFEpiJfr3ICzI 00ZC7WR9WbDFxzN2TPZJzDvpSCsbVPBEDJL6rhJzLQAyhDeOZcX/N87vQgnzHjKEojDi TqJg== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=arm.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id w9si9285792eds.193.2020.11.24.08.33.52; Tue, 24 Nov 2020 08:34:16 -0800 (PST) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=arm.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S2390487AbgKXQ31 (ORCPT + 99 others); Tue, 24 Nov 2020 11:29:27 -0500 Received: from foss.arm.com ([217.140.110.172]:42176 "EHLO foss.arm.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S2389808AbgKXQ30 (ORCPT ); Tue, 24 Nov 2020 11:29:26 -0500 Received: from usa-sjc-imap-foss1.foss.arm.com (unknown [10.121.207.14]) by usa-sjc-mx-foss1.foss.arm.com (Postfix) with ESMTP id A539E1396; Tue, 24 Nov 2020 08:29:25 -0800 (PST) Received: from [10.57.59.159] (unknown [10.57.59.159]) by usa-sjc-imap-foss1.foss.arm.com (Postfix) with ESMTPSA id 2DCF93F71F; Tue, 24 Nov 2020 08:29:22 -0800 (PST) Subject: Re: [PATCH 5/6] media: uvcvideo: Use dma_alloc_noncontiguos API To: Ricardo Ribalda , Christoph Hellwig , Mauro Carvalho Chehab , Marek Szyprowski , IOMMU DRIVERS , Joerg Roedel , Linux Doc Mailing List , Linux Kernel Mailing List , Linux Media Mailing List , Tomasz Figa , Sergey Senozhatsky References: <20201124153845.132207-1-ribalda@chromium.org> <20201124153845.132207-5-ribalda@chromium.org> From: Robin Murphy Message-ID: Date: Tue, 24 Nov 2020 16:29:20 +0000 User-Agent: Mozilla/5.0 (Windows NT 10.0; rv:78.0) Gecko/20100101 Thunderbird/78.5.0 MIME-Version: 1.0 In-Reply-To: <20201124153845.132207-5-ribalda@chromium.org> Content-Type: text/plain; charset=utf-8; format=flowed Content-Language: en-GB Content-Transfer-Encoding: 7bit Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On 2020-11-24 15:38, Ricardo Ribalda wrote: > On architectures where the is no coherent caching such as ARM use the > dma_alloc_noncontiguos API and handle manually the cache flushing using > dma_sync_single(). > > With this patch on the affected architectures we can measure up to 20x > performance improvement in uvc_video_copy_data_work(). > > Signed-off-by: Ricardo Ribalda > --- > drivers/media/usb/uvc/uvc_video.c | 74 ++++++++++++++++++++++++++----- > drivers/media/usb/uvc/uvcvideo.h | 1 + > 2 files changed, 63 insertions(+), 12 deletions(-) > > diff --git a/drivers/media/usb/uvc/uvc_video.c b/drivers/media/usb/uvc/uvc_video.c > index a6a441d92b94..9e90b261428a 100644 > --- a/drivers/media/usb/uvc/uvc_video.c > +++ b/drivers/media/usb/uvc/uvc_video.c > @@ -1490,6 +1490,11 @@ static void uvc_video_encode_bulk(struct uvc_urb *uvc_urb, > urb->transfer_buffer_length = stream->urb_size - len; > } > > +static inline struct device *stream_to_dmadev(struct uvc_streaming *stream) > +{ > + return stream->dev->udev->bus->controller->parent; > +} > + > static void uvc_video_complete(struct urb *urb) > { > struct uvc_urb *uvc_urb = urb->context; > @@ -1539,6 +1544,11 @@ static void uvc_video_complete(struct urb *urb) > * Process the URB headers, and optionally queue expensive memcpy tasks > * to be deferred to a work queue. > */ > + if (uvc_urb->pages) > + dma_sync_single_for_cpu(stream_to_dmadev(stream), > + urb->transfer_dma, > + urb->transfer_buffer_length, > + DMA_FROM_DEVICE); This doesn't work. Even in iommu-dma, the streaming API still expects to work on physically-contiguous memory that could have been passed to dma_map_single() in the first place. As-is, this will invalidate transfer_buffer_length bytes from the start of the *first* physical page, and thus destroy random other data if lines from subsequent unrelated pages are dirty in caches. The only feasible way to do a DMA sync on disjoint pages in a single call is with a scatterlist. Robin. > stream->decode(uvc_urb, buf, buf_meta); > > /* If no async work is needed, resubmit the URB immediately. */ > @@ -1566,8 +1576,15 @@ static void uvc_free_urb_buffers(struct uvc_streaming *stream) > continue; > > #ifndef CONFIG_DMA_NONCOHERENT > - usb_free_coherent(stream->dev->udev, stream->urb_size, > - uvc_urb->buffer, uvc_urb->dma); > + if (uvc_urb->pages) { > + vunmap(uvc_urb->buffer); > + dma_free_noncontiguous(stream_to_dmadev(stream), > + stream->urb_size, > + uvc_urb->pages, uvc_urb->dma); > + } else { > + usb_free_coherent(stream->dev->udev, stream->urb_size, > + uvc_urb->buffer, uvc_urb->dma); > + } > #else > kfree(uvc_urb->buffer); > #endif > @@ -1577,6 +1594,47 @@ static void uvc_free_urb_buffers(struct uvc_streaming *stream) > stream->urb_size = 0; > } > > +#ifndef CONFIG_DMA_NONCOHERENT > +static bool uvc_alloc_urb_buffer(struct uvc_streaming *stream, > + struct uvc_urb *uvc_urb, gfp_t gfp_flags) > +{ > + struct device *dma_dev = dma_dev = stream_to_dmadev(stream); > + > + if (!dma_can_alloc_noncontiguous(dma_dev)) { > + uvc_urb->buffer = usb_alloc_coherent(stream->dev->udev, > + stream->urb_size, > + gfp_flags | __GFP_NOWARN, > + &uvc_urb->dma); > + return uvc_urb->buffer != NULL; > + } > + > + uvc_urb->pages = dma_alloc_noncontiguous(dma_dev, stream->urb_size, > + &uvc_urb->dma, > + gfp_flags | __GFP_NOWARN, 0); > + if (!uvc_urb->pages) > + return false; > + > + uvc_urb->buffer = vmap(uvc_urb->pages, > + PAGE_ALIGN(stream->urb_size) >> PAGE_SHIFT, > + VM_DMA_COHERENT, PAGE_KERNEL); > + if (!uvc_urb->buffer) { > + dma_free_noncontiguous(dma_dev, stream->urb_size, > + uvc_urb->pages, uvc_urb->dma); > + return false; > + } > + > + return true; > +} > +#else > +static bool uvc_alloc_urb_buffer(struct uvc_streaming *stream, > + struct uvc_urb *uvc_urb, gfp_t gfp_flags) > +{ > + uvc_urb->buffer = kmalloc(stream->urb_size, gfp_flags | __GFP_NOWARN); > + > + return uvc_urb->buffer != NULL; > +} > +#endif > + > /* > * Allocate transfer buffers. This function can be called with buffers > * already allocated when resuming from suspend, in which case it will > @@ -1607,19 +1665,11 @@ static int uvc_alloc_urb_buffers(struct uvc_streaming *stream, > > /* Retry allocations until one succeed. */ > for (; npackets > 1; npackets /= 2) { > + stream->urb_size = psize * npackets; > for (i = 0; i < UVC_URBS; ++i) { > struct uvc_urb *uvc_urb = &stream->uvc_urb[i]; > > - stream->urb_size = psize * npackets; > -#ifndef CONFIG_DMA_NONCOHERENT > - uvc_urb->buffer = usb_alloc_coherent( > - stream->dev->udev, stream->urb_size, > - gfp_flags | __GFP_NOWARN, &uvc_urb->dma); > -#else > - uvc_urb->buffer = > - kmalloc(stream->urb_size, gfp_flags | __GFP_NOWARN); > -#endif > - if (!uvc_urb->buffer) { > + if (!uvc_alloc_urb_buffer(stream, uvc_urb, gfp_flags)) { > uvc_free_urb_buffers(stream); > break; > } > diff --git a/drivers/media/usb/uvc/uvcvideo.h b/drivers/media/usb/uvc/uvcvideo.h > index a3dfacf069c4..3e3ef1f1daa5 100644 > --- a/drivers/media/usb/uvc/uvcvideo.h > +++ b/drivers/media/usb/uvc/uvcvideo.h > @@ -532,6 +532,7 @@ struct uvc_urb { > > char *buffer; > dma_addr_t dma; > + struct page **pages; > > unsigned int async_operations; > struct uvc_copy_op copy_operations[UVC_MAX_PACKETS]; >