cregit-Linux how code gets into the kernel

Release 4.10 arch/powerpc/include/asm/spinlock.h

#ifndef __ASM_SPINLOCK_H

#define __ASM_SPINLOCK_H
#ifdef __KERNEL__

/*
 * Simple spin lock operations.  
 *
 * Copyright (C) 2001-2004 Paul Mackerras <paulus@au.ibm.com>, IBM
 * Copyright (C) 2001 Anton Blanchard <anton@au.ibm.com>, IBM
 * Copyright (C) 2002 Dave Engebretsen <engebret@us.ibm.com>, IBM
 *      Rework to support virtual processors
 *
 * Type of int is used as a full 64b word is not necessary.
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * as published by the Free Software Foundation; either version
 * 2 of the License, or (at your option) any later version.
 *
 * (the type definitions are in asm/spinlock_types.h)
 */
#include <linux/irqflags.h>
#ifdef CONFIG_PPC64
#include <asm/paca.h>
#include <asm/hvcall.h>
#endif
#include <asm/asm-compat.h>
#include <asm/synch.h>
#include <asm/ppc-opcode.h>

#ifdef CONFIG_PPC64
/* use 0x800000yy when locked, where yy == CPU number */
#ifdef __BIG_ENDIAN__

#define LOCK_TOKEN	(*(u32 *)(&get_paca()->lock_token))
#else

#define LOCK_TOKEN	(*(u32 *)(&get_paca()->paca_index))
#endif
#else

#define LOCK_TOKEN	1
#endif

#if defined(CONFIG_PPC64) && defined(CONFIG_SMP)

#define CLEAR_IO_SYNC	(get_paca()->io_sync = 0)

#define SYNC_IO		do {                                            \
                                if (unlikely(get_paca()->io_sync)) {    \
                                        mb();                           \
                                        get_paca()->io_sync = 0;        \
                                }                                       \
                        } while (0)
#else

#define CLEAR_IO_SYNC

#define SYNC_IO
#endif

#ifdef CONFIG_PPC_PSERIES

#define vcpu_is_preempted vcpu_is_preempted

static inline bool vcpu_is_preempted(int cpu) { return !!(be32_to_cpu(lppaca_of(cpu).yield_count) & 1); }

Contributors

PersonTokensPropCommitsCommitProp
pan xinhuipan xinhui27100.00%1100.00%
Total27100.00%1100.00%

#endif
static __always_inline int arch_spin_value_unlocked(arch_spinlock_t lock) { return lock.slock == 0; }

Contributors

PersonTokensPropCommitsCommitProp
michael ellermanmichael ellerman17100.00%1100.00%
Total17100.00%1100.00%


static inline int arch_spin_is_locked(arch_spinlock_t *lock) { smp_mb(); return !arch_spin_value_unlocked(*lock); }

Contributors

PersonTokensPropCommitsCommitProp
michael ellermanmichael ellerman22100.00%2100.00%
Total22100.00%2100.00%

/* * This returns the old value in the lock, so we succeeded * in getting the lock if the return value is 0. */
static inline unsigned long __arch_spin_trylock(arch_spinlock_t *lock) { unsigned long tmp, token; token = LOCK_TOKEN; __asm__ __volatile__( "1: " PPC_LWARX(%0,0,%2,1) "\n\ cmpwi 0,%0,0\n\ bne- 2f\n\ stwcx. %1,0,%2\n\ bne- 1b\n" PPC_ACQUIRE_BARRIER "2:" : "=&r" (tmp) : "r" (token), "r" (&lock->slock) : "cr0", "memory"); return tmp; }

Contributors

PersonTokensPropCommitsCommitProp
ingo molnaringo molnar1139.29%114.29%
andrew mortonandrew morton828.57%114.29%
paul mackerraspaul mackerras517.86%114.29%
thomas gleixnerthomas gleixner27.14%228.57%
bart van asschebart van assche13.57%114.29%
anton blanchardanton blanchard13.57%114.29%
Total28100.00%7100.00%


static inline int arch_spin_trylock(arch_spinlock_t *lock) { CLEAR_IO_SYNC; return __arch_spin_trylock(lock) == 0; }

Contributors

