2019-08-20 12:49:40 +10:00
/* SPDX-License-Identifier: GPL-2.0 */
/*
* This file provides wrappers with sanitizer instrumentation for atomic bit
* operations .
*
* To use this functionality , an arch ' s bitops . h file needs to define each of
* the below bit operations with an arch_ prefix ( e . g . arch_set_bit ( ) ,
* arch___set_bit ( ) , etc . ) .
*/
# ifndef _ASM_GENERIC_BITOPS_INSTRUMENTED_ATOMIC_H
# define _ASM_GENERIC_BITOPS_INSTRUMENTED_ATOMIC_H
2020-01-21 17:05:10 +01:00
# include <linux/instrumented.h>
2019-08-20 12:49:40 +10:00
/**
* set_bit - Atomically set a bit in memory
* @ nr : the bit to set
* @ addr : the address to start counting from
*
* This is a relaxed atomic operation ( no implied memory barriers ) .
*
* Note that @ nr may be almost arbitrarily large ; this function is not
* restricted to acting on a single - word quantity .
*/
2022-01-13 16:53:56 +01:00
static __always_inline void set_bit ( long nr , volatile unsigned long * addr )
2019-08-20 12:49:40 +10:00
{
2020-01-21 17:05:10 +01:00
instrument_atomic_write ( addr + BIT_WORD ( nr ) , sizeof ( long ) ) ;
2019-08-20 12:49:40 +10:00
arch_set_bit ( nr , addr ) ;
}
/**
* clear_bit - Clears a bit in memory
* @ nr : Bit to clear
* @ addr : Address to start counting from
*
* This is a relaxed atomic operation ( no implied memory barriers ) .
*/
2022-01-13 16:53:56 +01:00
static __always_inline void clear_bit ( long nr , volatile unsigned long * addr )
2019-08-20 12:49:40 +10:00
{
2020-01-21 17:05:10 +01:00
instrument_atomic_write ( addr + BIT_WORD ( nr ) , sizeof ( long ) ) ;
2019-08-20 12:49:40 +10:00
arch_clear_bit ( nr , addr ) ;
}
/**
* change_bit - Toggle a bit in memory
* @ nr : Bit to change
* @ addr : Address to start counting from
*
* This is a relaxed atomic operation ( no implied memory barriers ) .
*
* Note that @ nr may be almost arbitrarily large ; this function is not
* restricted to acting on a single - word quantity .
*/
2022-01-13 16:53:56 +01:00
static __always_inline void change_bit ( long nr , volatile unsigned long * addr )
2019-08-20 12:49:40 +10:00
{
2020-01-21 17:05:10 +01:00
instrument_atomic_write ( addr + BIT_WORD ( nr ) , sizeof ( long ) ) ;
2019-08-20 12:49:40 +10:00
arch_change_bit ( nr , addr ) ;
}
/**
* test_and_set_bit - Set a bit and return its old value
* @ nr : Bit to set
* @ addr : Address to count from
*
* This is an atomic fully - ordered operation ( implied full memory barrier ) .
*/
2022-01-13 16:53:56 +01:00
static __always_inline bool test_and_set_bit ( long nr , volatile unsigned long * addr )
2019-08-20 12:49:40 +10:00
{
2021-11-30 12:44:25 +01:00
kcsan_mb ( ) ;
2020-07-24 09:00:07 +02:00
instrument_atomic_read_write ( addr + BIT_WORD ( nr ) , sizeof ( long ) ) ;
2019-08-20 12:49:40 +10:00
return arch_test_and_set_bit ( nr , addr ) ;
}
/**
* test_and_clear_bit - Clear a bit and return its old value
* @ nr : Bit to clear
* @ addr : Address to count from
*
* This is an atomic fully - ordered operation ( implied full memory barrier ) .
*/
2022-01-13 16:53:56 +01:00
static __always_inline bool test_and_clear_bit ( long nr , volatile unsigned long * addr )
2019-08-20 12:49:40 +10:00
{
2021-11-30 12:44:25 +01:00
kcsan_mb ( ) ;
2020-07-24 09:00:07 +02:00
instrument_atomic_read_write ( addr + BIT_WORD ( nr ) , sizeof ( long ) ) ;
2019-08-20 12:49:40 +10:00
return arch_test_and_clear_bit ( nr , addr ) ;
}
/**
* test_and_change_bit - Change a bit and return its old value
* @ nr : Bit to change
* @ addr : Address to count from
*
* This is an atomic fully - ordered operation ( implied full memory barrier ) .
*/
2022-01-13 16:53:56 +01:00
static __always_inline bool test_and_change_bit ( long nr , volatile unsigned long * addr )
2019-08-20 12:49:40 +10:00
{
2021-11-30 12:44:25 +01:00
kcsan_mb ( ) ;
2020-07-24 09:00:07 +02:00
instrument_atomic_read_write ( addr + BIT_WORD ( nr ) , sizeof ( long ) ) ;
2019-08-20 12:49:40 +10:00
return arch_test_and_change_bit ( nr , addr ) ;
}
# endif /* _ASM_GENERIC_BITOPS_INSTRUMENTED_NON_ATOMIC_H */