2021-02-05 19:44:26 +01:00
/*
* Copyright ( c ) 2021 , Andreas Kling < kling @ serenityos . org >
*
2021-04-22 01:24:48 -07:00
* SPDX - License - Identifier : BSD - 2 - Clause
2021-02-05 19:44:26 +01:00
*/
# include <AK/Format.h>
# include <Kernel/KSyms.h>
2021-03-04 22:07:23 +01:00
# include <Kernel/Panic.h>
2021-02-05 19:44:26 +01:00
# include <Kernel/UBSanitizer.h>
using namespace Kernel ;
using namespace Kernel : : UBSanitizer ;
2021-03-04 22:07:23 +01:00
bool Kernel : : UBSanitizer : : g_ubsan_is_deadly { true } ;
2021-02-05 19:44:26 +01:00
extern " C " {
static void print_location ( const SourceLocation & location )
{
2021-02-11 20:58:01 +01:00
if ( ! location . filename ( ) ) {
dbgln ( " KUBSAN: in unknown file " ) ;
} else {
dbgln ( " KUBSAN: at {}, line {}, column: {} " , location . filename ( ) , location . line ( ) , location . column ( ) ) ;
}
2021-02-05 20:03:07 +01:00
dump_backtrace ( ) ;
2021-03-04 22:07:23 +01:00
if ( g_ubsan_is_deadly )
PANIC ( " UB is configured to be deadly. " ) ;
2021-02-05 19:44:26 +01:00
}
2021-02-11 20:58:01 +01:00
void __ubsan_handle_load_invalid_value ( const InvalidValueData & , ValueHandle ) ;
void __ubsan_handle_load_invalid_value ( const InvalidValueData & data , ValueHandle )
2021-02-05 19:44:26 +01:00
{
dbgln ( " KUBSAN: load-invalid-value: {} ({}-bit) " , data . type . name ( ) , data . type . bit_width ( ) ) ;
print_location ( data . location ) ;
}
2021-02-05 20:03:07 +01:00
void __ubsan_handle_nonnull_arg ( const NonnullArgData & ) ;
void __ubsan_handle_nonnull_arg ( const NonnullArgData & data )
2021-02-05 19:44:26 +01:00
{
dbgln ( " KUBSAN: null pointer passed as argument {}, which is declared to never be null " , data . argument_index ) ;
print_location ( data . location ) ;
2021-02-05 20:03:07 +01:00
}
2021-02-11 20:58:01 +01:00
void __ubsan_handle_nullability_arg ( const NonnullArgData & ) ;
void __ubsan_handle_nullability_arg ( const NonnullArgData & data )
{
dbgln ( " KUBSAN: null pointer passed as argument {}, which is declared to never be null " , data . argument_index ) ;
print_location ( data . location ) ;
}
2021-02-06 23:24:49 -08:00
void __ubsan_handle_nonnull_return_v1 ( const NonnullReturnData & , const SourceLocation & ) ;
void __ubsan_handle_nonnull_return_v1 ( const NonnullReturnData & , const SourceLocation & location )
{
dbgln ( " KUBSAN: null pointer return from function declared to never return null " ) ;
print_location ( location ) ;
}
2021-02-11 20:58:01 +01:00
void __ubsan_handle_nullability_return_v1 ( const NonnullReturnData & data , const SourceLocation & location ) ;
void __ubsan_handle_nullability_return_v1 ( const NonnullReturnData & , const SourceLocation & location )
{
dbgln ( " KUBSAN: null pointer return from function declared to never return null " ) ;
print_location ( location ) ;
}
void __ubsan_handle_vla_bound_not_positive ( const VLABoundData & , ValueHandle ) ;
void __ubsan_handle_vla_bound_not_positive ( const VLABoundData & data , ValueHandle )
2021-02-05 20:03:07 +01:00
{
dbgln ( " KUBSAN: VLA bound not positive {} ({}-bit) " , data . type . name ( ) , data . type . bit_width ( ) ) ;
print_location ( data . location ) ;
}
2021-02-11 20:58:01 +01:00
void __ubsan_handle_add_overflow ( const OverflowData & , ValueHandle lhs , ValueHandle rhs ) ;
void __ubsan_handle_add_overflow ( const OverflowData & data , ValueHandle , ValueHandle )
2021-02-05 20:03:07 +01:00
{
dbgln ( " KUBSAN: addition overflow, {} ({}-bit) " , data . type . name ( ) , data . type . bit_width ( ) ) ;
2021-02-11 20:58:01 +01:00
2021-02-05 20:03:07 +01:00
print_location ( data . location ) ;
}
2021-02-11 20:58:01 +01:00
void __ubsan_handle_sub_overflow ( const OverflowData & , ValueHandle lhs , ValueHandle rhs ) ;
void __ubsan_handle_sub_overflow ( const OverflowData & data , ValueHandle , ValueHandle )
2021-02-05 20:03:07 +01:00
{
dbgln ( " KUBSAN: subtraction overflow, {} ({}-bit) " , data . type . name ( ) , data . type . bit_width ( ) ) ;
2021-02-11 20:58:01 +01:00
2021-02-05 20:03:07 +01:00
print_location ( data . location ) ;
}
2021-02-11 20:58:01 +01:00
void __ubsan_handle_negate_overflow ( const OverflowData & , ValueHandle ) ;
void __ubsan_handle_negate_overflow ( const OverflowData & data , ValueHandle )
2021-02-05 20:03:07 +01:00
{
dbgln ( " KUBSAN: negation overflow, {} ({}-bit) " , data . type . name ( ) , data . type . bit_width ( ) ) ;
2021-02-11 20:58:01 +01:00
2021-02-05 20:03:07 +01:00
print_location ( data . location ) ;
}
2021-02-11 20:58:01 +01:00
void __ubsan_handle_mul_overflow ( const OverflowData & , ValueHandle lhs , ValueHandle rhs ) ;
void __ubsan_handle_mul_overflow ( const OverflowData & data , ValueHandle , ValueHandle )
2021-02-05 20:03:07 +01:00
{
dbgln ( " KUBSAN: multiplication overflow, {} ({}-bit) " , data . type . name ( ) , data . type . bit_width ( ) ) ;
print_location ( data . location ) ;
2021-02-05 19:44:26 +01:00
}
2021-02-06 16:08:30 +01:00
2021-02-11 20:58:01 +01:00
void __ubsan_handle_shift_out_of_bounds ( const ShiftOutOfBoundsData & , ValueHandle lhs , ValueHandle rhs ) ;
void __ubsan_handle_shift_out_of_bounds ( const ShiftOutOfBoundsData & data , ValueHandle , ValueHandle )
2021-02-06 16:08:30 +01:00
{
dbgln ( " KUBSAN: shift out of bounds, {} ({}-bit) shifted by {} ({}-bit) " , data . lhs_type . name ( ) , data . lhs_type . bit_width ( ) , data . rhs_type . name ( ) , data . rhs_type . bit_width ( ) ) ;
print_location ( data . location ) ;
}
2021-02-11 20:58:01 +01:00
void __ubsan_handle_divrem_overflow ( const OverflowData & , ValueHandle lhs , ValueHandle rhs ) ;
void __ubsan_handle_divrem_overflow ( const OverflowData & data , ValueHandle , ValueHandle )
2021-02-06 16:08:30 +01:00
{
2021-04-18 10:30:03 +02:00
dbgln ( " KUBSAN: divrem overflow, {} ({}-bit) " , data . type . name ( ) , data . type . bit_width ( ) ) ;
2021-02-06 16:08:30 +01:00
print_location ( data . location ) ;
}
2021-02-11 20:58:01 +01:00
void __ubsan_handle_out_of_bounds ( const OutOfBoundsData & , ValueHandle ) ;
void __ubsan_handle_out_of_bounds ( const OutOfBoundsData & data , ValueHandle )
2021-02-06 16:08:30 +01:00
{
dbgln ( " KUBSAN: out of bounds access into array of {} ({}-bit), index type {} ({}-bit) " , data . array_type . name ( ) , data . array_type . bit_width ( ) , data . index_type . name ( ) , data . index_type . bit_width ( ) ) ;
print_location ( data . location ) ;
}
2021-02-06 17:34:44 +01:00
2021-02-11 20:58:01 +01:00
void __ubsan_handle_type_mismatch_v1 ( const TypeMismatchData & , ValueHandle ) ;
2021-02-24 14:27:06 +01:00
void __ubsan_handle_type_mismatch_v1 ( const TypeMismatchData & data , ValueHandle ptr )
{
static const char * kinds [ ] = {
" load of " ,
" store to " ,
" reference binding to " ,
" member access within " ,
" member call on " ,
" constructor call on " ,
" downcast of " ,
" downcast of " ,
" upcast of " ,
" cast to virtual base of " ,
" _Nonnull binding to " ,
" dynamic operation on "
} ;
FlatPtr alignment = ( FlatPtr ) 1 < < data . log_alignment ;
auto * kind = kinds [ data . type_check_kind ] ;
if ( ! ptr ) {
dbgln ( " KUBSAN: {} null pointer of type {} " , kind , data . type . name ( ) ) ;
} else if ( ( FlatPtr ) ptr & ( alignment - 1 ) ) {
dbgln ( " KUBSAN: {} misaligned address {:p} of type {} " , kind , ptr , data . type . name ( ) ) ;
} else {
dbgln ( " KUBSAN: {} address {:p} with insufficient space for type {} " , kind , ptr , data . type . name ( ) ) ;
}
2021-02-06 17:34:44 +01:00
print_location ( data . location ) ;
}
2021-02-11 20:58:01 +01:00
// FIXME: Causes a triple fault on boot
void __ubsan_handle_alignment_assumption ( const AlignmentAssumptionData & , ValueHandle , ValueHandle , ValueHandle ) ;
void __ubsan_handle_alignment_assumption ( const AlignmentAssumptionData & data , ValueHandle pointer , ValueHandle alignment , ValueHandle offset )
{
if ( offset ) {
dbgln (
" KUBSAN: assumption of {:p} byte alignment (with offset of {:p} byte) for pointer {:p} "
" of type {} failed " ,
alignment , offset , pointer , data . type . name ( ) ) ;
} else {
dbgln ( " KUBSAN: assumption of {:p} byte alignment for pointer {:p} "
" of type {} failed " ,
alignment , pointer , data . type . name ( ) ) ;
}
2021-04-18 10:30:03 +02:00
// dbgln("KUBSAN: Assumption of pointer alignment failed");
2021-02-11 20:58:01 +01:00
print_location ( data . location ) ;
}
void __ubsan_handle_builtin_unreachable ( const UnreachableData & ) ;
void __ubsan_handle_builtin_unreachable ( const UnreachableData & data )
{
dbgln ( " KUBSAN: execution reached an unreachable program point " ) ;
print_location ( data . location ) ;
}
void __ubsan_handle_missing_return ( const UnreachableData & ) ;
void __ubsan_handle_missing_return ( const UnreachableData & data )
{
dbgln ( " KUBSAN: execution reached the end of a value-returning function without returning a value " ) ;
print_location ( data . location ) ;
}
void __ubsan_handle_implicit_conversion ( const ImplicitConversionData & , ValueHandle , ValueHandle ) ;
void __ubsan_handle_implicit_conversion ( const ImplicitConversionData & data , ValueHandle , ValueHandle )
{
const char * src_signed = data . from_type . is_signed ( ) ? " " : " un " ;
const char * dst_signed = data . to_type . is_signed ( ) ? " " : " un " ;
dbgln ( " KUBSAN: implicit conversion from type {} ({}-bit, {}signed) to type {} ({}-bit, {}signed) " ,
data . from_type . name ( ) , data . from_type . bit_width ( ) , src_signed , data . to_type . name ( ) , data . to_type . bit_width ( ) , dst_signed ) ;
print_location ( data . location ) ;
}
void __ubsan_handle_invalid_builtin ( const InvalidBuiltinData ) ;
void __ubsan_handle_invalid_builtin ( const InvalidBuiltinData data )
{
dbgln ( " KUBSAN: passing invalid argument " ) ;
print_location ( data . location ) ;
}
// FIXME: Causes a triple fault on boot
void __ubsan_handle_pointer_overflow ( const PointerOverflowData & , ValueHandle , ValueHandle ) ;
void __ubsan_handle_pointer_overflow ( const PointerOverflowData & data , ValueHandle base , ValueHandle result )
{
if ( base = = 0 & & result = = 0 ) {
dbgln ( " KUBSAN: applied zero offset to nullptr " ) ;
} else if ( base = = 0 & & result ! = 0 ) {
dbgln ( " KUBSAN: applied non-zero offset {:p} to nullptr " , result ) ;
} else if ( base ! = 0 & & result = = 0 ) {
dbgln ( " KUBSAN: applying non-zero offset to non-null pointer {:p} produced null pointer " , base ) ;
} else {
dbgln ( " KUBSAN: addition of unsigned offset to {:p} overflowed to {:p} " , base , result ) ;
}
print_location ( data . location ) ;
}
2021-02-05 19:44:26 +01:00
}