#include <iostream>
#include <stdexcept>
void erringFunc() { throw std::runtime_error("Hello"); }
void libCallA() { erringFunc(); }
void libCallB() { throw std::runtime_error("World"); }
void handleEventA() {
try {
libCallA();
} catch (std::runtime_error &E) {
std::cout << "handleEventA: unhandled error " << E.what() << "\n";
throw;
}
}
void handleEventB() {
try {
libCallB();
} catch (std::runtime_error &E) {
std::cout << "handleEventB: handle error " << E.what() << "\n";
}
}
class EventGen {
unsigned RemainingEvents = 5;
public:
int generateEvent() {
if (RemainingEvents > 0) {
--RemainingEvents;
return (RemainingEvents % 3) + 1;
}
return 0;
}
};
class TerminateException : public std::runtime_error {
public:
TerminateException() : std::runtime_error("Time to stop!") {}
};
void runEventLoop(EventGen &EG) {
while (true) {
try {
int Ev = EG.generateEvent();
switch (Ev) {
case 0:
throw TerminateException();
case 1:
handleEventA();
break;
case 2:
handleEventB();
break;
}
} catch (TerminateException &E) {
std::cout << "Terminated?\n";
throw;
} catch (std::runtime_error &E) {
std::cout << "Unhandled error: " << E.what() << "\n";
}
}
}
struct CleanUp {
~CleanUp() { std::cout << "Cleanup\n"; }
};
int main() {
EventGen EG;
try {
CleanUp CU;
runEventLoop(EG);
} catch (TerminateException &E) {
std::cout << "Terminated!\n";
}
return 0;
}
/*
REQUIRES: system-linux
RUN: %clang++ %cflags %s -o %t.exe -Wl,-q
RUN: llvm-bolt %t.exe --split-functions --split-strategy=randomN \
RUN: --split-all-cold --split-eh --bolt-seed=7 -o %t.bolt
RUN: %t.bolt | FileCheck %s
CHECK: handleEventB: handle error World
CHECK-NEXT: handleEventA: unhandled error Hello
CHECK-NEXT: Unhandled error: Hello
CHECK-NEXT: handleEventB: handle error World
CHECK-NEXT: handleEventA: unhandled error Hello
CHECK-NEXT: Unhandled error: Hello
CHECK-NEXT: Terminated?
CHECK-NEXT: Cleanup
CHECK-NEXT: Terminated!
*/