aboutsummaryrefslogtreecommitdiffhomepage
path: root/test/external/metalang99/examples/duffs_device.c
blob: cc5ade0ed40fadc1d9453dfd01ec276bf6ed6637 (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
/**
 * Duff's device [1] is a technique to implement loop unrolling through amalgamation of a switch
 * statement with a do-while loop.
 *
 * In this example, we are going to implement automatic generation of Duff's device.
 *
 * [1]: https://en.wikipedia.org/wiki/Duff's_device
 */

#include <metalang99.h>

#include <assert.h>

#define DUFFS_DEVICE(unrolling_factor, counter_ty, count, ...)                                     \
    do {                                                                                           \
        if ((count) > 0) {                                                                         \
            counter_ty DUFFS_DEVICE_n = ((count) + ML99_DEC(unrolling_factor)) / unrolling_factor; \
            switch ((count) % unrolling_factor) {                                                  \
            case 0:                                                                                \
                do {                                                                               \
                    __VA_ARGS__                                                                    \
                    ML99_EVAL(ML99_callUneval(genCases, ML99_DEC(unrolling_factor), __VA_ARGS__))  \
                } while (--DUFFS_DEVICE_n > 0);                                                    \
            }                                                                                      \
        }                                                                                          \
    } while (0)

#define genCases_IMPL(i, ...)                                                                      \
    ML99_IF(                                                                                       \
        ML99_NAT_EQ(i, 0),                                                                         \
        ML99_empty(),                                                                              \
        ML99_TERMS(                                                                                \
            v(/* FALLTHROUGH */ case i                                                             \
              : __VA_ARGS__),                                                                      \
            ML99_callUneval(genCases, ML99_DEC(i), __VA_ARGS__)))

int main(void) {
#define ARRAY_LEN        50
#define UNROLLING_FACTOR 3

    int array[ARRAY_LEN] = { ML99_EVAL(ML99_times(v(ARRAY_LEN), v(5, ))) };
    int *n_ptr = array;

    // Square all the elements in the array.
    DUFFS_DEVICE(UNROLLING_FACTOR, int, ARRAY_LEN, {
        *n_ptr *= *n_ptr;
        n_ptr++;
    });

    for (int i = 0; i < ARRAY_LEN; i++) {
        assert(25 == array[i]);
    }
}

/*
The generated Duff's device:

int DUFFS_DEVICE_n = ((50) + 2) / 3;
switch ((50) % 3) {
case 0:
    do {
        {
            *n_ptr *= *n_ptr;
            n_ptr++;
        }
    case 2: {
        *n_ptr *= *n_ptr;
        n_ptr++;
    }
    case 1: {
        *n_ptr *= *n_ptr;
        n_ptr++;
    }
    } while (--DUFFS_DEVICE_n > 0);
}
*/