All of lore.kernel.org
 help / color / mirror / Atom feed
* [PATCH] ARM: mm: Do not invoke OOM for higher order IOMMU DMA allocations
@ 2015-03-16  8:12 ` Tomasz Figa
  0 siblings, 0 replies; 6+ messages in thread
From: Tomasz Figa @ 2015-03-16  8:12 UTC (permalink / raw)
  To: linux-arm-kernel
  Cc: linux-kernel, Russell King, Marek Szyprowski, Will Deacon,
	Andrew Morton, Laura Abbott, Ritesh Harjani, Santosh Shilimkar,
	Carlo Caione, Laurent Pinchart, Douglas Anderson, Sonny Rao,
	Tomasz Figa

IOMMU should be able to use single pages as well as bigger blocks, so if
higher order allocations fail, we should not affect state of the system,
with events such as OOM killer, but rather fall back to order 0
allocations.

This patch changes the behavior of ARM IOMMU DMA allocator to use
__GFP_NORETRY, which bypasses OOM invocation, for positive orders and
only if that fails doing OOMable order 0 allocation as a fall back.

Signed-off-by: Tomasz Figa <tfiga@chromium.org>
---
 arch/arm/mm/dma-mapping.c | 29 +++++++++++++++++++++--------
 1 file changed, 21 insertions(+), 8 deletions(-)

diff --git a/arch/arm/mm/dma-mapping.c b/arch/arm/mm/dma-mapping.c
index 83cd5ac..f081e9e 100644
--- a/arch/arm/mm/dma-mapping.c
+++ b/arch/arm/mm/dma-mapping.c
@@ -1145,18 +1145,31 @@ static struct page **__iommu_alloc_buffer(struct device *dev, size_t size,
 	}
 
 	/*
-	 * IOMMU can map any pages, so himem can also be used here
+	 * IOMMU can map any pages, so himem can also be used here.
+	 * We do not want OOM killer to be invoked as long as we can fall back
+	 * to single pages, so we use __GFP_NORETRY for positive orders.
 	 */
-	gfp |= __GFP_NOWARN | __GFP_HIGHMEM;
+	gfp |= __GFP_NOWARN | __GFP_HIGHMEM | __GFP_NORETRY;
 
 	while (count) {
-		int j, order = __fls(count);
+		int j, order;
 
-		pages[i] = alloc_pages(gfp, order);
-		while (!pages[i] && order)
-			pages[i] = alloc_pages(gfp, --order);
-		if (!pages[i])
-			goto error;
+		for (order = __fls(count); order; --order) {
+			/* Will not trigger OOM. */
+			pages[i] = alloc_pages(gfp, order);
+			if (pages[i])
+				break;
+		}
+
+		if (!pages[i]) {
+			/*
+			 * Fall back to single page allocation.
+			 * Might invoke OOM killer as last resort.
+			 */
+			pages[i] = alloc_pages(gfp & ~__GFP_NORETRY, 0);
+			if (!pages[i])
+				goto error;
+		}
 
 		if (order) {
 			split_page(pages[i], order);
-- 
2.1.2


^ permalink raw reply related	[flat|nested] 6+ messages in thread

* [PATCH] ARM: mm: Do not invoke OOM for higher order IOMMU DMA allocations
@ 2015-03-16  8:12 ` Tomasz Figa
  0 siblings, 0 replies; 6+ messages in thread
From: Tomasz Figa @ 2015-03-16  8:12 UTC (permalink / raw)
  To: linux-arm-kernel

IOMMU should be able to use single pages as well as bigger blocks, so if
higher order allocations fail, we should not affect state of the system,
with events such as OOM killer, but rather fall back to order 0
allocations.

This patch changes the behavior of ARM IOMMU DMA allocator to use
__GFP_NORETRY, which bypasses OOM invocation, for positive orders and
only if that fails doing OOMable order 0 allocation as a fall back.

Signed-off-by: Tomasz Figa <tfiga@chromium.org>
---
 arch/arm/mm/dma-mapping.c | 29 +++++++++++++++++++++--------
 1 file changed, 21 insertions(+), 8 deletions(-)

diff --git a/arch/arm/mm/dma-mapping.c b/arch/arm/mm/dma-mapping.c
index 83cd5ac..f081e9e 100644
--- a/arch/arm/mm/dma-mapping.c
+++ b/arch/arm/mm/dma-mapping.c
@@ -1145,18 +1145,31 @@ static struct page **__iommu_alloc_buffer(struct device *dev, size_t size,
 	}
 
 	/*
-	 * IOMMU can map any pages, so himem can also be used here
+	 * IOMMU can map any pages, so himem can also be used here.
+	 * We do not want OOM killer to be invoked as long as we can fall back
+	 * to single pages, so we use __GFP_NORETRY for positive orders.
 	 */