PersonTokensPropCommitsCommitProp
ingo molnaringo molnar1152.38%116.67%
andrew mortonandrew morton419.05%116.67%
thomas gleixnerthomas gleixner314.29%233.33%
paul mackerraspaul mackerras29.52%116.67%
bart van asschebart van assche14.76%116.67%
Total21100.00%6100.00%

/* * On a system with shared processors (that is, where a physical * processor is multiplexed between several virtual processors), * there is no point spinning on a lock if the holder of the lock * isn't currently scheduled on a physical processor. Instead * we detect this situation and ask the hypervisor to give the * rest of our timeslice to the lock holder. * * So that we can tell which virtual processor is holding a lock, * we put 0x80000000 | smp_processor_id() in the lock when it is * held. Conveniently, we have a word in the paca that holds this * value. */ #if defined(CONFIG_PPC_SPLPAR) /* We only yield to the hypervisor if we are in shared processor mode */ #define SHARED_PROCESSOR (lppaca_shared_proc(local_paca->lppaca_ptr)) extern void __spin_yield(arch_spinlock_t *lock); extern void __rw_yield(arch_rwlock_t *lock); #else /* SPLPAR */ #define __spin_yield(x) barrier() #define __rw_yield(x) barrier() #define SHARED_PROCESSOR 0 #endif
static inline void arch_spin_lock(arch_spinlock_t *lock) { CLEAR_IO_SYNC; while (1) { if (likely(__arch_spin_trylock(lock) == 0)) break; do { HMT_low(); if (SHARED_PROCESSOR) __spin_yield(lock); } while (unlikely(lock->slock != 0)); HMT_medium(); } }

Contributors

PersonTokensPropCommitsCommitProp
zwane mwaikambozwane mwaikambo4674.19%112.50%
anton blanchardanton blanchard812.90%112.50%
thomas gleixnerthomas gleixner34.84%225.00%
paul mackerraspaul mackerras23.23%112.50%
jake moilanenjake moilanen11.61%112.50%
ingo molnaringo molnar11.61%112.50%
bart van asschebart van assche11.61%112.50%
Total62100.00%8100.00%


static inline void arch_spin_lock_flags(arch_spinlock_t *lock, unsigned long flags) { unsigned long flags_dis; CLEAR_IO_SYNC; while (1) { if (likely(__arch_spin_trylock(lock) == 0)) break; local_save_flags(flags_dis); local_irq_restore(flags); do { HMT_low(); if (SHARED_PROCESSOR) __spin_yield(lock); } while (unlikely(lock->slock != 0)); HMT_medium(); local_irq_restore(flags_dis); } }

Contributors

PersonTokensPropCommitsCommitProp
zwane mwaikambozwane mwaikambo6272.94%111.11%
anton blanchardanton blanchard89.41%111.11%
andrew mortonandrew morton78.24%111.11%
thomas gleixnerthomas gleixner33.53%222.22%
paul mackerraspaul mackerras22.35%111.11%
ingo molnaringo molnar11.18%111.11%
jake moilanenjake moilanen11.18%111.11%
bart van asschebart van assche11.18%111.11%
Total85100.00%9100.00%


static inline void arch_spin_unlock(arch_spinlock_t *lock) { SYNC_IO; __asm__ __volatile__("# arch_spin_unlock\n\t" PPC_RELEASE_BARRIER: : :"memory"); lock->slock = 0; }

Contributors

PersonTokensPropCommitsCommitProp
ingo molnaringo molnar1672.73%116.67%
thomas gleixnerthomas gleixner29.09%233.33%
paul mackerraspaul mackerras29.09%116.67%
anton blanchardanton blanchard14.55%116.67%
bart van asschebart van assche14.55%116.67%
Total22100.00%6100.00%


static inline void arch_spin_unlock_wait(arch_spinlock_t *lock) { arch_spinlock_t lock_val; smp_mb(); /* * Atomically load and store back the lock value (unchanged). This * ensures that our observation of the lock value is ordered with * respect to other lock operations. */ __asm__ __volatile__( "1: " PPC_LWARX(%0, 0, %2, 0) "\n" " stwcx. %0, 0, %2\n" " bne- 1b\n" : "=&r" (lock_val), "+m" (*lock) : "r" (lock) : "cr0", "xer"); if (arch_spin_value_unlocked(lock_val)) goto out; while (lock->slock) { HMT_low(); if (SHARED_PROCESSOR) __spin_yield(lock); } HMT_medium(); out: smp_mb(); }

