racket/collects/tests/racket/foreign-test.c
Matthew Flatt bba223a9fe ffi/unsafe: allow a constant result for asynch use of a callback
A constant result for foreign-thread use of a callback allows a
callback to return without synchronizing with the Racket thread.
A constant result is thus useful when a callback's work can simply
be skipped if the callback is applied in the "wrong" OS thread.
2013-04-23 13:48:15 -06:00

259 lines
6.3 KiB
C

#include <stdlib.h>
#ifdef USE_THREAD_TEST
#include <pthread.h>
#endif
typedef unsigned char byte;
#ifdef _WIN32
#define X __declspec(dllexport)
#else
#define X
#endif
X int add1_int_int (int x) { return x + 1; }
X int add1_byte_int (byte x) { return x + 1; }
X byte add1_int_byte (int x) { return x + 1; }
X byte add1_byte_byte (byte x) { return x + 1; }
X float add1_float_float (float x) { return x + 1; }
X double add1_double_double (double x) { return x + 1; }
X int decimal_int_int_int (int x, int y) { return 10*x + y; }
X int decimal_byte_int_int (byte x, int y) { return 10*x + y; }
X int decimal_int_byte_int (int x, byte y) { return 10*x + y; }
X int decimal_byte_byte_int (byte x, byte y) { return 10*x + y; }
X byte decimal_int_int_byte (int x, int y) { return 10*x + y; }
X byte decimal_byte_int_byte (byte x, int y) { return 10*x + y; }
X byte decimal_int_byte_byte (int x, byte y) { return 10*x + y; }
X byte decimal_byte_byte_byte (byte x, byte y) { return 10*x + y; }
X int callback3_int_int_int (int(*f)(int)) { if (f) return f(3); else return 79; }
X int callback3_byte_int_int (int(*f)(byte)) { return f(3); }
X int callback3_short_int_int (int(*f)(short)) { return f(3); }
X int callback3_int_byte_int (byte(*f)(int)) { return f(3); }
X int callback3_int_short_int (short(*f)(int)) { return f(3); }
X int callback3_byte_byte_int (byte(*f)(byte)) { return f(3); }
X int callback3_short_short_int (short(*f)(short)) { return f(3); }
X byte callback3_int_int_byte (int(*f)(int)) { return f(3); }
X short callback3_int_int_short (int(*f)(int)) { return f(3); }
X byte callback3_byte_int_byte (int(*f)(byte)) { return f(3); }
X short callback3_short_int_short (int(*f)(short)) { return f(3); }
X byte callback3_int_byte_byte (byte(*f)(int)) { return f(3); }
X short callback3_int_short_short (short(*f)(int)) { return f(3); }
X byte callback3_byte_byte_byte (byte(*f)(byte)) { return f(3); }
X short callback3_short_short_short(short(*f)(short)) { return f(3); }
X float callback3_float_float_float(float(*f)(float)) { return f(3.0); }
X double callback3_double_double_double(double(*f)(double)) { return f(3.0); }
X int g1;
X int curry_ret_int_int (int x) { return g1 + x; }
X int curry_ret_byte_int (byte x) { return g1 + x; }
X byte curry_ret_int_byte (int x) { return g1 + x; }
X byte curry_ret_byte_byte (byte x) { return g1 + x; }
X void* curry_int_int_int (int x) { g1 = x; return &curry_ret_int_int; }
X void* curry_byte_int_int (byte x) { g1 = x; return &curry_ret_int_int; }
X void* curry_int_byte_int (int x) { g1 = x; return &curry_ret_byte_int; }
X void* curry_byte_byte_int (byte x) { g1 = x; return &curry_ret_byte_int; }
X void* curry_int_int_byte (int x) { g1 = x; return &curry_ret_int_byte; }
X void* curry_byte_int_byte (byte x) { g1 = x; return &curry_ret_int_byte; }
X void* curry_int_byte_byte (int x) { g1 = x; return &curry_ret_byte_byte; }
X void* curry_byte_byte_byte(byte x) { g1 = x; return &curry_ret_byte_byte; }
X int g2;
X int ho_return(int x) { return g2 + x; }
X void* ho(int(*f)(int), int x) { g2 = f(x); return ho_return; }
X void *g3 = NULL;
X int use_g3(int x) { return ((int(*)(int))g3)(x); }
/* typedef int(*int2int)(int); */
/* typedef int2int(*int_to_int2int)(int); */
/* int hoho(int x, int_to_int2int f) { */
X int hoho(int x, int(*(*f)(int))(int)) { return (f(x+1))(x-1); }
X int grab7th(void *p) { return ((char *)p)[7]; }
X int vec4(int x[]) { return x[0]+x[1]+x[2]+x[3]; }
typedef struct _char_int { unsigned char a; int b; } char_int;
X int charint_to_int(char_int x) { return ((int)x.a) + x.b; }
X char_int int_to_charint(int x) {
char_int result;
result.a = (unsigned char)x;
result.b = x;
return result;
}
X char_int charint_swap(char_int x) {
char_int result;
result.a = (unsigned char)x.b;
result.b = (int)x.a;
return result;
}
int(*grabbed_callback)(int) = NULL;
X void grab_callback(int(*f)(int)) { grabbed_callback = f; }
X int use_grabbed_callback(int n) { return grabbed_callback(n); }
typedef char c7_array[7];
X char* increment_c_array(c7_array c) {
int i;
for (i = 0; i < 7; i++)
c[i]++;
return c;
}
struct char7 {
char c[7];
};
struct int_char7_int {
int i1;
struct char7 c7;
int i2;
};
X struct int_char7_int increment_ic7i(struct int_char7_int v)
{
int i;
v.i1++;
for (i = 0; i < 7; i++)
v.c7.c[i]++;
v.i2++;
return v;
}
X struct int_char7_int ic7i_cb(struct int_char7_int v,
struct int_char7_int (*cb)(struct int_char7_int))
{
int i;
--v.i1;
for (i = 0; i < 7; i++)
--v.c7.c[i];
--v.i2;
v = cb(v);
v.i1++;
for (i = 0; i < 7; i++)
v.c7.c[i]++;
v.i2++;
return v;
}
X char* increment_2d_array(char c[3][7]) {
int i, j;
for (i = 0; i < 7; i++) {
for (j = 0; j < 3; j++) {
c[j][i] += (i + j);
}
}
return (char *)c;
}
X int with_2d_array_cb(void (*cb)(char c[3][7]))
{
char c[3][7];
int i, j, r;
for (i = 0; i < 3; i++) {
for (j = 0; j < 7; j++) {
c[i][j] = (i + (2 * j));
}
}
cb(c);
r = 0;
for (i = 0; i < 3; i++) {
for (j = 0; j < 7; j++) {
r += (c[i][j] - ((2 * i) + (2 * j)));
}
}
return r;
}
union borl {
char b;
long l;
};
X union borl increment_borl(int which, union borl v)
{
int i;
if (which) {
v.l++;
} else {
v.b++;
}
return v;
}
union ic7iorl {
struct int_char7_int ic7i;
long l;
};
X union ic7iorl increment_ic7iorl(int which, union ic7iorl v)
{
int i;
if (which) {
v.l++;
} else {
v.ic7i.i1++;
for (i = 0; i < 7; i++)
v.ic7i.c7.c[i]++;
v.ic7i.i2++;
}
return v;
}
#ifdef USE_THREAD_TEST
typedef void* (*test_callback_t)(void*);
typedef void (*sleep_callback_t)();
void *do_f(void *_data)
{
test_callback_t f = ((void **)_data)[0];
void *data = ((void **)_data)[1];
data = f(data);
((void **)_data)[2] = (void *)1;
return data;
}
X void* foreign_thread_callback(test_callback_t f,
void *data,
sleep_callback_t s)
{
pthread_t th;
void *r, **d;
d = malloc(3 * sizeof(void*));
d[0] = f;
d[1] = data;
d[2] = NULL;
if (pthread_create(&th, NULL, do_f, d))
return NULL;
while (!d[2]) {
s();
}
if (pthread_join(th, &r))
return NULL;
return r;
}
#endif