-	gfp |= __GFP_NOWARN | __GFP_HIGHMEM;
+	gfp |= __GFP_NOWARN | __GFP_HIGHMEM | __GFP_NORETRY;
 
 	while (count) {
-		int j, order = __fls(count);
+		int j, order;
 
-		pages[i] = alloc_pages(gfp, order);
-		while (!pages[i] && order)
-			pages[i] = alloc_pages(gfp, --order);
-		if (!pages[i])
-			goto error;
+		for (order = __fls(count); order; --order) {
+			/* Will not trigger OOM. */
+			pages[i] = alloc_pages(gfp, order);
+			if (pages[i])
+				break;
+		}
+
+		if (!pages[i]) {
+			/*
+			 * Fall back to single page allocation.
+			 * Might invoke OOM killer as last resort.
+			 */
+			pages[i] = alloc_pages(gfp & ~__GFP_NORETRY, 0);
+			if (!pages[i])
+				goto error;
+		}
 
 		if (order) {
 			split_page(pages[i], order);
-- 
2.1.2

^ permalink raw reply related	[flat|nested] 6+ messages in thread

* Re: [PATCH] ARM: mm: Do not invoke OOM for higher order IOMMU DMA allocations
  2015-03-16  8:12 ` Tomasz Figa
@ 2015-03-16 23:32   ` David Rientjes
  -1 siblings, 0 replies; 6+ messages in thread
From: David Rientjes @ 2015-03-16 23:32 UTC (permalink / raw)
  To: Tomasz Figa
  Cc: linux-arm-kernel, linux-kernel, Russell King, Marek Szyprowski,
	Will Deacon, Andrew Morton, Laura Abbott, Ritesh Harjani,
	Santosh Shilimkar, Carlo Caione, Laurent Pinchart,
	Douglas Anderson, Sonny Rao

On Mon, 16 Mar 2015, Tomasz Figa wrote:

> diff --git a/arch/arm/mm/dma-mapping.c b/arch/arm/mm/dma-mapping.c
> index 83cd5ac..f081e9e 100644
> --- a/arch/arm/mm/dma-mapping.c
> +++ b/arch/arm/mm/dma-mapping.c
> @@ -1145,18 +1145,31 @@ static struct page **__iommu_alloc_buffer(struct device *dev, size_t size,
>  	}
>  
>  	/*
> -	 * IOMMU can map any pages, so himem can also be used here
> +	 * IOMMU can map any pages, so himem can also be used here.
> +	 * We do not want OOM killer to be invoked as long as we can fall back
> +	 * to single pages, so we use __GFP_NORETRY for positive orders.
>  	 */
> -	gfp |= __GFP_NOWARN | __GFP_HIGHMEM;
> +	gfp |= __GFP_NOWARN | __GFP_HIGHMEM | __GFP_NORETRY;
>  
>  	while (count) {
> -		int j, order = __fls(count);
> +		int j, order;
>  
> -		pages[i] = alloc_pages(gfp, order);
> -		while (!pages[i] && order)
> -			pages[i] = alloc_pages(gfp, --order);
> -		if (!pages[i])
> -			goto error;
> +		for (order = __fls(count); order; --order) {
> +			/* Will not trigger OOM. */
> +			pages[i] = alloc_pages(gfp, order);
> +			if (pages[i])
> +				break;
> +		}
> +
> +		if (!pages[i]) {
> +			/*
> +			 * Fall back to single page allocation.
> +			 * Might invoke OOM killer as last resort.
> +			 */
> +			pages[i] = alloc_pages(gfp & ~__GFP_NORETRY, 0);
> +			if (!pages[i])
> +				goto error;
> +		}
>  
>  		if (order) {
>  			split_page(pages[i], order);

I think this makes sense, but the problem is the unconditional setting and 
clearing of __GFP_NORETRY.  Strictly speaking, gfp may already have 
__GFP_NORETRY set when calling this function so it would be better to do 
the loop with alloc_pages(gfp | __GFP_NORETRY, order) and then the 
fallback as alloc_page(gfp).

^ permalink raw reply	[flat|nested] 6+ messages in thread

* [PATCH] ARM: mm: Do not invoke OOM for higher order IOMMU DMA allocations
@ 2015-03-16 23:32   ` David Rientjes
  0 siblings, 0 replies; 6+ messages in thread
From: David Rientjes @ 2015-03-16 23:32 UTC (permalink / raw)
  To: linux-arm-kernel

On Mon, 16 Mar 2015, Tomasz Figa wrote:

> diff --git a/arch/arm/mm/dma-mapping.c b/arch/arm/mm/dma-mapping.c
> index 83cd5ac..f081e9e 100644
> --- a/arch/arm/mm/dma-mapping.c
> +++ b/arch/arm/mm/dma-mapping.c
> @@ -1145,18 +1145,31 @@ static struct page **__iommu_alloc_buffer(struct device *dev, size_t size,
>  	}
>  
>  	/*
> -	 * IOMMU can map any pages, so himem can also be used here
> +	 * IOMMU can map any pages, so himem can also be used here.
> +	 * We do not want OOM killer to be invoked as long as we can fall back
> +	 * to single pages, so we use __GFP_NORETRY for positive orders.
>  	 */
> -	gfp |= __GFP_NOWARN | __GFP_HIGHMEM;
> +	gfp |= __GFP_NOWARN | __GFP_HIGHMEM | __GFP_NORETRY;
>  
>  	while (count) {
> -		int j, order = __fls(count);
> +		int j, order;
>  
> -		pages[i] = alloc_pages(gfp, order);
> -		while (!pages[i] && order)
> -			pages[i] = alloc_pages(gfp, --order);
> -		if (!pages[i])
> -			goto error;
> +		for (order = __fls(count); order; --order) {
> +			/* Will not trigger OOM. */
> +			pages[i] = alloc_pages(gfp, order);
> +			if (pages[i])
> +				break;
> +		}
> +
> +		if (!pages[i]) {
> +			/*
> +			 * Fall back to single page allocation.
> +			 * Might invoke OOM killer as last resort.
> +			 */
> +			pages[i] = alloc_pages(gfp & ~__GFP_NORETRY, 0);
> +			if (!pages[i])
> +				goto error;
> +		}
>  
>  		if (order) {
>  			split_page(pages[i], order);

I think this makes sense, but the problem is the unconditional setting and 
clearing of __GFP_NORETRY.  Strictly speaking, gfp may already have 
__GFP_NORETRY set when calling this function so it would be better to do 
the loop with alloc_pages(gfp | __GFP_NORETRY, order) and then the 
fallback as alloc_page(gfp).

^ permalink raw reply	[flat|nested] 6+ messages in thread

* Re: [PATCH] ARM: mm: Do not invoke OOM for higher order IOMMU DMA allocations
  2015-03-16 23:32   ` David Rientjes
@ 2015-03-17  1:19     ` Tomasz Figa
  -1 siblings, 0 replies; 6+ messages in thread
From: Tomasz Figa @ 2015-03-17  1:19 UTC (permalink / raw)
  To: David Rientjes
  Cc: linux-arm-kernel, linux-kernel, Russell King, Marek Szyprowski,
	Will Deacon, Andrew Morton, Laura Abbott, Ritesh Harjani,
	Santosh Shilimkar, Carlo Caione, Laurent Pinchart,
	Douglas Anderson, Sonny Rao

Hi David,

On Tue, Mar 17, 2015 at 8:32 AM, David Rientjes <rientjes@google.com> wrote:
> On Mon, 16 Mar 2015, Tomasz Figa wrote:
>
>> diff --git a/arch/arm/mm/dma-mapping.c b/arch/arm/mm/dma-mapping.c
>> index 83cd5ac..f081e9e 100644
>> --- a/arch/arm/mm/dma-mapping.c
>> +++ b/arch/arm/mm/dma-mapping.c
>> @@ -1145,18 +1145,31 @@ static struct page **__iommu_alloc_buffer(struct device *dev, size_t size,
>>       }
>>
>>       /*
>> -      * IOMMU can map any pages, so himem can also be used here
>> +      * IOMMU can map any pages, so himem can also be used here.
>> +      * We do not want OOM killer to be invoked as long as we can fall back
>> +      * to single pages, so we use __GFP_NORETRY for positive orders.
>>        */
>> -     gfp |= __GFP_NOWARN | __GFP_HIGHMEM;
>> +     gfp |= __GFP_NOWARN | __GFP_HIGHMEM | __GFP_NORETRY;
>>
>>       while (count) {
>> -             int j, order = __fls(count);
>> +             int j, order;
>>
>> -             pages[i] = alloc_pages(gfp, order);
>> -             while (!pages[i] && order)
>> -                     pages[i] = alloc_pages(gfp, --order);
>> -             if (!pages[i])
>> -                     goto error;
>> +             for (order = __fls(count); order; --order) {
>> +                     /* Will not trigger OOM. */
>> +                     pages[i] = alloc_pages(gfp, order);
>> +                     if (pages[i])
>> +                             break;
>> +             }
>> +
>> +             if (!pages[i]) {
>> +                     /*
>> +                      * Fall back to single page allocation.
>> +                      * Might invoke OOM killer as last resort.
>> +                      */
>> +                     pages[i] = alloc_pages(gfp & ~__GFP_NORETRY, 0);
>> +                     if (!pages[i])
>> +                             goto error;
>> +             }
>>
>>               if (order) {
>>                       split_page(pages[i], order);
>
> I think this makes sense, but the problem is the unconditional setting and
> clearing of __GFP_NORETRY.  Strictly speaking, gfp may already have
> __GFP_NORETRY set when calling this function so it would be better to do
> the loop with alloc_pages(gfp | __GFP_NORETRY, order) and then the
> fallback as alloc_page(gfp).

Good point. I'll change it to that in next version.

Best regards,
Tomasz

^ permalink raw reply	[flat|nested] 6+ messages in thread

* [PATCH] ARM: mm: Do not invoke OOM for higher order IOMMU DMA allocations
@ 2015-03-17  1:19     ` Tomasz Figa
  0 siblings, 0 replies; 6+ messages in thread
From: Tomasz Figa @ 2015-03-17  1:19 UTC (permalink / raw)
  To: linux-arm-kernel

Hi David,

On Tue, Mar 17, 2015 at 8:32 AM, David Rientjes <rientjes@google.com> wrote:
> On Mon, 16 Mar 2015, Tomasz Figa wrote:
>
>> diff --git a/arch/arm/mm/dma-mapping.c b/arch/arm/mm/dma-mapping.c
>> index 83cd5ac..f081e9e 100644
>> --- a/arch/arm/mm/dma-mapping.c
>> +++ b/arch/arm/mm/dma-mapping.c
>> @@ -1145,18 +1145,31 @@ static struct page **__iommu_alloc_buffer(struct device *dev, size_t size,
>>       }
>>
>>       /*
>> -      * IOMMU can map any pages, so himem can also be used here
>> +      * IOMMU can map any pages, so himem can also be used here.
>> +      * We do not want OOM killer to be invoked as long as we can fall back
>> +      * to single pages, so we use __GFP_NORETRY for positive orders.
>>        */
>> -     gfp |= __GFP_NOWARN | __GFP_HIGHMEM;
>> +     gfp |= __GFP_NOWARN | __GFP_HIGHMEM | __GFP_NORETRY;
>>
>>       while (count) {
>> -             int j, order = __fls(count);
>> +             int j, order;
>>
>> -             pages[i] = alloc_pages(gfp, order);
>> -             while (!pages[i] && order)
>> -                     pages[i] = alloc_pages(gfp, --order);
>> -             if (!pages[i])
>> -                     goto error;
>> +             for (order = __fls(count); order; --order) {
>> +                     /* Will not trigger OOM. */
>> +                     pages[i] = alloc_pages(gfp, order);
>> +                     if (pages[i])
>> +                             break;
>> +             }
>> +
>> +             if (!pages[i]) {
>> +                     /*
>> +                      * Fall back to single page allocation.
>> +                      * Might invoke OOM killer as last resort.
>> +                      */
>> +                     pages[i] = alloc_pages(gfp & ~__GFP_NORETRY, 0);
>> +                     if (!pages[i])
>> +                             goto error;
>> +             }
>>
>>               if (order) {
>>                       split_page(pages[i], order);
>
> I think this makes sense, but the problem is the unconditional setting and
> clearing of __GFP_NORETRY.  Strictly speaking, gfp may already have
> __GFP_NORETRY set when calling this function so it would be better to do
> the loop with alloc_pages(gfp | __GFP_NORETRY, order) and then the
> fallback as alloc_page(gfp).

Good point. I'll change it to that in next version.

Best regards,
Tomasz

^ permalink raw reply	[flat|nested] 6+ messages in thread

end of thread, other threads:[~2015-03-17  1:20 UTC | newest]

Thread overview: 6+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2015-03-16  8:12 [PATCH] ARM: mm: Do not invoke OOM for higher order IOMMU DMA allocations Tomasz Figa
2015-03-16  8:12 ` Tomasz Figa
2015-03-16 23:32 ` David Rientjes
2015-03-16 23:32   ` David Rientjes
2015-03-17  1:19   ` Tomasz Figa
2015-03-17  1:19     ` Tomasz Figa

This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.