From: Josh Aas <josha@sgi.com>
To: akpm@osdl.org, linux-kernel@vger.kernel.org, linux-ia64@vger.kernel.org
Subject: [PATCH] improve speed of freeing bootmem
Date: Fri, 06 Aug 2004 14:26:27 -0500 [thread overview]
Message-ID: <4113DB63.9020706@sgi.com> (raw)
[-- Attachment #1: Type: text/plain, Size: 1361 bytes --]
Attached is a patch that greatly improves the speed of freeing boot
memory. On ia64 machines with 2GB or more memory (I didn't test with
less, but I can't imagine there being a problem), the speed improvement
is about 75% for the function free_all_bootmem_core. This translates to
savings on the order of 1 minute / TB of memory during boot time. That
number comes from testing on a machine with 512GB, and extrapolating
based on profiling of an unpatched 4TB machine. For 4 and 8 TB machines,
the time spent in this function is about 1 minutes/TB, which is painful
especially given that there is no indication of what is going on put to
the console (this issue to possibly be addressed later).
The basic idea is to free higher order pages instead of going through
every single one. Also, some unnecessary atomic operations are done away
with and replaced with non-atomic equivalents, and prefetching is done
where it helps the most. For a more in-depth discusion of this patch,
please see the linux-ia64 archives (topic is "free bootmem feedback patch").
The patch is originally Tony Luck's, and I added some further
optimizations (non-atomic ops improvements and prefetching).
Signed-off-by: Tony Luck <tony.luck@intel.com>
Signed-off-by: Josh Aas <josha@sgi.com>
--
Josh Aas
Silicon Graphics, Inc. (SGI)
Linux System Software
651-683-3068
[-- Attachment #2: bootmem4.patch --]
[-- Type: text/x-patch, Size: 2397 bytes --]
--- a/mm/bootmem.c 2004-08-05 15:33:39.000000000 -0500
+++ b/mm/bootmem.c 2004-08-06 13:42:33.000000000 -0500
@@ -259,6 +259,7 @@ static unsigned long __init free_all_boo
unsigned long i, count, total = 0;
unsigned long idx;
unsigned long *map;
+ int gofast = 0;
BUG_ON(!bdata->node_bootmem_map);
@@ -267,14 +268,32 @@ static unsigned long __init free_all_boo
page = virt_to_page(phys_to_virt(bdata->node_boot_start));
idx = bdata->node_low_pfn - (bdata->node_boot_start >> PAGE_SHIFT);
map = bdata->node_bootmem_map;
+ if (bdata->node_boot_start == 0 ||
+ ffs(bdata->node_boot_start) - PAGE_SHIFT > ffs(BITS_PER_LONG))
+ gofast = 1;
for (i = 0; i < idx; ) {
unsigned long v = ~map[i / BITS_PER_LONG];
- if (v) {
+ if (gofast && v == ~0UL) {
+ int j;
+
+ count += BITS_PER_LONG;
+ ClearPageReservedNoAtomic(page);
+ set_page_count(page, 1);
+ for (j = 1; j < BITS_PER_LONG; j++) {
+ if (j + 16 < BITS_PER_LONG) {
+ prefetchw(page + j + 16);
+ }
+ ClearPageReservedNoAtomic(page + j);
+ }
+ __free_pages(page, ffs(BITS_PER_LONG)-1);
+ i += BITS_PER_LONG;
+ page += BITS_PER_LONG;
+ } else if (v) {
unsigned long m;
for (m = 1; m && i < idx; m<<=1, page++, i++) {
if (v & m) {
count++;
- ClearPageReserved(page);
+ ClearPageReservedNoAtomic(page);
set_page_count(page, 1);
__free_page(page);
}
@@ -294,7 +313,7 @@ static unsigned long __init free_all_boo
count = 0;
for (i = 0; i < ((bdata->node_low_pfn-(bdata->node_boot_start >> PAGE_SHIFT))/8 + PAGE_SIZE-1)/PAGE_SIZE; i++,page++) {
count++;
- ClearPageReserved(page);
+ ClearPageReservedNoAtomic(page);
set_page_count(page, 1);
__free_page(page);
}
--- a/include/linux/page-flags.h 2004-08-06 13:43:36.000000000 -0500
+++ b/include/linux/page-flags.h 2004-08-06 13:43:27.000000000 -0500
@@ -236,6 +236,7 @@ extern unsigned long __read_page_state(u
#define PageReserved(page) test_bit(PG_reserved, &(page)->flags)
#define SetPageReserved(page) set_bit(PG_reserved, &(page)->flags)
#define ClearPageReserved(page) clear_bit(PG_reserved, &(page)->flags)
+#define ClearPageReservedNoAtomic(page) (page)->flags &= ~(1UL << PG_reserved)
#define SetPagePrivate(page) set_bit(PG_private, &(page)->flags)
#define ClearPagePrivate(page) clear_bit(PG_private, &(page)->flags)
next reply other threads:[~2004-08-06 19:24 UTC|newest]
Thread overview: 4+ messages / expand[flat|nested] mbox.gz Atom feed top
2004-08-06 19:26 Josh Aas [this message]
2004-08-06 19:52 ` [PATCH] improve speed of freeing bootmem Andrew Morton
2004-08-06 20:46 ` William Lee Irwin III
2004-08-06 22:02 ` Josh Aas
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=4113DB63.9020706@sgi.com \
--to=josha@sgi.com \
--cc=akpm@osdl.org \
--cc=linux-ia64@vger.kernel.org \
--cc=linux-kernel@vger.kernel.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox