[SPARC64]: Make debugging spinlocks usable again.
When the spinlock routines were moved out of line into kernel/spinlock.c this made it so that the debugging spinlocks record lock acquisition program counts in the kernel/spinlock.c functions not in their callers. This makes the debugging info kind of useless. So record the correct caller's program counter and now this feature is useful once more. Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
ca7c8d2c1e
commit
442464a500
3 changed files with 37 additions and 64 deletions
|
@ -99,17 +99,6 @@ extern int __ashrdi3(int, int);
|
||||||
extern void dump_thread(struct pt_regs *, struct user *);
|
extern void dump_thread(struct pt_regs *, struct user *);
|
||||||
extern int dump_fpu (struct pt_regs * regs, elf_fpregset_t * fpregs);
|
extern int dump_fpu (struct pt_regs * regs, elf_fpregset_t * fpregs);
|
||||||
|
|
||||||
#if defined(CONFIG_SMP) && defined(CONFIG_DEBUG_SPINLOCK)
|
|
||||||
extern void _do_spin_lock (spinlock_t *lock, char *str);
|
|
||||||
extern void _do_spin_unlock (spinlock_t *lock);
|
|
||||||
extern int _spin_trylock (spinlock_t *lock);
|
|
||||||
extern void _do_read_lock(rwlock_t *rw, char *str);
|
|
||||||
extern void _do_read_unlock(rwlock_t *rw, char *str);
|
|
||||||
extern void _do_write_lock(rwlock_t *rw, char *str);
|
|
||||||
extern void _do_write_unlock(rwlock_t *rw);
|
|
||||||
extern int _do_write_trylock(rwlock_t *rw, char *str);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
extern unsigned long phys_base;
|
extern unsigned long phys_base;
|
||||||
extern unsigned long pfn_base;
|
extern unsigned long pfn_base;
|
||||||
|
|
||||||
|
@ -152,18 +141,6 @@ EXPORT_SYMBOL(_mcount);
|
||||||
EXPORT_SYMBOL(cpu_online_map);
|
EXPORT_SYMBOL(cpu_online_map);
|
||||||
EXPORT_SYMBOL(phys_cpu_present_map);
|
EXPORT_SYMBOL(phys_cpu_present_map);
|
||||||
|
|
||||||
/* Spinlock debugging library, optional. */
|
|
||||||
#ifdef CONFIG_DEBUG_SPINLOCK
|
|
||||||
EXPORT_SYMBOL(_do_spin_lock);
|
|
||||||
EXPORT_SYMBOL(_do_spin_unlock);
|
|
||||||
EXPORT_SYMBOL(_spin_trylock);
|
|
||||||
EXPORT_SYMBOL(_do_read_lock);
|
|
||||||
EXPORT_SYMBOL(_do_read_unlock);
|
|
||||||
EXPORT_SYMBOL(_do_write_lock);
|
|
||||||
EXPORT_SYMBOL(_do_write_unlock);
|
|
||||||
EXPORT_SYMBOL(_do_write_trylock);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
EXPORT_SYMBOL(smp_call_function);
|
EXPORT_SYMBOL(smp_call_function);
|
||||||
#endif /* CONFIG_SMP */
|
#endif /* CONFIG_SMP */
|
||||||
|
|
||||||
|
|
|
@ -12,8 +12,6 @@
|
||||||
|
|
||||||
#ifdef CONFIG_SMP
|
#ifdef CONFIG_SMP
|
||||||
|
|
||||||
#define GET_CALLER(PC) __asm__ __volatile__("mov %%i7, %0" : "=r" (PC))
|
|
||||||
|
|
||||||
static inline void show (char *str, spinlock_t *lock, unsigned long caller)
|
static inline void show (char *str, spinlock_t *lock, unsigned long caller)
|
||||||
{
|
{
|
||||||
int cpu = smp_processor_id();
|
int cpu = smp_processor_id();
|
||||||
|
@ -51,14 +49,13 @@ static inline void show_write (char *str, rwlock_t *lock, unsigned long caller)
|
||||||
#undef INIT_STUCK
|
#undef INIT_STUCK
|
||||||
#define INIT_STUCK 100000000
|
#define INIT_STUCK 100000000
|
||||||
|
|
||||||
void _do_spin_lock(spinlock_t *lock, char *str)
|
void _do_spin_lock(spinlock_t *lock, char *str, unsigned long caller)
|
||||||
{
|
{
|
||||||
unsigned long caller, val;
|
unsigned long val;
|
||||||
int stuck = INIT_STUCK;
|
int stuck = INIT_STUCK;
|
||||||
int cpu = get_cpu();
|
int cpu = get_cpu();
|
||||||
int shown = 0;
|
int shown = 0;
|
||||||
|
|
||||||
GET_CALLER(caller);
|
|
||||||
again:
|
again:
|
||||||
__asm__ __volatile__("ldstub [%1], %0"
|
__asm__ __volatile__("ldstub [%1], %0"
|
||||||
: "=r" (val)
|
: "=r" (val)
|
||||||
|
@ -84,12 +81,11 @@ void _do_spin_lock(spinlock_t *lock, char *str)
|
||||||
put_cpu();
|
put_cpu();
|
||||||
}
|
}
|
||||||
|
|
||||||
int _do_spin_trylock(spinlock_t *lock)
|
int _do_spin_trylock(spinlock_t *lock, unsigned long caller)
|
||||||
{
|
{
|
||||||
unsigned long val, caller;
|
unsigned long val;
|
||||||
int cpu = get_cpu();
|
int cpu = get_cpu();
|
||||||
|
|
||||||
GET_CALLER(caller);
|
|
||||||
__asm__ __volatile__("ldstub [%1], %0"
|
__asm__ __volatile__("ldstub [%1], %0"
|
||||||
: "=r" (val)
|
: "=r" (val)
|
||||||
: "r" (&(lock->lock))
|
: "r" (&(lock->lock))
|
||||||
|
@ -118,14 +114,13 @@ void _do_spin_unlock(spinlock_t *lock)
|
||||||
|
|
||||||
/* Keep INIT_STUCK the same... */
|
/* Keep INIT_STUCK the same... */
|
||||||
|
|
||||||
void _do_read_lock (rwlock_t *rw, char *str)
|
void _do_read_lock(rwlock_t *rw, char *str, unsigned long caller)
|
||||||
{
|
{
|
||||||
unsigned long caller, val;
|
unsigned long val;
|
||||||
int stuck = INIT_STUCK;
|
int stuck = INIT_STUCK;
|
||||||
int cpu = get_cpu();
|
int cpu = get_cpu();
|
||||||
int shown = 0;
|
int shown = 0;
|
||||||
|
|
||||||
GET_CALLER(caller);
|
|
||||||
wlock_again:
|
wlock_again:
|
||||||
/* Wait for any writer to go away. */
|
/* Wait for any writer to go away. */
|
||||||
while (((long)(rw->lock)) < 0) {
|
while (((long)(rw->lock)) < 0) {
|
||||||
|
@ -157,15 +152,13 @@ void _do_read_lock (rwlock_t *rw, char *str)
|
||||||
put_cpu();
|
put_cpu();
|
||||||
}
|
}
|
||||||
|
|
||||||
void _do_read_unlock (rwlock_t *rw, char *str)
|
void _do_read_unlock(rwlock_t *rw, char *str, unsigned long caller)
|
||||||
{
|
{
|
||||||
unsigned long caller, val;
|
unsigned long val;
|
||||||
int stuck = INIT_STUCK;
|
int stuck = INIT_STUCK;
|
||||||
int cpu = get_cpu();
|
int cpu = get_cpu();
|
||||||
int shown = 0;
|
int shown = 0;
|
||||||
|
|
||||||
GET_CALLER(caller);
|
|
||||||
|
|
||||||
/* Drop our identity _first_. */
|
/* Drop our identity _first_. */
|
||||||
rw->reader_pc[cpu] = 0;
|
rw->reader_pc[cpu] = 0;
|
||||||
current->thread.smp_lock_count--;
|
current->thread.smp_lock_count--;
|
||||||
|
@ -193,14 +186,13 @@ void _do_read_unlock (rwlock_t *rw, char *str)
|
||||||
put_cpu();
|
put_cpu();
|
||||||
}
|
}
|
||||||
|
|
||||||
void _do_write_lock (rwlock_t *rw, char *str)
|
void _do_write_lock(rwlock_t *rw, char *str, unsigned long caller)
|
||||||
{
|
{
|
||||||
unsigned long caller, val;
|
unsigned long val;
|
||||||
int stuck = INIT_STUCK;
|
int stuck = INIT_STUCK;
|
||||||
int cpu = get_cpu();
|
int cpu = get_cpu();
|
||||||
int shown = 0;
|
int shown = 0;
|
||||||
|
|
||||||
GET_CALLER(caller);
|
|
||||||
wlock_again:
|
wlock_again:
|
||||||
/* Spin while there is another writer. */
|
/* Spin while there is another writer. */
|
||||||
while (((long)rw->lock) < 0) {
|
while (((long)rw->lock) < 0) {
|
||||||
|
@ -278,14 +270,12 @@ void _do_write_lock (rwlock_t *rw, char *str)
|
||||||
put_cpu();
|
put_cpu();
|
||||||
}
|
}
|
||||||
|
|
||||||
void _do_write_unlock(rwlock_t *rw)
|
void _do_write_unlock(rwlock_t *rw, unsigned long caller)
|
||||||
{
|
{
|
||||||
unsigned long caller, val;
|
unsigned long val;
|
||||||
int stuck = INIT_STUCK;
|
int stuck = INIT_STUCK;
|
||||||
int shown = 0;
|
int shown = 0;
|
||||||
|
|
||||||
GET_CALLER(caller);
|
|
||||||
|
|
||||||
/* Drop our identity _first_ */
|
/* Drop our identity _first_ */
|
||||||
rw->writer_pc = 0;
|
rw->writer_pc = 0;
|
||||||
rw->writer_cpu = NO_PROC_ID;
|
rw->writer_cpu = NO_PROC_ID;
|
||||||
|
@ -313,13 +303,11 @@ void _do_write_unlock(rwlock_t *rw)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int _do_write_trylock (rwlock_t *rw, char *str)
|
int _do_write_trylock(rwlock_t *rw, char *str, unsigned long caller)
|
||||||
{
|
{
|
||||||
unsigned long caller, val;
|
unsigned long val;
|
||||||
int cpu = get_cpu();
|
int cpu = get_cpu();
|
||||||
|
|
||||||
GET_CALLER(caller);
|
|
||||||
|
|
||||||
/* Try to acuire the write bit. */
|
/* Try to acuire the write bit. */
|
||||||
__asm__ __volatile__(
|
__asm__ __volatile__(
|
||||||
" mov 1, %%g3\n"
|
" mov 1, %%g3\n"
|
||||||
|
|
|
@ -132,12 +132,15 @@ do { \
|
||||||
membar("#LoadLoad"); \
|
membar("#LoadLoad"); \
|
||||||
} while((__lock)->lock)
|
} while((__lock)->lock)
|
||||||
|
|
||||||
extern void _do_spin_lock (spinlock_t *lock, char *str);
|
extern void _do_spin_lock(spinlock_t *lock, char *str, unsigned long caller);
|
||||||
extern void _do_spin_unlock (spinlock_t *lock);
|
extern void _do_spin_unlock(spinlock_t *lock);
|
||||||
extern int _do_spin_trylock (spinlock_t *lock);
|
extern int _do_spin_trylock(spinlock_t *lock, unsigned long caller);
|
||||||
|
|
||||||
#define _raw_spin_trylock(lp) _do_spin_trylock(lp)
|
#define _raw_spin_trylock(lp) \
|
||||||
#define _raw_spin_lock(lock) _do_spin_lock(lock, "spin_lock")
|
_do_spin_trylock(lp, (unsigned long) __builtin_return_address(0))
|
||||||
|
#define _raw_spin_lock(lock) \
|
||||||
|
_do_spin_lock(lock, "spin_lock", \
|
||||||
|
(unsigned long) __builtin_return_address(0))
|
||||||
#define _raw_spin_unlock(lock) _do_spin_unlock(lock)
|
#define _raw_spin_unlock(lock) _do_spin_unlock(lock)
|
||||||
#define _raw_spin_lock_flags(lock, flags) _raw_spin_lock(lock)
|
#define _raw_spin_lock_flags(lock, flags) _raw_spin_lock(lock)
|
||||||
|
|
||||||
|
@ -279,37 +282,41 @@ typedef struct {
|
||||||
#define RW_LOCK_UNLOCKED (rwlock_t) { 0, 0, 0xff, { } }
|
#define RW_LOCK_UNLOCKED (rwlock_t) { 0, 0, 0xff, { } }
|
||||||
#define rwlock_init(lp) do { *(lp) = RW_LOCK_UNLOCKED; } while(0)
|
#define rwlock_init(lp) do { *(lp) = RW_LOCK_UNLOCKED; } while(0)
|
||||||
|
|
||||||
extern void _do_read_lock(rwlock_t *rw, char *str);
|
extern void _do_read_lock(rwlock_t *rw, char *str, unsigned long caller);
|
||||||
extern void _do_read_unlock(rwlock_t *rw, char *str);
|
extern void _do_read_unlock(rwlock_t *rw, char *str, unsigned long caller);
|
||||||
extern void _do_write_lock(rwlock_t *rw, char *str);
|
extern void _do_write_lock(rwlock_t *rw, char *str, unsigned long caller);
|
||||||
extern void _do_write_unlock(rwlock_t *rw);
|
extern void _do_write_unlock(rwlock_t *rw, unsigned long caller);
|
||||||
extern int _do_write_trylock(rwlock_t *rw, char *str);
|
extern int _do_write_trylock(rwlock_t *rw, char *str, unsigned long caller);
|
||||||
|
|
||||||
#define _raw_read_lock(lock) \
|
#define _raw_read_lock(lock) \
|
||||||
do { unsigned long flags; \
|
do { unsigned long flags; \
|
||||||
local_irq_save(flags); \
|
local_irq_save(flags); \
|
||||||
_do_read_lock(lock, "read_lock"); \
|
_do_read_lock(lock, "read_lock", \
|
||||||
|
(unsigned long) __builtin_return_address(0)); \
|
||||||
local_irq_restore(flags); \
|
local_irq_restore(flags); \
|
||||||
} while(0)
|
} while(0)
|
||||||
|
|
||||||
#define _raw_read_unlock(lock) \
|
#define _raw_read_unlock(lock) \
|
||||||
do { unsigned long flags; \
|
do { unsigned long flags; \
|
||||||
local_irq_save(flags); \
|
local_irq_save(flags); \
|
||||||
_do_read_unlock(lock, "read_unlock"); \
|
_do_read_unlock(lock, "read_unlock", \
|
||||||
|
(unsigned long) __builtin_return_address(0)); \
|
||||||
local_irq_restore(flags); \
|
local_irq_restore(flags); \
|
||||||
} while(0)
|
} while(0)
|
||||||
|
|
||||||
#define _raw_write_lock(lock) \
|
#define _raw_write_lock(lock) \
|
||||||
do { unsigned long flags; \
|
do { unsigned long flags; \
|
||||||
local_irq_save(flags); \
|
local_irq_save(flags); \
|
||||||
_do_write_lock(lock, "write_lock"); \
|
_do_write_lock(lock, "write_lock", \
|
||||||
|
(unsigned long) __builtin_return_address(0)); \
|
||||||
local_irq_restore(flags); \
|
local_irq_restore(flags); \
|
||||||
} while(0)
|
} while(0)
|
||||||
|
|
||||||
#define _raw_write_unlock(lock) \
|
#define _raw_write_unlock(lock) \
|
||||||
do { unsigned long flags; \
|
do { unsigned long flags; \
|
||||||
local_irq_save(flags); \
|
local_irq_save(flags); \
|
||||||
_do_write_unlock(lock); \
|
_do_write_unlock(lock, \
|
||||||
|
(unsigned long) __builtin_return_address(0)); \
|
||||||
local_irq_restore(flags); \
|
local_irq_restore(flags); \
|
||||||
} while(0)
|
} while(0)
|
||||||
|
|
||||||
|
@ -317,7 +324,8 @@ do { unsigned long flags; \
|
||||||
({ unsigned long flags; \
|
({ unsigned long flags; \
|
||||||
int val; \
|
int val; \
|
||||||
local_irq_save(flags); \
|
local_irq_save(flags); \
|
||||||
val = _do_write_trylock(lock, "write_trylock"); \
|
val = _do_write_trylock(lock, "write_trylock", \
|
||||||
|
(unsigned long) __builtin_return_address(0)); \
|
||||||
local_irq_restore(flags); \
|
local_irq_restore(flags); \
|
||||||
val; \
|
val; \
|
||||||
})
|
})
|
||||||
|
|
Loading…
Reference in a new issue