Skip to content

C++ error handling

C++ has two main mechanisms to handle runtime errors: 1. Exceptions
1. Error codes - extra: with std::optional nullable return type

Exceptions

The most common way of handling runtime errors is using exceptions - with try, throw and catch statements.

Exceptions work well for desktop applications but have certain issues which make it undesirable for writing real-time code running on embedded hardware (e.g. medical devices or vehicle software).

Issues

  • Breaks codeflow by potentially creating multiple exits
  • Performance overhead - takes non-deterministic time for stack unwinding
  • Can cause resource leaks, especially when calling legacy C++ code with raw pointers
  • Requires heap memory allocation - which is usually a no-no for embedded standards compliance (e.g. AUTOSAR)

Many companies have a no exception policy.

Error codes

Returning error codes are a legacy from C. A function would return a sentinal value indicating an error and the caller is responsible for checking this.

Some common patterns:

bool func1() { return true; }
int func2() { return 2; }

enum class ErrorCode { SUCCESS = 1, INVALID_INPUT = 2}; // expressive + type-safe
ErrorCode func3() { return ErrorCode::SUCCESS; }

Issues

  • At the call-site, a if..else chain is needed to handle errors
  • Return codes can be silently ignored

+ std::optional

🏗️

std::optional<T> was introduced in C++17 and expresses nullability ("either a <T> or nothing"). So the function can either (a) return null if there was an error or (b) the return value. However, we would lose information on the type of error.