Page MenuHomeFreeBSD

LinuxKPI: make __kmalloc() play by the rules
AcceptedPublic

Authored by bz on Sep 12 2024, 6:25 PM.
Tags
None
Referenced Files
F108449205: D46656.diff
Fri, Jan 24, 9:43 PM
F108433982: D46656.id143295.diff
Fri, Jan 24, 5:53 PM
Unknown Object (File)
Thu, Jan 16, 12:40 AM
Unknown Object (File)
Wed, Jan 8, 10:00 AM
Unknown Object (File)
Wed, Jan 8, 9:59 AM
Unknown Object (File)
Wed, Jan 8, 9:59 AM
Unknown Object (File)
Wed, Jan 8, 8:57 AM
Unknown Object (File)
Sat, Jan 4, 4:35 AM
Subscribers

Details

Reviewers
jhb
emaste
Group Reviewers
linuxkpi
Summary

According to Documentation/core-api/dma-api.rst kmalloc() is supposd
to provide physically contiguous memory. [1]

In order to guarantee that allocations are contiguous even if using
larger than PAGE_SIZE check the size and use contigmalloc if needed.
This makes use of 9e6544dd6e02 (and following) allowing free(9) to
also work for contigmalloced memory.

Sponsored by: The FreeBSD Foundation
Pointed out by: jhb [1]
MFC after: 3 days

Diff Detail

Repository
rG FreeBSD src repository
Lint
Lint Passed
Unit
No Test Coverage
Build Status
Buildable 59463
Build 56350: arc lint + arc unit

Event Timeline

bz requested review of this revision.Sep 12 2024, 6:25 PM
bz edited reviewers, added: jhb; removed: jh.Sep 12 2024, 6:28 PM

Looks reasonable to me. Aside, it seems we don't document boundary=0 in contigmalloc(9).

Looks reasonable to me. Aside, it seems we don't document boundary=0 in contigmalloc(9).

Yes, though most drivers in sys/dev do not set it; vm_page_alloc_contig documents it as

"If the boundary
parameter is non-zero, the pages constituting the run will not cross a
physical address that is a multiple of the parameter value, which must be
a power of two."

I followed a very long call chain to arrive at vm_addr_bound_ok, which confirmed -- and the nice thing is boundary == 0 doesn't even need to be a special case:

/*
 * Do the first and last addresses of a range match in all bits except the ones
 * in -boundary (a power-of-two)?  For boundary == 0, all addresses match.
 */
static inline bool
vm_addr_bound_ok(vm_paddr_t pa, vm_paddr_t size, vm_paddr_t boundary)
{                               
        KASSERT(powerof2(boundary), ("%s: boundary is not a power of 2: %#jx",
            __func__, (uintmax_t)boundary));
        return (((pa ^ (pa + size - 1)) & -boundary) == 0);
}

If I have a chance I'll propose a contigmalloc(9) addition.

sys/compat/linuxkpi/common/src/linux_slab.c
221

is this needed? (either for contigmalloc, or to match Linux KPI?)

Looking, kmem_alloc_contig_domain does asize = round_page(size); so unless we need it to be explicit about Linux behaviour perhaps just passing _s through is sensible?

bz marked an inline comment as done.Sep 12 2024, 9:03 PM
bz added inline comments.
sys/compat/linuxkpi/common/src/linux_slab.c
221

[ and so does contigmalloc itself: malloc_type_allocated(type, round_page(size)); ]

I think this is no longer needed; I'll remove it.

bz marked an inline comment as done.

simplify as @emaste pointed out; we no longer need to round up to page size.

LGTM, we can wait for @jhb as well though

This revision is now accepted and ready to land.Sep 13 2024, 1:06 AM
sys/compat/linuxkpi/common/src/linux_slab.c
218–224

Given we have some traction on these reviews and I looked again; this should be <= as 0..PAGE_SIZE is the first page and a single page. @emaste @jhb

Note my comment in the other review. The kmalloc_node family of functions needs the same fix.