Merge branches 'x86/asm', 'x86/cleanups', 'x86/cpudetect', 'x86/debug', 'x86/doc...
authorIngo Molnar <mingo@elte.hu>
Wed, 28 Jan 2009 22:13:53 +0000 (23:13 +0100)
committerIngo Molnar <mingo@elte.hu>
Wed, 28 Jan 2009 22:13:53 +0000 (23:13 +0100)
17 files changed:
1  2  3  4  5  6  7  8  9  10  11  12  13  14 
arch/x86/include/asm/io.h
arch/x86/include/asm/paravirt.h
arch/x86/include/asm/processor.h
arch/x86/include/asm/setup.h
arch/x86/include/asm/spinlock.h
arch/x86/include/asm/system.h
arch/x86/include/asm/thread_info.h
arch/x86/kernel/cpu/common.c
arch/x86/kernel/cpu/intel.c
arch/x86/kernel/entry_64.S
arch/x86/kernel/head_32.S
arch/x86/kernel/io_apic.c
arch/x86/kernel/signal.c
arch/x86/mm/fault.c
arch/x86/mm/ioremap.c
arch/x86/mm/pat.c
arch/x86/xen/enlighten.c

Simple merge
Simple merge
Simple merge
Simple merge
index d17c91981da2f1c94a5fb045067f821ece61b4fe,4d3dcc51cacdfde16bc2fe8888c803fd866b4049,2bd6b111a414970cb7fe96f786751b69169dd737,d17c91981da2f1c94a5fb045067f821ece61b4fe,d17c91981da2f1c94a5fb045067f821ece61b4fe,d17c91981da2f1c94a5fb045067f821ece61b4fe,d17c91981da2f1c94a5fb045067f821ece61b4fe,d17c91981da2f1c94a5fb045067f821ece61b4fe,d17c91981da2f1c94a5fb045067f821ece61b4fe,d17c91981da2f1c94a5fb045067f821ece61b4fe,d17c91981da2f1c94a5fb045067f821ece61b4fe,d17c91981da2f1c94a5fb045067f821ece61b4fe,2bd6b111a414970cb7fe96f786751b69169dd737,d17c91981da2f1c94a5fb045067f821ece61b4fe..139b4249a5ec4fcb09294bb58e4eaa7f0848be1a
@@@@@@@@@@@@@@@ -172,70 -172,70 -172,8 -172,70 -172,70 -172,70 -172,70 -172,70 -172,70 -172,70 -172,70 -172,70 -172,8 -172,70 +172,8 @@@@@@@@@@@@@@@ static inline int __ticket_spin_is_cont
                return (((tmp >> TICKET_SHIFT) - tmp) & ((1 << TICKET_SHIFT) - 1)) > 1;
              }
              
