mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-12 08:09:56 +00:00
[PATCH] x86-64: Speed and clean up cache flushing in change_page_attr
CLFLUSH is a lot faster than WBINVD so avoid the later if at all possible. Always pass the complete list of pages to other CPUs to cut down the number of IPIs. Minor other cleanup and sync with i386 version. Signed-off-by: Andi Kleen <ak@suse.de>
This commit is contained in:
parent
74b47a7844
commit
ea7322decb
@ -61,34 +61,40 @@ static struct page *split_large_page(unsigned long address, pgprot_t prot,
|
||||
return base;
|
||||
}
|
||||
|
||||
|
||||
static void flush_kernel_map(void *address)
|
||||
static void cache_flush_page(void *adr)
|
||||
{
|
||||
if (0 && address && cpu_has_clflush) {
|
||||
/* is this worth it? */
|
||||
int i;
|
||||
for (i = 0; i < PAGE_SIZE; i += boot_cpu_data.x86_clflush_size)
|
||||
asm volatile("clflush (%0)" :: "r" (address + i));
|
||||
} else
|
||||
asm volatile("wbinvd":::"memory");
|
||||
if (address)
|
||||
__flush_tlb_one(address);
|
||||
else
|
||||
__flush_tlb_all();
|
||||
asm volatile("clflush (%0)" :: "r" (adr + i));
|
||||
}
|
||||
|
||||
|
||||
static inline void flush_map(unsigned long address)
|
||||
static void flush_kernel_map(void *arg)
|
||||
{
|
||||
on_each_cpu(flush_kernel_map, (void *)address, 1, 1);
|
||||
struct list_head *l = (struct list_head *)arg;
|
||||
struct page *pg;
|
||||
|
||||
/* When clflush is available always use it because it is
|
||||
much cheaper than WBINVD */
|
||||
if (!cpu_has_clflush)
|
||||
asm volatile("wbinvd" ::: "memory");
|
||||
list_for_each_entry(pg, l, lru) {
|
||||
void *adr = page_address(pg);
|
||||
if (cpu_has_clflush)
|
||||
cache_flush_page(adr);
|
||||
__flush_tlb_one(adr);
|
||||
}
|
||||
}
|
||||
|
||||
static struct page *deferred_pages; /* protected by init_mm.mmap_sem */
|
||||
static inline void flush_map(struct list_head *l)
|
||||
{
|
||||
on_each_cpu(flush_kernel_map, l, 1, 1);
|
||||
}
|
||||
|
||||
static LIST_HEAD(deferred_pages); /* protected by init_mm.mmap_sem */
|
||||
|
||||
static inline void save_page(struct page *fpage)
|
||||
{
|
||||
fpage->lru.next = (struct list_head *)deferred_pages;
|
||||
deferred_pages = fpage;
|
||||
list_add(&fpage->lru, &deferred_pages);
|
||||
}
|
||||
|
||||
/*
|
||||
@ -207,18 +213,18 @@ int change_page_attr(struct page *page, int numpages, pgprot_t prot)
|
||||
|
||||
void global_flush_tlb(void)
|
||||
{
|
||||
struct page *dpage;
|
||||
struct page *pg, *next;
|
||||
struct list_head l;
|
||||
|
||||
down_read(&init_mm.mmap_sem);
|
||||
dpage = xchg(&deferred_pages, NULL);
|
||||
list_replace_init(&deferred_pages, &l);
|
||||
up_read(&init_mm.mmap_sem);
|
||||
|
||||
flush_map((dpage && !dpage->lru.next) ? (unsigned long)page_address(dpage) : 0);
|
||||
while (dpage) {
|
||||
struct page *tmp = dpage;
|
||||
dpage = (struct page *)dpage->lru.next;
|
||||
ClearPagePrivate(tmp);
|
||||
__free_page(tmp);
|
||||
flush_map(&l);
|
||||
|
||||
list_for_each_entry_safe(pg, next, &l, lru) {
|
||||
ClearPagePrivate(pg);
|
||||
__free_page(pg);
|
||||
}
|
||||
}
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user