Commit 63096c89 authored by Matthew Wilcox's avatar Matthew Wilcox Committed by Linus Torvalds

[PATCH] parisc: debuglock fixes

 - pdc_printf from Randolph Chung
 - implement memory barriers in debuglocks
Committed-by: default avatarThibaut Varene <varenet@parisc-linux.org>
parent 54064592
...@@ -71,6 +71,19 @@ void pdc_outc(unsigned char c) ...@@ -71,6 +71,19 @@ void pdc_outc(unsigned char c)
pdc_iodc_outc(c); pdc_iodc_outc(c);
} }
void pdc_printf(const char *fmt, ...)
{
va_list args;
char buf[1024];
int i, len;
va_start(args, fmt);
len = vscnprintf(buf, sizeof(buf), fmt, args);
va_end(args);
for (i = 0; i < len; i++)
pdc_iodc_outc(buf[i]);
}
int pdc_console_poll_key(struct console *co) int pdc_console_poll_key(struct console *co)
{ {
......
...@@ -18,6 +18,11 @@ ...@@ -18,6 +18,11 @@
* You should have received a copy of the GNU General Public License * You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software * along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
* We use pdc_printf() throughout the file for all output messages, to avoid
* losing messages because of disabled interrupts. Since we're using these
* messages for debugging purposes, it makes sense not to send them to the
* linux console.
*/ */
...@@ -27,12 +32,15 @@ ...@@ -27,12 +32,15 @@
#include <linux/spinlock.h> #include <linux/spinlock.h>
#include <linux/hardirq.h> /* in_interrupt() */ #include <linux/hardirq.h> /* in_interrupt() */
#include <asm/system.h> #include <asm/system.h>
#include <asm/hardirq.h> /* in_interrupt() */
#include <asm/pdc.h>
#undef INIT_STUCK #undef INIT_STUCK
#define INIT_STUCK 1L << 30 #define INIT_STUCK 1L << 30
#ifdef CONFIG_DEBUG_SPINLOCK #ifdef CONFIG_DEBUG_SPINLOCK
void _dbg_spin_lock(spinlock_t * lock, const char *base_file, int line_no) void _dbg_spin_lock(spinlock_t * lock, const char *base_file, int line_no)
{ {
volatile unsigned int *a; volatile unsigned int *a;
...@@ -59,12 +67,14 @@ void _dbg_spin_lock(spinlock_t * lock, const char *base_file, int line_no) ...@@ -59,12 +67,14 @@ void _dbg_spin_lock(spinlock_t * lock, const char *base_file, int line_no)
* <tausq> __ldcw() returns 1 if we get the lock; otherwise we * <tausq> __ldcw() returns 1 if we get the lock; otherwise we
* spin until the value of the lock changes, or we time out. * spin until the value of the lock changes, or we time out.
*/ */
mb();
a = __ldcw_align(lock); a = __ldcw_align(lock);
while (stuck && (__ldcw(a) == 0)) while (stuck && (__ldcw(a) == 0))
while ((*a == 0) && --stuck); while ((*a == 0) && --stuck);
mb();
if (unlikely(stuck <= 0)) { if (unlikely(stuck <= 0)) {
printk(KERN_WARNING pdc_printf(
"%s:%d: spin_lock(%s/%p) stuck in %s at %p(%d)" "%s:%d: spin_lock(%s/%p) stuck in %s at %p(%d)"
" owned by %s:%d in %s at %p(%d)\n", " owned by %s:%d in %s at %p(%d)\n",
base_file, line_no, lock->module, lock, base_file, line_no, lock->module, lock,
...@@ -84,7 +94,7 @@ void _dbg_spin_lock(spinlock_t * lock, const char *base_file, int line_no) ...@@ -84,7 +94,7 @@ void _dbg_spin_lock(spinlock_t * lock, const char *base_file, int line_no)
lock->bline = line_no; lock->bline = line_no;
if (unlikely(printed)) { if (unlikely(printed)) {
printk(KERN_WARNING pdc_printf(
"%s:%d: spin_lock grabbed in %s at %p(%d) %ld ticks\n", "%s:%d: spin_lock grabbed in %s at %p(%d) %ld ticks\n",
base_file, line_no, current->comm, inline_pc, base_file, line_no, current->comm, inline_pc,
cpu, jiffies - started); cpu, jiffies - started);
...@@ -94,21 +104,28 @@ void _dbg_spin_lock(spinlock_t * lock, const char *base_file, int line_no) ...@@ -94,21 +104,28 @@ void _dbg_spin_lock(spinlock_t * lock, const char *base_file, int line_no)
void _dbg_spin_unlock(spinlock_t * lock, const char *base_file, int line_no) void _dbg_spin_unlock(spinlock_t * lock, const char *base_file, int line_no)
{ {
CHECK_LOCK(lock); CHECK_LOCK(lock);
volatile unsigned int *a = __ldcw_align(lock); volatile unsigned int *a;
mb();
a = __ldcw_align(lock);
if (unlikely((*a != 0) && lock->babble)) { if (unlikely((*a != 0) && lock->babble)) {
lock->babble--; lock->babble--;
printk(KERN_WARNING pdc_printf(
"%s:%d: spin_unlock(%s:%p) not locked\n", "%s:%d: spin_unlock(%s:%p) not locked\n",
base_file, line_no, lock->module, lock); base_file, line_no, lock->module, lock);
} }
*a = 1; *a = 1;
mb();
} }
int _dbg_spin_trylock(spinlock_t * lock, const char *base_file, int line_no) int _dbg_spin_trylock(spinlock_t * lock, const char *base_file, int line_no)
{ {
int ret; int ret;
volatile unsigned int *a = __ldcw_align(lock); volatile unsigned int *a;
if ((ret = (__ldcw(a) != 0))) { mb();
a = __ldcw_align(lock);
ret = (__ldcw(a) != 0);
mb();
if (ret) {
lock->oncpu = smp_processor_id(); lock->oncpu = smp_processor_id();
lock->previous = __builtin_return_address(0); lock->previous = __builtin_return_address(0);
lock->task = current; lock->task = current;
...@@ -150,7 +167,7 @@ void _dbg_write_lock(rwlock_t *rw, const char *bfile, int bline) ...@@ -150,7 +167,7 @@ void _dbg_write_lock(rwlock_t *rw, const char *bfile, int bline)
int cpu = smp_processor_id(); int cpu = smp_processor_id();
if(unlikely(in_interrupt())) { /* acquiring write lock in interrupt context, bad idea */ if(unlikely(in_interrupt())) { /* acquiring write lock in interrupt context, bad idea */
printk(KERN_WARNING "write_lock caller: %s:%d, IRQs enabled,\n", bfile, bline); pdc_printf("write_lock caller: %s:%d, IRQs enabled,\n", bfile, bline);
BUG(); BUG();
} }
...@@ -167,7 +184,7 @@ void _dbg_write_lock(rwlock_t *rw, const char *bfile, int bline) ...@@ -167,7 +184,7 @@ void _dbg_write_lock(rwlock_t *rw, const char *bfile, int bline)
stuck--; stuck--;
if ((unlikely(stuck <= 0)) && (rw->counter < 0)) { if ((unlikely(stuck <= 0)) && (rw->counter < 0)) {
printk(KERN_WARNING pdc_printf(
"%s:%d: write_lock stuck on writer" "%s:%d: write_lock stuck on writer"
" in %s at %p(%d) %ld ticks\n", " in %s at %p(%d) %ld ticks\n",
bfile, bline, current->comm, inline_pc, bfile, bline, current->comm, inline_pc,
...@@ -176,7 +193,7 @@ void _dbg_write_lock(rwlock_t *rw, const char *bfile, int bline) ...@@ -176,7 +193,7 @@ void _dbg_write_lock(rwlock_t *rw, const char *bfile, int bline)
printed = 1; printed = 1;
} }
else if (unlikely(stuck <= 0)) { else if (unlikely(stuck <= 0)) {
printk(KERN_WARNING pdc_printf(
"%s:%d: write_lock stuck on reader" "%s:%d: write_lock stuck on reader"
" in %s at %p(%d) %ld ticks\n", " in %s at %p(%d) %ld ticks\n",
bfile, bline, current->comm, inline_pc, bfile, bline, current->comm, inline_pc,
...@@ -194,7 +211,7 @@ void _dbg_write_lock(rwlock_t *rw, const char *bfile, int bline) ...@@ -194,7 +211,7 @@ void _dbg_write_lock(rwlock_t *rw, const char *bfile, int bline)
rw->counter = -1; /* remember we are locked */ rw->counter = -1; /* remember we are locked */
if (unlikely(printed)) { if (unlikely(printed)) {
printk(KERN_WARNING pdc_printf(
"%s:%d: write_lock grabbed in %s at %p(%d) %ld ticks\n", "%s:%d: write_lock grabbed in %s at %p(%d) %ld ticks\n",
bfile, bline, current->comm, inline_pc, bfile, bline, current->comm, inline_pc,
cpu, jiffies - started); cpu, jiffies - started);
...@@ -215,7 +232,7 @@ void _dbg_read_lock(rwlock_t * rw, const char *bfile, int bline) ...@@ -215,7 +232,7 @@ void _dbg_read_lock(rwlock_t * rw, const char *bfile, int bline)
rw->counter++; rw->counter++;
#if 0 #if 0
printk(KERN_WARNING pdc_printf(
"%s:%d: read_lock grabbed in %s at %p(%d) %ld ticks\n", "%s:%d: read_lock grabbed in %s at %p(%d) %ld ticks\n",
bfile, bline, current->comm, inline_pc, bfile, bline, current->comm, inline_pc,
cpu, jiffies - started); cpu, jiffies - started);
......
...@@ -754,6 +754,7 @@ void pdc_io_reset_devices(void); ...@@ -754,6 +754,7 @@ void pdc_io_reset_devices(void);
int pdc_iodc_getc(void); int pdc_iodc_getc(void);
void pdc_iodc_putc(unsigned char c); void pdc_iodc_putc(unsigned char c);
void pdc_iodc_outc(unsigned char c); void pdc_iodc_outc(unsigned char c);
void pdc_printf(const char *fmt, ...);
void pdc_emergency_unlock(void); void pdc_emergency_unlock(void);
int pdc_sti_call(unsigned long func, unsigned long flags, int pdc_sti_call(unsigned long func, unsigned long flags,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment