2019-06-04 10:11:33 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2016-01-20 15:00:55 -08:00
/*
* UBSAN error reporting functions
*
* Copyright ( c ) 2014 Samsung Electronics Co . , Ltd .
* Author : Andrey Ryabinin < ryabinin . a . a @ gmail . com >
*/
# include <linux/bitops.h>
# include <linux/bug.h>
# include <linux/ctype.h>
# include <linux/init.h>
# include <linux/kernel.h>
# include <linux/types.h>
# include <linux/sched.h>
2019-04-03 09:40:16 +02:00
# include <linux/uaccess.h>
2021-04-02 14:21:31 -07:00
# include <kunit/test-bug.h>
2016-01-20 15:00:55 -08:00
# include "ubsan.h"
2020-12-15 20:46:50 -08:00
static const char * const type_check_kinds [ ] = {
2016-01-20 15:00:55 -08:00
" load of " ,
" store to " ,
" reference binding to " ,
" member access within " ,
" member call on " ,
" constructor call on " ,
" downcast of " ,
" downcast of "
} ;
# define REPORTED_BIT 31
# if (BITS_PER_LONG == 64) && defined(__BIG_ENDIAN)
# define COLUMN_MASK (~(1U << REPORTED_BIT))
# define LINE_MASK (~0U)
# else
# define COLUMN_MASK (~0U)
# define LINE_MASK (~(1U << REPORTED_BIT))
# endif
# define VALUE_LENGTH 40
static bool was_reported ( struct source_location * location )
{
return test_and_set_bit ( REPORTED_BIT , & location - > reported ) ;
}
static bool suppress_report ( struct source_location * loc )
{
return current - > in_ubsan | | was_reported ( loc ) ;
}
static bool type_is_int ( struct type_descriptor * type )
{
return type - > type_kind = = type_kind_int ;
}
static bool type_is_signed ( struct type_descriptor * type )
{
WARN_ON ( ! type_is_int ( type ) ) ;
return type - > type_info & 1 ;
}
static unsigned type_bit_width ( struct type_descriptor * type )
{
return 1 < < ( type - > type_info > > 1 ) ;
}
static bool is_inline_int ( struct type_descriptor * type )
{
unsigned inline_bits = sizeof ( unsigned long ) * 8 ;
unsigned bits = type_bit_width ( type ) ;
WARN_ON ( ! type_is_int ( type ) ) ;
return bits < = inline_bits ;
}
2019-05-06 13:45:26 +03:00
static s_max get_signed_val ( struct type_descriptor * type , void * val )
2016-01-20 15:00:55 -08:00
{
if ( is_inline_int ( type ) ) {
unsigned extra_bits = sizeof ( s_max ) * 8 - type_bit_width ( type ) ;
2019-05-06 13:45:26 +03:00
unsigned long ulong_val = ( unsigned long ) val ;
return ( ( s_max ) ulong_val ) < < extra_bits > > extra_bits ;
2016-01-20 15:00:55 -08:00
}
if ( type_bit_width ( type ) = = 64 )
return * ( s64 * ) val ;
return * ( s_max * ) val ;
}
2019-05-06 13:45:26 +03:00
static bool val_is_negative ( struct type_descriptor * type , void * val )
2016-01-20 15:00:55 -08:00
{
return type_is_signed ( type ) & & get_signed_val ( type , val ) < 0 ;
}
2019-05-06 13:45:26 +03:00
static u_max get_unsigned_val ( struct type_descriptor * type , void * val )
2016-01-20 15:00:55 -08:00
{
if ( is_inline_int ( type ) )
2019-05-06 13:45:26 +03:00
return ( unsigned long ) val ;
2016-01-20 15:00:55 -08:00
if ( type_bit_width ( type ) = = 64 )
return * ( u64 * ) val ;
return * ( u_max * ) val ;
}
static void val_to_string ( char * str , size_t size , struct type_descriptor * type ,
2019-05-06 13:45:26 +03:00
void * value )
2016-01-20 15:00:55 -08:00
{
if ( type_is_int ( type ) ) {
if ( type_bit_width ( type ) = = 128 ) {
2019-11-08 13:22:27 +01:00
# if defined(CONFIG_ARCH_SUPPORTS_INT128)
2016-01-20 15:00:55 -08:00
u_max val = get_unsigned_val ( type , value ) ;
scnprintf ( str , size , " 0x%08x%08x%08x%08x " ,
( u32 ) ( val > > 96 ) ,
( u32 ) ( val > > 64 ) ,
( u32 ) ( val > > 32 ) ,
( u32 ) ( val ) ) ;
# else
WARN_ON ( 1 ) ;
# endif
} else if ( type_is_signed ( type ) ) {
scnprintf ( str , size , " %lld " ,
( s64 ) get_signed_val ( type , value ) ) ;
} else {
scnprintf ( str , size , " %llu " ,
( u64 ) get_unsigned_val ( type , value ) ) ;
}
}
}
2020-04-06 20:12:45 -07:00
static void ubsan_prologue ( struct source_location * loc , const char * reason )
2016-01-20 15:00:55 -08:00
{
current - > in_ubsan + + ;
pr_err ( " ======================================== "
" ======================================== \n " ) ;
2020-04-06 20:12:45 -07:00
pr_err ( " UBSAN: %s in %s:%d:%d \n " , reason , loc - > file_name ,
loc - > line & LINE_MASK , loc - > column & COLUMN_MASK ) ;
2021-04-02 14:21:31 -07:00
kunit_fail_current_test ( " %s in %s " , reason , loc - > file_name ) ;
2016-01-20 15:00:55 -08:00
}
2019-12-04 16:52:53 -08:00
static void ubsan_epilogue ( void )
2016-01-20 15:00:55 -08:00
{
dump_stack ( ) ;
pr_err ( " ======================================== "
" ======================================== \n " ) ;
2019-12-04 16:52:53 -08:00
2016-01-20 15:00:55 -08:00
current - > in_ubsan - - ;
2020-04-06 20:12:38 -07:00
if ( panic_on_warn ) {
/*
* This thread may hit another WARN ( ) in the panic path .
* Resetting this prevents additional WARN ( ) from panicking the
* system on this thread . Other threads are blocked by the
* panic_mutex in panic ( ) .
*/
panic_on_warn = 0 ;
panic ( " panic_on_warn set ... \n " ) ;
}
2016-01-20 15:00:55 -08:00
}
lib/ubsan.c: fix gcc-10 warnings
The latest compiler expects slightly different function prototypes
for the ubsan helpers:
lib/ubsan.c:192:6: error: conflicting types for built-in function '__ubsan_handle_add_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
192 | void __ubsan_handle_add_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:200:6: error: conflicting types for built-in function '__ubsan_handle_sub_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
200 | void __ubsan_handle_sub_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:207:6: error: conflicting types for built-in function '__ubsan_handle_mul_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
207 | void __ubsan_handle_mul_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:214:6: error: conflicting types for built-in function '__ubsan_handle_negate_overflow'; expected 'void(void *, void *)' [-Werror=builtin-declaration-mismatch]
214 | void __ubsan_handle_negate_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:234:6: error: conflicting types for built-in function '__ubsan_handle_divrem_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
234 | void __ubsan_handle_divrem_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Change the Linux implementation to match these, using a local typed
pointer.
Signed-off-by: Arnd Bergmann <arnd@arndb.de>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Reviewed-by: Kees Cook <keescook@chromium.org>
Cc: Andrey Ryabinin <aryabinin@virtuozzo.com>
Cc: Herbert Xu <herbert@gondor.apana.org.au>
Cc: Julien Grall <julien.grall@arm.com>
Link: http://lkml.kernel.org/r/20200429185948.4189600-1-arnd@arndb.de
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2020-06-04 16:53:00 -07:00
void __ubsan_handle_divrem_overflow ( void * _data , void * lhs , void * rhs )
2016-01-20 15:00:55 -08:00
{
lib/ubsan.c: fix gcc-10 warnings
The latest compiler expects slightly different function prototypes
for the ubsan helpers:
lib/ubsan.c:192:6: error: conflicting types for built-in function '__ubsan_handle_add_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
192 | void __ubsan_handle_add_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:200:6: error: conflicting types for built-in function '__ubsan_handle_sub_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
200 | void __ubsan_handle_sub_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:207:6: error: conflicting types for built-in function '__ubsan_handle_mul_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
207 | void __ubsan_handle_mul_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:214:6: error: conflicting types for built-in function '__ubsan_handle_negate_overflow'; expected 'void(void *, void *)' [-Werror=builtin-declaration-mismatch]
214 | void __ubsan_handle_negate_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:234:6: error: conflicting types for built-in function '__ubsan_handle_divrem_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
234 | void __ubsan_handle_divrem_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Change the Linux implementation to match these, using a local typed
pointer.
Signed-off-by: Arnd Bergmann <arnd@arndb.de>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Reviewed-by: Kees Cook <keescook@chromium.org>
Cc: Andrey Ryabinin <aryabinin@virtuozzo.com>
Cc: Herbert Xu <herbert@gondor.apana.org.au>
Cc: Julien Grall <julien.grall@arm.com>
Link: http://lkml.kernel.org/r/20200429185948.4189600-1-arnd@arndb.de
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2020-06-04 16:53:00 -07:00
struct overflow_data * data = _data ;
2016-01-20 15:00:55 -08:00
char rhs_val_str [ VALUE_LENGTH ] ;
if ( suppress_report ( & data - > location ) )
return ;
2020-04-06 20:12:45 -07:00
ubsan_prologue ( & data - > location , " division-overflow " ) ;
2016-01-20 15:00:55 -08:00
val_to_string ( rhs_val_str , sizeof ( rhs_val_str ) , data - > type , rhs ) ;
if ( type_is_signed ( data - > type ) & & get_signed_val ( data - > type , rhs ) = = - 1 )
pr_err ( " division of %s by -1 cannot be represented in type %s \n " ,
rhs_val_str , data - > type - > type_name ) ;
else
pr_err ( " division by zero \n " ) ;
2019-12-04 16:52:53 -08:00
ubsan_epilogue ( ) ;
2016-01-20 15:00:55 -08:00
}
EXPORT_SYMBOL ( __ubsan_handle_divrem_overflow ) ;
2018-02-06 15:40:42 -08:00
static void handle_null_ptr_deref ( struct type_mismatch_data_common * data )
2016-01-20 15:00:55 -08:00
{
2018-02-06 15:40:42 -08:00
if ( suppress_report ( data - > location ) )
2016-01-20 15:00:55 -08:00
return ;
2020-04-06 20:12:45 -07:00
ubsan_prologue ( data - > location , " null-ptr-deref " ) ;
2016-01-20 15:00:55 -08:00
pr_err ( " %s null pointer of type %s \n " ,
type_check_kinds [ data - > type_check_kind ] ,
data - > type - > type_name ) ;
2019-12-04 16:52:53 -08:00
ubsan_epilogue ( ) ;
2016-01-20 15:00:55 -08:00
}
2018-02-06 15:40:42 -08:00
static void handle_misaligned_access ( struct type_mismatch_data_common * data ,
2016-01-20 15:00:55 -08:00
unsigned long ptr )
{
2018-02-06 15:40:42 -08:00
if ( suppress_report ( data - > location ) )
2016-01-20 15:00:55 -08:00
return ;
2020-04-06 20:12:45 -07:00
ubsan_prologue ( data - > location , " misaligned-access " ) ;
2016-01-20 15:00:55 -08:00
pr_err ( " %s misaligned address %p for type %s \n " ,
type_check_kinds [ data - > type_check_kind ] ,
( void * ) ptr , data - > type - > type_name ) ;
pr_err ( " which requires %ld byte alignment \n " , data - > alignment ) ;
2019-12-04 16:52:53 -08:00
ubsan_epilogue ( ) ;
2016-01-20 15:00:55 -08:00
}
2018-02-06 15:40:42 -08:00
static void handle_object_size_mismatch ( struct type_mismatch_data_common * data ,
2016-01-20 15:00:55 -08:00
unsigned long ptr )
{
2018-02-06 15:40:42 -08:00
if ( suppress_report ( data - > location ) )
2016-01-20 15:00:55 -08:00
return ;
2020-04-06 20:12:45 -07:00
ubsan_prologue ( data - > location , " object-size-mismatch " ) ;
2016-08-02 14:03:10 -07:00
pr_err ( " %s address %p with insufficient space \n " ,
2016-01-20 15:00:55 -08:00
type_check_kinds [ data - > type_check_kind ] ,
( void * ) ptr ) ;
pr_err ( " for an object of type %s \n " , data - > type - > type_name ) ;
2019-12-04 16:52:53 -08:00
ubsan_epilogue ( ) ;
2016-01-20 15:00:55 -08:00
}
2018-02-06 15:40:42 -08:00
static void ubsan_type_mismatch_common ( struct type_mismatch_data_common * data ,
2016-01-20 15:00:55 -08:00
unsigned long ptr )
{
2019-04-03 09:40:16 +02:00
unsigned long flags = user_access_save ( ) ;
2016-01-20 15:00:55 -08:00
if ( ! ptr )
handle_null_ptr_deref ( data ) ;
else if ( data - > alignment & & ! IS_ALIGNED ( ptr , data - > alignment ) )
2018-02-06 15:40:38 -08:00
handle_misaligned_access ( data , ptr ) ;
2016-01-20 15:00:55 -08:00
else
handle_object_size_mismatch ( data , ptr ) ;
2019-04-03 09:40:16 +02:00
user_access_restore ( flags ) ;
2016-01-20 15:00:55 -08:00
}
2018-02-06 15:40:42 -08:00
void __ubsan_handle_type_mismatch ( struct type_mismatch_data * data ,
2019-05-06 13:45:26 +03:00
void * ptr )
2018-02-06 15:40:42 -08:00
{
struct type_mismatch_data_common common_data = {
. location = & data - > location ,
. type = data - > type ,
. alignment = data - > alignment ,
. type_check_kind = data - > type_check_kind
} ;
2019-05-06 13:45:26 +03:00
ubsan_type_mismatch_common ( & common_data , ( unsigned long ) ptr ) ;
2018-02-06 15:40:42 -08:00
}
2016-01-20 15:00:55 -08:00
EXPORT_SYMBOL ( __ubsan_handle_type_mismatch ) ;
lib/ubsan.c: fix gcc-10 warnings
The latest compiler expects slightly different function prototypes
for the ubsan helpers:
lib/ubsan.c:192:6: error: conflicting types for built-in function '__ubsan_handle_add_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
192 | void __ubsan_handle_add_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:200:6: error: conflicting types for built-in function '__ubsan_handle_sub_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
200 | void __ubsan_handle_sub_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:207:6: error: conflicting types for built-in function '__ubsan_handle_mul_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
207 | void __ubsan_handle_mul_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:214:6: error: conflicting types for built-in function '__ubsan_handle_negate_overflow'; expected 'void(void *, void *)' [-Werror=builtin-declaration-mismatch]
214 | void __ubsan_handle_negate_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:234:6: error: conflicting types for built-in function '__ubsan_handle_divrem_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
234 | void __ubsan_handle_divrem_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Change the Linux implementation to match these, using a local typed
pointer.
Signed-off-by: Arnd Bergmann <arnd@arndb.de>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Reviewed-by: Kees Cook <keescook@chromium.org>
Cc: Andrey Ryabinin <aryabinin@virtuozzo.com>
Cc: Herbert Xu <herbert@gondor.apana.org.au>
Cc: Julien Grall <julien.grall@arm.com>
Link: http://lkml.kernel.org/r/20200429185948.4189600-1-arnd@arndb.de
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2020-06-04 16:53:00 -07:00
void __ubsan_handle_type_mismatch_v1 ( void * _data , void * ptr )
2018-02-06 15:40:42 -08:00
{
lib/ubsan.c: fix gcc-10 warnings
The latest compiler expects slightly different function prototypes
for the ubsan helpers:
lib/ubsan.c:192:6: error: conflicting types for built-in function '__ubsan_handle_add_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
192 | void __ubsan_handle_add_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:200:6: error: conflicting types for built-in function '__ubsan_handle_sub_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
200 | void __ubsan_handle_sub_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:207:6: error: conflicting types for built-in function '__ubsan_handle_mul_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
207 | void __ubsan_handle_mul_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:214:6: error: conflicting types for built-in function '__ubsan_handle_negate_overflow'; expected 'void(void *, void *)' [-Werror=builtin-declaration-mismatch]
214 | void __ubsan_handle_negate_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:234:6: error: conflicting types for built-in function '__ubsan_handle_divrem_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
234 | void __ubsan_handle_divrem_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Change the Linux implementation to match these, using a local typed
pointer.
Signed-off-by: Arnd Bergmann <arnd@arndb.de>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Reviewed-by: Kees Cook <keescook@chromium.org>
Cc: Andrey Ryabinin <aryabinin@virtuozzo.com>
Cc: Herbert Xu <herbert@gondor.apana.org.au>
Cc: Julien Grall <julien.grall@arm.com>
Link: http://lkml.kernel.org/r/20200429185948.4189600-1-arnd@arndb.de
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2020-06-04 16:53:00 -07:00
struct type_mismatch_data_v1 * data = _data ;
2018-02-06 15:40:42 -08:00
struct type_mismatch_data_common common_data = {
. location = & data - > location ,
. type = data - > type ,
. alignment = 1UL < < data - > log_alignment ,
. type_check_kind = data - > type_check_kind
} ;
2019-05-06 13:45:26 +03:00
ubsan_type_mismatch_common ( & common_data , ( unsigned long ) ptr ) ;
2018-02-06 15:40:42 -08:00
}
EXPORT_SYMBOL ( __ubsan_handle_type_mismatch_v1 ) ;
lib/ubsan.c: fix gcc-10 warnings
The latest compiler expects slightly different function prototypes
for the ubsan helpers:
lib/ubsan.c:192:6: error: conflicting types for built-in function '__ubsan_handle_add_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
192 | void __ubsan_handle_add_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:200:6: error: conflicting types for built-in function '__ubsan_handle_sub_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
200 | void __ubsan_handle_sub_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:207:6: error: conflicting types for built-in function '__ubsan_handle_mul_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
207 | void __ubsan_handle_mul_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:214:6: error: conflicting types for built-in function '__ubsan_handle_negate_overflow'; expected 'void(void *, void *)' [-Werror=builtin-declaration-mismatch]
214 | void __ubsan_handle_negate_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:234:6: error: conflicting types for built-in function '__ubsan_handle_divrem_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
234 | void __ubsan_handle_divrem_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Change the Linux implementation to match these, using a local typed
pointer.
Signed-off-by: Arnd Bergmann <arnd@arndb.de>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Reviewed-by: Kees Cook <keescook@chromium.org>
Cc: Andrey Ryabinin <aryabinin@virtuozzo.com>
Cc: Herbert Xu <herbert@gondor.apana.org.au>
Cc: Julien Grall <julien.grall@arm.com>
Link: http://lkml.kernel.org/r/20200429185948.4189600-1-arnd@arndb.de
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2020-06-04 16:53:00 -07:00
void __ubsan_handle_out_of_bounds ( void * _data , void * index )
2016-01-20 15:00:55 -08:00
{
lib/ubsan.c: fix gcc-10 warnings
The latest compiler expects slightly different function prototypes
for the ubsan helpers:
lib/ubsan.c:192:6: error: conflicting types for built-in function '__ubsan_handle_add_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
192 | void __ubsan_handle_add_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:200:6: error: conflicting types for built-in function '__ubsan_handle_sub_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
200 | void __ubsan_handle_sub_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:207:6: error: conflicting types for built-in function '__ubsan_handle_mul_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
207 | void __ubsan_handle_mul_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:214:6: error: conflicting types for built-in function '__ubsan_handle_negate_overflow'; expected 'void(void *, void *)' [-Werror=builtin-declaration-mismatch]
214 | void __ubsan_handle_negate_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:234:6: error: conflicting types for built-in function '__ubsan_handle_divrem_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
234 | void __ubsan_handle_divrem_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Change the Linux implementation to match these, using a local typed
pointer.
Signed-off-by: Arnd Bergmann <arnd@arndb.de>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Reviewed-by: Kees Cook <keescook@chromium.org>
Cc: Andrey Ryabinin <aryabinin@virtuozzo.com>
Cc: Herbert Xu <herbert@gondor.apana.org.au>
Cc: Julien Grall <julien.grall@arm.com>
Link: http://lkml.kernel.org/r/20200429185948.4189600-1-arnd@arndb.de
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2020-06-04 16:53:00 -07:00
struct out_of_bounds_data * data = _data ;
2016-01-20 15:00:55 -08:00
char index_str [ VALUE_LENGTH ] ;
if ( suppress_report ( & data - > location ) )
return ;
2020-04-06 20:12:45 -07:00
ubsan_prologue ( & data - > location , " array-index-out-of-bounds " ) ;
2016-01-20 15:00:55 -08:00
val_to_string ( index_str , sizeof ( index_str ) , data - > index_type , index ) ;
pr_err ( " index %s is out of range for type %s \n " , index_str ,
data - > array_type - > type_name ) ;
2019-12-04 16:52:53 -08:00
ubsan_epilogue ( ) ;
2016-01-20 15:00:55 -08:00
}
EXPORT_SYMBOL ( __ubsan_handle_out_of_bounds ) ;
lib/ubsan.c: fix gcc-10 warnings
The latest compiler expects slightly different function prototypes
for the ubsan helpers:
lib/ubsan.c:192:6: error: conflicting types for built-in function '__ubsan_handle_add_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
192 | void __ubsan_handle_add_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:200:6: error: conflicting types for built-in function '__ubsan_handle_sub_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
200 | void __ubsan_handle_sub_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:207:6: error: conflicting types for built-in function '__ubsan_handle_mul_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
207 | void __ubsan_handle_mul_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:214:6: error: conflicting types for built-in function '__ubsan_handle_negate_overflow'; expected 'void(void *, void *)' [-Werror=builtin-declaration-mismatch]
214 | void __ubsan_handle_negate_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:234:6: error: conflicting types for built-in function '__ubsan_handle_divrem_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
234 | void __ubsan_handle_divrem_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Change the Linux implementation to match these, using a local typed
pointer.
Signed-off-by: Arnd Bergmann <arnd@arndb.de>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Reviewed-by: Kees Cook <keescook@chromium.org>
Cc: Andrey Ryabinin <aryabinin@virtuozzo.com>
Cc: Herbert Xu <herbert@gondor.apana.org.au>
Cc: Julien Grall <julien.grall@arm.com>
Link: http://lkml.kernel.org/r/20200429185948.4189600-1-arnd@arndb.de
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2020-06-04 16:53:00 -07:00
void __ubsan_handle_shift_out_of_bounds ( void * _data , void * lhs , void * rhs )
2016-01-20 15:00:55 -08:00
{
lib/ubsan.c: fix gcc-10 warnings
The latest compiler expects slightly different function prototypes
for the ubsan helpers:
lib/ubsan.c:192:6: error: conflicting types for built-in function '__ubsan_handle_add_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
192 | void __ubsan_handle_add_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:200:6: error: conflicting types for built-in function '__ubsan_handle_sub_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
200 | void __ubsan_handle_sub_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:207:6: error: conflicting types for built-in function '__ubsan_handle_mul_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
207 | void __ubsan_handle_mul_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:214:6: error: conflicting types for built-in function '__ubsan_handle_negate_overflow'; expected 'void(void *, void *)' [-Werror=builtin-declaration-mismatch]
214 | void __ubsan_handle_negate_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:234:6: error: conflicting types for built-in function '__ubsan_handle_divrem_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
234 | void __ubsan_handle_divrem_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Change the Linux implementation to match these, using a local typed
pointer.
Signed-off-by: Arnd Bergmann <arnd@arndb.de>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Reviewed-by: Kees Cook <keescook@chromium.org>
Cc: Andrey Ryabinin <aryabinin@virtuozzo.com>
Cc: Herbert Xu <herbert@gondor.apana.org.au>
Cc: Julien Grall <julien.grall@arm.com>
Link: http://lkml.kernel.org/r/20200429185948.4189600-1-arnd@arndb.de
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2020-06-04 16:53:00 -07:00
struct shift_out_of_bounds_data * data = _data ;
2016-01-20 15:00:55 -08:00
struct type_descriptor * rhs_type = data - > rhs_type ;
struct type_descriptor * lhs_type = data - > lhs_type ;
char rhs_str [ VALUE_LENGTH ] ;
char lhs_str [ VALUE_LENGTH ] ;
2019-10-21 15:11:49 +02:00
unsigned long ua_flags = user_access_save ( ) ;
2016-01-20 15:00:55 -08:00
if ( suppress_report ( & data - > location ) )
2019-10-21 15:11:49 +02:00
goto out ;
2016-01-20 15:00:55 -08:00
2020-04-06 20:12:45 -07:00
ubsan_prologue ( & data - > location , " shift-out-of-bounds " ) ;
2016-01-20 15:00:55 -08:00
val_to_string ( rhs_str , sizeof ( rhs_str ) , rhs_type , rhs ) ;
val_to_string ( lhs_str , sizeof ( lhs_str ) , lhs_type , lhs ) ;
if ( val_is_negative ( rhs_type , rhs ) )
pr_err ( " shift exponent %s is negative \n " , rhs_str ) ;
else if ( get_unsigned_val ( rhs_type , rhs ) > =
type_bit_width ( lhs_type ) )
pr_err ( " shift exponent %s is too large for %u-bit type %s \n " ,
rhs_str ,
type_bit_width ( lhs_type ) ,
lhs_type - > type_name ) ;
else if ( val_is_negative ( lhs_type , lhs ) )
pr_err ( " left shift of negative value %s \n " ,
lhs_str ) ;
else
pr_err ( " left shift of %s by %s places cannot be "
" represented in type %s \n " ,
lhs_str , rhs_str ,
lhs_type - > type_name ) ;
2019-12-04 16:52:53 -08:00
ubsan_epilogue ( ) ;
2019-10-21 15:11:49 +02:00
out :
user_access_restore ( ua_flags ) ;
2016-01-20 15:00:55 -08:00
}
EXPORT_SYMBOL ( __ubsan_handle_shift_out_of_bounds ) ;
lib/ubsan.c: fix gcc-10 warnings
The latest compiler expects slightly different function prototypes
for the ubsan helpers:
lib/ubsan.c:192:6: error: conflicting types for built-in function '__ubsan_handle_add_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
192 | void __ubsan_handle_add_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:200:6: error: conflicting types for built-in function '__ubsan_handle_sub_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
200 | void __ubsan_handle_sub_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:207:6: error: conflicting types for built-in function '__ubsan_handle_mul_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
207 | void __ubsan_handle_mul_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:214:6: error: conflicting types for built-in function '__ubsan_handle_negate_overflow'; expected 'void(void *, void *)' [-Werror=builtin-declaration-mismatch]
214 | void __ubsan_handle_negate_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:234:6: error: conflicting types for built-in function '__ubsan_handle_divrem_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
234 | void __ubsan_handle_divrem_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Change the Linux implementation to match these, using a local typed
pointer.
Signed-off-by: Arnd Bergmann <arnd@arndb.de>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Reviewed-by: Kees Cook <keescook@chromium.org>
Cc: Andrey Ryabinin <aryabinin@virtuozzo.com>
Cc: Herbert Xu <herbert@gondor.apana.org.au>
Cc: Julien Grall <julien.grall@arm.com>
Link: http://lkml.kernel.org/r/20200429185948.4189600-1-arnd@arndb.de
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2020-06-04 16:53:00 -07:00
void __ubsan_handle_builtin_unreachable ( void * _data )
2016-01-20 15:00:55 -08:00
{
lib/ubsan.c: fix gcc-10 warnings
The latest compiler expects slightly different function prototypes
for the ubsan helpers:
lib/ubsan.c:192:6: error: conflicting types for built-in function '__ubsan_handle_add_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
192 | void __ubsan_handle_add_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:200:6: error: conflicting types for built-in function '__ubsan_handle_sub_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
200 | void __ubsan_handle_sub_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:207:6: error: conflicting types for built-in function '__ubsan_handle_mul_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
207 | void __ubsan_handle_mul_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:214:6: error: conflicting types for built-in function '__ubsan_handle_negate_overflow'; expected 'void(void *, void *)' [-Werror=builtin-declaration-mismatch]
214 | void __ubsan_handle_negate_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:234:6: error: conflicting types for built-in function '__ubsan_handle_divrem_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
234 | void __ubsan_handle_divrem_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Change the Linux implementation to match these, using a local typed
pointer.
Signed-off-by: Arnd Bergmann <arnd@arndb.de>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Reviewed-by: Kees Cook <keescook@chromium.org>
Cc: Andrey Ryabinin <aryabinin@virtuozzo.com>
Cc: Herbert Xu <herbert@gondor.apana.org.au>
Cc: Julien Grall <julien.grall@arm.com>
Link: http://lkml.kernel.org/r/20200429185948.4189600-1-arnd@arndb.de
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2020-06-04 16:53:00 -07:00
struct unreachable_data * data = _data ;
2020-04-06 20:12:45 -07:00
ubsan_prologue ( & data - > location , " unreachable " ) ;
2016-01-20 15:00:55 -08:00
pr_err ( " calling __builtin_unreachable() \n " ) ;
2019-12-04 16:52:53 -08:00
ubsan_epilogue ( ) ;
2016-01-20 15:00:55 -08:00
panic ( " can't return from __builtin_unreachable() " ) ;
}
EXPORT_SYMBOL ( __ubsan_handle_builtin_unreachable ) ;
lib/ubsan.c: fix gcc-10 warnings
The latest compiler expects slightly different function prototypes
for the ubsan helpers:
lib/ubsan.c:192:6: error: conflicting types for built-in function '__ubsan_handle_add_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
192 | void __ubsan_handle_add_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:200:6: error: conflicting types for built-in function '__ubsan_handle_sub_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
200 | void __ubsan_handle_sub_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:207:6: error: conflicting types for built-in function '__ubsan_handle_mul_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
207 | void __ubsan_handle_mul_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:214:6: error: conflicting types for built-in function '__ubsan_handle_negate_overflow'; expected 'void(void *, void *)' [-Werror=builtin-declaration-mismatch]
214 | void __ubsan_handle_negate_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:234:6: error: conflicting types for built-in function '__ubsan_handle_divrem_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
234 | void __ubsan_handle_divrem_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Change the Linux implementation to match these, using a local typed
pointer.
Signed-off-by: Arnd Bergmann <arnd@arndb.de>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Reviewed-by: Kees Cook <keescook@chromium.org>
Cc: Andrey Ryabinin <aryabinin@virtuozzo.com>
Cc: Herbert Xu <herbert@gondor.apana.org.au>
Cc: Julien Grall <julien.grall@arm.com>
Link: http://lkml.kernel.org/r/20200429185948.4189600-1-arnd@arndb.de
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2020-06-04 16:53:00 -07:00
void __ubsan_handle_load_invalid_value ( void * _data , void * val )
2016-01-20 15:00:55 -08:00
{
lib/ubsan.c: fix gcc-10 warnings
The latest compiler expects slightly different function prototypes
for the ubsan helpers:
lib/ubsan.c:192:6: error: conflicting types for built-in function '__ubsan_handle_add_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
192 | void __ubsan_handle_add_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:200:6: error: conflicting types for built-in function '__ubsan_handle_sub_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
200 | void __ubsan_handle_sub_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:207:6: error: conflicting types for built-in function '__ubsan_handle_mul_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
207 | void __ubsan_handle_mul_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:214:6: error: conflicting types for built-in function '__ubsan_handle_negate_overflow'; expected 'void(void *, void *)' [-Werror=builtin-declaration-mismatch]
214 | void __ubsan_handle_negate_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
lib/ubsan.c:234:6: error: conflicting types for built-in function '__ubsan_handle_divrem_overflow'; expected 'void(void *, void *, void *)' [-Werror=builtin-declaration-mismatch]
234 | void __ubsan_handle_divrem_overflow(struct overflow_data *data,
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Change the Linux implementation to match these, using a local typed
pointer.
Signed-off-by: Arnd Bergmann <arnd@arndb.de>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Reviewed-by: Kees Cook <keescook@chromium.org>
Cc: Andrey Ryabinin <aryabinin@virtuozzo.com>
Cc: Herbert Xu <herbert@gondor.apana.org.au>
Cc: Julien Grall <julien.grall@arm.com>
Link: http://lkml.kernel.org/r/20200429185948.4189600-1-arnd@arndb.de
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2020-06-04 16:53:00 -07:00
struct invalid_value_data * data = _data ;
2016-01-20 15:00:55 -08:00
char val_str [ VALUE_LENGTH ] ;
if ( suppress_report ( & data - > location ) )
return ;
2020-04-06 20:12:45 -07:00
ubsan_prologue ( & data - > location , " invalid-load " ) ;
2016-01-20 15:00:55 -08:00
val_to_string ( val_str , sizeof ( val_str ) , data - > type , val ) ;
pr_err ( " load of value %s is not a valid value for type %s \n " ,
val_str , data - > type - > type_name ) ;
2019-12-04 16:52:53 -08:00
ubsan_epilogue ( ) ;
2016-01-20 15:00:55 -08:00
}
EXPORT_SYMBOL ( __ubsan_handle_load_invalid_value ) ;
2021-02-04 18:32:57 -08:00
void __ubsan_handle_alignment_assumption ( void * _data , unsigned long ptr ,
unsigned long align ,
unsigned long offset ) ;
void __ubsan_handle_alignment_assumption ( void * _data , unsigned long ptr ,
unsigned long align ,
unsigned long offset )
{
struct alignment_assumption_data * data = _data ;
unsigned long real_ptr ;
if ( suppress_report ( & data - > location ) )
return ;
ubsan_prologue ( & data - > location , " alignment-assumption " ) ;
if ( offset )
pr_err ( " assumption of %lu byte alignment (with offset of %lu byte) for pointer of type %s failed " ,
align , offset , data - > type - > type_name ) ;
else
pr_err ( " assumption of %lu byte alignment for pointer of type %s failed " ,
align , data - > type - > type_name ) ;
real_ptr = ptr - offset ;
pr_err ( " %saddress is %lu aligned, misalignment offset is %lu bytes " ,
offset ? " offset " : " " , BIT ( real_ptr ? __ffs ( real_ptr ) : 0 ) ,
real_ptr & ( align - 1 ) ) ;
ubsan_epilogue ( ) ;
}
EXPORT_SYMBOL ( __ubsan_handle_alignment_assumption ) ;