2  *  include/asm-s390/system.h
 
   5  *    Copyright (C) 1999 IBM Deutschland Entwicklung GmbH, IBM Corporation
 
   6  *    Author(s): Martin Schwidefsky (schwidefsky@de.ibm.com),
 
   8  *  Derived from "include/asm-i386/system.h"
 
  11 #ifndef __ASM_SYSTEM_H
 
  12 #define __ASM_SYSTEM_H
 
  14 #include <linux/kernel.h>
 
  15 #include <asm/types.h>
 
  16 #include <asm/ptrace.h>
 
  17 #include <asm/setup.h>
 
  18 #include <asm/processor.h>
 
  24 extern struct task_struct *__switch_to(void *, void *);
 
  26 static inline void save_fp_regs(s390_fp_regs *fpregs)
 
  33                 : "=m" (*fpregs) : "a" (fpregs), "m" (*fpregs) : "memory");
 
  34         if (!MACHINE_HAS_IEEE)
 
  50                 : "=m" (*fpregs) : "a" (fpregs), "m" (*fpregs) : "memory");
 
  53 static inline void restore_fp_regs(s390_fp_regs *fpregs)
 
  60                 : : "a" (fpregs), "m" (*fpregs));
 
  61         if (!MACHINE_HAS_IEEE)
 
  77                 : : "a" (fpregs), "m" (*fpregs));
 
  80 static inline void save_access_regs(unsigned int *acrs)
 
  82         asm volatile("stam 0,15,0(%0)" : : "a" (acrs) : "memory");
 
  85 static inline void restore_access_regs(unsigned int *acrs)
 
  87         asm volatile("lam 0,15,0(%0)" : : "a" (acrs));
 
  90 #define switch_to(prev,next,last) do {                                       \
 
  93         save_fp_regs(&prev->thread.fp_regs);                                 \
 
  94         restore_fp_regs(&next->thread.fp_regs);                              \
 
  95         save_access_regs(&prev->thread.acrs[0]);                             \
 
  96         restore_access_regs(&next->thread.acrs[0]);                          \
 
  97         prev = __switch_to(prev,next);                                       \
 
 101  * On SMP systems, when the scheduler does migration-cost autodetection,
 
 102  * it needs a way to flush as much of the CPU's caches as possible.
 
 104  * TODO: fill this in!
 
 106 static inline void sched_cacheflush(void)
 
 110 #ifdef CONFIG_VIRT_CPU_ACCOUNTING
 
 111 extern void account_vtime(struct task_struct *);
 
 112 extern void account_tick_vtime(struct task_struct *);
 
 113 extern void account_system_vtime(struct task_struct *);
 
 115 #define account_vtime(x) do { /* empty */ } while (0)
 
 119 extern void pfault_irq_init(void);
 
 120 extern int pfault_init(void);
 
 121 extern void pfault_fini(void);
 
 122 #else /* CONFIG_PFAULT */
 
 123 #define pfault_irq_init()       do { } while (0)
 
 124 #define pfault_init()           ({-1;})
 
 125 #define pfault_fini()           do { } while (0)
 
 126 #endif /* CONFIG_PFAULT */
 
 128 #define finish_arch_switch(prev) do {                                        \
 
 129         set_fs(current->thread.mm_segment);                                  \
 
 130         account_vtime(prev);                                                 \
 
 133 #define nop() asm volatile("nop")
 
 135 #define xchg(ptr,x)                                                       \
 
 137         __typeof__(*(ptr)) __ret;                                         \
 
 138         __ret = (__typeof__(*(ptr)))                                      \
 
 139                 __xchg((unsigned long)(x), (void *)(ptr),sizeof(*(ptr))); \
 
 143 static inline unsigned long __xchg(unsigned long x, void * ptr, int size)
 
 145         unsigned long addr, old;
 
 150                 addr = (unsigned long) ptr;
 
 151                 shift = (3 ^ (addr & 3)) << 3;
 
 160                         : "=&d" (old), "=m" (*(int *) addr)
 
 161                         : "d" (x << shift), "d" (~(255 << shift)), "a" (addr),
 
 162                           "m" (*(int *) addr) : "memory", "cc", "0");
 
 166                 addr = (unsigned long) ptr;
 
 167                 shift = (2 ^ (addr & 2)) << 3;
 
 176                         : "=&d" (old), "=m" (*(int *) addr)
 
 177                         : "d" (x << shift), "d" (~(65535 << shift)), "a" (addr),
 
 178                           "m" (*(int *) addr) : "memory", "cc", "0");
 
 184                         "0:     cs      %0,%2,0(%3)\n"
 
 186                         : "=&d" (old), "=m" (*(int *) ptr)
 
 187                         : "d" (x), "a" (ptr), "m" (*(int *) ptr)
 
 195                         "0:     csg     %0,%2,0(%3)\n"
 
 197                         : "=&d" (old), "=m" (*(long *) ptr)
 
 198                         : "d" (x), "a" (ptr), "m" (*(long *) ptr)
 
 202 #endif /* __s390x__ */
 
 208  * Atomic compare and exchange.  Compare OLD with MEM, if identical,
 
 209  * store NEW in MEM.  Return the initial value in MEM.  Success is
 
 210  * indicated by comparing RETURN with OLD.
 
 213 #define __HAVE_ARCH_CMPXCHG 1
 
 215 #define cmpxchg(ptr,o,n)\
 
 216         ((__typeof__(*(ptr)))__cmpxchg((ptr),(unsigned long)(o),\
 
 217                                         (unsigned long)(n),sizeof(*(ptr))))
 
 219 static inline unsigned long
 
 220 __cmpxchg(volatile void *ptr, unsigned long old, unsigned long new, int size)
 
 222         unsigned long addr, prev, tmp;
 
 227                 addr = (unsigned long) ptr;
 
 228                 shift = (3 ^ (addr & 3)) << 3;
 
 242                         : "=&d" (prev), "=&d" (tmp)
 
 243                         : "d" (old << shift), "d" (new << shift), "a" (ptr),
 
 244                           "d" (~(255 << shift))
 
 246                 return prev >> shift;
 
 248                 addr = (unsigned long) ptr;
 
 249                 shift = (2 ^ (addr & 2)) << 3;
 
 263                         : "=&d" (prev), "=&d" (tmp)
 
 264                         : "d" (old << shift), "d" (new << shift), "a" (ptr),
 
 265                           "d" (~(65535 << shift))
 
 267                 return prev >> shift;
 
 271                         : "=&d" (prev) : "0" (old), "d" (new), "a" (ptr)
 
 278                         : "=&d" (prev) : "0" (old), "d" (new), "a" (ptr)
 
 281 #endif /* __s390x__ */
 
 287  * Force strict CPU ordering.
 
 288  * And yes, this is required on UP too when we're talking
 
 291  * This is very similar to the ppc eieio/sync instruction in that is
 
 292  * does a checkpoint syncronisation & makes sure that 
 
 293  * all memory ops have completed wrt other CPU's ( see 7-15 POP  DJB ).
 
 296 #define eieio() asm volatile("bcr 15,0" : : : "memory")
 
 297 #define SYNC_OTHER_CORES(x)   eieio()
 
 299 #define rmb()   eieio()
 
 300 #define wmb()   eieio()
 
 301 #define read_barrier_depends() do { } while(0)
 
 302 #define smp_mb()       mb()
 
 303 #define smp_rmb()      rmb()
 
 304 #define smp_wmb()      wmb()
 
 305 #define smp_read_barrier_depends()    read_barrier_depends()
 
 306 #define smp_mb__before_clear_bit()     smp_mb()
 
 307 #define smp_mb__after_clear_bit()      smp_mb()
 
 310 #define set_mb(var, value)      do { var = value; mb(); } while (0)
 
 314 #define __ctl_load(array, low, high) ({                         \
 
 315         typedef struct { char _[sizeof(array)]; } addrtype;     \
 
 317                 "       lctlg   %1,%2,0(%0)\n"                  \
 
 318                 : : "a" (&array), "i" (low), "i" (high),        \
 
 319                     "m" (*(addrtype *)(array)));                \
 
 322 #define __ctl_store(array, low, high) ({                        \
 
 323         typedef struct { char _[sizeof(array)]; } addrtype;     \
 
 325                 "       stctg   %2,%3,0(%1)\n"                  \
 
 326                 : "=m" (*(addrtype *)(array))                   \
 
 327                 : "a" (&array), "i" (low), "i" (high));         \
 
 330 #else /* __s390x__ */
 
 332 #define __ctl_load(array, low, high) ({                         \
 
 333         typedef struct { char _[sizeof(array)]; } addrtype;     \
 
 335                 "       lctl    %1,%2,0(%0)\n"                  \
 
 336                 : : "a" (&array), "i" (low), "i" (high),        \
 
 337                     "m" (*(addrtype *)(array)));                \
 
 340 #define __ctl_store(array, low, high) ({                        \
 
 341         typedef struct { char _[sizeof(array)]; } addrtype;     \
 
 343                 "       stctl   %2,%3,0(%1)\n"                  \
 
 344                 : "=m" (*(addrtype *)(array))                   \
 
 345                 : "a" (&array), "i" (low), "i" (high));         \
 
 348 #endif /* __s390x__ */
 
 350 #define __ctl_set_bit(cr, bit) ({       \
 
 351         unsigned long __dummy;          \
 
 352         __ctl_store(__dummy, cr, cr);   \
 
 353         __dummy |= 1UL << (bit);        \
 
 354         __ctl_load(__dummy, cr, cr);    \
 
 357 #define __ctl_clear_bit(cr, bit) ({     \
 
 358         unsigned long __dummy;          \
 
 359         __ctl_store(__dummy, cr, cr);   \
 
 360         __dummy &= ~(1UL << (bit));     \
 
 361         __ctl_load(__dummy, cr, cr);    \
 
 364 #include <linux/irqflags.h>
 
 367  * Use to set psw mask except for the first byte which
 
 368  * won't be changed by this function.
 
 371 __set_psw_mask(unsigned long mask)
 
 373         __load_psw_mask(mask | (__raw_local_irq_stosm(0x00) & ~(-1UL >> 8)));
 
 376 #define local_mcck_enable()  __set_psw_mask(psw_kernel_bits)
 
 377 #define local_mcck_disable() __set_psw_mask(psw_kernel_bits & ~PSW_MASK_MCHECK)
 
 381 extern void smp_ctl_set_bit(int cr, int bit);
 
 382 extern void smp_ctl_clear_bit(int cr, int bit);
 
 383 #define ctl_set_bit(cr, bit) smp_ctl_set_bit(cr, bit)
 
 384 #define ctl_clear_bit(cr, bit) smp_ctl_clear_bit(cr, bit)
 
 388 #define ctl_set_bit(cr, bit) __ctl_set_bit(cr, bit)
 
 389 #define ctl_clear_bit(cr, bit) __ctl_clear_bit(cr, bit)
 
 391 #endif /* CONFIG_SMP */
 
 393 extern void (*_machine_restart)(char *command);
 
 394 extern void (*_machine_halt)(void);
 
 395 extern void (*_machine_power_off)(void);
 
 397 #define arch_align_stack(x) (x)
 
 399 #endif /* __KERNEL__ */