1. Pageblock size
There are a couple of features that rely on the pageblock size to be reasonably small to
work as expected. One example is virtio-balloon free page reporting, then there is
virtio-mem (still also glued MAX_ORDER) and we have CMA (still also glued to MAX_ORDER).
Most probably there are more. We track movability/ page isolation per pageblock; it's the
smallest granularity you can effectively isolate pages or mark them as CMA
(MIGRATE_ISOLATE, MIGRATE_CMA). Well, and there are "ordinary" THP / huge pages
most of our applications use and will use, especially on smallish systems.
Assume you bump up the pageblock order to 1G. Small VMs won't be able to report any free
pages to the hypervisor. You'll take the "fine-grained" out of virtio-mem. Each
CMA area will have to be at least 1G big, which turns CMA essentially useless on smallish
systems (like we have on arm64 with 64k base pages -- pageblock_size is 512MB and I hate
it).
I understand the issue of having large pageblock in small systems. My plan for
this issue is to make MAX_ORDER a variable (pageblock size would be set
according to MAX_ORDER) that can be adjusted based on total memory and via boot
time parameter. My apology since I did not state this clearly in my cover
letter and it confused you. When we have a boot time adjustable MAX_ORDER,
large pageblock like 1GB would only appear for systems with large memory. For
small VMs, pageblock size would stay at 2MB, so all your concerns on smallish
systems should go away.
I have to admit that I am not really a friend of that. I still think our
target goal should be to have gigantic THP *in addition to* ordinary
THP. Use gigantic THP where enabled and possible, and just use ordinary
THP everywhere else. Having one pageblock granularity is a real
limitation IMHO and requires us to hack the system to support it to some
degree.
Then, imagine systems that have like 4G of main memory. By stopping grouping at
2M and instead grouping at 1G you can very easily find yourself in the system
where all your 4 pageblocks are unmovable and you essentially don't optimize
for huge pages in that environment any more.
Long story short: we need a different mechanism on top and shall leave the
pageblock size untouched, it's too tightly integrated with page isolation,
ordinary THP, and CMA.
I think it is better to make pageblock size adjustable based on total memory of a
system. It is not reasonable to have the same pageblock size across systems with
memory sizes from <1GB to several TBs. Do you agree?
I suggest an additional mechanism on top. Please bear in mind that
ordinary THP will most probably be still the default for 99.9% of all
application/library cases, even when you have gigantic THP around.
2. Section size
I assume the only reason you want to touch that is because pageblock_size <=
section_size, and I guess that's one of the reasons I dislike it so much. Messing
with the section size really only makes sense when we want to manage metadata for
larger granularity within a section.
Perhaps it is worth checking if it is feasible to make pageblock_size >
section_size, so we can still have small sections when pageblock_size are large.
One potential issue for that is when PFN discontinues at section boundary, we
might have partial pageblock when pageblock_size is big. I guess supporting
partial pageblock (or different pageblock sizes like you mentioned below ) would
be the right solution.
We allocate metadata per section. We mark whole sections
early/online/present/.... Yes, in case of vmemmap, we manage the memmap in
smaller granularity using the sub-section map, some kind of hack to support
some ZONE_DEVICE cases better.
Let's assume we introduce something new "gigapage_order", corresponding to 1G.
We could either decide to squeeze the metadata into sections, having to increase the
section size, or manage that metadata differently.
Managing it differently certainly makes the necessary changes easier. Instead
of adding more hacks into sections, rather manage that metadata at differently
place / in a different way.
Can you elaborate on managing it differently?
Let's keep it simple. Assume you track on a 1G gigpageblock MOVABLE vs.
!movable in addition to existing pageblocks. A 64 TB system would have
64*1024 gigpageblocks. One bit per gigapageblock would require 8k a.k.a.
2 pages. If you need more states, it would maybe double. No need to
manage that using sparse memory sections IMHO. Just allocate 2/4 pages
during boot for the bitmap.
See [1] for an alternative. Not necessarily what I would dream off, but just to
showcase that there might be alternative to group pages.
I saw this patch too. It is an interesting idea to separate different
allocation orders into different regions, but it would not work for gigantic
page allocations unless we have large pageblock size to utilize existing
anti-fragmentation mechanisms.
Right, any anti-fragmentation mechanism on top.
3. Grouping pages > pageblock_order
There are other approaches that would benefit from grouping at >
pageblock_order and having bigger MAX_ORDER. And that doesn't necessarily mean to
form gigantic pages only, we might want to group in multiple granularity on a
single system. Memory hot(un)plug is one example, but also optimizing memory
consumption by powering down DIMM banks. Also, some architectures support
differing huge page sizes (aarch64) that could be improved without CMA. Why not
have more than 2 THP sizes on these systems?
Ideally, we'd have a mechanism that tries grouping on different granularity,
like for every order in pageblock_order ... max_pageblock_order (e.g., 1 GiB),
and not only add one new level of grouping (or increase the single grouping
size).
I agree. In some sense, supporting partial pageblock and increasing pageblock
size (e.g., to 1GB) is, at the high level, quite similar to having multiple
pageblock sizes. But I am not sure if we really want to support multiple
pageblock sizes, since it creates pageblock fragmentation when we want to
change migratetype for part of a pageblock. This means we would break a large
pageblock into small ones if we just want to steal a subset of pages from
MOVEABLE for UNMOVABLE allocations. Then pageblock loses its most useful
anti-fragmentation feature. Also it seems to be a replication of buddy
allocator functionalities when it comes to pageblock split and merge.
Let's assume for simplicity that you have a 4G machine, maximum 4
gigantic pages. The first gigantic page will be impossible either way
due to the kernel, boot time allocations etc. So you're left with 3
gigantic pages you could use at best.
Obviously, you want to make sure that the remaining parts of the first
gigantic page are used as best as possible for ordinary huge pages, so
you would actually want to group them in 2 MiB chunks and avoid
fragmentation there.
Obviously, supporting two pageblock types would require core
modifications to support it natively. (not pushing for the idea of two
pageblock orders, just motivating why we actually want to keep grouping
for ordinary THP).
The above is really a nice discussion with you on pageblock, section, memory
hotplug/hotremove, which also helps me understand more on the issues with
increasing MAX_ORDER to enable 1GB page allocation.
In sum, if I get it correctly, the issues I need to address are:
1. large pageblock size (which is needed when we bump MAX_ORDER for gigantic
page allocation from buddy allocator) is not good for machines with small
memory.
2. pageblock size is currently tied with section size (which made me want to
bump section size).
For 1, I think making MAX_ORDER a variable that can be set based on total
memory size and adjustable via boot time parameter should solve the problem.
For small machines, we will keep MAX_ORDER as small as we have now like 4MB,
whereas for large machines, we can increase MAX_ORDER to utilize gigantic pages.
For 2, supporting partial pageblock and allow a pageblock to cross multiple
sections would break the tie between pageblock size and section to solve the
issue.
I am going to look into them. What do you think?
I am not sure that's really the right direction as stated above.
--
Thanks,
David / dhildenb