Contributors

PersonTokensPropCommitsCommitProp
boqun fengboqun feng5186.44%120.00%
ingo molnaringo molnar58.47%120.00%
thomas gleixnerthomas gleixner23.39%240.00%
paul mackerraspaul mackerras11.69%120.00%
Total59100.00%5100.00%

/* * Read-write spinlocks, allowing multiple readers * but only one writer. * * NOTE! it is quite common to have readers in interrupts * but no interrupt writers. For those circumstances we * can "mix" irq-safe locks - any writer needs to get a * irq-safe write-lock, but readers can get non-irqsafe * read-locks. */ #define arch_read_can_lock(rw) ((rw)->lock >= 0) #define arch_write_can_lock(rw) (!(rw)->lock) #ifdef CONFIG_PPC64 #define __DO_SIGN_EXTEND "extsw %0,%0\n" #define WRLOCK_TOKEN LOCK_TOKEN /* it's negative */ #else #define __DO_SIGN_EXTEND #define WRLOCK_TOKEN (-1) #endif /* * This returns the old value in the lock + 1, * so we got a read lock if the return value is > 0. */
static inline long __arch_read_trylock(arch_rwlock_t *rw) { long tmp; __asm__ __volatile__( "1: " PPC_LWARX(%0,0,%1,1) "\n" __DO_SIGN_EXTEND " addic. %0,%0,1\n\ ble- 2f\n" PPC405_ERR77(0,%1) " stwcx. %0,0,%1\n\ bne- 1b\n" PPC_ACQUIRE_BARRIER "2:" : "=&r" (tmp) : "r" (&rw->lock) : "cr0", "xer", "memory"); return tmp; }

Contributors

PersonTokensPropCommitsCommitProp
zwane mwaikambozwane mwaikambo1260.00%116.67%
andrew mortonandrew morton420.00%116.67%
thomas gleixnerthomas gleixner210.00%233.33%
bart van asschebart van assche15.00%116.67%
anton blanchardanton blanchard15.00%116.67%
Total20100.00%6100.00%

/* * This returns the old value in the lock, * so we got the write lock if the return value is 0. */
static inline long __arch_write_trylock(arch_rwlock_t *rw) { long tmp, token; token = WRLOCK_TOKEN; __asm__ __volatile__( "1: " PPC_LWARX(%0,0,%2,1) "\n\ cmpwi 0,%0,0\n\ bne- 2f\n" PPC405_ERR77(0,%1) " stwcx. %1,0,%2\n\ bne- 1b\n" PPC_ACQUIRE_BARRIER "2:" : "=&r" (tmp) : "r" (token), "r" (&rw->lock) : "cr0", "memory"); return tmp; }

Contributors

PersonTokensPropCommitsCommitProp
ingo molnaringo molnar830.77%112.50%
paul mackerraspaul mackerras519.23%112.50%
zwane mwaikambozwane mwaikambo519.23%112.50%
andrew mortonandrew morton415.38%112.50%
thomas gleixnerthomas gleixner27.69%225.00%
bart van asschebart van assche13.85%112.50%
anton blanchardanton blanchard13.85%112.50%
Total26100.00%8100.00%


static inline void arch_read_lock(arch_rwlock_t *rw) { while (1) { if (likely(__arch_read_trylock(rw) > 0)) break; do { HMT_low(); if (SHARED_PROCESSOR) __rw_yield(rw); } while (unlikely(rw->lock < 0)); HMT_medium(); } }

Contributors

PersonTokensPropCommitsCommitProp
zwane mwaikambozwane mwaikambo4880.00%116.67%
anton blanchardanton blanchard711.67%116.67%
thomas gleixnerthomas gleixner35.00%233.33%
jake moilanenjake moilanen11.67%116.67%
bart van asschebart van assche11.67%116.67%
Total60100.00%6100.00%


