Received: by 2002:a05:6a10:f347:0:0:0:0 with SMTP id d7csp1814000pxu; Tue, 24 Nov 2020 09:29:26 -0800 (PST) X-Google-Smtp-Source: ABdhPJzlSI+Ae1JD3g5qQ85O2tx936diBEoHsPy164ZZZzqOw+o2KHn8jAjGz+pbnRtv36QPlH9Q X-Received: by 2002:a17:906:854c:: with SMTP id h12mr5128217ejy.212.1606238966055; Tue, 24 Nov 2020 09:29:26 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1606238966; cv=none; d=google.com; s=arc-20160816; b=XoqQ9p53XM8PqkDVKrkx3tAtSEtmduOEwddcIBkUnKqvBemwF/DepsF4buxGvI8ip3 PIOmILELYw/AC4M9+a4Ezu9k8rx/rFcBOCmUVMIQ8X3fLGnnDuT98rFEYK6xcUxbNhhq 7Q/BF2xv4BkLE8qc3/LyVv5nwtDvXNVTzlaZBCtGvZqczyJ5cXd184yHrm+CCxpVHrgb cy+fHCjuqJZEo59ShTFL98mvxA7zuBzvaVu1X5xBeDu28mptg+o9dp2UDPph//TUuKxG 6nxXZVPxtYOPW31njiNPmbsjgAiO7JJlKNoX1X/0caBXRg0geUHW0AHCASSqZz2K8lhU 2Osw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:cc:to:subject:message-id:date:from:in-reply-to :references:mime-version:dkim-signature; bh=Z6tlEwV2va2PDMozZGMdT895LM4tZsvIbgGRKWaEoV4=; b=DsjoBlqKamqRgzdXy21DHO9/R37mtZr0jxB9Ko1T7QbHWn4B8eM/9wAKxtQbFoxCI4 J8SpWwaPQX4pVbyozbkARpLk7Bi8/LOPOj1xn0G4zC/A/hyjcW9nLc7ccfSHKxcuTKLk WH4931F6t4BGagnjwwa/3iKR0EWGupwN5viyGqKBAZ97R8EMrZgedJfc9zjiDKCv0Kpp 7QfyRHjHbacvYlEHOjD4tD7pdhQkDH2RhnYxW3I5VW8CayVtJqOv39uq1mIULiAbfsXx xAhXddHDVNy1vklDCbZQzyP7ubqAjcvTf55wKatPM0D4/6x5px/XR8JfU+CTlQVpjrAr aVeA== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@chromium.org header.s=google header.b=O1otos6r; 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=pass (p=NONE sp=NONE dis=NONE) header.from=chromium.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id s12si9022878ejd.753.2020.11.24.09.29.02; Tue, 24 Nov 2020 09:29:26 -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; dkim=pass header.i=@chromium.org header.s=google header.b=O1otos6r; 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=pass (p=NONE sp=NONE dis=NONE) header.from=chromium.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S2390800AbgKXRZq (ORCPT + 99 others); Tue, 24 Nov 2020 12:25:46 -0500 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:42398 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S2390794AbgKXRZq (ORCPT ); Tue, 24 Nov 2020 12:25:46 -0500 Received: from mail-il1-x142.google.com (mail-il1-x142.google.com [IPv6:2607:f8b0:4864:20::142]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 945C0C0617A6 for ; Tue, 24 Nov 2020 09:25:44 -0800 (PST) Received: by mail-il1-x142.google.com with SMTP id a19so14235553ilm.3 for ; Tue, 24 Nov 2020 09:25:44 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=chromium.org; s=google; h=mime-version:references:in-reply-to:from:date:message-id:subject:to :cc; bh=Z6tlEwV2va2PDMozZGMdT895LM4tZsvIbgGRKWaEoV4=; b=O1otos6rFw3st6xur2OtmZ59kmbeoRug0Ytr6D48haYnFV/4i5YTnbi7zgXShcxOYw qW4Lxi/kxwNU/9Q+YW3586WdhZjbM0GSSV2zXURBFZGjJuH+gfoFZj+6309L5sfOP7Uz ZNE/98YEWZrRlCURKbfdWnqDbiBV906YsvYKY= X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:mime-version:references:in-reply-to:from:date :message-id:subject:to:cc; bh=Z6tlEwV2va2PDMozZGMdT895LM4tZsvIbgGRKWaEoV4=; b=Zw7KeN5XrtAz6KBmwGHI0Wui/qx2TGH44B7utzUlZSz3YHCH1dLEEeG6G7jIL1N/Y2 l30pijExwVhUDYe8uBZgiouJYyHlw+KlrtkcSy47/pmuXDdqdUorrfhE3/MMeL/yl1Fc 99SflrsiUbslacLc/xA0+hzx/pHBzWyiGCrr43rR94vx83v+eUIZT6nF+1+3W2QbAnRY KMUtHgBsM4UnNlgkoylfqjRXNuyvLbLPJFCZdgwyHNWybmXrLSRtZe5MZNl+1XTj/65N kZNglLI7GbMBRLdugPukitVATWjClUDGxkd9K3eMLqlYX+KO+lsVjAZKw6Fr7ZRx3Zmh +ljw== X-Gm-Message-State: AOAM5300AClXiEXDuz8P4EBgyf8S/ItcsUB/PeLkvJwu/JwrstlJVQ0D ztD9Kn14CMwQGEmw4bx6rAx+Cat+aySu8VV5 X-Received: by 2002:a92:5:: with SMTP id 5mr4995143ila.150.1606238743226; Tue, 24 Nov 2020 09:25:43 -0800 (PST) Received: from mail-io1-f43.google.com (mail-io1-f43.google.com. [209.85.166.43]) by smtp.gmail.com with ESMTPSA id k13sm5166036ioh.30.2020.11.24.09.25.42 for (version=TLS1_3 cipher=TLS_AES_128_GCM_SHA256 bits=128/128); Tue, 24 Nov 2020 09:25:42 -0800 (PST) Received: by mail-io1-f43.google.com with SMTP id t8so22691173iov.8 for ; Tue, 24 Nov 2020 09:25:42 -0800 (PST) X-Received: by 2002:a02:b144:: with SMTP id s4mr5587937jah.32.1606238741494; Tue, 24 Nov 2020 09:25:41 -0800 (PST) MIME-Version: 1.0 References: <20201124153845.132207-1-ribalda@chromium.org> <20201124153845.132207-5-ribalda@chromium.org> In-Reply-To: From: Ricardo Ribalda Date: Tue, 24 Nov 2020 18:25:30 +0100 X-Gmail-Original-Message-ID: Message-ID: Subject: Re: [PATCH 5/6] media: uvcvideo: Use dma_alloc_noncontiguos API To: Robin Murphy Cc: 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 Content-Type: text/plain; charset="UTF-8" Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Hi Robin On Tue, Nov 24, 2020 at 5:29 PM Robin Murphy wrote: > > 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. Thanks for pointing this out. I guess I was lucky on my hardware and the areas were always contiguous. Will rework and send back to the list. Thanks again. > > 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]; > > -- Ricardo Ribalda