va_struct3.c 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. /* Area: ffi_call
  2. Purpose: Test passing struct in variable argument lists.
  3. Limitations: none.
  4. PR: none.
  5. Originator: ARM Ltd. */
  6. /* { dg-do run } */
  7. /* { dg-output "" { xfail avr32*-*-* } } */
  8. #include "ffitest.h"
  9. #include <stdarg.h>
  10. struct small_tag
  11. {
  12. unsigned char a;
  13. unsigned char b;
  14. };
  15. struct large_tag
  16. {
  17. unsigned a;
  18. unsigned b;
  19. unsigned c;
  20. unsigned d;
  21. unsigned e;
  22. };
  23. static struct large_tag
  24. test_fn (int n, ...)
  25. {
  26. va_list ap;
  27. struct small_tag s1;
  28. struct small_tag s2;
  29. struct large_tag l;
  30. va_start (ap, n);
  31. s1 = va_arg (ap, struct small_tag);
  32. l = va_arg (ap, struct large_tag);
  33. s2 = va_arg (ap, struct small_tag);
  34. printf ("%u %u %u %u %u %u %u %u %u\n", s1.a, s1.b, l.a, l.b, l.c, l.d, l.e,
  35. s2.a, s2.b);
  36. va_end (ap);
  37. l.a += s1.a;
  38. l.b += s1.b;
  39. l.c += s2.a;
  40. l.d += s2.b;
  41. return l;
  42. }
  43. int
  44. main (void)
  45. {
  46. ffi_cif cif;
  47. void* args[5];
  48. ffi_type* arg_types[5];
  49. ffi_type s_type;
  50. ffi_type *s_type_elements[3];
  51. ffi_type l_type;
  52. ffi_type *l_type_elements[6];
  53. struct small_tag s1;
  54. struct small_tag s2;
  55. struct large_tag l1;
  56. int n;
  57. struct large_tag res;
  58. s_type.size = 0;
  59. s_type.alignment = 0;
  60. s_type.type = FFI_TYPE_STRUCT;
  61. s_type.elements = s_type_elements;
  62. s_type_elements[0] = &ffi_type_uchar;
  63. s_type_elements[1] = &ffi_type_uchar;
  64. s_type_elements[2] = NULL;
  65. l_type.size = 0;
  66. l_type.alignment = 0;
  67. l_type.type = FFI_TYPE_STRUCT;
  68. l_type.elements = l_type_elements;
  69. l_type_elements[0] = &ffi_type_uint;
  70. l_type_elements[1] = &ffi_type_uint;
  71. l_type_elements[2] = &ffi_type_uint;
  72. l_type_elements[3] = &ffi_type_uint;
  73. l_type_elements[4] = &ffi_type_uint;
  74. l_type_elements[5] = NULL;
  75. arg_types[0] = &ffi_type_sint;
  76. arg_types[1] = &s_type;
  77. arg_types[2] = &l_type;
  78. arg_types[3] = &s_type;
  79. arg_types[4] = NULL;
  80. CHECK(ffi_prep_cif_var(&cif, FFI_DEFAULT_ABI, 1, 4, &l_type, arg_types) == FFI_OK);
  81. s1.a = 5;
  82. s1.b = 6;
  83. l1.a = 10;
  84. l1.b = 11;
  85. l1.c = 12;
  86. l1.d = 13;
  87. l1.e = 14;
  88. s2.a = 7;
  89. s2.b = 8;
  90. n = 41;
  91. args[0] = &n;
  92. args[1] = &s1;
  93. args[2] = &l1;
  94. args[3] = &s2;
  95. args[4] = NULL;
  96. ffi_call(&cif, FFI_FN(test_fn), &res, args);
  97. /* { dg-output "5 6 10 11 12 13 14 7 8" } */
  98. printf("res: %d %d %d %d %d\n", res.a, res.b, res.c, res.d, res.e);
  99. /* { dg-output "\nres: 15 17 19 21 14" } */
  100. return 0;
  101. }