2015-04-28 22:58:25 +00:00
|
|
|
// RUN: %clang_cc1 -triple x86_64-windows -fborland-extensions -DBORLAND -fsyntax-only -verify -fblocks %s
|
|
|
|
// RUN: %clang_cc1 -triple x86_64-windows -fms-extensions -fsyntax-only -verify -fblocks %s
|
2011-04-28 01:08:34 +00:00
|
|
|
|
|
|
|
#define JOIN2(x,y) x ## y
|
|
|
|
#define JOIN(x,y) JOIN2(x,y)
|
|
|
|
#define TEST2(name) JOIN(name,__LINE__)
|
|
|
|
#define TEST TEST2(test)
|
|
|
|
typedef int DWORD;
|
|
|
|
|
|
|
|
#pragma sysheader begin
|
|
|
|
|
|
|
|
struct EXCEPTION_INFO{};
|
|
|
|
|
2022-02-03 16:39:21 -05:00
|
|
|
unsigned long __exception_code(void);
|
Initial support for Win64 SEH IR emission
The lowering looks a lot like normal EH lowering, with the exception
that the exceptions are caught by executing filter expression code
instead of matching typeinfo globals. The filter expressions are
outlined into functions which are used in landingpad clauses where
typeinfo would normally go.
Major aspects that still need work:
- Non-call exceptions in __try bodies won't work yet. The plan is to
outline the __try block in the frontend to keep things simple.
- Filter expressions cannot use local variables until capturing is
implemented.
- __finally blocks will not run after exceptions. Fixing this requires
work in the LLVM SEH preparation pass.
The IR lowering looks like this:
// C code:
bool safe_div(int n, int d, int *r) {
__try {
*r = normal_div(n, d);
} __except(_exception_code() == EXCEPTION_INT_DIVIDE_BY_ZERO) {
return false;
}
return true;
}
; LLVM IR:
define i32 @filter(i8* %e, i8* %fp) {
%ehptrs = bitcast i8* %e to i32**
%ehrec = load i32** %ehptrs
%code = load i32* %ehrec
%matches = icmp eq i32 %code, i32 u0xC0000094
%matches.i32 = zext i1 %matches to i32
ret i32 %matches.i32
}
define i1 zeroext @safe_div(i32 %n, i32 %d, i32* %r) {
%rr = invoke i32 @normal_div(i32 %n, i32 %d)
to label %normal unwind to label %lpad
normal:
store i32 %rr, i32* %r
ret i1 1
lpad:
%ehvals = landingpad {i8*, i32} personality i32 (...)* @__C_specific_handler
catch i8* bitcast (i32 (i8*, i8*)* @filter to i8*)
%ehptr = extractvalue {i8*, i32} %ehvals, i32 0
%sel = extractvalue {i8*, i32} %ehvals, i32 1
%filter_sel = call i32 @llvm.eh.seh.typeid.for(i8* bitcast (i32 (i8*, i8*)* @filter to i8*))
%matches = icmp eq i32 %sel, %filter_sel
br i1 %matches, label %eh.except, label %eh.resume
eh.except:
ret i1 false
eh.resume:
resume
}
Reviewers: rjmccall, rsmith, majnemer
Differential Revision: http://reviews.llvm.org/D5607
llvm-svn: 226760
2015-01-22 01:36:17 +00:00
|
|
|
#ifdef BORLAND
|
2022-02-03 16:39:21 -05:00
|
|
|
struct EXCEPTION_INFO* __exception_info(void);
|
Initial support for Win64 SEH IR emission
The lowering looks a lot like normal EH lowering, with the exception
that the exceptions are caught by executing filter expression code
instead of matching typeinfo globals. The filter expressions are
outlined into functions which are used in landingpad clauses where
typeinfo would normally go.
Major aspects that still need work:
- Non-call exceptions in __try bodies won't work yet. The plan is to
outline the __try block in the frontend to keep things simple.
- Filter expressions cannot use local variables until capturing is
implemented.
- __finally blocks will not run after exceptions. Fixing this requires
work in the LLVM SEH preparation pass.
The IR lowering looks like this:
// C code:
bool safe_div(int n, int d, int *r) {
__try {
*r = normal_div(n, d);
} __except(_exception_code() == EXCEPTION_INT_DIVIDE_BY_ZERO) {
return false;
}
return true;
}
; LLVM IR:
define i32 @filter(i8* %e, i8* %fp) {
%ehptrs = bitcast i8* %e to i32**
%ehrec = load i32** %ehptrs
%code = load i32* %ehrec
%matches = icmp eq i32 %code, i32 u0xC0000094
%matches.i32 = zext i1 %matches to i32
ret i32 %matches.i32
}
define i1 zeroext @safe_div(i32 %n, i32 %d, i32* %r) {
%rr = invoke i32 @normal_div(i32 %n, i32 %d)
to label %normal unwind to label %lpad
normal:
store i32 %rr, i32* %r
ret i1 1
lpad:
%ehvals = landingpad {i8*, i32} personality i32 (...)* @__C_specific_handler
catch i8* bitcast (i32 (i8*, i8*)* @filter to i8*)
%ehptr = extractvalue {i8*, i32} %ehvals, i32 0
%sel = extractvalue {i8*, i32} %ehvals, i32 1
%filter_sel = call i32 @llvm.eh.seh.typeid.for(i8* bitcast (i32 (i8*, i8*)* @filter to i8*))
%matches = icmp eq i32 %sel, %filter_sel
br i1 %matches, label %eh.except, label %eh.resume
eh.except:
ret i1 false
eh.resume:
resume
}
Reviewers: rjmccall, rsmith, majnemer
Differential Revision: http://reviews.llvm.org/D5607
llvm-svn: 226760
2015-01-22 01:36:17 +00:00
|
|
|
#endif
|
2022-02-03 16:39:21 -05:00
|
|
|
int __abnormal_termination(void);
|
2011-04-28 01:08:34 +00:00
|
|
|
|
|
|
|
#define GetExceptionCode __exception_code
|
|
|
|
#define GetExceptionInformation __exception_info
|
|
|
|
#define AbnormalTermination __abnormal_termination
|
|
|
|
|
|
|
|
#pragma sysheader end
|
|
|
|
|
2011-10-21 03:57:52 +00:00
|
|
|
DWORD FilterExpression(int); // expected-note{{declared here}}
|
2011-04-28 01:08:34 +00:00
|
|
|
DWORD FilterExceptionInformation(struct EXCEPTION_INFO*);
|
|
|
|
|
2022-02-03 16:39:21 -05:00
|
|
|
const char * NotFilterExpression(void);
|
2011-04-28 01:08:34 +00:00
|
|
|
|
2022-02-03 16:39:21 -05:00
|
|
|
void TEST(void) {
|
2011-04-28 01:08:34 +00:00
|
|
|
__try {
|
|
|
|
__try {
|
|
|
|
__try {
|
|
|
|
}
|
|
|
|
__finally{
|
|
|
|
}
|
|
|
|
}
|
|
|
|
__finally{
|
|
|
|
}
|
|
|
|
}
|
|
|
|
__finally{
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-03 16:39:21 -05:00
|
|
|
void TEST(void) {
|
2011-04-28 01:08:34 +00:00
|
|
|
__try {
|
|
|
|
|
|
|
|
}
|
|
|
|
} // expected-error{{expected '__except' or '__finally' block}}
|
|
|
|
|
2022-02-03 16:39:21 -05:00
|
|
|
void TEST(void) {
|
[C11/C2x] Change the behavior of the implicit function declaration warning
C89 had a questionable feature where the compiler would implicitly
declare a function that the user called but was never previously
declared. The resulting function would be globally declared as
extern int func(); -- a function without a prototype which accepts zero
or more arguments.
C99 removed support for this questionable feature due to severe
security concerns. However, there was no deprecation period; C89 had
the feature, C99 didn't. So Clang (and GCC) both supported the
functionality as an extension in C99 and later modes.
C2x no longer supports that function signature as it now requires all
functions to have a prototype, and given the known security issues with
the feature, continuing to support it as an extension is not tenable.
This patch changes the diagnostic behavior for the
-Wimplicit-function-declaration warning group depending on the language
mode in effect. We continue to warn by default in C89 mode (due to the
feature being dangerous to use). However, because this feature will not
be supported in C2x mode, we've diagnosed it as being invalid for so
long, the security concerns with the feature, and the trivial
workaround for users (declare the function), we now default the
extension warning to an error in C99-C17 mode. This still gives users
an easy workaround if they are extensively using the extension in those
modes (they can disable the warning or use -Wno-error to downgrade the
error), but the new diagnostic makes it more clear that this feature is
not supported and should be avoided. In C2x mode, we no longer allow an
implicit function to be defined and treat the situation the same as any
other lookup failure.
Differential Revision: https://reviews.llvm.org/D122983
2022-04-20 11:25:35 -04:00
|
|
|
__except (FilterExpression()) { // expected-error{{call to undeclared function '__except'; ISO C99 and later do not support implicit function declarations}} \
|
2020-11-23 10:57:27 +01:00
|
|
|
// expected-error{{too few arguments to function call, expected 1, have 0}} \
|
|
|
|
// expected-error{{expected ';' after expression}}
|
2011-04-28 01:08:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-03 16:39:21 -05:00
|
|
|
void TEST(void) {
|
2011-04-28 01:08:34 +00:00
|
|
|
__finally { } // expected-error{{}}
|
|
|
|
}
|
|
|
|
|
2022-02-03 16:39:21 -05:00
|
|
|
void TEST(void) {
|
2011-04-28 01:08:34 +00:00
|
|
|
__try{
|
|
|
|
int try_scope = 0;
|
|
|
|
} // TODO: expected expression is an extra error
|
|
|
|
__except( try_scope ? 1 : -1 ) // expected-error{{undeclared identifier 'try_scope'}} expected-error{{expected expression}}
|
|
|
|
{}
|
|
|
|
}
|
|
|
|
|
2022-02-03 16:39:21 -05:00
|
|
|
void TEST(void) {
|
2011-04-28 01:08:34 +00:00
|
|
|
__try {
|
|
|
|
|
|
|
|
}
|
|
|
|
// TODO: Why are there two errors?
|
|
|
|
__except( ) { // expected-error{{expected expression}} expected-error{{expected expression}}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-03 16:39:21 -05:00
|
|
|
void TEST(void) {
|
2011-04-28 01:08:34 +00:00
|
|
|
__try {
|
|
|
|
|
|
|
|
}
|
|
|
|
__except ( FilterExpression(GetExceptionCode()) ) {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
__try {
|
|
|
|
|
|
|
|
}
|
|
|
|
__except( FilterExpression(__exception_code()) ) {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
__try {
|
|
|
|
|
|
|
|
}
|
|
|
|
__except( FilterExceptionInformation(__exception_info()) ) {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
__try {
|
|
|
|
|
|
|
|
}
|
|
|
|
__except(FilterExceptionInformation( GetExceptionInformation() ) ) {
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-03 16:39:21 -05:00
|
|
|
void TEST(void) {
|
2011-04-28 01:08:34 +00:00
|
|
|
__try {
|
|
|
|
|
|
|
|
}
|
2019-11-07 14:13:26 -08:00
|
|
|
__except ( NotFilterExpression() ) { // expected-error{{filter expression has non-integral type 'const char *'}}
|
2011-04-28 01:08:34 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-03 16:39:21 -05:00
|
|
|
void TEST(void) {
|
2011-04-28 01:08:34 +00:00
|
|
|
int function_scope = 0;
|
|
|
|
__try {
|
|
|
|
int try_scope = 0;
|
|
|
|
}
|
|
|
|
__except ( FilterExpression(GetExceptionCode()) ) {
|
|
|
|
(void)function_scope;
|
|
|
|
(void)try_scope; // expected-error{{undeclared identifier}}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-03 16:39:21 -05:00
|
|
|
void TEST(void) {
|
2011-04-28 01:08:34 +00:00
|
|
|
int function_scope = 0;
|
|
|
|
__try {
|
|
|
|
int try_scope = 0;
|
|
|
|
}
|
|
|
|
__finally {
|
|
|
|
(void)function_scope;
|
|
|
|
(void)try_scope; // expected-error{{undeclared identifier}}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-03 16:39:21 -05:00
|
|
|
void TEST(void) {
|
2011-04-28 01:08:34 +00:00
|
|
|
int function_scope = 0;
|
|
|
|
__try {
|
|
|
|
|
|
|
|
}
|
|
|
|
__except( function_scope ? 1 : -1 ) {}
|
|
|
|
}
|
|
|
|
|
Initial support for Win64 SEH IR emission
The lowering looks a lot like normal EH lowering, with the exception
that the exceptions are caught by executing filter expression code
instead of matching typeinfo globals. The filter expressions are
outlined into functions which are used in landingpad clauses where
typeinfo would normally go.
Major aspects that still need work:
- Non-call exceptions in __try bodies won't work yet. The plan is to
outline the __try block in the frontend to keep things simple.
- Filter expressions cannot use local variables until capturing is
implemented.
- __finally blocks will not run after exceptions. Fixing this requires
work in the LLVM SEH preparation pass.
The IR lowering looks like this:
// C code:
bool safe_div(int n, int d, int *r) {
__try {
*r = normal_div(n, d);
} __except(_exception_code() == EXCEPTION_INT_DIVIDE_BY_ZERO) {
return false;
}
return true;
}
; LLVM IR:
define i32 @filter(i8* %e, i8* %fp) {
%ehptrs = bitcast i8* %e to i32**
%ehrec = load i32** %ehptrs
%code = load i32* %ehrec
%matches = icmp eq i32 %code, i32 u0xC0000094
%matches.i32 = zext i1 %matches to i32
ret i32 %matches.i32
}
define i1 zeroext @safe_div(i32 %n, i32 %d, i32* %r) {
%rr = invoke i32 @normal_div(i32 %n, i32 %d)
to label %normal unwind to label %lpad
normal:
store i32 %rr, i32* %r
ret i1 1
lpad:
%ehvals = landingpad {i8*, i32} personality i32 (...)* @__C_specific_handler
catch i8* bitcast (i32 (i8*, i8*)* @filter to i8*)
%ehptr = extractvalue {i8*, i32} %ehvals, i32 0
%sel = extractvalue {i8*, i32} %ehvals, i32 1
%filter_sel = call i32 @llvm.eh.seh.typeid.for(i8* bitcast (i32 (i8*, i8*)* @filter to i8*))
%matches = icmp eq i32 %sel, %filter_sel
br i1 %matches, label %eh.except, label %eh.resume
eh.except:
ret i1 false
eh.resume:
resume
}
Reviewers: rjmccall, rsmith, majnemer
Differential Revision: http://reviews.llvm.org/D5607
llvm-svn: 226760
2015-01-22 01:36:17 +00:00
|
|
|
#ifdef BORLAND
|
2022-02-03 16:39:21 -05:00
|
|
|
void TEST(void) {
|
Initial support for Win64 SEH IR emission
The lowering looks a lot like normal EH lowering, with the exception
that the exceptions are caught by executing filter expression code
instead of matching typeinfo globals. The filter expressions are
outlined into functions which are used in landingpad clauses where
typeinfo would normally go.
Major aspects that still need work:
- Non-call exceptions in __try bodies won't work yet. The plan is to
outline the __try block in the frontend to keep things simple.
- Filter expressions cannot use local variables until capturing is
implemented.
- __finally blocks will not run after exceptions. Fixing this requires
work in the LLVM SEH preparation pass.
The IR lowering looks like this:
// C code:
bool safe_div(int n, int d, int *r) {
__try {
*r = normal_div(n, d);
} __except(_exception_code() == EXCEPTION_INT_DIVIDE_BY_ZERO) {
return false;
}
return true;
}
; LLVM IR:
define i32 @filter(i8* %e, i8* %fp) {
%ehptrs = bitcast i8* %e to i32**
%ehrec = load i32** %ehptrs
%code = load i32* %ehrec
%matches = icmp eq i32 %code, i32 u0xC0000094
%matches.i32 = zext i1 %matches to i32
ret i32 %matches.i32
}
define i1 zeroext @safe_div(i32 %n, i32 %d, i32* %r) {
%rr = invoke i32 @normal_div(i32 %n, i32 %d)
to label %normal unwind to label %lpad
normal:
store i32 %rr, i32* %r
ret i1 1
lpad:
%ehvals = landingpad {i8*, i32} personality i32 (...)* @__C_specific_handler
catch i8* bitcast (i32 (i8*, i8*)* @filter to i8*)
%ehptr = extractvalue {i8*, i32} %ehvals, i32 0
%sel = extractvalue {i8*, i32} %ehvals, i32 1
%filter_sel = call i32 @llvm.eh.seh.typeid.for(i8* bitcast (i32 (i8*, i8*)* @filter to i8*))
%matches = icmp eq i32 %sel, %filter_sel
br i1 %matches, label %eh.except, label %eh.resume
eh.except:
ret i1 false
eh.resume:
resume
}
Reviewers: rjmccall, rsmith, majnemer
Differential Revision: http://reviews.llvm.org/D5607
llvm-svn: 226760
2015-01-22 01:36:17 +00:00
|
|
|
(void)__abnormal_termination(); // expected-error{{only allowed in __finally block}}
|
|
|
|
(void)AbnormalTermination(); // expected-error{{only allowed in __finally block}}
|
|
|
|
|
2011-04-28 01:08:34 +00:00
|
|
|
__try {
|
|
|
|
(void)AbnormalTermination; // expected-error{{only allowed in __finally block}}
|
|
|
|
(void)__abnormal_termination; // expected-error{{only allowed in __finally block}}
|
|
|
|
}
|
|
|
|
__except( 1 ) {
|
|
|
|
(void)AbnormalTermination; // expected-error{{only allowed in __finally block}}
|
|
|
|
(void)__abnormal_termination; // expected-error{{only allowed in __finally block}}
|
|
|
|
}
|
|
|
|
|
|
|
|
__try {
|
|
|
|
}
|
|
|
|
__finally {
|
|
|
|
AbnormalTermination();
|
|
|
|
__abnormal_termination();
|
|
|
|
}
|
|
|
|
}
|
Initial support for Win64 SEH IR emission
The lowering looks a lot like normal EH lowering, with the exception
that the exceptions are caught by executing filter expression code
instead of matching typeinfo globals. The filter expressions are
outlined into functions which are used in landingpad clauses where
typeinfo would normally go.
Major aspects that still need work:
- Non-call exceptions in __try bodies won't work yet. The plan is to
outline the __try block in the frontend to keep things simple.
- Filter expressions cannot use local variables until capturing is
implemented.
- __finally blocks will not run after exceptions. Fixing this requires
work in the LLVM SEH preparation pass.
The IR lowering looks like this:
// C code:
bool safe_div(int n, int d, int *r) {
__try {
*r = normal_div(n, d);
} __except(_exception_code() == EXCEPTION_INT_DIVIDE_BY_ZERO) {
return false;
}
return true;
}
; LLVM IR:
define i32 @filter(i8* %e, i8* %fp) {
%ehptrs = bitcast i8* %e to i32**
%ehrec = load i32** %ehptrs
%code = load i32* %ehrec
%matches = icmp eq i32 %code, i32 u0xC0000094
%matches.i32 = zext i1 %matches to i32
ret i32 %matches.i32
}
define i1 zeroext @safe_div(i32 %n, i32 %d, i32* %r) {
%rr = invoke i32 @normal_div(i32 %n, i32 %d)
to label %normal unwind to label %lpad
normal:
store i32 %rr, i32* %r
ret i1 1
lpad:
%ehvals = landingpad {i8*, i32} personality i32 (...)* @__C_specific_handler
catch i8* bitcast (i32 (i8*, i8*)* @filter to i8*)
%ehptr = extractvalue {i8*, i32} %ehvals, i32 0
%sel = extractvalue {i8*, i32} %ehvals, i32 1
%filter_sel = call i32 @llvm.eh.seh.typeid.for(i8* bitcast (i32 (i8*, i8*)* @filter to i8*))
%matches = icmp eq i32 %sel, %filter_sel
br i1 %matches, label %eh.except, label %eh.resume
eh.except:
ret i1 false
eh.resume:
resume
}
Reviewers: rjmccall, rsmith, majnemer
Differential Revision: http://reviews.llvm.org/D5607
llvm-svn: 226760
2015-01-22 01:36:17 +00:00
|
|
|
#endif
|
2011-04-28 01:08:34 +00:00
|
|
|
|
2022-02-03 16:39:21 -05:00
|
|
|
void TEST(void) {
|
Initial support for Win64 SEH IR emission
The lowering looks a lot like normal EH lowering, with the exception
that the exceptions are caught by executing filter expression code
instead of matching typeinfo globals. The filter expressions are
outlined into functions which are used in landingpad clauses where
typeinfo would normally go.
Major aspects that still need work:
- Non-call exceptions in __try bodies won't work yet. The plan is to
outline the __try block in the frontend to keep things simple.
- Filter expressions cannot use local variables until capturing is
implemented.
- __finally blocks will not run after exceptions. Fixing this requires
work in the LLVM SEH preparation pass.
The IR lowering looks like this:
// C code:
bool safe_div(int n, int d, int *r) {
__try {
*r = normal_div(n, d);
} __except(_exception_code() == EXCEPTION_INT_DIVIDE_BY_ZERO) {
return false;
}
return true;
}
; LLVM IR:
define i32 @filter(i8* %e, i8* %fp) {
%ehptrs = bitcast i8* %e to i32**
%ehrec = load i32** %ehptrs
%code = load i32* %ehrec
%matches = icmp eq i32 %code, i32 u0xC0000094
%matches.i32 = zext i1 %matches to i32
ret i32 %matches.i32
}
define i1 zeroext @safe_div(i32 %n, i32 %d, i32* %r) {
%rr = invoke i32 @normal_div(i32 %n, i32 %d)
to label %normal unwind to label %lpad
normal:
store i32 %rr, i32* %r
ret i1 1
lpad:
%ehvals = landingpad {i8*, i32} personality i32 (...)* @__C_specific_handler
catch i8* bitcast (i32 (i8*, i8*)* @filter to i8*)
%ehptr = extractvalue {i8*, i32} %ehvals, i32 0
%sel = extractvalue {i8*, i32} %ehvals, i32 1
%filter_sel = call i32 @llvm.eh.seh.typeid.for(i8* bitcast (i32 (i8*, i8*)* @filter to i8*))
%matches = icmp eq i32 %sel, %filter_sel
br i1 %matches, label %eh.except, label %eh.resume
eh.except:
ret i1 false
eh.resume:
resume
}
Reviewers: rjmccall, rsmith, majnemer
Differential Revision: http://reviews.llvm.org/D5607
llvm-svn: 226760
2015-01-22 01:36:17 +00:00
|
|
|
(void)__exception_info(); // expected-error{{only allowed in __except filter expression}}
|
2011-04-28 01:08:34 +00:00
|
|
|
(void)GetExceptionInformation(); // expected-error{{only allowed in __except filter expression}}
|
Initial support for Win64 SEH IR emission
The lowering looks a lot like normal EH lowering, with the exception
that the exceptions are caught by executing filter expression code
instead of matching typeinfo globals. The filter expressions are
outlined into functions which are used in landingpad clauses where
typeinfo would normally go.
Major aspects that still need work:
- Non-call exceptions in __try bodies won't work yet. The plan is to
outline the __try block in the frontend to keep things simple.
- Filter expressions cannot use local variables until capturing is
implemented.
- __finally blocks will not run after exceptions. Fixing this requires
work in the LLVM SEH preparation pass.
The IR lowering looks like this:
// C code:
bool safe_div(int n, int d, int *r) {
__try {
*r = normal_div(n, d);
} __except(_exception_code() == EXCEPTION_INT_DIVIDE_BY_ZERO) {
return false;
}
return true;
}
; LLVM IR:
define i32 @filter(i8* %e, i8* %fp) {
%ehptrs = bitcast i8* %e to i32**
%ehrec = load i32** %ehptrs
%code = load i32* %ehrec
%matches = icmp eq i32 %code, i32 u0xC0000094
%matches.i32 = zext i1 %matches to i32
ret i32 %matches.i32
}
define i1 zeroext @safe_div(i32 %n, i32 %d, i32* %r) {
%rr = invoke i32 @normal_div(i32 %n, i32 %d)
to label %normal unwind to label %lpad
normal:
store i32 %rr, i32* %r
ret i1 1
lpad:
%ehvals = landingpad {i8*, i32} personality i32 (...)* @__C_specific_handler
catch i8* bitcast (i32 (i8*, i8*)* @filter to i8*)
%ehptr = extractvalue {i8*, i32} %ehvals, i32 0
%sel = extractvalue {i8*, i32} %ehvals, i32 1
%filter_sel = call i32 @llvm.eh.seh.typeid.for(i8* bitcast (i32 (i8*, i8*)* @filter to i8*))
%matches = icmp eq i32 %sel, %filter_sel
br i1 %matches, label %eh.except, label %eh.resume
eh.except:
ret i1 false
eh.resume:
resume
}
Reviewers: rjmccall, rsmith, majnemer
Differential Revision: http://reviews.llvm.org/D5607
llvm-svn: 226760
2015-01-22 01:36:17 +00:00
|
|
|
}
|
|
|
|
|
2022-02-03 16:39:21 -05:00
|
|
|
void TEST(void) {
|
Initial support for Win64 SEH IR emission
The lowering looks a lot like normal EH lowering, with the exception
that the exceptions are caught by executing filter expression code
instead of matching typeinfo globals. The filter expressions are
outlined into functions which are used in landingpad clauses where
typeinfo would normally go.
Major aspects that still need work:
- Non-call exceptions in __try bodies won't work yet. The plan is to
outline the __try block in the frontend to keep things simple.
- Filter expressions cannot use local variables until capturing is
implemented.
- __finally blocks will not run after exceptions. Fixing this requires
work in the LLVM SEH preparation pass.
The IR lowering looks like this:
// C code:
bool safe_div(int n, int d, int *r) {
__try {
*r = normal_div(n, d);
} __except(_exception_code() == EXCEPTION_INT_DIVIDE_BY_ZERO) {
return false;
}
return true;
}
; LLVM IR:
define i32 @filter(i8* %e, i8* %fp) {
%ehptrs = bitcast i8* %e to i32**
%ehrec = load i32** %ehptrs
%code = load i32* %ehrec
%matches = icmp eq i32 %code, i32 u0xC0000094
%matches.i32 = zext i1 %matches to i32
ret i32 %matches.i32
}
define i1 zeroext @safe_div(i32 %n, i32 %d, i32* %r) {
%rr = invoke i32 @normal_div(i32 %n, i32 %d)
to label %normal unwind to label %lpad
normal:
store i32 %rr, i32* %r
ret i1 1
lpad:
%ehvals = landingpad {i8*, i32} personality i32 (...)* @__C_specific_handler
catch i8* bitcast (i32 (i8*, i8*)* @filter to i8*)
%ehptr = extractvalue {i8*, i32} %ehvals, i32 0
%sel = extractvalue {i8*, i32} %ehvals, i32 1
%filter_sel = call i32 @llvm.eh.seh.typeid.for(i8* bitcast (i32 (i8*, i8*)* @filter to i8*))
%matches = icmp eq i32 %sel, %filter_sel
br i1 %matches, label %eh.except, label %eh.resume
eh.except:
ret i1 false
eh.resume:
resume
}
Reviewers: rjmccall, rsmith, majnemer
Differential Revision: http://reviews.llvm.org/D5607
llvm-svn: 226760
2015-01-22 01:36:17 +00:00
|
|
|
#ifndef BORLAND
|
|
|
|
(void)__exception_code; // expected-error{{builtin functions must be directly called}}
|
|
|
|
#endif
|
|
|
|
(void)__exception_code(); // expected-error{{only allowed in __except block or filter expression}}
|
|
|
|
(void)GetExceptionCode(); // expected-error{{only allowed in __except block or filter expression}}
|
|
|
|
}
|
|
|
|
|
2022-02-03 16:39:21 -05:00
|
|
|
void TEST(void) {
|
Initial support for Win64 SEH IR emission
The lowering looks a lot like normal EH lowering, with the exception
that the exceptions are caught by executing filter expression code
instead of matching typeinfo globals. The filter expressions are
outlined into functions which are used in landingpad clauses where
typeinfo would normally go.
Major aspects that still need work:
- Non-call exceptions in __try bodies won't work yet. The plan is to
outline the __try block in the frontend to keep things simple.
- Filter expressions cannot use local variables until capturing is
implemented.
- __finally blocks will not run after exceptions. Fixing this requires
work in the LLVM SEH preparation pass.
The IR lowering looks like this:
// C code:
bool safe_div(int n, int d, int *r) {
__try {
*r = normal_div(n, d);
} __except(_exception_code() == EXCEPTION_INT_DIVIDE_BY_ZERO) {
return false;
}
return true;
}
; LLVM IR:
define i32 @filter(i8* %e, i8* %fp) {
%ehptrs = bitcast i8* %e to i32**
%ehrec = load i32** %ehptrs
%code = load i32* %ehrec
%matches = icmp eq i32 %code, i32 u0xC0000094
%matches.i32 = zext i1 %matches to i32
ret i32 %matches.i32
}
define i1 zeroext @safe_div(i32 %n, i32 %d, i32* %r) {
%rr = invoke i32 @normal_div(i32 %n, i32 %d)
to label %normal unwind to label %lpad
normal:
store i32 %rr, i32* %r
ret i1 1
lpad:
%ehvals = landingpad {i8*, i32} personality i32 (...)* @__C_specific_handler
catch i8* bitcast (i32 (i8*, i8*)* @filter to i8*)
%ehptr = extractvalue {i8*, i32} %ehvals, i32 0
%sel = extractvalue {i8*, i32} %ehvals, i32 1
%filter_sel = call i32 @llvm.eh.seh.typeid.for(i8* bitcast (i32 (i8*, i8*)* @filter to i8*))
%matches = icmp eq i32 %sel, %filter_sel
br i1 %matches, label %eh.except, label %eh.resume
eh.except:
ret i1 false
eh.resume:
resume
}
Reviewers: rjmccall, rsmith, majnemer
Differential Revision: http://reviews.llvm.org/D5607
llvm-svn: 226760
2015-01-22 01:36:17 +00:00
|
|
|
__try {
|
|
|
|
} __except(1) {
|
|
|
|
GetExceptionCode(); // valid
|
|
|
|
GetExceptionInformation(); // expected-error{{only allowed in __except filter expression}}
|
|
|
|
}
|
2011-04-28 01:08:34 +00:00
|
|
|
}
|
2014-07-06 22:32:59 +00:00
|
|
|
|
2022-02-03 16:39:21 -05:00
|
|
|
void test_seh_leave_stmt(void) {
|
2014-07-06 22:53:19 +00:00
|
|
|
__leave; // expected-error{{'__leave' statement not in __try block}}
|
|
|
|
|
2014-07-06 22:32:59 +00:00
|
|
|
__try {
|
2014-07-07 00:12:30 +00:00
|
|
|
__leave;
|
|
|
|
__leave 4; // expected-error{{expected ';' after __leave statement}}
|
2014-07-06 22:32:59 +00:00
|
|
|
} __except(1) {
|
2014-07-06 22:53:19 +00:00
|
|
|
__leave; // expected-error{{'__leave' statement not in __try block}}
|
2014-07-06 22:32:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
__try {
|
2014-07-07 00:12:30 +00:00
|
|
|
__leave;
|
2014-07-06 22:32:59 +00:00
|
|
|
} __finally {
|
2014-07-06 22:53:19 +00:00
|
|
|
__leave; // expected-error{{'__leave' statement not in __try block}}
|
2014-07-06 22:32:59 +00:00
|
|
|
}
|
2014-07-06 22:53:19 +00:00
|
|
|
__leave; // expected-error{{'__leave' statement not in __try block}}
|
2014-07-06 22:32:59 +00:00
|
|
|
}
|
Warn when jumping out of a __finally block via continue, break, return, __leave.
Since continue, break, return are much more common than __finally, this tries
to keep the work for continue, break, return O(1). Sema keeps a stack of active
__finally scopes (to do this, ActOnSEHFinally() is split into
ActOnStartSEHFinally() and ActOnFinishSEHFinally()), and the various jump
statements then check if the current __finally scope (if present) is deeper
than then destination scope of the jump.
The same warning for goto statements is still missing.
This is the moral equivalent of MSVC's C4532.
llvm-svn: 231623
2015-03-09 02:47:59 +00:00
|
|
|
|
2022-02-03 16:39:21 -05:00
|
|
|
void test_jump_out_of___finally(void) {
|
Warn when jumping out of a __finally block via continue, break, return, __leave.
Since continue, break, return are much more common than __finally, this tries
to keep the work for continue, break, return O(1). Sema keeps a stack of active
__finally scopes (to do this, ActOnSEHFinally() is split into
ActOnStartSEHFinally() and ActOnFinishSEHFinally()), and the various jump
statements then check if the current __finally scope (if present) is deeper
than then destination scope of the jump.
The same warning for goto statements is still missing.
This is the moral equivalent of MSVC's C4532.
llvm-svn: 231623
2015-03-09 02:47:59 +00:00
|
|
|
while(1) {
|
|
|
|
__try {
|
|
|
|
} __finally {
|
|
|
|
continue; // expected-warning{{jump out of __finally block has undefined behavior}}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
__try {
|
|
|
|
} __finally {
|
|
|
|
while (1) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that a deep __finally containing a block with a shallow continue
|
|
|
|
// doesn't trigger the warning.
|
|
|
|
while(1) {{{{
|
|
|
|
__try {
|
|
|
|
} __finally {
|
|
|
|
^{
|
|
|
|
while(1)
|
|
|
|
continue;
|
|
|
|
}();
|
|
|
|
}
|
|
|
|
}}}}
|
|
|
|
|
|
|
|
while(1) {
|
|
|
|
__try {
|
|
|
|
} __finally {
|
|
|
|
break; // expected-warning{{jump out of __finally block has undefined behavior}}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
switch(1) {
|
|
|
|
case 1:
|
|
|
|
__try {
|
|
|
|
} __finally {
|
|
|
|
break; // expected-warning{{jump out of __finally block has undefined behavior}}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
__try {
|
|
|
|
} __finally {
|
|
|
|
while (1) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
__try {
|
|
|
|
__try {
|
|
|
|
} __finally {
|
|
|
|
__leave; // expected-warning{{jump out of __finally block has undefined behavior}}
|
|
|
|
}
|
|
|
|
} __finally {
|
|
|
|
}
|
|
|
|
__try {
|
|
|
|
} __finally {
|
|
|
|
__try {
|
|
|
|
__leave;
|
|
|
|
} __finally {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
__try {
|
|
|
|
} __finally {
|
|
|
|
return; // expected-warning{{jump out of __finally block has undefined behavior}}
|
|
|
|
}
|
|
|
|
|
|
|
|
__try {
|
|
|
|
} __finally {
|
|
|
|
^{
|
|
|
|
return;
|
|
|
|
}();
|
|
|
|
}
|
|
|
|
}
|
2015-04-02 22:09:32 +00:00
|
|
|
|
2022-02-03 16:39:21 -05:00
|
|
|
void test_typo_in_except(void) {
|
2015-04-02 22:09:32 +00:00
|
|
|
__try {
|
|
|
|
} __except(undeclared_identifier) { // expected-error {{use of undeclared identifier 'undeclared_identifier'}} expected-error {{expected expression}}
|
|
|
|
}
|
|
|
|
}
|