static inline void arch_write_lock(arch_rwlock_t *rw) { while (1) { if (likely(__arch_write_trylock(rw) == 0)) break; do { HMT_low(); if (SHARED_PROCESSOR) __rw_yield(rw); } while (unlikely(rw->lock != 0)); HMT_medium(); } }

Contributors

PersonTokensPropCommitsCommitProp
zwane mwaikambozwane mwaikambo4778.33%114.29%
anton blanchardanton blanchard711.67%114.29%
thomas gleixnerthomas gleixner35.00%228.57%
jake moilanenjake moilanen11.67%114.29%
bart van asschebart van assche11.67%114.29%
andrew mortonandrew morton11.67%114.29%
Total60100.00%7100.00%


static inline int arch_read_trylock(arch_rwlock_t *rw) { return __arch_read_trylock(rw) > 0; }

Contributors

PersonTokensPropCommitsCommitProp
ingo molnaringo molnar1578.95%125.00%
thomas gleixnerthomas gleixner315.79%250.00%
bart van asschebart van assche15.26%125.00%
Total19100.00%4100.00%


static inline int arch_write_trylock(arch_rwlock_t *rw) { return __arch_write_trylock(rw) == 0; }

Contributors

PersonTokensPropCommitsCommitProp
ingo molnaringo molnar1578.95%125.00%
thomas gleixnerthomas gleixner315.79%250.00%
bart van asschebart van assche15.26%125.00%
Total19100.00%4100.00%


static inline void arch_read_unlock(arch_rwlock_t *rw) { long tmp; __asm__ __volatile__( "# read_unlock\n\t" PPC_RELEASE_BARRIER "1: lwarx %0,0,%1\n\ addic %0,%0,-1\n" PPC405_ERR77(0,%1) " stwcx. %0,0,%1\n\ bne- 1b" : "=&r"(tmp) : "r"(&rw->lock) : "cr0", "xer", "memory"); }

Contributors

PersonTokensPropCommitsCommitProp
ingo molnaringo molnar1376.47%120.00%
thomas gleixnerthomas gleixner211.76%240.00%
anton blanchardanton blanchard15.88%120.00%
bart van asschebart van assche15.88%120.00%
Total17100.00%5100.00%


static inline void arch_write_unlock(arch_rwlock_t *rw) { __asm__ __volatile__("# write_unlock\n\t" PPC_RELEASE_BARRIER: : :"memory"); rw->lock = 0; }

Contributors

PersonTokensPropCommitsCommitProp
ingo molnaringo molnar1680.00%120.00%
thomas gleixnerthomas gleixner210.00%240.00%
anton blanchardanton blanchard15.00%120.00%
bart van asschebart van assche15.00%120.00%
Total20100.00%5100.00%

#define arch_read_lock_flags(lock, flags) arch_read_lock(lock) #define arch_write_lock_flags(lock, flags) arch_write_lock(lock) #define arch_spin_relax(lock) __spin_yield(lock) #define arch_read_relax(lock) __rw_yield(lock) #define arch_write_relax(lock) __rw_yield(lock) #endif /* __KERNEL__ */ #endif /* __ASM_SPINLOCK_H */

Overall Contributors

PersonTokensPropCommitsCommitProp
zwane mwaikambozwane mwaikambo25731.30%13.33%
ingo molnaringo molnar11514.01%13.33%
paul mackerraspaul mackerras9811.94%26.67%
anton blanchardanton blanchard698.40%516.67%
boqun fengboqun feng516.21%13.33%
thomas gleixnerthomas gleixner465.60%413.33%
andrew mortonandrew morton455.48%26.67%
michael ellermanmichael ellerman394.75%310.00%
pan xinhuipan xinhui364.38%13.33%
martin schwidefskymartin schwidefsky182.19%26.67%
robin holtrobin holt141.71%13.33%
bart van asschebart van assche131.58%13.33%
arnd bergmannarnd bergmann60.73%13.33%
jake moilanenjake moilanen40.49%13.33%
linus torvaldslinus torvalds40.49%13.33%
benjamin herrenschmidtbenjamin herrenschmidt30.37%13.33%
david gibsondavid gibson20.24%13.33%
stephen rothwellstephen rothwell10.12%13.33%
Total821100.00%30100.00%
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.