dma-mapping: introduce dma_get_seg_boundary_nr_pages()
authorNicolin Chen <nicoleotsuka@gmail.com>
Tue, 1 Sep 2020 22:16:45 +0000 (15:16 -0700)
committerChristoph Hellwig <hch@lst.de>
Thu, 3 Sep 2020 16:12:15 +0000 (18:12 +0200)
commit1e9d90dbed120ec98517428ffff4dacd9797e39d
treed743be6b54427bb62e36e6d1ae2f6fc243cd70b7
parent2281f797f5524abb8fff66bf8540b4f4687332a2
dma-mapping: introduce dma_get_seg_boundary_nr_pages()

We found that callers of dma_get_seg_boundary mostly do an ALIGN
with page mask and then do a page shift to get number of pages:
    ALIGN(boundary + 1, 1 << shift) >> shift

However, the boundary might be as large as ULONG_MAX, which means
that a device has no specific boundary limit. So either "+ 1" or
passing it to ALIGN() would potentially overflow.

According to kernel defines:
    #define ALIGN_MASK(x, mask) (((x) + (mask)) & ~(mask))
    #define ALIGN(x, a) ALIGN_MASK(x, (typeof(x))(a) - 1)

We can simplify the logic here into a helper function doing:
  ALIGN(boundary + 1, 1 << shift) >> shift
= ALIGN_MASK(b + 1, (1 << s) - 1) >> s
= {[b + 1 + (1 << s) - 1] & ~[(1 << s) - 1]} >> s
= [b + 1 + (1 << s) - 1] >> s
= [b + (1 << s)] >> s
= (b >> s) + 1

This patch introduces and applies dma_get_seg_boundary_nr_pages()
as an overflow-free helper for the dma_get_seg_boundary() callers
to get numbers of pages. It also takes care of the NULL dev case
for non-DMA API callers.

Suggested-by: Christoph Hellwig <hch@lst.de>
Signed-off-by: Nicolin Chen <nicoleotsuka@gmail.com>
Acked-by: Niklas Schnelle <schnelle@linux.ibm.com>
Acked-by: Michael Ellerman <mpe@ellerman.id.au> (powerpc)
Signed-off-by: Christoph Hellwig <hch@lst.de>
arch/alpha/kernel/pci_iommu.c
arch/ia64/hp/common/sba_iommu.c
arch/powerpc/kernel/iommu.c
arch/s390/pci/pci_dma.c
arch/sparc/kernel/iommu-common.c
arch/sparc/kernel/iommu.c
arch/sparc/kernel/pci_sun4v.c
arch/x86/kernel/amd_gart_64.c
drivers/parisc/ccio-dma.c
drivers/parisc/sba_iommu.c
include/linux/dma-mapping.h