-- --------- -#ifdef CONFIG_PARAVIRT
-- --------- -/*
-- --------- - * Define virtualization-friendly old-style lock byte lock, for use in
-- --------- - * pv_lock_ops if desired.
-- --------- - *
-- --------- - * This differs from the pre-2.6.24 spinlock by always using xchgb
-- --------- - * rather than decb to take the lock; this allows it to use a
-- --------- - * zero-initialized lock structure.  It also maintains a 1-byte
-- --------- - * contention counter, so that we can implement
-- --------- - * __byte_spin_is_contended.
-- --------- - */
-- --------- -struct __byte_spinlock {
-- --------- -  s8 lock;
-- --------- -  s8 spinners;
-- --------- -};
-- --------- -
-- --------- -static inline int __byte_spin_is_locked(raw_spinlock_t *lock)
-- --------- -{
-- --------- -  struct __byte_spinlock *bl = (struct __byte_spinlock *)lock;
-- --------- -  return bl->lock != 0;
-- --------- -}
-- --------- -
-- --------- -static inline int __byte_spin_is_contended(raw_spinlock_t *lock)
-- --------- -{
-- --------- -  struct __byte_spinlock *bl = (struct __byte_spinlock *)lock;
-- --------- -  return bl->spinners != 0;
-- --------- -}
-- --------- -
-- --------- -static inline void __byte_spin_lock(raw_spinlock_t *lock)
-- --------- -{
-- --------- -  struct __byte_spinlock *bl = (struct __byte_spinlock *)lock;
-- --------- -  s8 val = 1;
-  --------- -
-  --------- -  asm("1: xchgb %1, %0\n"
-  --------- -      "   test %1,%1\n"
-  --------- -      "   jz 3f\n"
-  --------- -      "   " LOCK_PREFIX "incb %2\n"
-  --------- -      "2: rep;nop\n"
-  --------- -      "   cmpb $1, %0\n"
-  --------- -      "   je 2b\n"
-  --------- -      "   " LOCK_PREFIX "decb %2\n"
-  --------- -      "   jmp 1b\n"
-  --------- -      "3:"
-  --------- -      : "+m" (bl->lock), "+q" (val), "+m" (bl->spinners): : "memory");
-  --------- -}
-  --------- -
-  --------- -static inline int __byte_spin_trylock(raw_spinlock_t *lock)
-  --------- -{
-  --------- -  struct __byte_spinlock *bl = (struct __byte_spinlock *)lock;
-  --------- -  u8 old = 1;
-  --------- -
-  --------- -  asm("xchgb %1,%0"
-  --------- -      : "+m" (bl->lock), "+q" (old) : : "memory");
++ +++++++++ +#ifndef CONFIG_PARAVIRT
              
 -              asm("1: xchgb %1, %0\n"
 -                  "   test %1,%1\n"
 -                  "   jz 3f\n"
 -                  "   " LOCK_PREFIX "incb %2\n"
 -                  "2: rep;nop\n"
 -                  "   cmpb $1, %0\n"
 -                  "   je 2b\n"
 -                  "   " LOCK_PREFIX "decb %2\n"
 -                  "   jmp 1b\n"
 -                  "3:"
 -                  : "+m" (bl->lock), "+q" (val), "+m" (bl->spinners): : "memory");
 -            }
 -            
 -            static inline int __byte_spin_trylock(raw_spinlock_t *lock)
 -            {
 -              struct __byte_spinlock *bl = (struct __byte_spinlock *)lock;
 -              u8 old = 1;
 -            
 -              asm("xchgb %1,%0"
 -                  : "+m" (bl->lock), "+q" (old) : : "memory");
 -            
-- --------- -  return old == 0;
-- --------- -}
-- --------- -
-- --------- -static inline void __byte_spin_unlock(raw_spinlock_t *lock)
-- --------- -{
-- --------- -  struct __byte_spinlock *bl = (struct __byte_spinlock *)lock;
-- --------- -  smp_wmb();
-- --------- -  bl->lock = 0;
-- --------- -}
-- --------- -#else  /* !CONFIG_PARAVIRT */
              static inline int __raw_spin_is_locked(raw_spinlock_t *lock)
              {
                return __ticket_spin_is_locked(lock);
index 2fcc70bc85f3b52950f7b731c67bdb93a38cf552,fa47b1e6a86651872c98ce2e2c144f05ecda92d3,8e626ea33a1a64d1565c10a0784200ef92ec9a31,8e626ea33a1a64d1565c10a0784200ef92ec9a31,8e626ea33a1a64d1565c10a0784200ef92ec9a31,8e626ea33a1a64d1565c10a0784200ef92ec9a31,8e626ea33a1a64d1565c10a0784200ef92ec9a31,8e626ea33a1a64d1565c10a0784200ef92ec9a31,8e626ea33a1a64d1565c10a0784200ef92ec9a31,8e626ea33a1a64d1565c10a0784200ef92ec9a31,8e626ea33a1a64d1565c10a0784200ef92ec9a31,8e626ea33a1a64d1565c10a0784200ef92ec9a31,8e626ea33a1a64d1565c10a0784200ef92ec9a31,8e626ea33a1a64d1565c10a0784200ef92ec9a31..c22383743f36d22c02d33431c8165c68114d4240
@@@@@@@@@@@@@@@ -108,22 -94,19 -94,19 -94,19 -94,19 -94,19 -94,19 -94,19 -94,19 -94,19 -94,19 -94,19 -94,19 -94,19 +108,22 @@@@@@@@@@@@@@@ do {                                                                 
                     "call __switch_to\n\t"                                       \
                     ".globl thread_return\n"                                     \
                     "thread_return:\n\t"                                         \
 -------------       "movq %%gs:%P[pda_pcurrent],%%rsi\n\t"                       \
 +++++++++++++       "movq "__percpu_arg([current_task])",%%rsi\n\t"              \
 +++++++++++++       __switch_canary                                              \
                     "movq %P[thread_info](%%rsi),%%r8\n\t"                       \
- ------------       LOCK_PREFIX "btr  %[tif_fork],%P[ti_flags](%%r8)\n\t"        \
                     "movq %%rax,%%rdi\n\t"                                       \
- ------------       "jc   ret_from_fork\n\t"                                     \
+ ++++++++++++       "testl  %[_tif_fork],%P[ti_flags](%%r8)\n\t"         \
+ ++++++++++++       "jnz   ret_from_fork\n\t"                                    \
                     RESTORE_CONTEXT                                              \
                     : "=a" (last)                                                \
 +++++++++++++         __switch_canary_oparam                                     \
                     : [next] "S" (next), [prev] "D" (prev),                      \
                       [threadrsp] "i" (offsetof(struct task_struct, thread.sp)), \
                       [ti_flags] "i" (offsetof(struct thread_info, flags)),      \
- ------------         [tif_fork] "i" (TIF_FORK),                                 \
+ ++++++++++++         [_tif_fork] "i" (_TIF_FORK),                               \
                       [thread_info] "i" (offsetof(struct task_struct, stack)),   \
 -------------         [pda_pcurrent] "i" (offsetof(struct x8664_pda, pcurrent))  \
 +++++++++++++         [current_task] "m" (per_cpu_var(current_task))             \
 +++++++++++++         __switch_canary_iparam                                     \
                     : "memory", "cc" __EXTRA_CLOBBER)
              #endif
              
Simple merge
Simple merge
index 549f2ada55f584bd045e13fb1e0c65a029f317e1,8ea6929e974c090142ed0d3744167792ce04d777,8ea6929e974c090142ed0d3744167792ce04d777,20ce03acf04bd329ef21e1a649ed99ed152e716b,8ea6929e974c090142ed0d3744167792ce04d777,8ea6929e974c090142ed0d3744167792ce04d777,8ea6929e974c090142ed0d3744167792ce04d777,8ea6929e974c090142ed0d3744167792ce04d777,8ea6929e974c090142ed0d3744167792ce04d777,8ea6929e974c090142ed0d3744167792ce04d777,8ea6929e974c090142ed0d3744167792ce04d777,8ea6929e974c090142ed0d3744167792ce04d777,8ea6929e974c090142ed0d3744167792ce04d777,430e5c38a5449a93c65e0877d780bbf2cac97884..5deefae9064dcba2fdfa11265abf37f79c4b3978
              
              static void __cpuinit early_init_intel(struct cpuinfo_x86 *c)
              {
-               if (c->x86 == 6 && c->x86_model >= 15) {
 ++++++++++++   /* Unmask CPUID levels if masked: */
+++++++++++++   if (c->x86 > 6 || (c->x86 == 6 && c->x86_model >= 0xd)) {
 ++++++++++++           u64 misc_enable;
 ++++++++++++ 
 ++++++++++++           rdmsrl(MSR_IA32_MISC_ENABLE, misc_enable);
 ++++++++++++ 
 ++++++++++++           if (misc_enable & MSR_IA32_MISC_ENABLE_LIMIT_CPUID) {
 ++++++++++++                   misc_enable &= ~MSR_IA32_MISC_ENABLE_LIMIT_CPUID;
 ++++++++++++                   wrmsrl(MSR_IA32_MISC_ENABLE, misc_enable);
 ++++++++++++                   c->cpuid_level = cpuid_eax(0);
 ++++++++++++           }
 ++++++++++++   }
 ++++++++++++ 
                if ((c->x86 == 0xf && c->x86_model >= 0x03) ||
                        (c->x86 == 0x6 && c->x86_model >= 0x0e))
                        set_cpu_cap(c, X86_FEATURE_CONSTANT_TSC);
Simple merge
Simple merge
Simple merge
Simple merge
index 65709a6aa6ee046dab8e3368075a371cd4e2a6a0,9e268b6b204e1e3b34d639061aa43fcd9e0bbc3f,9e268b6b204e1e3b34d639061aa43fcd9e0bbc3f,90dfae511a41e1dcf7fcf188b94d13c123780606,90dfae511a41e1dcf7fcf188b94d13c123780606,90dfae511a41e1dcf7fcf188b94d13c123780606,9e268b6b204e1e3b34d639061aa43fcd9e0bbc3f,8f4b859a04b3ff95caabfcfce54523e3adc1c313,90dfae511a41e1dcf7fcf188b94d13c123780606,90dfae511a41e1dcf7fcf188b94d13c123780606,90dfae511a41e1dcf7fcf188b94d13c123780606,90dfae511a41e1dcf7fcf188b94d13c123780606,9e268b6b204e1e3b34d639061aa43fcd9e0bbc3f,90dfae511a41e1dcf7fcf188b94d13c123780606..d3eee74f830ad71bbc2fdad868e9f3549a581d96
@@@@@@@@@@@@@@@ -415,12 -414,12 -414,12 -414,12 -414,12 -414,12 -414,12 -414,11 -414,12 -414,12 -414,12 -414,12 -414,12 -414,12 +415,11 @@@@@@@@@@@@@@@ static noinline void pgtable_bad(struc
              {
                unsigned long flags = oops_begin();
                int sig = SIGKILL;
 ------ ------  struct task_struct *tsk;
 ++++++ ++++++  struct task_struct *tsk = current;
              
                printk(KERN_ALERT "%s: Corrupted page table at address %lx\n",
 ------ ------         current->comm, address);
 ++++++ ++++++         tsk->comm, address);
                dump_pagetable(address);
------- ------  tsk = current;
                tsk->thread.cr2 = address;
                tsk->thread.trap_no = 14;
                tsk->thread.error_code = error_code;
              }
              #endif
              
 ++++++ ++++++static noinline void no_context(struct pt_regs *regs,
 ++++++ ++++++                  unsigned long error_code, unsigned long address)
 ++++++ ++++++{
 ++++++ ++++++  struct task_struct *tsk = current;
 +++++++++++++  unsigned long *stackend;
 +++++++++++++
 ++++++ ++++++#ifdef CONFIG_X86_64
 ++++++ ++++++  unsigned long flags;
 ++++++ ++++++  int sig;
 ++++++ ++++++#endif
 ++++++ ++++++
 ++++++ ++++++  /* Are we prepared to handle this kernel fault?  */
 ++++++ ++++++  if (fixup_exception(regs))
 ++++++ ++++++          return;
 ++++++ ++++++
 ++++++ ++++++  /*
 ++++++ ++++++   * X86_32
 ++++++ ++++++   * Valid to do another page fault here, because if this fault
 ++++++ ++++++   * had been triggered by is_prefetch fixup_exception would have
 ++++++ ++++++   * handled it.
 ++++++ ++++++   *
 ++++++ ++++++   * X86_64
 ++++++ ++++++   * Hall of shame of CPU/BIOS bugs.
 ++++++ ++++++   */
 ++++++ ++++++  if (is_prefetch(regs, error_code, address))
 ++++++ ++++++          return;
 ++++++ ++++++
 ++++++ ++++++  if (is_errata93(regs, address))
 ++++++ ++++++          return;
 ++++++ ++++++
 ++++++ ++++++  /*
 ++++++ ++++++   * Oops. The kernel tried to access some bad page. We'll have to
 ++++++ ++++++   * terminate things with extreme prejudice.
 ++++++ ++++++   */
 ++++++ ++++++#ifdef CONFIG_X86_32
 ++++++ ++++++  bust_spinlocks(1);
 ++++++ ++++++#else
 ++++++ ++++++  flags = oops_begin();
 ++++++ ++++++#endif
 ++++++ ++++++
 ++++++ ++++++  show_fault_oops(regs, error_code, address);
 ++++++ ++++++
 +++++++++++++  stackend = end_of_stack(tsk);
 +++++++++++++  if (*stackend != STACK_END_MAGIC)
 +++++++++++++          printk(KERN_ALERT "Thread overran stack, or stack corrupted\n");
 +++++++++++++
 ++++++ ++++++  tsk->thread.cr2 = address;
 ++++++ ++++++  tsk->thread.trap_no = 14;
 ++++++ ++++++  tsk->thread.error_code = error_code;
 ++++++ ++++++
 ++++++ ++++++#ifdef CONFIG_X86_32
 ++++++ ++++++  die("Oops", regs, error_code);
 ++++++ ++++++  bust_spinlocks(0);
 ++++++ ++++++  do_exit(SIGKILL);
 ++++++ ++++++#else
 ++++++ ++++++  sig = SIGKILL;
 ++++++ ++++++  if (__die("Oops", regs, error_code))
 ++++++ ++++++          sig = 0;
 ++++++ ++++++  /* Executive summary in case the body of the oops scrolled away */
 ++++++ ++++++  printk(KERN_EMERG "CR2: %016lx\n", address);
 ++++++ ++++++  oops_end(flags, regs, sig);
 ++++++ ++++++#endif
 ++++++ ++++++}
 ++++++ ++++++
 ++++++ ++++++static void __bad_area_nosemaphore(struct pt_regs *regs,
 ++++++ ++++++                  unsigned long error_code, unsigned long address,
 ++++++ ++++++                  int si_code)
 ++++++ ++++++{
 ++++++ ++++++  struct task_struct *tsk = current;
 ++++++ ++++++
 ++++++ ++++++  /* User mode accesses just cause a SIGSEGV */
 ++++++ ++++++  if (error_code & PF_USER) {
 ++++++ ++++++          /*
 ++++++ ++++++           * It's possible to have interrupts off here.
 ++++++ ++++++           */
 ++++++ ++++++          local_irq_enable();
 ++++++ ++++++
 ++++++ ++++++          /*
 ++++++ ++++++           * Valid to do another page fault here because this one came
 ++++++ ++++++           * from user space.
 ++++++ ++++++           */
 ++++++ ++++++          if (is_prefetch(regs, error_code, address))
 ++++++ ++++++                  return;
 ++++++ ++++++
 ++++++ ++++++          if (is_errata100(regs, address))
 ++++++ ++++++                  return;
 ++++++ ++++++
 ++++++ ++++++          if (show_unhandled_signals && unhandled_signal(tsk, SIGSEGV) &&
 ++++++ ++++++              printk_ratelimit()) {
 ++++++ ++++++                  printk(
 ++++++ ++++++                  "%s%s[%d]: segfault at %lx ip %p sp %p error %lx",
 ++++++ ++++++                  task_pid_nr(tsk) > 1 ? KERN_INFO : KERN_EMERG,
 ++++++ ++++++                  tsk->comm, task_pid_nr(tsk), address,
 ++++++ ++++++                  (void *) regs->ip, (void *) regs->sp, error_code);
 ++++++ ++++++                  print_vma_addr(" in ", regs->ip);
 ++++++ ++++++                  printk("\n");
 ++++++ ++++++          }
 ++++++ ++++++
 ++++++ ++++++          tsk->thread.cr2 = address;
 ++++++ ++++++          /* Kernel addresses are always protection faults */
 ++++++ ++++++          tsk->thread.error_code = error_code | (address >= TASK_SIZE);
 ++++++ ++++++          tsk->thread.trap_no = 14;
 ++++++ ++++++          force_sig_info_fault(SIGSEGV, si_code, address, tsk);
 ++++++ ++++++          return;
 ++++++ ++++++  }
 ++++++ ++++++
 ++++++ ++++++  if (is_f00f_bug(regs, address))
 ++++++ ++++++          return;
 ++++++ ++++++
 ++++++ ++++++  no_context(regs, error_code, address);
 ++++++ ++++++}
 ++++++ ++++++
 ++++++ ++++++static noinline void bad_area_nosemaphore(struct pt_regs *regs,
 ++++++ ++++++                  unsigned long error_code, unsigned long address)
 ++++++ ++++++{
 ++++++ ++++++  __bad_area_nosemaphore(regs, error_code, address, SEGV_MAPERR);
 ++++++ ++++++}
 ++++++ ++++++
 ++++++ ++++++static void __bad_area(struct pt_regs *regs,
 ++++++ ++++++                  unsigned long error_code, unsigned long address,
 ++++++ ++++++                  int si_code)
 ++++++ ++++++{
 ++++++ ++++++  struct mm_struct *mm = current->mm;
 ++++++ ++++++
 ++++++ ++++++  /*
 ++++++ ++++++   * Something tried to access memory that isn't in our memory map..
 ++++++ ++++++   * Fix it, but check if it's kernel or user first..
 ++++++ ++++++   */
 ++++++ ++++++  up_read(&mm->mmap_sem);
 ++++++ ++++++
 ++++++ ++++++  __bad_area_nosemaphore(regs, error_code, address, si_code);
 ++++++ ++++++}
 ++++++ ++++++
 ++++++ ++++++static noinline void bad_area(struct pt_regs *regs,
 ++++++ ++++++                  unsigned long error_code, unsigned long address)
 ++++++ ++++++{
 ++++++ ++++++  __bad_area(regs, error_code, address, SEGV_MAPERR);
 ++++++ ++++++}
 ++++++ ++++++
 ++++++ ++++++static noinline void bad_area_access_error(struct pt_regs *regs,
 ++++++ ++++++                  unsigned long error_code, unsigned long address)
 ++++++ ++++++{
 ++++++ ++++++  __bad_area(regs, error_code, address, SEGV_ACCERR);
 ++++++ ++++++}
 ++++++ ++++++
 ++++++ ++++++/* TODO: fixup for "mm-invoke-oom-killer-from-page-fault.patch" */
 ++++++ ++++++static void out_of_memory(struct pt_regs *regs,
 ++++++ ++++++                  unsigned long error_code, unsigned long address)
 ++++++ ++++++{
 ++++++ ++++++  /*
 ++++++ ++++++   * We ran out of memory, call the OOM killer, and return the userspace
 ++++++ ++++++   * (which will retry the fault, or kill us if we got oom-killed).
 ++++++ ++++++   */
 ++++++ ++++++  up_read(&current->mm->mmap_sem);
 ++++++ ++++++  pagefault_out_of_memory();
 ++++++ ++++++}
 ++++++ ++++++
 ++++++ ++++++static void do_sigbus(struct pt_regs *regs,
 ++++++ ++++++                  unsigned long error_code, unsigned long address)
 ++++++ ++++++{
 ++++++ ++++++  struct task_struct *tsk = current;
 ++++++ ++++++  struct mm_struct *mm = tsk->mm;
 ++++++ ++++++
 ++++++ ++++++  up_read(&mm->mmap_sem);
 ++++++ ++++++
 ++++++ ++++++  /* Kernel mode? Handle exceptions or die */
 ++++++ ++++++  if (!(error_code & PF_USER))
 ++++++ ++++++          no_context(regs, error_code, address);
 ++++++ ++++++#ifdef CONFIG_X86_32
 ++++++ ++++++  /* User space => ok to do another page fault */
 ++++++ ++++++  if (is_prefetch(regs, error_code, address))
 ++++++ ++++++          return;
 ++++++ ++++++#endif
 ++++++ ++++++  tsk->thread.cr2 = address;
 ++++++ ++++++  tsk->thread.error_code = error_code;
 ++++++ ++++++  tsk->thread.trap_no = 14;
 ++++++ ++++++  force_sig_info_fault(SIGBUS, BUS_ADRERR, address, tsk);
 ++++++ ++++++}
 ++++++ ++++++
 ++++++ ++++++static noinline void mm_fault_error(struct pt_regs *regs,
 ++++++ ++++++          unsigned long error_code, unsigned long address, unsigned int fault)
 ++++++ ++++++{
 ++++++ ++++++  if (fault & VM_FAULT_OOM)
 ++++++ ++++++          out_of_memory(regs, error_code, address);
 ++++++ ++++++  else if (fault & VM_FAULT_SIGBUS)
 ++++++ ++++++          do_sigbus(regs, error_code, address);
 ++++++ ++++++  else
 ++++++ ++++++          BUG();
 ++++++ ++++++}
 ++++++ ++++++
              static int spurious_fault_check(unsigned long error_code, pte_t *pte)
              {
                if ((error_code & PF_WRITE) && !pte_write(*pte))
Simple merge
index 7b61036427df69d11a33044d60bc85102931e385,85cbd3cd3723698ed51707d01fa75d8000f81477,85cbd3cd3723698ed51707d01fa75d8000f81477,430cb44dd3f4b03864972ad3e143ff9f80e5604b,8b08fb955274148454a2acaa538c78215322e655,160c42d3eb8f0a5393140b43ebb496cf8fe200d1,85cbd3cd3723698ed51707d01fa75d8000f81477,8b08fb955274148454a2acaa538c78215322e655,8b08fb955274148454a2acaa538c78215322e655,8b08fb955274148454a2acaa538c78215322e655,8b08fb955274148454a2acaa538c78215322e655,8b08fb955274148454a2acaa538c78215322e655,85cbd3cd3723698ed51707d01fa75d8000f81477,ffc88cc00fda01be5aeabdea047cdd3fc9512d66..9127e31c726866d462d0da61d03ceade383ced4e
@@@@@@@@@@@@@@@ -618,13 -601,12 -601,12 -610,13 -601,13 -628,13 -601,12 -601,13 -601,13 -601,13 -601,13 -601,13 -601,12 -618,12 +627,13 @@@@@@@@@@@@@@@ void unmap_devmem(unsigned long pfn, un
               * Reserved non RAM regions only and after successful reserve_memtype,
               * this func also keeps identity mapping (if any) in sync with this new prot.
               */
 --   -     --static int reserve_pfn_range(u64 paddr, unsigned long size, pgprot_t vma_prot)
 ++   +     ++static int reserve_pfn_range(u64 paddr, unsigned long size, pgprot_t *vma_prot,
 ++   +     ++                          int strict_prot)
              {
                int is_ram = 0;
     -          int ret;
     +          int id_sz, ret;
                unsigned long flags;
 --   -     --  unsigned long want_flags = (pgprot_val(vma_prot) & _PAGE_CACHE_MASK);
 ++   +     ++  unsigned long want_flags = (pgprot_val(*vma_prot) & _PAGE_CACHE_MASK);
              
                is_ram = pagerange_is_ram(paddr, paddr + size);
              
                        return ret;
              
                if (flags != want_flags) {
 --   -     --          free_memtype(paddr, paddr + size);
 --   -     --          printk(KERN_ERR
 --   -     --          "%s:%d map pfn expected mapping type %s for %Lx-%Lx, got %s\n",
 --   -     --                  current->comm, current->pid,
 --   -     --                  cattr_name(want_flags),
 --   -     --                  (unsigned long long)paddr,
 --   -     --                  (unsigned long long)(paddr + size),
 --   -     --                  cattr_name(flags));
 --   -     --          return -EINVAL;
 ++   +     ++          if (strict_prot || !is_new_memtype_allowed(want_flags, flags)) {
 ++   +     ++                  free_memtype(paddr, paddr + size);
 ++   +     ++                  printk(KERN_ERR "%s:%d map pfn expected mapping type %s"
 ++   +     ++                          " for %Lx-%Lx, got %s\n",
 ++   +     ++                          current->comm, current->pid,
 ++   +     ++                          cattr_name(want_flags),
 ++   +     ++                          (unsigned long long)paddr,
 ++   +     ++                          (unsigned long long)(paddr + size),
 ++   +     ++                          cattr_name(flags));
 ++   +     ++                  return -EINVAL;
 ++   +     ++          }
 ++   +     ++          /*
 ++   +     ++           * We allow returning different type than the one requested in
 ++   +     ++           * non strict case.
 ++   +     ++           */
 ++   +     ++          *vma_prot = __pgprot((pgprot_val(*vma_prot) &
 ++   +     ++                                (~_PAGE_CACHE_MASK)) |
 ++   +     ++                               flags);
                }
              
     -          if (kernel_map_sync_memtype(paddr, size, flags)) {
     +          /* Need to keep identity mapping in sync */
     +          if (paddr >= __pa(high_memory))
     +                  return 0;
     +        
     +          id_sz = (__pa(high_memory) < paddr + size) ?
     +                                  __pa(high_memory) - paddr :
     +                                  size;
     +        
     +          if (ioremap_change_attr((unsigned long)__va(paddr), id_sz, flags) < 0) {
                        free_memtype(paddr, paddr + size);
                        printk(KERN_ERR
                                "%s:%d reserve_pfn_range ioremap_change_attr failed %s "
Simple merge