aboutsummaryrefslogtreecommitdiffstats
path: root/contrib/restricted/libffi/testsuite/libffi.closures/nested_struct2.c
blob: 90b7c91178fadefad47c286809dd8c5487686a76 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
/* Area:	ffi_call, closure_call 
   Purpose:	Check structure passing with different structure size. 
		Contains structs as parameter of the struct itself. 
		Sample taken from Alan Modras patch to src/prep_cif.c. 
   Limitations:	none. 
   PR:		none. 
   Originator:	<andreast@gcc.gnu.org> 20030911	 */ 
 
/* { dg-do run } */ 
#include "ffitest.h" 
 
typedef struct A { 
  unsigned long a; 
  unsigned char b; 
} A; 
 
typedef struct B { 
  struct A x; 
  unsigned char y; 
} B; 
 
B B_fn(struct A b0, struct B b1) 
{ 
  struct B result; 
 
  result.x.a = b0.a + b1.x.a; 
  result.x.b = b0.b + b1.x.b + b1.y; 
  result.y = b0.b + b1.x.b; 
 
  printf("%lu %d %lu %d %d: %lu %d %d\n", b0.a, b0.b, b1.x.a, b1.x.b, b1.y, 
	 result.x.a, result.x.b, result.y); 
 
  return result; 
} 
 
static void 
B_gn(ffi_cif* cif __UNUSED__, void* resp, void** args, 
     void* userdata __UNUSED__) 
{ 
  struct A b0; 
  struct B b1; 
 
  b0 = *(struct A*)(args[0]); 
  b1 = *(struct B*)(args[1]); 
 
  *(B*)resp = B_fn(b0, b1); 
} 
 
int main (void) 
{ 
  ffi_cif cif; 
  void *code; 
  ffi_closure *pcl = ffi_closure_alloc(sizeof(ffi_closure), &code); 
  void* args_dbl[3]; 
  ffi_type* cls_struct_fields[3]; 
  ffi_type* cls_struct_fields1[3]; 
  ffi_type cls_struct_type, cls_struct_type1; 
  ffi_type* dbl_arg_types[3]; 
 
  struct A e_dbl = { 1, 7}; 
  struct B f_dbl = {{12 , 127}, 99}; 
 
  struct B res_dbl; 
 
  cls_struct_type.size = 0; 
  cls_struct_type.alignment = 0; 
  cls_struct_type.type = FFI_TYPE_STRUCT; 
  cls_struct_type.elements = cls_struct_fields; 
 
  cls_struct_type1.size = 0; 
  cls_struct_type1.alignment = 0; 
  cls_struct_type1.type = FFI_TYPE_STRUCT; 
  cls_struct_type1.elements = cls_struct_fields1; 
 
  cls_struct_fields[0] = &ffi_type_ulong; 
  cls_struct_fields[1] = &ffi_type_uchar; 
  cls_struct_fields[2] = NULL; 
 
  cls_struct_fields1[0] = &cls_struct_type; 
  cls_struct_fields1[1] = &ffi_type_uchar; 
  cls_struct_fields1[2] = NULL; 
 
 
  dbl_arg_types[0] = &cls_struct_type; 
  dbl_arg_types[1] = &cls_struct_type1; 
  dbl_arg_types[2] = NULL; 
 
  CHECK(ffi_prep_cif(&cif, FFI_DEFAULT_ABI, 2, &cls_struct_type1, 
		     dbl_arg_types) == FFI_OK); 
 
  args_dbl[0] = &e_dbl; 
  args_dbl[1] = &f_dbl; 
  args_dbl[2] = NULL; 
 
  ffi_call(&cif, FFI_FN(B_fn), &res_dbl, args_dbl); 
  /* { dg-output "1 7 12 127 99: 13 233 134" } */ 
  CHECK( res_dbl.x.a == (e_dbl.a + f_dbl.x.a)); 
  CHECK( res_dbl.x.b == (e_dbl.b + f_dbl.x.b + f_dbl.y)); 
  CHECK( res_dbl.y == (e_dbl.b + f_dbl.x.b)); 
 
  CHECK(ffi_prep_closure_loc(pcl, &cif, B_gn, NULL, code) == FFI_OK); 
 
  res_dbl = ((B(*)(A, B))(code))(e_dbl, f_dbl); 
  /* { dg-output "\n1 7 12 127 99: 13 233 134" } */ 
  CHECK( res_dbl.x.a == (e_dbl.a + f_dbl.x.a)); 
  CHECK( res_dbl.x.b == (e_dbl.b + f_dbl.x.b + f_dbl.y)); 
  CHECK( res_dbl.y == (e_dbl.b + f_dbl.x.b)); 
 
  exit(0); 
}