Commit aabd1260 authored by Petr Tesarik's avatar Petr Tesarik Committed by Christoph Hellwig

swiotlb: always set the number of areas before allocating the pool

The number of areas defaults to the number of possible CPUs. However, the
total number of slots may have to be increased after adjusting the number
of areas. Consequently, the number of areas must be determined before
allocating the memory pool. This is even explained with a comment in
swiotlb_init_remap(), but swiotlb_init_late() adjusts the number of areas
after slots are already allocated. The areas may end up being smaller than
IO_TLB_SEGSIZE, which breaks per-area locking.

While fixing swiotlb_init_late(), move all relevant comments before the
definition of swiotlb_adjust_nareas() and convert them to kernel-doc.

Fixes: 20347fca ("swiotlb: split up the global swiotlb lock")
Signed-off-by: default avatarPetr Tesarik <petr.tesarik.ext@huawei.com>
Reviewed-by: default avatarRoberto Sassu <roberto.sassu@huawei.com>
Signed-off-by: default avatarChristoph Hellwig <hch@lst.de>
parent 0a2f6372
...@@ -115,9 +115,16 @@ static bool round_up_default_nslabs(void) ...@@ -115,9 +115,16 @@ static bool round_up_default_nslabs(void)
return true; return true;
} }
/**
* swiotlb_adjust_nareas() - adjust the number of areas and slots
* @nareas: Desired number of areas. Zero is treated as 1.
*
* Adjust the default number of areas in a memory pool.
* The default size of the memory pool may also change to meet minimum area
* size requirements.
*/
static void swiotlb_adjust_nareas(unsigned int nareas) static void swiotlb_adjust_nareas(unsigned int nareas)
{ {
/* use a single area when non is specified */
if (!nareas) if (!nareas)
nareas = 1; nareas = 1;
else if (!is_power_of_2(nareas)) else if (!is_power_of_2(nareas))
...@@ -298,10 +305,6 @@ void __init swiotlb_init_remap(bool addressing_limit, unsigned int flags, ...@@ -298,10 +305,6 @@ void __init swiotlb_init_remap(bool addressing_limit, unsigned int flags,
if (swiotlb_force_disable) if (swiotlb_force_disable)
return; return;
/*
* default_nslabs maybe changed when adjust area number.
* So allocate bounce buffer after adjusting area number.
*/
if (!default_nareas) if (!default_nareas)
swiotlb_adjust_nareas(num_possible_cpus()); swiotlb_adjust_nareas(num_possible_cpus());
...@@ -363,6 +366,9 @@ int swiotlb_init_late(size_t size, gfp_t gfp_mask, ...@@ -363,6 +366,9 @@ int swiotlb_init_late(size_t size, gfp_t gfp_mask,
if (swiotlb_force_disable) if (swiotlb_force_disable)
return 0; return 0;
if (!default_nareas)
swiotlb_adjust_nareas(num_possible_cpus());
retry: retry:
order = get_order(nslabs << IO_TLB_SHIFT); order = get_order(nslabs << IO_TLB_SHIFT);
nslabs = SLABS_PER_PAGE << order; nslabs = SLABS_PER_PAGE << order;
...@@ -397,9 +403,6 @@ int swiotlb_init_late(size_t size, gfp_t gfp_mask, ...@@ -397,9 +403,6 @@ int swiotlb_init_late(size_t size, gfp_t gfp_mask,
(PAGE_SIZE << order) >> 20); (PAGE_SIZE << order) >> 20);
} }
if (!default_nareas)
swiotlb_adjust_nareas(num_possible_cpus());
area_order = get_order(array_size(sizeof(*mem->areas), area_order = get_order(array_size(sizeof(*mem->areas),
default_nareas)); default_nareas));
mem->areas = (struct io_tlb_area *) mem->areas = (struct io_tlb_area *)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment