mirror of
https://github.com/frida/tinycc
synced 2024-12-23 05:26:49 +03:00
f85b1e393f
tcc would reject e.g., void f(){ struct {_Bool x:_Generic(({0;}),default:1);} my_x; } with `expected constant`. This patch makes it accept it. (The patch also makes tcc's _Generic a little more "generic" than that of gcc and clang in that that tcc now also accepts `struct {_Bool x:_Generic(({0;}),default:1);} my_x;` in file scope while gcc and clang don't, but I think there's no harm in that and gcc and clang might as well accept it in filescope too, given that they have no problem with e.g., `/*filescope:*/int x=1, y=2, z=_Generic(x+y, int:3);`)
79 lines
1.6 KiB
C
79 lines
1.6 KiB
C
#include <stdio.h>
|
|
|
|
const int a = 0;
|
|
|
|
struct a {
|
|
int a;
|
|
};
|
|
|
|
struct b {
|
|
int a;
|
|
};
|
|
|
|
int a_f()
|
|
{
|
|
return 20;
|
|
}
|
|
|
|
int b_f()
|
|
{
|
|
return 10;
|
|
}
|
|
|
|
typedef int (*fptr)(int);
|
|
int foo(int i)
|
|
{
|
|
return i;
|
|
}
|
|
|
|
typedef int int_type1;
|
|
|
|
#define gen_sw(a) _Generic(a, const char *: 1, default: 8, int: 123);
|
|
|
|
int main()
|
|
{
|
|
int i = 0;
|
|
signed long int l = 2;
|
|
struct b titi;
|
|
const int * const ptr;
|
|
const char *ti;
|
|
int_type1 i2;
|
|
|
|
i = _Generic(a, int: a_f, const int: b_f)();
|
|
printf("%d\n", i);
|
|
i = _Generic(a, int: a_f() / 2, const int: b_f() / 2);
|
|
printf("%d\n", i);
|
|
i = _Generic(ptr, int *:1, int * const:2, default:20);
|
|
printf("%d\n", i);
|
|
i = gen_sw(a);
|
|
printf("%d\n", i);
|
|
i = _Generic(titi, struct a:1, struct b:2, default:20);
|
|
printf("%d\n", i);
|
|
i = _Generic(i2, char: 1, int : 0);
|
|
printf("%d\n", i);
|
|
i = _Generic(a, char:1, int[4]:2, default:5);
|
|
printf("%d\n", i);
|
|
i = _Generic(17, int :1, int **:2);
|
|
printf("%d\n", i);
|
|
i = _Generic(17L, int :1, long :2, long long : 3);
|
|
printf("%d\n", i);
|
|
i = _Generic("17, io", char *: 3, const char *: 1);
|
|
printf("%d\n", i);
|
|
i = _Generic(ti, const unsigned char *:1, const char *:4, char *:3,
|
|
const signed char *:2);
|
|
printf("%d\n", i);
|
|
printf("%s\n", _Generic(i + 2L, long: "long", int: "int",
|
|
long long: "long long"));
|
|
i = _Generic(l, long: 1, int: 2);
|
|
printf("%d\n", i);
|
|
i = _Generic(foo, fptr: 3, int: 4);
|
|
printf("%d\n", i);
|
|
|
|
(void)_Generic((int(*)[2]){0}, int(*)[2]:0, int(*)[4]:0); //shouldn't match twice
|
|
|
|
//should accept ({ }) in the controlling expr of _Generic even in const_wanted contexts
|
|
struct { _Bool x_0: _Generic(({0;}),default:1); } my_x;
|
|
|
|
return 0;
|
|
}
|