2005-04-16 15:20:36 -07:00
/*
* linux / include / asm - arm / locks . h
*
* Copyright ( C ) 2000 Russell King
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*
* Interrupt safe locking assembler .
*/
# ifndef __ASM_PROC_LOCKS_H
# define __ASM_PROC_LOCKS_H
# if __LINUX_ARM_ARCH__ >= 6
# define __down_op(ptr,fail) \
( { \
__asm__ __volatile__ ( \
" @ down_op \n " \
" 1: ldrex lr, [%0] \n " \
" sub lr, lr, %1 \n " \
" strex ip, lr, [%0] \n " \
" teq ip, #0 \n " \
" bne 1b \n " \
" teq lr, #0 \n " \
" movmi ip, %0 \n " \
" blmi " # fail \
: \
: " r " ( ptr ) , " I " ( 1 ) \
2005-07-26 19:44:26 +01:00
: " ip " , " lr " , " cc " ) ; \
smp_mb ( ) ; \
2005-04-16 15:20:36 -07:00
} )
# define __down_op_ret(ptr,fail) \
( { \
unsigned int ret ; \
__asm__ __volatile__ ( \
" @ down_op_ret \n " \
" 1: ldrex lr, [%1] \n " \
" sub lr, lr, %2 \n " \
" strex ip, lr, [%1] \n " \
" teq ip, #0 \n " \
" bne 1b \n " \
" teq lr, #0 \n " \
" movmi ip, %1 \n " \
" movpl ip, #0 \n " \
" blmi " # fail " \n " \
" mov %0, ip " \
: " =&r " ( ret ) \
: " r " ( ptr ) , " I " ( 1 ) \
2005-07-26 19:44:26 +01:00
: " ip " , " lr " , " cc " ) ; \
smp_mb ( ) ; \
2005-04-16 15:20:36 -07:00
ret ; \
} )
# define __up_op(ptr,wake) \
( { \
2005-07-26 19:44:26 +01:00
smp_mb ( ) ; \
2005-04-16 15:20:36 -07:00
__asm__ __volatile__ ( \
" @ up_op \n " \
" 1: ldrex lr, [%0] \n " \
" add lr, lr, %1 \n " \
" strex ip, lr, [%0] \n " \
" teq ip, #0 \n " \
" bne 1b \n " \
2005-07-24 12:13:40 +01:00
" cmp lr, #0 \n " \
2005-04-16 15:20:36 -07:00
" movle ip, %0 \n " \
" blle " # wake \
: \
: " r " ( ptr ) , " I " ( 1 ) \
2005-07-26 19:44:26 +01:00
: " ip " , " lr " , " cc " ) ; \
2005-04-16 15:20:36 -07:00
} )
/*
* The value 0x01000000 supports up to 128 processors and
* lots of processes . BIAS must be chosen such that sub ' ing
* BIAS once per CPU will result in the long remaining
* negative .
*/
# define RW_LOCK_BIAS 0x01000000
# define RW_LOCK_BIAS_STR "0x01000000"
# define __down_op_write(ptr,fail) \
( { \
__asm__ __volatile__ ( \
" @ down_op_write \n " \
" 1: ldrex lr, [%0] \n " \
" sub lr, lr, %1 \n " \
" strex ip, lr, [%0] \n " \
" teq ip, #0 \n " \
" bne 1b \n " \
" teq lr, #0 \n " \
" movne ip, %0 \n " \
" blne " # fail \
: \
: " r " ( ptr ) , " I " ( RW_LOCK_BIAS ) \
2005-07-26 19:44:26 +01:00
: " ip " , " lr " , " cc " ) ; \
smp_mb ( ) ; \
2005-04-16 15:20:36 -07:00
} )
# define __up_op_write(ptr,wake) \
( { \
2005-07-26 19:44:26 +01:00
smp_mb ( ) ; \
2005-04-16 15:20:36 -07:00
__asm__ __volatile__ ( \
2005-10-18 07:51:34 +01:00
" @ up_op_write \n " \
2005-04-16 15:20:36 -07:00
" 1: ldrex lr, [%0] \n " \
2005-07-24 12:13:40 +01:00
" adds lr, lr, %1 \n " \
2005-04-16 15:20:36 -07:00
" strex ip, lr, [%0] \n " \
" teq ip, #0 \n " \
" bne 1b \n " \
" movcs ip, %0 \n " \
" blcs " # wake \
: \
: " r " ( ptr ) , " I " ( RW_LOCK_BIAS ) \
2005-07-26 19:44:26 +01:00
: " ip " , " lr " , " cc " ) ; \
2005-04-16 15:20:36 -07:00
} )
# define __down_op_read(ptr,fail) \
__down_op ( ptr , fail )
# define __up_op_read(ptr,wake) \
( { \
2005-07-26 19:44:26 +01:00
smp_mb ( ) ; \
2005-04-16 15:20:36 -07:00
__asm__ __volatile__ ( \
" @ up_op_read \n " \
" 1: ldrex lr, [%0] \n " \
" add lr, lr, %1 \n " \
" strex ip, lr, [%0] \n " \
" teq ip, #0 \n " \
" bne 1b \n " \
" teq lr, #0 \n " \
" moveq ip, %0 \n " \
" bleq " # wake \
: \
: " r " ( ptr ) , " I " ( 1 ) \
2005-07-26 19:44:26 +01:00
: " ip " , " lr " , " cc " ) ; \
2005-04-16 15:20:36 -07:00
} )
# else
# define __down_op(ptr,fail) \
( { \
__asm__ __volatile__ ( \
" @ down_op \n " \
" mrs ip, cpsr \n " \
" orr lr, ip, #128 \n " \
" msr cpsr_c, lr \n " \
" ldr lr, [%0] \n " \
" subs lr, lr, %1 \n " \
" str lr, [%0] \n " \
" msr cpsr_c, ip \n " \
" movmi ip, %0 \n " \
" blmi " # fail \
: \
: " r " ( ptr ) , " I " ( 1 ) \
2005-07-26 19:44:26 +01:00
: " ip " , " lr " , " cc " ) ; \
smp_mb ( ) ; \
2005-04-16 15:20:36 -07:00
} )
# define __down_op_ret(ptr,fail) \
( { \
unsigned int ret ; \
__asm__ __volatile__ ( \
" @ down_op_ret \n " \
" mrs ip, cpsr \n " \
" orr lr, ip, #128 \n " \
" msr cpsr_c, lr \n " \
" ldr lr, [%1] \n " \
" subs lr, lr, %2 \n " \
" str lr, [%1] \n " \
" msr cpsr_c, ip \n " \
" movmi ip, %1 \n " \
" movpl ip, #0 \n " \
" blmi " # fail " \n " \
" mov %0, ip " \
: " =&r " ( ret ) \
: " r " ( ptr ) , " I " ( 1 ) \
2005-07-26 19:44:26 +01:00
: " ip " , " lr " , " cc " ) ; \
smp_mb ( ) ; \
2005-04-16 15:20:36 -07:00
ret ; \
} )
# define __up_op(ptr,wake) \
( { \
2005-07-26 19:44:26 +01:00
smp_mb ( ) ; \
2005-04-16 15:20:36 -07:00
__asm__ __volatile__ ( \
" @ up_op \n " \
" mrs ip, cpsr \n " \
" orr lr, ip, #128 \n " \
" msr cpsr_c, lr \n " \
" ldr lr, [%0] \n " \
" adds lr, lr, %1 \n " \
" str lr, [%0] \n " \
" msr cpsr_c, ip \n " \
" movle ip, %0 \n " \
" blle " # wake \
: \
: " r " ( ptr ) , " I " ( 1 ) \
2005-07-26 19:44:26 +01:00
: " ip " , " lr " , " cc " ) ; \
2005-04-16 15:20:36 -07:00
} )
/*
* The value 0x01000000 supports up to 128 processors and
* lots of processes . BIAS must be chosen such that sub ' ing
* BIAS once per CPU will result in the long remaining
* negative .
*/
# define RW_LOCK_BIAS 0x01000000
# define RW_LOCK_BIAS_STR "0x01000000"
# define __down_op_write(ptr,fail) \
( { \
__asm__ __volatile__ ( \
" @ down_op_write \n " \
" mrs ip, cpsr \n " \
" orr lr, ip, #128 \n " \
" msr cpsr_c, lr \n " \
" ldr lr, [%0] \n " \
" subs lr, lr, %1 \n " \
" str lr, [%0] \n " \
" msr cpsr_c, ip \n " \
" movne ip, %0 \n " \
" blne " # fail \
: \
: " r " ( ptr ) , " I " ( RW_LOCK_BIAS ) \
2005-07-26 19:44:26 +01:00
: " ip " , " lr " , " cc " ) ; \
smp_mb ( ) ; \
2005-04-16 15:20:36 -07:00
} )
# define __up_op_write(ptr,wake) \
( { \
__asm__ __volatile__ ( \
2005-10-18 07:51:34 +01:00
" @ up_op_write \n " \
2005-04-16 15:20:36 -07:00
" mrs ip, cpsr \n " \
" orr lr, ip, #128 \n " \
" msr cpsr_c, lr \n " \
" ldr lr, [%0] \n " \
" adds lr, lr, %1 \n " \
" str lr, [%0] \n " \
" msr cpsr_c, ip \n " \
" movcs ip, %0 \n " \
" blcs " # wake \
: \
: " r " ( ptr ) , " I " ( RW_LOCK_BIAS ) \
2005-07-26 19:44:26 +01:00
: " ip " , " lr " , " cc " ) ; \
smp_mb ( ) ; \
2005-04-16 15:20:36 -07:00
} )
# define __down_op_read(ptr,fail) \
__down_op ( ptr , fail )
# define __up_op_read(ptr,wake) \
( { \
2005-07-26 19:44:26 +01:00
smp_mb ( ) ; \
2005-04-16 15:20:36 -07:00
__asm__ __volatile__ ( \
" @ up_op_read \n " \
" mrs ip, cpsr \n " \
" orr lr, ip, #128 \n " \
" msr cpsr_c, lr \n " \
" ldr lr, [%0] \n " \
" adds lr, lr, %1 \n " \
" str lr, [%0] \n " \
" msr cpsr_c, ip \n " \
" moveq ip, %0 \n " \
" bleq " # wake \
: \
: " r " ( ptr ) , " I " ( 1 ) \
2005-07-26 19:44:26 +01:00
: " ip " , " lr " , " cc " ) ; \
2005-04-16 15:20:36 -07:00
} )
# endif
# endif