123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411 |
- //===-- ubsan_handlers.cc -------------------------------------------------===//
- //
- // This file is distributed under the University of Illinois Open Source
- // License. See LICENSE.TXT for details.
- //
- //===----------------------------------------------------------------------===//
- //
- // Error logging entry points for the UBSan runtime.
- //
- //===----------------------------------------------------------------------===//
- #include "ubsan_handlers.h"
- #include "ubsan_diag.h"
- #include "sanitizer_common/sanitizer_common.h"
- using namespace __sanitizer;
- using namespace __ubsan;
- static bool ignoreReport(SourceLocation SLoc, ReportOptions Opts) {
- // If source location is already acquired, we don't need to print an error
- // report for the second time. However, if we're in an unrecoverable handler,
- // it's possible that location was required by concurrently running thread.
- // In this case, we should continue the execution to ensure that any of
- // threads will grab the report mutex and print the report before
- // crashing the program.
- return SLoc.isDisabled() && !Opts.DieAfterReport;
- }
- namespace __ubsan {
- const char *TypeCheckKinds[] = {
- "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"};
- }
- static void handleTypeMismatchImpl(TypeMismatchData *Data, ValueHandle Pointer,
- Location FallbackLoc, ReportOptions Opts) {
- Location Loc = Data->Loc.acquire();
- // Use the SourceLocation from Data to track deduplication, even if 'invalid'
- if (ignoreReport(Loc.getSourceLocation(), Opts))
- return;
- if (Data->Loc.isInvalid())
- Loc = FallbackLoc;
- ScopedReport R(Opts, Loc);
- if (!Pointer)
- Diag(Loc, DL_Error, "%0 null pointer of type %1")
- << TypeCheckKinds[Data->TypeCheckKind] << Data->Type;
- else if (Data->Alignment && (Pointer & (Data->Alignment - 1)))
- Diag(Loc, DL_Error, "%0 misaligned address %1 for type %3, "
- "which requires %2 byte alignment")
- << TypeCheckKinds[Data->TypeCheckKind] << (void*)Pointer
- << Data->Alignment << Data->Type;
- else
- Diag(Loc, DL_Error, "%0 address %1 with insufficient space "
- "for an object of type %2")
- << TypeCheckKinds[Data->TypeCheckKind] << (void*)Pointer << Data->Type;
- if (Pointer)
- Diag(Pointer, DL_Note, "pointer points here");
- }
- void __ubsan::__ubsan_handle_type_mismatch(TypeMismatchData *Data,
- ValueHandle Pointer) {
- GET_REPORT_OPTIONS(false);
- handleTypeMismatchImpl(Data, Pointer, getCallerLocation(), Opts);
- }
- void __ubsan::__ubsan_handle_type_mismatch_abort(TypeMismatchData *Data,
- ValueHandle Pointer) {
- GET_REPORT_OPTIONS(true);
- handleTypeMismatchImpl(Data, Pointer, getCallerLocation(), Opts);
- Die();
- }
- /// \brief Common diagnostic emission for various forms of integer overflow.
- template <typename T>
- static void handleIntegerOverflowImpl(OverflowData *Data, ValueHandle LHS,
- const char *Operator, T RHS,
- ReportOptions Opts) {
- SourceLocation Loc = Data->Loc.acquire();
- if (ignoreReport(Loc, Opts))
- return;
- ScopedReport R(Opts, Loc);
- Diag(Loc, DL_Error, "%0 integer overflow: "
- "%1 %2 %3 cannot be represented in type %4")
- << (Data->Type.isSignedIntegerTy() ? "signed" : "unsigned")
- << Value(Data->Type, LHS) << Operator << RHS << Data->Type;
- }
- #define UBSAN_OVERFLOW_HANDLER(handler_name, op, abort) \
- void __ubsan::handler_name(OverflowData *Data, ValueHandle LHS, \
- ValueHandle RHS) { \
- GET_REPORT_OPTIONS(abort); \
- handleIntegerOverflowImpl(Data, LHS, op, Value(Data->Type, RHS), Opts); \
- if (abort) Die(); \
- }
- UBSAN_OVERFLOW_HANDLER(__ubsan_handle_add_overflow, "+", false)
- UBSAN_OVERFLOW_HANDLER(__ubsan_handle_add_overflow_abort, "+", true)
- UBSAN_OVERFLOW_HANDLER(__ubsan_handle_sub_overflow, "-", false)
- UBSAN_OVERFLOW_HANDLER(__ubsan_handle_sub_overflow_abort, "-", true)
- UBSAN_OVERFLOW_HANDLER(__ubsan_handle_mul_overflow, "*", false)
- UBSAN_OVERFLOW_HANDLER(__ubsan_handle_mul_overflow_abort, "*", true)
- static void handleNegateOverflowImpl(OverflowData *Data, ValueHandle OldVal,
- ReportOptions Opts) {
- SourceLocation Loc = Data->Loc.acquire();
- if (ignoreReport(Loc, Opts))
- return;
- ScopedReport R(Opts, Loc);
- if (Data->Type.isSignedIntegerTy())
- Diag(Loc, DL_Error,
- "negation of %0 cannot be represented in type %1; "
- "cast to an unsigned type to negate this value to itself")
- << Value(Data->Type, OldVal) << Data->Type;
- else
- Diag(Loc, DL_Error,
- "negation of %0 cannot be represented in type %1")
- << Value(Data->Type, OldVal) << Data->Type;
- }
- void __ubsan::__ubsan_handle_negate_overflow(OverflowData *Data,
- ValueHandle OldVal) {
- GET_REPORT_OPTIONS(false);
- handleNegateOverflowImpl(Data, OldVal, Opts);
- }
- void __ubsan::__ubsan_handle_negate_overflow_abort(OverflowData *Data,
- ValueHandle OldVal) {
- GET_REPORT_OPTIONS(true);
- handleNegateOverflowImpl(Data, OldVal, Opts);
- Die();
- }
- static void handleDivremOverflowImpl(OverflowData *Data, ValueHandle LHS,
- ValueHandle RHS, ReportOptions Opts) {
- SourceLocation Loc = Data->Loc.acquire();
- if (ignoreReport(Loc, Opts))
- return;
- ScopedReport R(Opts, Loc);
- Value LHSVal(Data->Type, LHS);
- Value RHSVal(Data->Type, RHS);
- if (RHSVal.isMinusOne())
- Diag(Loc, DL_Error,
- "division of %0 by -1 cannot be represented in type %1")
- << LHSVal << Data->Type;
- else
- Diag(Loc, DL_Error, "division by zero");
- }
- void __ubsan::__ubsan_handle_divrem_overflow(OverflowData *Data,
- ValueHandle LHS, ValueHandle RHS) {
- GET_REPORT_OPTIONS(false);
- handleDivremOverflowImpl(Data, LHS, RHS, Opts);
- }
- void __ubsan::__ubsan_handle_divrem_overflow_abort(OverflowData *Data,
- ValueHandle LHS,
- ValueHandle RHS) {
- GET_REPORT_OPTIONS(true);
- handleDivremOverflowImpl(Data, LHS, RHS, Opts);
- Die();
- }
- static void handleShiftOutOfBoundsImpl(ShiftOutOfBoundsData *Data,
- ValueHandle LHS, ValueHandle RHS,
- ReportOptions Opts) {
- SourceLocation Loc = Data->Loc.acquire();
- if (ignoreReport(Loc, Opts))
- return;
- ScopedReport R(Opts, Loc);
- Value LHSVal(Data->LHSType, LHS);
- Value RHSVal(Data->RHSType, RHS);
- if (RHSVal.isNegative())
- Diag(Loc, DL_Error, "shift exponent %0 is negative") << RHSVal;
- else if (RHSVal.getPositiveIntValue() >= Data->LHSType.getIntegerBitWidth())
- Diag(Loc, DL_Error,
- "shift exponent %0 is too large for %1-bit type %2")
- << RHSVal << Data->LHSType.getIntegerBitWidth() << Data->LHSType;
- else if (LHSVal.isNegative())
- Diag(Loc, DL_Error, "left shift of negative value %0") << LHSVal;
- else
- Diag(Loc, DL_Error,
- "left shift of %0 by %1 places cannot be represented in type %2")
- << LHSVal << RHSVal << Data->LHSType;
- }
- void __ubsan::__ubsan_handle_shift_out_of_bounds(ShiftOutOfBoundsData *Data,
- ValueHandle LHS,
- ValueHandle RHS) {
- GET_REPORT_OPTIONS(false);
- handleShiftOutOfBoundsImpl(Data, LHS, RHS, Opts);
- }
- void __ubsan::__ubsan_handle_shift_out_of_bounds_abort(
- ShiftOutOfBoundsData *Data,
- ValueHandle LHS,
- ValueHandle RHS) {
- GET_REPORT_OPTIONS(true);
- handleShiftOutOfBoundsImpl(Data, LHS, RHS, Opts);
- Die();
- }
- static void handleOutOfBoundsImpl(OutOfBoundsData *Data, ValueHandle Index,
- ReportOptions Opts) {
- SourceLocation Loc = Data->Loc.acquire();
- if (ignoreReport(Loc, Opts))
- return;
- ScopedReport R(Opts, Loc);
- Value IndexVal(Data->IndexType, Index);
- Diag(Loc, DL_Error, "index %0 out of bounds for type %1")
- << IndexVal << Data->ArrayType;
- }
- void __ubsan::__ubsan_handle_out_of_bounds(OutOfBoundsData *Data,
- ValueHandle Index) {
- GET_REPORT_OPTIONS(false);
- handleOutOfBoundsImpl(Data, Index, Opts);
- }
- void __ubsan::__ubsan_handle_out_of_bounds_abort(OutOfBoundsData *Data,
- ValueHandle Index) {
- GET_REPORT_OPTIONS(true);
- handleOutOfBoundsImpl(Data, Index, Opts);
- Die();
- }
- static void handleBuiltinUnreachableImpl(UnreachableData *Data,
- ReportOptions Opts) {
- ScopedReport R(Opts, Data->Loc);
- Diag(Data->Loc, DL_Error, "execution reached a __builtin_unreachable() call");
- }
- void __ubsan::__ubsan_handle_builtin_unreachable(UnreachableData *Data) {
- GET_REPORT_OPTIONS(true);
- handleBuiltinUnreachableImpl(Data, Opts);
- Die();
- }
- static void handleMissingReturnImpl(UnreachableData *Data, ReportOptions Opts) {
- ScopedReport R(Opts, Data->Loc);
- Diag(Data->Loc, DL_Error,
- "execution reached the end of a value-returning function "
- "without returning a value");
- }
- void __ubsan::__ubsan_handle_missing_return(UnreachableData *Data) {
- GET_REPORT_OPTIONS(true);
- handleMissingReturnImpl(Data, Opts);
- Die();
- }
- static void handleVLABoundNotPositive(VLABoundData *Data, ValueHandle Bound,
- ReportOptions Opts) {
- SourceLocation Loc = Data->Loc.acquire();
- if (ignoreReport(Loc, Opts))
- return;
- ScopedReport R(Opts, Loc);
- Diag(Loc, DL_Error, "variable length array bound evaluates to "
- "non-positive value %0")
- << Value(Data->Type, Bound);
- }
- void __ubsan::__ubsan_handle_vla_bound_not_positive(VLABoundData *Data,
- ValueHandle Bound) {
- GET_REPORT_OPTIONS(false);
- handleVLABoundNotPositive(Data, Bound, Opts);
- }
- void __ubsan::__ubsan_handle_vla_bound_not_positive_abort(VLABoundData *Data,
- ValueHandle Bound) {
- GET_REPORT_OPTIONS(true);
- handleVLABoundNotPositive(Data, Bound, Opts);
- Die();
- }
- static void handleFloatCastOverflow(FloatCastOverflowData *Data,
- ValueHandle From, ReportOptions Opts) {
- // TODO: Add deduplication once a SourceLocation is generated for this check.
- Location Loc = getCallerLocation();
- ScopedReport R(Opts, Loc);
- Diag(Loc, DL_Error,
- "value %0 is outside the range of representable values of type %2")
- << Value(Data->FromType, From) << Data->FromType << Data->ToType;
- }
- void __ubsan::__ubsan_handle_float_cast_overflow(FloatCastOverflowData *Data,
- ValueHandle From) {
- GET_REPORT_OPTIONS(false);
- handleFloatCastOverflow(Data, From, Opts);
- }
- void
- __ubsan::__ubsan_handle_float_cast_overflow_abort(FloatCastOverflowData *Data,
- ValueHandle From) {
- GET_REPORT_OPTIONS(true);
- handleFloatCastOverflow(Data, From, Opts);
- Die();
- }
- static void handleLoadInvalidValue(InvalidValueData *Data, ValueHandle Val,
- ReportOptions Opts) {
- SourceLocation Loc = Data->Loc.acquire();
- if (ignoreReport(Loc, Opts))
- return;
- ScopedReport R(Opts, Loc);
- Diag(Loc, DL_Error,
- "load of value %0, which is not a valid value for type %1")
- << Value(Data->Type, Val) << Data->Type;
- }
- void __ubsan::__ubsan_handle_load_invalid_value(InvalidValueData *Data,
- ValueHandle Val) {
- GET_REPORT_OPTIONS(false);
- handleLoadInvalidValue(Data, Val, Opts);
- }
- void __ubsan::__ubsan_handle_load_invalid_value_abort(InvalidValueData *Data,
- ValueHandle Val) {
- GET_REPORT_OPTIONS(true);
- handleLoadInvalidValue(Data, Val, Opts);
- Die();
- }
- static void handleFunctionTypeMismatch(FunctionTypeMismatchData *Data,
- ValueHandle Function,
- ReportOptions Opts) {
- const char *FName = "(unknown)";
- Location Loc = getFunctionLocation(Function, &FName);
- ScopedReport R(Opts, Loc);
- Diag(Data->Loc, DL_Error,
- "call to function %0 through pointer to incorrect function type %1")
- << FName << Data->Type;
- Diag(Loc, DL_Note, "%0 defined here") << FName;
- }
- void
- __ubsan::__ubsan_handle_function_type_mismatch(FunctionTypeMismatchData *Data,
- ValueHandle Function) {
- GET_REPORT_OPTIONS(false);
- handleFunctionTypeMismatch(Data, Function, Opts);
- }
- void __ubsan::__ubsan_handle_function_type_mismatch_abort(
- FunctionTypeMismatchData *Data, ValueHandle Function) {
- GET_REPORT_OPTIONS(true);
- handleFunctionTypeMismatch(Data, Function, Opts);
- Die();
- }
- static void handleNonNullReturn(NonNullReturnData *Data, ReportOptions Opts) {
- SourceLocation Loc = Data->Loc.acquire();
- if (ignoreReport(Loc, Opts))
- return;
- ScopedReport R(Opts, Loc);
- Diag(Loc, DL_Error, "null pointer returned from function declared to never "
- "return null");
- if (!Data->AttrLoc.isInvalid())
- Diag(Data->AttrLoc, DL_Note, "returns_nonnull attribute specified here");
- }
- void __ubsan::__ubsan_handle_nonnull_return(NonNullReturnData *Data) {
- GET_REPORT_OPTIONS(false);
- handleNonNullReturn(Data, Opts);
- }
- void __ubsan::__ubsan_handle_nonnull_return_abort(NonNullReturnData *Data) {
- GET_REPORT_OPTIONS(true);
- handleNonNullReturn(Data, Opts);
- Die();
- }
- static void handleNonNullArg(NonNullArgData *Data, ReportOptions Opts) {
- SourceLocation Loc = Data->Loc.acquire();
- if (ignoreReport(Loc, Opts))
- return;
- ScopedReport R(Opts, Loc);
- Diag(Loc, DL_Error, "null pointer passed as argument %0, which is declared to "
- "never be null") << Data->ArgIndex;
- if (!Data->AttrLoc.isInvalid())
- Diag(Data->AttrLoc, DL_Note, "nonnull attribute specified here");
- }
- void __ubsan::__ubsan_handle_nonnull_arg(NonNullArgData *Data) {
- GET_REPORT_OPTIONS(false);
- handleNonNullArg(Data, Opts);
- }
- void __ubsan::__ubsan_handle_nonnull_arg_abort(NonNullArgData *Data) {
- GET_REPORT_OPTIONS(true);
- handleNonNullArg(Data, Opts);
- Die();
- }
|