reactos/modules/rostests/apitests/compiler/pseh.c

2902 lines
50 KiB
C

/*
Copyright (c) 2008 KJK::Hyperion
Permission is hereby granted, free of charge, to any person obtaining a
copy of this software and associated documentation files (the "Software"),
to deal in the Software without restriction, including without limitation
the rights to use, copy, modify, merge, publish, distribute, sublicense,
and/or sell copies of the Software, and to permit persons to whom the
Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
DEALINGS IN THE SOFTWARE.
*/
#include <pseh/pseh2.h>
#ifdef __cplusplus
extern "C" {
#endif
#include <wine/test.h>
#undef subtest
extern void no_op(void);
extern int return_arg(int);
extern int return_zero(void);
extern int return_positive(void);
extern int return_negative(void);
extern int return_one(void);
extern int return_minusone(void);
extern int return_zero_2(void *);
extern int return_positive_2(void *);
extern int return_negative_2(void *);
extern int return_one_2(void *);
extern int return_minusone_2(void *);
extern int return_zero_3(int);
extern int return_positive_3(int);
extern int return_negative_3(int);
extern int return_one_3(int);
extern int return_minusone_3(int);
extern int return_zero_4(void *, int);
extern int return_positive_4(void *, int);
extern int return_negative_4(void *, int);
extern int return_one_4(void *, int);
extern int return_minusone_4(void *, int);
extern void set_positive(int *);
//static int call_test(int (*)(void));
#ifdef __cplusplus
} // extern "C"
#endif
#define DEFINE_TEST(NAME_) static int NAME_(void)
/* Empty statements *///{{{
DEFINE_TEST(test_empty_1)
{
_SEH2_TRY { } _SEH2_EXCEPT(0) { } _SEH2_END;
return 1;
}
DEFINE_TEST(test_empty_2)
{
_SEH2_TRY { } _SEH2_EXCEPT(-1) { } _SEH2_END;
return 1;
}
DEFINE_TEST(test_empty_3)
{
_SEH2_TRY { } _SEH2_EXCEPT(1) { } _SEH2_END;
return 1;
}
DEFINE_TEST(test_empty_4)
{
_SEH2_TRY { } _SEH2_FINALLY { } _SEH2_END;
return 1;
}
DEFINE_TEST(test_empty_5)
{
_SEH2_TRY { _SEH2_TRY { } _SEH2_EXCEPT(0) { } _SEH2_END; } _SEH2_FINALLY { } _SEH2_END;
return 1;
}
DEFINE_TEST(test_empty_6)
{
_SEH2_TRY { _SEH2_TRY { } _SEH2_EXCEPT(-1) { } _SEH2_END; } _SEH2_FINALLY { } _SEH2_END;
return 1;
}
DEFINE_TEST(test_empty_7)
{
_SEH2_TRY { _SEH2_TRY { } _SEH2_EXCEPT(1) { } _SEH2_END; } _SEH2_FINALLY { } _SEH2_END;
return 1;
}
DEFINE_TEST(test_empty_8)
{
_SEH2_TRY { _SEH2_TRY { } _SEH2_FINALLY { } _SEH2_END; } _SEH2_FINALLY { } _SEH2_END;
return 1;
}
//}}}
/* Static exception filters *///{{{
DEFINE_TEST(test_execute_handler_1)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_positive();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_continue_execution_1)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_positive();
}
_SEH2_EXCEPT(EXCEPTION_CONTINUE_EXECUTION)
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_continue_search_1)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_EXCEPT(EXCEPTION_CONTINUE_SEARCH)
{
ret = return_zero();
}
_SEH2_END;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_positive();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_execute_handler_2)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_EXCEPT(12345)
{
ret = return_positive();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_continue_execution_2)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_positive();
}
_SEH2_EXCEPT(-12345)
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
//}}}
/* Dynamic exception filters *///{{{
DEFINE_TEST(test_execute_handler_3)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_EXCEPT(return_one())
{
ret = return_positive();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_continue_execution_3)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_positive();
}
_SEH2_EXCEPT(return_minusone())
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_continue_search_2)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_EXCEPT(return_zero())
{
ret = return_zero();
}
_SEH2_END;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_positive();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_execute_handler_4)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_EXCEPT(return_positive())
{
ret = return_positive();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_continue_execution_4)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_positive();
}
_SEH2_EXCEPT(return_negative())
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
//}}}
/* Dynamic exception filters, using _SEH2_GetExceptionInformation() *///{{{
DEFINE_TEST(test_execute_handler_5)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_EXCEPT(return_one_2(_SEH2_GetExceptionInformation()))
{
ret = return_positive();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_continue_execution_5)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_positive();
}
_SEH2_EXCEPT(return_minusone_2(_SEH2_GetExceptionInformation()))
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_continue_search_3)
{
static int ret;
ret = return_positive();
_SEH2_TRY
{
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_EXCEPT(return_zero_2(_SEH2_GetExceptionInformation()))
{
ret = return_zero();
}
_SEH2_END;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_arg(ret);
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_execute_handler_6)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_EXCEPT(return_positive_2(_SEH2_GetExceptionInformation()))
{
ret = return_positive();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_continue_execution_6)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_positive();
}
_SEH2_EXCEPT(return_negative_2(_SEH2_GetExceptionInformation()))
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
//}}}
/* Dynamic exception filters, using _SEH2_GetExceptionCode() *///{{{
DEFINE_TEST(test_execute_handler_7)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_EXCEPT(return_one_3(_SEH2_GetExceptionCode()))
{
ret = return_positive();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_continue_execution_7)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_positive();
}
_SEH2_EXCEPT(return_minusone_3(_SEH2_GetExceptionCode()))
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_continue_search_4)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_EXCEPT(return_zero_3(_SEH2_GetExceptionCode()))
{
ret = return_zero();
}
_SEH2_END;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_positive();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_execute_handler_8)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_EXCEPT(return_positive_3(_SEH2_GetExceptionCode()))
{
ret = return_positive();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_continue_execution_8)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_positive();
}
_SEH2_EXCEPT(return_negative_3(_SEH2_GetExceptionCode()))
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
//}}}
/* Dynamic exception filters, using _SEH2_GetExceptionInformation() and _SEH2_GetExceptionCode() *///{{{
DEFINE_TEST(test_execute_handler_9)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_EXCEPT(return_one_4(_SEH2_GetExceptionInformation(), _SEH2_GetExceptionCode()))
{
ret = return_positive();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_continue_execution_9)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_positive();
}
_SEH2_EXCEPT(return_minusone_4(_SEH2_GetExceptionInformation(), _SEH2_GetExceptionCode()))
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_continue_search_5)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_EXCEPT(return_zero_4(_SEH2_GetExceptionInformation(), _SEH2_GetExceptionCode()))
{
ret = return_zero();
}
_SEH2_END;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_positive();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_execute_handler_10)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_EXCEPT(return_positive_4(_SEH2_GetExceptionInformation(), _SEH2_GetExceptionCode()))
{
ret = return_positive();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_continue_execution_10)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_positive();
}
_SEH2_EXCEPT(return_negative_4(_SEH2_GetExceptionInformation(), _SEH2_GetExceptionCode()))
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
//}}}
/* Constant exception filters with side effects *///{{{
DEFINE_TEST(test_execute_handler_11)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_EXCEPT(set_positive(&ret), EXCEPTION_EXECUTE_HANDLER)
{
ret = ret ? return_positive() : return_zero();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_continue_execution_11)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = ret ? return_positive() : return_zero();
}
_SEH2_EXCEPT(set_positive(&ret), EXCEPTION_CONTINUE_EXECUTION)
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_continue_search_6)
{
static int ret;
static int ret2;
ret = return_zero();
ret2 = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
ret2 = return_zero();
}
_SEH2_EXCEPT(set_positive(&ret), EXCEPTION_CONTINUE_SEARCH)
{
ret = return_zero();
ret2 = return_zero();
}
_SEH2_END;
}
_SEH2_EXCEPT(set_positive(&ret2), EXCEPTION_EXECUTE_HANDLER)
{
ret = return_arg(ret);
ret2 = return_arg(ret2);
}
_SEH2_END;
return ret == return_positive() && ret2 == return_positive();
}
DEFINE_TEST(test_execute_handler_12)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_EXCEPT(set_positive(&ret), 12345)
{
ret = return_arg(ret);
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_continue_execution_12)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_arg(ret);
}
_SEH2_EXCEPT(set_positive(&ret), -12345)
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
//}}}
/* _SEH2_LEAVE *///{{{
DEFINE_TEST(test_leave_1)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
ret = return_positive();
_SEH2_LEAVE;
ret = return_zero();
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_leave_2)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
ret = return_positive();
_SEH2_LEAVE;
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_leave_3)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
ret = return_positive();
if(return_one())
_SEH2_LEAVE;
ret = return_zero();
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_leave_4)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
int i;
int n = return_one() + return_one();
for(i = return_zero(); i < n; ++ i)
{
if(i == return_one())
{
ret = return_positive();
_SEH2_LEAVE;
}
}
ret = return_zero();
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_leave_5)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
switch(return_one())
{
case 0: ret = return_zero();
case 1: ret = return_positive(); _SEH2_LEAVE;
case 2: ret = return_zero();
}
ret = return_zero();
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_leave_6)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
_SEH2_LEAVE;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_zero();
}
_SEH2_END;
ret = return_positive();
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
//}}}
/* _SEH2_YIELD() *///{{{
static
int test_yield_1_helper(void)
{
_SEH2_TRY
{
_SEH2_YIELD(return return_positive());
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
_SEH2_YIELD(return return_zero());
}
_SEH2_END;
return return_zero();
}
DEFINE_TEST(test_yield_1)
{
return test_yield_1_helper() == return_positive();
}
static
int test_yield_2_helper(void)
{
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
_SEH2_YIELD(return return_zero());
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
_SEH2_YIELD(return return_positive());
}
_SEH2_END;
return return_zero();
}
DEFINE_TEST(test_yield_2)
{
return test_yield_2_helper() == return_positive();
}
static
int test_yield_3_helper(void)
{
_SEH2_TRY
{
_SEH2_TRY
{
_SEH2_YIELD(return return_positive());
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
_SEH2_YIELD(return return_zero());
}
_SEH2_END;
_SEH2_YIELD(return return_zero());
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
_SEH2_YIELD(return return_zero());
}
_SEH2_END;
return return_zero();
}
DEFINE_TEST(test_yield_3)
{
return test_yield_3_helper() == return_positive();
}
static
int test_yield_4_helper(void)
{
_SEH2_TRY
{
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
_SEH2_YIELD(return return_zero());
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
_SEH2_YIELD(return return_positive());
}
_SEH2_END;
_SEH2_YIELD(return return_zero());
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
_SEH2_YIELD(return return_zero());
}
_SEH2_END;
return return_zero();
}
DEFINE_TEST(test_yield_4)
{
return test_yield_4_helper() == return_positive();
}
static int test_yield_5_ret;
static
int test_yield_5_helper(void)
{
test_yield_5_ret = return_zero();
_SEH2_TRY
{
_SEH2_YIELD(return return_positive());
}
_SEH2_FINALLY
{
test_yield_5_ret = return_positive();
}
_SEH2_END;
return return_zero();
}
DEFINE_TEST(test_yield_5)
{
return test_yield_5_helper() == return_positive() && test_yield_5_ret == return_positive();
}
static int test_yield_6_ret;
static
int test_yield_6_helper(void)
{
test_yield_6_ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
_SEH2_YIELD(return return_positive());
}
_SEH2_FINALLY
{
test_yield_6_ret = return_positive();
}
_SEH2_END;
}
_SEH2_FINALLY
{
test_yield_6_ret += return_one();
}
_SEH2_END;
return return_zero();
}
DEFINE_TEST(test_yield_6)
{
return test_yield_6_helper() == return_positive() && test_yield_6_ret == return_positive() + return_one();
}
//}}}
/* Termination blocks *///{{{
DEFINE_TEST(test_finally_1)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
ret = return_arg(ret);
}
_SEH2_FINALLY
{
ret = return_positive();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_finally_2)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
ret = return_arg(ret);
_SEH2_LEAVE;
}
_SEH2_FINALLY
{
ret = return_positive();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_finally_3)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
ret = return_arg(ret);
_SEH2_YIELD(goto leave);
}
_SEH2_FINALLY
{
ret = return_positive();
}
_SEH2_END;
leave:
return ret == return_positive();
}
static int test_finally_4_ret;
static int test_finally_4_helper(void)
{
test_finally_4_ret = return_zero();
_SEH2_TRY
{
test_finally_4_ret = return_arg(test_finally_4_ret);
_SEH2_YIELD(return return_positive());
}
_SEH2_FINALLY
{
test_finally_4_ret = return_positive();
}
_SEH2_END;
return return_zero();
}
DEFINE_TEST(test_finally_4)
{
return test_finally_4_helper() == return_positive() && test_finally_4_ret;
}
DEFINE_TEST(test_finally_5)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_FINALLY
{
ret = return_positive();
}
_SEH2_END;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_arg(ret);
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_finally_6)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
ret = return_arg(ret);
}
_SEH2_FINALLY
{
if(ret == return_zero())
ret = return_positive();
}
_SEH2_END;
}
_SEH2_FINALLY
{
if(ret == return_positive())
ret = return_positive() + return_one();
}
_SEH2_END;
return ret == return_positive() + return_one();
}
DEFINE_TEST(test_finally_7)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
ret = return_arg(ret);
_SEH2_LEAVE;
}
_SEH2_FINALLY
{
if(ret == return_zero())
ret = return_positive();
}
_SEH2_END;
}
_SEH2_FINALLY
{
if(ret == return_positive())
ret = return_positive() + return_one();
}
_SEH2_END;
return ret == return_positive() + return_one();
}
DEFINE_TEST(test_finally_8)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
ret = return_arg(ret);
_SEH2_YIELD(goto leave);
}
_SEH2_FINALLY
{
if(ret == return_zero())
ret = return_positive();
}
_SEH2_END;
}
_SEH2_FINALLY
{
if(ret == return_positive())
ret = return_positive() + return_one();
}
_SEH2_END;
leave:
return ret == return_positive() + return_one();
}
static int test_finally_9_ret;
static int test_finally_9_helper(void)
{
test_finally_9_ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
test_finally_9_ret = return_arg(test_finally_9_ret);
_SEH2_YIELD(return return_positive());
}
_SEH2_FINALLY
{
if(test_finally_9_ret == return_zero())
test_finally_9_ret = return_positive();
}
_SEH2_END;
}
_SEH2_FINALLY
{
if(test_finally_9_ret == return_positive())
test_finally_9_ret = return_positive() + return_one();
}
_SEH2_END;
return return_zero();
}
DEFINE_TEST(test_finally_9)
{
return test_finally_9_helper() == return_positive() && test_finally_9_ret == return_positive() + return_one();
}
DEFINE_TEST(test_finally_10)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_FINALLY
{
if(ret == return_zero())
ret = return_positive();
}
_SEH2_END;
}
_SEH2_FINALLY
{
if(ret == return_positive())
ret = return_positive() + return_one();
}
_SEH2_END;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_arg(ret);
}
_SEH2_END;
return ret == return_positive() + return_one();
}
DEFINE_TEST(test_finally_11)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
_SEH2_TRY
{
ret = return_arg(ret);
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_zero();
}
_SEH2_END;
}
_SEH2_FINALLY
{
ret = return_positive();
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_END;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
if(ret == return_positive())
ret += return_one();
}
_SEH2_END;
return ret == return_positive() + return_one();
}
DEFINE_TEST(test_finally_12)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
_SEH2_TRY
{
ret = return_arg(ret);
}
_SEH2_FINALLY
{
ret = return_positive();
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_END;
}
_SEH2_FINALLY
{
if(ret == return_positive())
ret += return_one();
}
_SEH2_END;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
if(ret == return_positive() + return_one())
ret += return_one();
}
_SEH2_END;
return ret == return_positive() + return_one() + return_one();
}
static int test_finally_13_ret;
static
void test_finally_13_helper(void)
{
test_finally_13_ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
test_finally_13_ret = return_positive();
_SEH2_YIELD(return);
test_finally_13_ret = return_zero();
}
_SEH2_FINALLY
{
if(test_finally_13_ret == return_positive())
test_finally_13_ret += return_one();
}
_SEH2_END;
}
_SEH2_FINALLY
{
if(test_finally_13_ret == return_positive() + return_one())
test_finally_13_ret += return_one();
RaiseException(0xE00DEAD0, 0, 0, NULL);
test_finally_13_ret = return_zero();
}
_SEH2_END;
test_finally_13_ret = return_zero();
}
DEFINE_TEST(test_finally_13)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
ret = return_arg(ret);
test_finally_13_helper();
ret = return_zero();
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_positive();
}
_SEH2_END;
return ret == return_positive() && test_finally_13_ret == return_positive() + return_one() + return_one();
}
static int test_finally_14_ret;
static
void test_finally_14_helper(void)
{
test_finally_14_ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
_SEH2_TRY
{
test_finally_14_ret = return_positive();
RaiseException(0xE00DEAD0, 0, 0, NULL);
test_finally_14_ret = return_zero();
}
_SEH2_FINALLY
{
if(test_finally_14_ret == return_positive())
test_finally_14_ret += return_one();
}
_SEH2_END;
}
_SEH2_FINALLY
{
if(test_finally_14_ret == return_positive() + return_one())
test_finally_14_ret += return_one();
RaiseException(0xE00DEAD0, 0, 0, NULL);
test_finally_14_ret = return_zero();
}
_SEH2_END;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
if(test_finally_14_ret == return_positive() + return_one() + return_one())
test_finally_14_ret += return_one();
}
_SEH2_END;
test_finally_14_ret = return_arg(test_finally_14_ret);
}
DEFINE_TEST(test_finally_14)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
ret = return_arg(ret);
test_finally_14_helper();
ret = return_positive();
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive() && test_finally_14_ret == return_positive() + return_one() + return_one() + return_one();
}
//}}}
/* _SEH2_GetExceptionInformation() *///{{{
static
int verify_xpointers(struct _EXCEPTION_POINTERS * ep, DWORD code, DWORD flags, DWORD argc, const ULONG_PTR * argv, int * ret, int filter)
{
*ret =
ep &&
ep->ExceptionRecord &&
ep->ContextRecord &&
ep->ExceptionRecord->ExceptionCode == code &&
ep->ExceptionRecord->ExceptionFlags == flags &&
ep->ExceptionRecord->NumberParameters == argc &&
(argv || !argc) &&
memcmp(ep->ExceptionRecord->ExceptionInformation, argv, sizeof(argv[0]) * argc) == 0;
if(*ret)
*ret = return_positive();
return filter;
}
DEFINE_TEST(test_xpointers_1)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
}
_SEH2_EXCEPT(verify_xpointers(_SEH2_GetExceptionInformation(), 0xE00DEAD0, 0, 0, NULL, &ret, EXCEPTION_EXECUTE_HANDLER))
{
ret = return_arg(ret);
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_xpointers_2)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, EXCEPTION_NONCONTINUABLE, 0, NULL);
}
_SEH2_EXCEPT(verify_xpointers(_SEH2_GetExceptionInformation(), 0xE00DEAD0, EXCEPTION_NONCONTINUABLE, 0, NULL, &ret, EXCEPTION_EXECUTE_HANDLER))
{
ret = return_arg(ret);
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_xpointers_3)
{
static int ret;
static const ULONG_PTR args[] = { 1, 2, 12345 };
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, EXCEPTION_NONCONTINUABLE, 0, args);
}
_SEH2_EXCEPT(verify_xpointers(_SEH2_GetExceptionInformation(), 0xE00DEAD0, EXCEPTION_NONCONTINUABLE, 0, args, &ret, EXCEPTION_EXECUTE_HANDLER))
{
ret = return_arg(ret);
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_xpointers_4)
{
static int ret;
static const ULONG_PTR args[] = { 1, 2, 12345 };
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, EXCEPTION_NONCONTINUABLE, 1, args);
}
_SEH2_EXCEPT(verify_xpointers(_SEH2_GetExceptionInformation(), 0xE00DEAD0, EXCEPTION_NONCONTINUABLE, 1, args, &ret, EXCEPTION_EXECUTE_HANDLER))
{
ret = return_arg(ret);
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_xpointers_5)
{
static int ret;
static const ULONG_PTR args[] = { 1, 2, 12345 };
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, EXCEPTION_NONCONTINUABLE, 2, args);
}
_SEH2_EXCEPT(verify_xpointers(_SEH2_GetExceptionInformation(), 0xE00DEAD0, EXCEPTION_NONCONTINUABLE, 2, args, &ret, EXCEPTION_EXECUTE_HANDLER))
{
ret = return_arg(ret);
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_xpointers_6)
{
static int ret;
static const ULONG_PTR args[] = { 1, 2, 12345 };
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, EXCEPTION_NONCONTINUABLE, 3, args);
}
_SEH2_EXCEPT(verify_xpointers(_SEH2_GetExceptionInformation(), 0xE00DEAD0, EXCEPTION_NONCONTINUABLE, 3, args, &ret, EXCEPTION_EXECUTE_HANDLER))
{
ret = return_arg(ret);
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_xpointers_7)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_arg(ret);
}
_SEH2_EXCEPT(verify_xpointers(_SEH2_GetExceptionInformation(), 0xE00DEAD0, 0, 0, NULL, &ret, EXCEPTION_CONTINUE_EXECUTION))
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_xpointers_8)
{
static int ret;
static const ULONG_PTR args[] = { 1, 2, 12345 };
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, args);
ret = return_arg(ret);
}
_SEH2_EXCEPT(verify_xpointers(_SEH2_GetExceptionInformation(), 0xE00DEAD0, 0, 0, args, &ret, EXCEPTION_CONTINUE_EXECUTION))
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_xpointers_9)
{
static int ret;
static const ULONG_PTR args[] = { 1, 2, 12345 };
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 1, args);
ret = return_arg(ret);
}
_SEH2_EXCEPT(verify_xpointers(_SEH2_GetExceptionInformation(), 0xE00DEAD0, 0, 1, args, &ret, EXCEPTION_CONTINUE_EXECUTION))
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_xpointers_10)
{
static int ret;
static const ULONG_PTR args[] = { 1, 2, 12345 };
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 2, args);
ret = return_arg(ret);
}
_SEH2_EXCEPT(verify_xpointers(_SEH2_GetExceptionInformation(), 0xE00DEAD0, 0, 2, args, &ret, EXCEPTION_CONTINUE_EXECUTION))
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_xpointers_11)
{
static int ret;
static const ULONG_PTR args[] = { 1, 2, 12345 };
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 3, args);
ret = return_arg(ret);
}
_SEH2_EXCEPT(verify_xpointers(_SEH2_GetExceptionInformation(), 0xE00DEAD0, 0, 3, args, &ret, EXCEPTION_CONTINUE_EXECUTION))
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_xpointers_12)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
RaiseException(0xE00DEAD0, EXCEPTION_NONCONTINUABLE, 0, NULL);
}
_SEH2_EXCEPT(verify_xpointers(_SEH2_GetExceptionInformation(), 0xE00DEAD0, EXCEPTION_NONCONTINUABLE, 0, NULL, &ret, EXCEPTION_CONTINUE_SEARCH))
{
ret = return_zero();
}
_SEH2_END;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_arg(ret);
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_xpointers_13)
{
static int ret;
static const ULONG_PTR args[] = { 1, 2, 12345 };
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
RaiseException(0xE00DEAD0, EXCEPTION_NONCONTINUABLE, 0, args);
}
_SEH2_EXCEPT(verify_xpointers(_SEH2_GetExceptionInformation(), 0xE00DEAD0, EXCEPTION_NONCONTINUABLE, 0, args, &ret, EXCEPTION_CONTINUE_SEARCH))
{
ret = return_zero();
}
_SEH2_END;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_arg(ret);
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_xpointers_14)
{
static int ret;
static const ULONG_PTR args[] = { 1, 2, 12345 };
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
RaiseException(0xE00DEAD0, EXCEPTION_NONCONTINUABLE, 1, args);
}
_SEH2_EXCEPT(verify_xpointers(_SEH2_GetExceptionInformation(), 0xE00DEAD0, EXCEPTION_NONCONTINUABLE, 1, args, &ret, EXCEPTION_CONTINUE_SEARCH))
{
ret = return_zero();
}
_SEH2_END;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_arg(ret);
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_xpointers_15)
{
static int ret;
static const ULONG_PTR args[] = { 1, 2, 12345 };
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
RaiseException(0xE00DEAD0, EXCEPTION_NONCONTINUABLE, 2, args);
}
_SEH2_EXCEPT(verify_xpointers(_SEH2_GetExceptionInformation(), 0xE00DEAD0, EXCEPTION_NONCONTINUABLE, 2, args, &ret, EXCEPTION_CONTINUE_SEARCH))
{
ret = return_zero();
}
_SEH2_END;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_arg(ret);
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_xpointers_16)
{
static int ret;
static const ULONG_PTR args[] = { 1, 2, 12345 };
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
RaiseException(0xE00DEAD0, EXCEPTION_NONCONTINUABLE, 3, args);
}
_SEH2_EXCEPT(verify_xpointers(_SEH2_GetExceptionInformation(), 0xE00DEAD0, EXCEPTION_NONCONTINUABLE, 3, args, &ret, EXCEPTION_CONTINUE_SEARCH))
{
ret = return_zero();
}
_SEH2_END;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_arg(ret);
}
_SEH2_END;
return ret == return_positive();
}
//}}}
/* _SEH2_GetExceptionCode() *///{{{
static
int verify_xcode(int code, int xcode, int * ret, int filter)
{
*ret = code == xcode;
if(*ret)
*ret = return_positive();
return filter;
}
DEFINE_TEST(test_xcode_1)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_EXCEPT(verify_xcode(_SEH2_GetExceptionCode(), 0xE00DEAD0, &ret, EXCEPTION_EXECUTE_HANDLER))
{
ret = return_arg(ret);
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_xcode_2)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_arg(ret);
}
_SEH2_EXCEPT(verify_xcode(_SEH2_GetExceptionCode(), 0xE00DEAD0, &ret, EXCEPTION_CONTINUE_EXECUTION))
{
ret = return_zero();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_xcode_3)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_EXCEPT(verify_xcode(_SEH2_GetExceptionCode(), 0xE00DEAD0, &ret, EXCEPTION_CONTINUE_SEARCH))
{
ret = return_zero();
}
_SEH2_END;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_arg(ret);
}
_SEH2_END;
return ret == return_positive();
}
//}}}
/* _SEH2_AbnormalTermination() *///{{{
DEFINE_TEST(test_abnorm_1)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
ret = return_arg(ret);
}
_SEH2_FINALLY
{
ret = _SEH2_AbnormalTermination() ? return_zero() : return_positive();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_abnorm_2)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
_SEH2_LEAVE;
}
_SEH2_FINALLY
{
ret = _SEH2_AbnormalTermination() ? return_zero() : return_positive();
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_abnorm_3)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
_SEH2_YIELD(goto leave);
}
_SEH2_FINALLY
{
ret = _SEH2_AbnormalTermination() ? return_positive() : return_zero();
}
_SEH2_END;
leave:
return ret == return_positive();
}
DEFINE_TEST(test_abnorm_4)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_FINALLY
{
ret = _SEH2_AbnormalTermination() ? return_positive() : return_zero();
}
_SEH2_END;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_arg(ret);
}
_SEH2_END;
return ret == return_positive();
}
DEFINE_TEST(test_abnorm_5)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
ret = return_arg(ret);
}
_SEH2_FINALLY
{
ret = _SEH2_AbnormalTermination() ? return_zero() : return_positive();
}
_SEH2_END;
}
_SEH2_FINALLY
{
ret = ret == return_positive() && !_SEH2_AbnormalTermination() ? return_positive() + return_one() : ret;
}
_SEH2_END;
return ret == return_positive() + return_one();
}
DEFINE_TEST(test_abnorm_6)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
_SEH2_LEAVE;
}
_SEH2_FINALLY
{
ret = _SEH2_AbnormalTermination() ? return_zero() : return_positive();
}
_SEH2_END;
}
_SEH2_FINALLY
{
ret = ret == return_positive() && !_SEH2_AbnormalTermination() ? return_positive() + return_one() : ret;
}
_SEH2_END;
return ret == return_positive() + return_one();
}
DEFINE_TEST(test_abnorm_7)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
_SEH2_YIELD(goto leave);
}
_SEH2_FINALLY
{
ret = _SEH2_AbnormalTermination() ? return_positive() : return_zero();
}
_SEH2_END;
}
_SEH2_FINALLY
{
ret = ret == return_positive() && _SEH2_AbnormalTermination() ? return_positive() + return_one() : ret;
}
_SEH2_END;
leave:
return ret == return_positive() + return_one();
}
DEFINE_TEST(test_abnorm_8)
{
static int ret;
ret = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
ret = return_zero();
}
_SEH2_FINALLY
{
ret = _SEH2_AbnormalTermination() ? return_positive() : return_zero();
}
_SEH2_END;
}
_SEH2_FINALLY
{
ret = ret == return_positive() && _SEH2_AbnormalTermination() ? return_positive() + return_one() : ret;
}
_SEH2_END;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
ret = return_arg(ret);
}
_SEH2_END;
return ret == return_positive() + return_one();
}
//}}}
/* Use of local variables from _SEH2_EXCEPT(...) and _SEH2_FINALLY { ... } *///{{{
DEFINE_TEST(test_nested_locals_1)
{
int var1 = return_one();
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, 0);
}
_SEH2_EXCEPT((var1 = (var1 == return_one() ? return_positive() : var1)), EXCEPTION_EXECUTE_HANDLER)
{
if(var1 == return_positive())
var1 = return_positive() + 1;
}
_SEH2_END;
return var1 == return_positive() + 1;
}
DEFINE_TEST(test_nested_locals_2)
{
int var1 = return_positive();
_SEH2_TRY
{
}
_SEH2_FINALLY
{
if(var1 == return_positive())
var1 = return_positive() + 1;
}
_SEH2_END;
return var1 == return_positive() + 1;
}
DEFINE_TEST(test_nested_locals_3)
{
int var1 = return_zero();
_SEH2_TRY
{
_SEH2_TRY
{
var1 = return_one();
RaiseException(0xE00DEAD0, 0, 0, 0);
}
_SEH2_FINALLY
{
if(var1 == return_one())
var1 = return_positive();
}
_SEH2_END;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
if(var1 == return_positive())
var1 = return_positive() + 1;
}
_SEH2_END;
return var1 == return_positive() + 1;
}
//}}}
/* System support *///{{{
// TODO
//}}}
/* CPU faults *///{{{
// TODO
//}}}
/* Past bugs, to detect regressions *///{{{
/* #4004: volatile registers clobbered when catching across frames (originally misreported) *///{{{
static
void test_bug_4004_helper_1(void)
{
int i1, i2, i3;
i1 = return_positive();
i2 = return_positive();
i3 = return_positive();
(void)return_arg(i1 + i2 + i3);
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, NULL);
}
_SEH2_FINALLY
{
}
_SEH2_END;
}
static
void test_bug_4004_helper_2(void)
{
_SEH2_TRY
{
test_bug_4004_helper_1();
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
}
_SEH2_END;
}
DEFINE_TEST(test_bug_4004)
{
int i1, i2, i3;
i1 = return_positive();
i2 = return_positive();
i3 = return_positive();
test_bug_4004_helper_2();
return return_arg(i1) + return_arg(i2) + return_arg(i3) == return_positive() * 3;
}
//}}}
/* #4663: *///{{{
DEFINE_TEST(test_bug_4663)
{
int i1, i2;
i1 = return_positive();
i2 = return_positive();
_SEH2_TRY
{
_SEH2_TRY
{
RaiseException(0xE00DEAD0, 0, 0, 0);
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
if (i1 == return_positive())
{
i1 = return_positive() + 1;
}
}
_SEH2_END;
if (i1 == return_positive() + 1)
{
i1 = return_negative();
RaiseException(0xE00DEAD0, 0, 0, 0);
}
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
i2 = return_negative();
}
_SEH2_END;
return ((i1 == return_negative()) && (i2 == return_negative()));
}
//}}}
//}}}
DEFINE_TEST(test_unvolatile)
{
int val = 0;
_SEH2_TRY
{
val = return_one();
*((char*)(intptr_t)0xc0000000) = 0;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
val = val + 3;
}
_SEH2_END;
/* This works with a proper SEH implementation, but not with our hacked PSEH */
#ifdef _USE_NATIVE_SEH
return (val == 4);
#else
return (val == 4 || val == 3);
#endif
}
DEFINE_TEST(test_unvolatile_2)
{
int val = 0;
_SEH2_TRY
{
val = 1;
*((char*)(intptr_t)0xc0000000) = 0;
val = 2;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
val = val + 3;
}
_SEH2_END;
return (val == 3) || (val == 4) || (val == 5);
}
/* This test is mainly for documentation purpose. As can be seen it doesn't
provide a satisfying result. In fact the compiler could do even more
crazy things like reusing val1 between the assignment to 0 and the last
assignment to 3. This DOES happen with C++ and it's NOT a PSEH bug, but
rather an unavoidable consequence of how the compiler works.
The conclusion: Do not use assignments to a variable inside a __try block
that is being used later inside the __except block, unless it is declared
volatile! */
#ifndef __cplusplus
DEFINE_TEST(test_unvolatile_3)
{
register int val1 = 0, val2 = 0;
_SEH2_TRY
{
val1 = 1;
_SEH2_TRY
{
val2 = 1;
*((char*)(intptr_t)0xc0000000) = 0;
val2 = 2;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
val2 |= 4;
}
_SEH2_END;
val1 = 2;
*((int*)(intptr_t)0xc0000000) = 1;
val1 = 3;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
val1 = val1 * val2;
}
_SEH2_END;
/* The expected case */
if ((val1 == 10) && (val2 == 5))
return TRUE;
/* The compiler can optimize away "val1 = 1" and "val1 = 2" and
only use the last "val1 = 3", in this case val1 is still 0
when the outer exception handler kicks in */
if ((val1 == 0) && (val2 == 5))
return TRUE;
/* Same as above, but this time val2 optimized away */
if (((val1 == 8) && (val2 == 4)) ||
((val1 == 0) && (val2 == 4)))
return TRUE;
return FALSE;
}
#endif // __cplusplus
DEFINE_TEST(test_unvolatile_4)
{
unsigned result = 0xdeadbeef;
_SEH2_TRY
{
*(char*)(intptr_t)0x80000000 = 1;
}
_SEH2_EXCEPT(result == 0xdeadbeef)
{
result = 2;
}
_SEH2_END;
result = (result == 0xdeadbeef) ? 0 : result + 1;
return result == 3;
}
DEFINE_TEST(test_finally_goto)
{
volatile int val = 0;
_SEH2_TRY
{
val |= 1;
_SEH2_TRY
{
val |= 2;
goto next;
}
_SEH2_FINALLY
{
val |= 4;
*((char*)(intptr_t)0xdeadc0de) = 0;
val |= 8;
}
_SEH2_END;
val |= 16;
next:
val |= 32;
*((char*)(intptr_t)0xdeadc0de) = 0;
val |= 64;
}
_SEH2_EXCEPT(1)
{
val |= 128;
}
_SEH2_END;
return (val == (128|4|2|1));
}
DEFINE_TEST(test_nested_exception)
{
volatile int val = 0;
_SEH2_TRY
{
val |= 1;
_SEH2_TRY
{
val |= 2;
*((char*)(intptr_t)0xdeadc0de) = 0;
val |= 4;
}
_SEH2_EXCEPT(1)
{
val |= 8;
*((char*)(intptr_t)0xdeadc0de) = 0;
val |= 16;
}
_SEH2_END;
val |= 32;
*((char*)(intptr_t)0xdeadc0de) = 0;
val |= 64;
}
_SEH2_EXCEPT(1)
{
val |= 128;
}
_SEH2_END;
return (val == (1|2|8|128));
}
static
LONG WINAPI unhandled_exception(PEXCEPTION_POINTERS ExceptionInfo)
{
trace("unhandled exception %08lX thrown from %p\n", ExceptionInfo->ExceptionRecord->ExceptionCode, ExceptionInfo->ExceptionRecord->ExceptionAddress);
return EXCEPTION_CONTINUE_SEARCH;
}
#if defined(_M_IX86)
struct volatile_context
{
void * esp;
void * ebp;
void * ebx;
void * esi;
void * edi;
};
#else
struct volatile_context
{
int _ignore;
};
#endif
static
DECLSPEC_NOINLINE
int sanity_check(int ret, struct volatile_context * before, struct volatile_context * after)
{
if(ret && memcmp(before, after, sizeof(*before)))
{
trace("volatile context corrupted\n");
return 0;
}
return ret;
}
#ifndef _PSEH3_H_
static
int passthrough_handler(struct _EXCEPTION_RECORD * e, void * f, struct _CONTEXT * c, void * d)
{
return ExceptionContinueSearch;
}
#endif
static
DECLSPEC_NOINLINE
int call_test(int (* func)(void))
{
static int ret;
static struct volatile_context before, after;
static LPTOP_LEVEL_EXCEPTION_FILTER prev_unhandled_exception;
#if defined(_X86_) && !defined(_PSEH3_H_) && !defined(_MSC_VER)
static _SEH2Registration_t * prev_frame;
_SEH2Registration_t passthrough_frame;
#endif
prev_unhandled_exception = SetUnhandledExceptionFilter(&unhandled_exception);
#if defined(_X86_) && !defined(_PSEH3_H_) && !defined(_MSC_VER)
prev_frame = (_SEH2Registration_t *)__readfsdword(0);
passthrough_frame.SER_Prev = prev_frame;
passthrough_frame.SER_Handler = passthrough_handler;
__writefsdword(0, (unsigned long)&passthrough_frame);
#endif
#if defined(__GNUC__) && defined(__i386__)
__asm__ __volatile__
(
"mov %%esp, 0x00 + %c[before]\n"
"mov %%ebp, 0x04 + %c[before]\n"
"mov %%ebx, 0x08 + %c[before]\n"
"mov %%esi, 0x0c + %c[before]\n"
"mov %%edi, 0x10 + %c[before]\n"
"call *%[test]\n"
"mov %%esp, 0x00 + %c[after]\n"
"mov %%ebp, 0x04 + %c[after]\n"
"mov %%ebx, 0x08 + %c[after]\n"
"mov %%esi, 0x0c + %c[after]\n"
"mov %%edi, 0x10 + %c[after]\n"
"push %[after]\n"
"push %[before]\n"
"push %[ret]\n"
"call %c[sanity_check]\n"
"pop %%ecx\n"
"pop %%ecx\n"
"pop %%ecx\n" :
[ret] "=a" (ret) :
[test] "r" (func), [before] "i" (&before), [after] "i" (&after), [sanity_check] "i" (&sanity_check) :
"ebx", "ecx", "edx", "esi", "edi", "flags", "memory"
);
#else
ret = func();
#endif
#if defined(_X86_) && !defined(_PSEH3_H_) && !defined(_MSC_VER)
if((_SEH2Registration_t *)__readfsdword(0) != &passthrough_frame || passthrough_frame.SER_Prev != prev_frame)
{
trace("exception registration list corrupted\n");
ret = 0;
}
__writefsdword(0, (unsigned long)prev_frame);
#endif
SetUnhandledExceptionFilter(prev_unhandled_exception);
return ret;
}
DEFINE_TEST(test_PSEH3_bug)
{
volatile int count = 0;
int dummy = 0;
_SEH2_TRY
{
if (count++ == 0)
{
*(volatile int*)0x12345678 = 0x12345678;
}
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
dummy = 0;
}
_SEH2_END;
(void)dummy;
return (count == 1);
}
void
use_lots_of_stack(void)
{
int i;
volatile int arr[512];
for (i = 0; i < 512; i++)
arr[i] = 123;
(void)arr;
}
DEFINE_TEST(test_PSEH3_bug2)
{
unsigned long status = 0;
_SEH2_TRY
{
*(volatile int*)0x12345678 = 0x12345678;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
use_lots_of_stack();
status = _SEH2_GetExceptionCode();
}
_SEH2_END;
return (status == STATUS_ACCESS_VIOLATION);
}
#define USE_TEST_NAME_(NAME_) # NAME_
#define USE_TEST_NAME(NAME_) USE_TEST_NAME_(NAME_)
#define USE_TEST(NAME_) { USE_TEST_NAME(NAME_), NAME_ }
struct subtest
{
const char * name;
int (* func)(void);
};
START_TEST(pseh)
{
const struct subtest testsuite[] =
{
USE_TEST(test_empty_1),
USE_TEST(test_empty_2),
USE_TEST(test_empty_3),
USE_TEST(test_empty_4),
USE_TEST(test_empty_5),
USE_TEST(test_empty_6),
USE_TEST(test_empty_7),
USE_TEST(test_empty_8),
USE_TEST(test_execute_handler_1),
USE_TEST(test_continue_execution_1),
USE_TEST(test_continue_search_1),
USE_TEST(test_execute_handler_2),
USE_TEST(test_continue_execution_2),
USE_TEST(test_execute_handler_3),
USE_TEST(test_continue_execution_3),
USE_TEST(test_continue_search_2),
USE_TEST(test_execute_handler_4),
USE_TEST(test_continue_execution_4),
USE_TEST(test_execute_handler_5),
USE_TEST(test_continue_execution_5),
USE_TEST(test_continue_search_3),
USE_TEST(test_execute_handler_6),
USE_TEST(test_continue_execution_6),
USE_TEST(test_execute_handler_7),
USE_TEST(test_continue_execution_7),
USE_TEST(test_continue_search_4),
USE_TEST(test_execute_handler_8),
USE_TEST(test_continue_execution_8),
USE_TEST(test_execute_handler_9),
USE_TEST(test_continue_execution_9),
USE_TEST(test_continue_search_5),
USE_TEST(test_execute_handler_10),
USE_TEST(test_continue_execution_10),
USE_TEST(test_execute_handler_11),
USE_TEST(test_continue_execution_11),
USE_TEST(test_continue_search_6),
USE_TEST(test_execute_handler_12),
USE_TEST(test_continue_execution_12),
USE_TEST(test_leave_1),
USE_TEST(test_leave_2),
USE_TEST(test_leave_3),
USE_TEST(test_leave_4),
USE_TEST(test_leave_5),
USE_TEST(test_leave_6),
USE_TEST(test_yield_1),
USE_TEST(test_yield_2),
USE_TEST(test_yield_3),
USE_TEST(test_yield_4),
USE_TEST(test_yield_5),
USE_TEST(test_yield_6),
USE_TEST(test_finally_1),
USE_TEST(test_finally_2),
USE_TEST(test_finally_3),
USE_TEST(test_finally_4),
USE_TEST(test_finally_5),
USE_TEST(test_finally_6),
USE_TEST(test_finally_7),
USE_TEST(test_finally_8),
USE_TEST(test_finally_9),
USE_TEST(test_finally_10),
USE_TEST(test_finally_11),
USE_TEST(test_finally_12),
USE_TEST(test_finally_13),
USE_TEST(test_finally_14),
USE_TEST(test_xpointers_1),
USE_TEST(test_xpointers_2),
USE_TEST(test_xpointers_3),
USE_TEST(test_xpointers_4),
USE_TEST(test_xpointers_5),
USE_TEST(test_xpointers_6),
USE_TEST(test_xpointers_7),
USE_TEST(test_xpointers_8),
USE_TEST(test_xpointers_9),
USE_TEST(test_xpointers_10),
USE_TEST(test_xpointers_11),
USE_TEST(test_xpointers_12),
USE_TEST(test_xpointers_13),
USE_TEST(test_xpointers_14),
USE_TEST(test_xpointers_15),
USE_TEST(test_xpointers_16),
USE_TEST(test_xcode_1),
USE_TEST(test_xcode_2),
USE_TEST(test_xcode_3),
USE_TEST(test_abnorm_1),
USE_TEST(test_abnorm_2),
USE_TEST(test_abnorm_3),
USE_TEST(test_abnorm_4),
USE_TEST(test_abnorm_5),
USE_TEST(test_abnorm_6),
USE_TEST(test_abnorm_7),
USE_TEST(test_abnorm_8),
USE_TEST(test_nested_locals_1),
USE_TEST(test_nested_locals_2),
USE_TEST(test_nested_locals_3),
USE_TEST(test_bug_4004),
USE_TEST(test_bug_4663),
USE_TEST(test_unvolatile),
USE_TEST(test_unvolatile_2),
#ifndef __cplusplus
USE_TEST(test_unvolatile_3),
#endif
USE_TEST(test_unvolatile_4),
USE_TEST(test_finally_goto),
USE_TEST(test_nested_exception),
USE_TEST(test_PSEH3_bug),
USE_TEST(test_PSEH3_bug2),
};
size_t i;
for(i = 0; i < sizeof(testsuite) / sizeof(testsuite[0]); ++ i)
ok(call_test(testsuite[i].func), "%s failed\n", testsuite[i].name);
}
/* EOF */