ARM: I-cache: flush executable mappings in flush_cache_range()
authorRussell King <rmk+kernel@arm.linux.org.uk>
Sun, 25 Oct 2009 14:12:27 +0000 (14:12 +0000)
committerRussell King <rmk+kernel@arm.linux.org.uk>
Fri, 4 Dec 2009 14:58:51 +0000 (14:58 +0000)
Dirk Behme reported instability on ARM11 SMP (VIPT non-aliasing cache)
caused by the dynamic linker changing protection on text pages to write
GOT entries.  The problem is due to an interaction between the write
faulting code providing new anonymous pages which are incoherent with
the I-cache due to write buffering, and the I-cache not having been
invalidated.

a4db94d plugs the hole with the data cache coherency.  This patch
provides the other half of the fix by flushing the I-cache in
flush_cache_range() for VM_EXEC VMAs (which is what we have when the
region is being made executable again.)  This ensures that the I-cache
will be up to date with the newly COW'd pages.

Note: if users are writing instructions, then they still need to use
the ARM sys_cacheflush API to ensure that the caches are correctly
synchronized.

Signed-off-by: Russell King <rmk+kernel@arm.linux.org.uk>
arch/arm/mm/flush.c

index f8feb5d919fe1e76541dc271a168b9f1ba8b472d..329594e760cdb09e868b1e1ad42a30dfda78f429 100644 (file)
@@ -66,10 +66,9 @@ void flush_cache_range(struct vm_area_struct *vma, unsigned long start, unsigned
                    :
                    : "r" (0)
                    : "cc");
-               __flush_icache_all();
        }
 
-       if (vma->vm_flags & VM_EXEC && icache_is_vivt_asid_tagged())
+       if (vma->vm_flags & VM_EXEC)
                __flush_icache_all();
 }