Loading...
1// SPDX-License-Identifier: GPL-2.0
2#include <api/fd/array.h>
3#include <poll.h>
4#include "util/debug.h"
5#include "tests/tests.h"
6
7static void fdarray__init_revents(struct fdarray *fda, short revents)
8{
9 int fd;
10
11 fda->nr = fda->nr_alloc;
12
13 for (fd = 0; fd < fda->nr; ++fd) {
14 fda->entries[fd].fd = fda->nr - fd;
15 fda->entries[fd].revents = revents;
16 }
17}
18
19static int fdarray__fprintf_prefix(struct fdarray *fda, const char *prefix, FILE *fp)
20{
21 int printed = 0;
22
23 if (verbose <= 0)
24 return 0;
25
26 printed += fprintf(fp, "\n%s: ", prefix);
27 return printed + fdarray__fprintf(fda, fp);
28}
29
30int test__fdarray__filter(struct test *test __maybe_unused, int subtest __maybe_unused)
31{
32 int nr_fds, expected_fd[2], fd, err = TEST_FAIL;
33 struct fdarray *fda = fdarray__new(5, 5);
34
35 if (fda == NULL) {
36 pr_debug("\nfdarray__new() failed!");
37 goto out;
38 }
39
40 fdarray__init_revents(fda, POLLIN);
41 nr_fds = fdarray__filter(fda, POLLHUP, NULL, NULL);
42 if (nr_fds != fda->nr_alloc) {
43 pr_debug("\nfdarray__filter()=%d != %d shouldn't have filtered anything",
44 nr_fds, fda->nr_alloc);
45 goto out_delete;
46 }
47
48 fdarray__init_revents(fda, POLLHUP);
49 nr_fds = fdarray__filter(fda, POLLHUP, NULL, NULL);
50 if (nr_fds != 0) {
51 pr_debug("\nfdarray__filter()=%d != %d, should have filtered all fds",
52 nr_fds, fda->nr_alloc);
53 goto out_delete;
54 }
55
56 fdarray__init_revents(fda, POLLHUP);
57 fda->entries[2].revents = POLLIN;
58 expected_fd[0] = fda->entries[2].fd;
59
60 pr_debug("\nfiltering all but fda->entries[2]:");
61 fdarray__fprintf_prefix(fda, "before", stderr);
62 nr_fds = fdarray__filter(fda, POLLHUP, NULL, NULL);
63 fdarray__fprintf_prefix(fda, " after", stderr);
64 if (nr_fds != 1) {
65 pr_debug("\nfdarray__filter()=%d != 1, should have left just one event", nr_fds);
66 goto out_delete;
67 }
68
69 if (fda->entries[0].fd != expected_fd[0]) {
70 pr_debug("\nfda->entries[0].fd=%d != %d\n",
71 fda->entries[0].fd, expected_fd[0]);
72 goto out_delete;
73 }
74
75 fdarray__init_revents(fda, POLLHUP);
76 fda->entries[0].revents = POLLIN;
77 expected_fd[0] = fda->entries[0].fd;
78 fda->entries[3].revents = POLLIN;
79 expected_fd[1] = fda->entries[3].fd;
80
81 pr_debug("\nfiltering all but (fda->entries[0], fda->entries[3]):");
82 fdarray__fprintf_prefix(fda, "before", stderr);
83 nr_fds = fdarray__filter(fda, POLLHUP, NULL, NULL);
84 fdarray__fprintf_prefix(fda, " after", stderr);
85 if (nr_fds != 2) {
86 pr_debug("\nfdarray__filter()=%d != 2, should have left just two events",
87 nr_fds);
88 goto out_delete;
89 }
90
91 for (fd = 0; fd < 2; ++fd) {
92 if (fda->entries[fd].fd != expected_fd[fd]) {
93 pr_debug("\nfda->entries[%d].fd=%d != %d\n", fd,
94 fda->entries[fd].fd, expected_fd[fd]);
95 goto out_delete;
96 }
97 }
98
99 pr_debug("\n");
100
101 err = 0;
102out_delete:
103 fdarray__delete(fda);
104out:
105 return err;
106}
107
108int test__fdarray__add(struct test *test __maybe_unused, int subtest __maybe_unused)
109{
110 int err = TEST_FAIL;
111 struct fdarray *fda = fdarray__new(2, 2);
112
113 if (fda == NULL) {
114 pr_debug("\nfdarray__new() failed!");
115 goto out;
116 }
117
118#define FDA_CHECK(_idx, _fd, _revents) \
119 if (fda->entries[_idx].fd != _fd) { \
120 pr_debug("\n%d: fda->entries[%d](%d) != %d!", \
121 __LINE__, _idx, fda->entries[1].fd, _fd); \
122 goto out_delete; \
123 } \
124 if (fda->entries[_idx].events != (_revents)) { \
125 pr_debug("\n%d: fda->entries[%d].revents(%d) != %d!", \
126 __LINE__, _idx, fda->entries[_idx].fd, _revents); \
127 goto out_delete; \
128 }
129
130#define FDA_ADD(_idx, _fd, _revents, _nr) \
131 if (fdarray__add(fda, _fd, _revents) < 0) { \
132 pr_debug("\n%d: fdarray__add(fda, %d, %d) failed!", \
133 __LINE__,_fd, _revents); \
134 goto out_delete; \
135 } \
136 if (fda->nr != _nr) { \
137 pr_debug("\n%d: fdarray__add(fda, %d, %d)=%d != %d", \
138 __LINE__,_fd, _revents, fda->nr, _nr); \
139 goto out_delete; \
140 } \
141 FDA_CHECK(_idx, _fd, _revents)
142
143 FDA_ADD(0, 1, POLLIN, 1);
144 FDA_ADD(1, 2, POLLERR, 2);
145
146 fdarray__fprintf_prefix(fda, "before growing array", stderr);
147
148 FDA_ADD(2, 35, POLLHUP, 3);
149
150 if (fda->entries == NULL) {
151 pr_debug("\nfdarray__add(fda, 35, POLLHUP) should have allocated fda->pollfd!");
152 goto out_delete;
153 }
154
155 fdarray__fprintf_prefix(fda, "after 3rd add", stderr);
156
157 FDA_ADD(3, 88, POLLIN | POLLOUT, 4);
158
159 fdarray__fprintf_prefix(fda, "after 4th add", stderr);
160
161 FDA_CHECK(0, 1, POLLIN);
162 FDA_CHECK(1, 2, POLLERR);
163 FDA_CHECK(2, 35, POLLHUP);
164 FDA_CHECK(3, 88, POLLIN | POLLOUT);
165
166#undef FDA_ADD
167#undef FDA_CHECK
168
169 pr_debug("\n");
170
171 err = 0;
172out_delete:
173 fdarray__delete(fda);
174out:
175 return err;
176}
1#include <api/fd/array.h>
2#include "util/debug.h"
3#include "tests/tests.h"
4
5static void fdarray__init_revents(struct fdarray *fda, short revents)
6{
7 int fd;
8
9 fda->nr = fda->nr_alloc;
10
11 for (fd = 0; fd < fda->nr; ++fd) {
12 fda->entries[fd].fd = fda->nr - fd;
13 fda->entries[fd].revents = revents;
14 }
15}
16
17static int fdarray__fprintf_prefix(struct fdarray *fda, const char *prefix, FILE *fp)
18{
19 int printed = 0;
20
21 if (!verbose)
22 return 0;
23
24 printed += fprintf(fp, "\n%s: ", prefix);
25 return printed + fdarray__fprintf(fda, fp);
26}
27
28int test__fdarray__filter(int subtest __maybe_unused)
29{
30 int nr_fds, expected_fd[2], fd, err = TEST_FAIL;
31 struct fdarray *fda = fdarray__new(5, 5);
32
33 if (fda == NULL) {
34 pr_debug("\nfdarray__new() failed!");
35 goto out;
36 }
37
38 fdarray__init_revents(fda, POLLIN);
39 nr_fds = fdarray__filter(fda, POLLHUP, NULL);
40 if (nr_fds != fda->nr_alloc) {
41 pr_debug("\nfdarray__filter()=%d != %d shouldn't have filtered anything",
42 nr_fds, fda->nr_alloc);
43 goto out_delete;
44 }
45
46 fdarray__init_revents(fda, POLLHUP);
47 nr_fds = fdarray__filter(fda, POLLHUP, NULL);
48 if (nr_fds != 0) {
49 pr_debug("\nfdarray__filter()=%d != %d, should have filtered all fds",
50 nr_fds, fda->nr_alloc);
51 goto out_delete;
52 }
53
54 fdarray__init_revents(fda, POLLHUP);
55 fda->entries[2].revents = POLLIN;
56 expected_fd[0] = fda->entries[2].fd;
57
58 pr_debug("\nfiltering all but fda->entries[2]:");
59 fdarray__fprintf_prefix(fda, "before", stderr);
60 nr_fds = fdarray__filter(fda, POLLHUP, NULL);
61 fdarray__fprintf_prefix(fda, " after", stderr);
62 if (nr_fds != 1) {
63 pr_debug("\nfdarray__filter()=%d != 1, should have left just one event", nr_fds);
64 goto out_delete;
65 }
66
67 if (fda->entries[0].fd != expected_fd[0]) {
68 pr_debug("\nfda->entries[0].fd=%d != %d\n",
69 fda->entries[0].fd, expected_fd[0]);
70 goto out_delete;
71 }
72
73 fdarray__init_revents(fda, POLLHUP);
74 fda->entries[0].revents = POLLIN;
75 expected_fd[0] = fda->entries[0].fd;
76 fda->entries[3].revents = POLLIN;
77 expected_fd[1] = fda->entries[3].fd;
78
79 pr_debug("\nfiltering all but (fda->entries[0], fda->entries[3]):");
80 fdarray__fprintf_prefix(fda, "before", stderr);
81 nr_fds = fdarray__filter(fda, POLLHUP, NULL);
82 fdarray__fprintf_prefix(fda, " after", stderr);
83 if (nr_fds != 2) {
84 pr_debug("\nfdarray__filter()=%d != 2, should have left just two events",
85 nr_fds);
86 goto out_delete;
87 }
88
89 for (fd = 0; fd < 2; ++fd) {
90 if (fda->entries[fd].fd != expected_fd[fd]) {
91 pr_debug("\nfda->entries[%d].fd=%d != %d\n", fd,
92 fda->entries[fd].fd, expected_fd[fd]);
93 goto out_delete;
94 }
95 }
96
97 pr_debug("\n");
98
99 err = 0;
100out_delete:
101 fdarray__delete(fda);
102out:
103 return err;
104}
105
106int test__fdarray__add(int subtest __maybe_unused)
107{
108 int err = TEST_FAIL;
109 struct fdarray *fda = fdarray__new(2, 2);
110
111 if (fda == NULL) {
112 pr_debug("\nfdarray__new() failed!");
113 goto out;
114 }
115
116#define FDA_CHECK(_idx, _fd, _revents) \
117 if (fda->entries[_idx].fd != _fd) { \
118 pr_debug("\n%d: fda->entries[%d](%d) != %d!", \
119 __LINE__, _idx, fda->entries[1].fd, _fd); \
120 goto out_delete; \
121 } \
122 if (fda->entries[_idx].events != (_revents)) { \
123 pr_debug("\n%d: fda->entries[%d].revents(%d) != %d!", \
124 __LINE__, _idx, fda->entries[_idx].fd, _revents); \
125 goto out_delete; \
126 }
127
128#define FDA_ADD(_idx, _fd, _revents, _nr) \
129 if (fdarray__add(fda, _fd, _revents) < 0) { \
130 pr_debug("\n%d: fdarray__add(fda, %d, %d) failed!", \
131 __LINE__,_fd, _revents); \
132 goto out_delete; \
133 } \
134 if (fda->nr != _nr) { \
135 pr_debug("\n%d: fdarray__add(fda, %d, %d)=%d != %d", \
136 __LINE__,_fd, _revents, fda->nr, _nr); \
137 goto out_delete; \
138 } \
139 FDA_CHECK(_idx, _fd, _revents)
140
141 FDA_ADD(0, 1, POLLIN, 1);
142 FDA_ADD(1, 2, POLLERR, 2);
143
144 fdarray__fprintf_prefix(fda, "before growing array", stderr);
145
146 FDA_ADD(2, 35, POLLHUP, 3);
147
148 if (fda->entries == NULL) {
149 pr_debug("\nfdarray__add(fda, 35, POLLHUP) should have allocated fda->pollfd!");
150 goto out_delete;
151 }
152
153 fdarray__fprintf_prefix(fda, "after 3rd add", stderr);
154
155 FDA_ADD(3, 88, POLLIN | POLLOUT, 4);
156
157 fdarray__fprintf_prefix(fda, "after 4th add", stderr);
158
159 FDA_CHECK(0, 1, POLLIN);
160 FDA_CHECK(1, 2, POLLERR);
161 FDA_CHECK(2, 35, POLLHUP);
162 FDA_CHECK(3, 88, POLLIN | POLLOUT);
163
164#undef FDA_ADD
165#undef FDA_CHECK
166
167 pr_debug("\n");
168
169 err = 0;
170out_delete:
171 fdarray__delete(fda);
172out:
173 return err;
174}