summaryrefslogtreecommitdiff
path: root/ccan/take/test/run.c
blob: 0c8ca2a1d6ae5207e76fabb249f2737327c5595b (plain)
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
#include <stdlib.h>
#include <stdbool.h>

static bool fail_realloc;
static void *my_realloc(void *p, size_t len)
{
	if (fail_realloc)
		return NULL;
	return realloc(p, len);
}
#define realloc my_realloc

#include <ccan/take/take.h>
#include <ccan/take/take.c>
#include <ccan/tap/tap.h>

static int my_allocfail_called;
static void my_allocfail(const void *p)
{
	my_allocfail_called++;
}	

static void recurse(const char *takeme, int count)
{
	if (count < 1000)
		recurse(take(strdup(takeme)), count+1);
	if (taken(takeme))
		free((char *)takeme);
}

int main(void)
{
	const char *p = "hi";

	plan_tests(43);

	/* We can take NULL. */
	ok1(take(NULL) == NULL);
	ok1(is_taken(NULL));
	ok1(taken_any());
	ok1(taken(NULL)); /* Undoes take() */
	ok1(!is_taken(NULL));
	ok1(!taken(NULL));

	/* We can take NULL twice! */
	ok1(take(NULL) == NULL);
	ok1(take(NULL) == NULL);
	ok1(is_taken(NULL));
	ok1(taken_any());
	ok1(taken(NULL)); /* Undoes take() */
	ok1(is_taken(NULL));
	ok1(taken_any());
	ok1(taken(NULL)); /* Undoes take() */
	ok1(!is_taken(NULL));
	ok1(!taken(NULL));
	ok1(!taken_any());

	/* We can take a real pointer. */
	ok1(take(p) == p);
	ok1(is_taken(p));
	ok1(taken_any());
	ok1(taken(p)); /* Undoes take() */
	ok1(!is_taken(p));
	ok1(!taken(p));
	ok1(!taken_any());

	/* Force a failure. */
	ok1(!my_allocfail_called);
	ok1(take(p) == p);
	ok1(take(p+1) == p+1);

	fail_realloc = true;
	/* Without a handler, must pass through and leak. */
	ok1(take(p+2) == p+2);
	ok1(!taken(p+2));

	/* Now, with a handler. */
	take_allocfail(my_allocfail);
	ok1(take(p+2) == NULL);

	ok1(my_allocfail_called == 1);
	ok1(taken_any());
	ok1(taken(p));
	ok1(taken(p+1));
	ok1(is_taken(NULL));
	ok1(taken(NULL));
	ok1(!taken(NULL));
	ok1(!taken_any());

	/* Test some deep nesting. */
	fail_realloc = false;
	recurse("hello", 0);
	ok1(max_taken == 1000);
	ok1(!taken_any());

	take_cleanup();
	ok1(num_taken == 0);
	ok1(max_taken == 0);
	ok1(takenarr == NULL);

	return exit_status();
}