source: mainline/uspace/lib/ddev/test/ddev.c@ 85b41bc

ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since 85b41bc was 4c6fd56, checked in by Jiri Svoboda <jiri@…>, 22 months ago

loc_server_register() should be callable more than once (API only)

Now loc_server_register() returns a pointer to a loc_srv_t object,
that is then passed to loc_service_register() and
loc_service_add_to_cat().

Added loc_server_unregister() that unregisters the server
and frees the loc_srv_t object.

Updated all callers. The implementation, however, is a stub.
It is not actually possible to call loc_server_register() more
than once, yet.

  • Property mode set to 100644
File size: 8.2 KB
Line 
1/*
2 * Copyright (c) 2023 Jiri Svoboda
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 *
9 * - Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * - Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * - The name of the author may not be used to endorse or promote products
15 * derived from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 */
28
29#include <async.h>
30#include <errno.h>
31#include <ddev.h>
32#include <ddev_srv.h>
33#include <fibril_synch.h>
34#include <gfx/color.h>
35#include <gfx/context.h>
36#include <gfx/render.h>
37#include <ipcgfx/server.h>
38#include <loc.h>
39#include <pcut/pcut.h>
40
41PCUT_INIT;
42
43PCUT_TEST_SUITE(ddev);
44
45static const char *test_ddev_server = "test-ddev";
46static const char *test_ddev_svc = "test/ddev";
47
48static void test_ddev_conn(ipc_call_t *, void *);
49
50static errno_t test_get_gc(void *, sysarg_t *, sysarg_t *);
51static errno_t test_get_info(void *, ddev_info_t *);
52static errno_t test_gc_set_color(void *, gfx_color_t *);
53
54static ddev_ops_t test_ddev_ops = {
55 .get_gc = test_get_gc,
56 .get_info = test_get_info
57};
58
59static gfx_context_ops_t test_gc_ops = {
60 .set_color = test_gc_set_color
61};
62
63/** Describes to the server how to respond to our request and pass tracking
64 * data back to the client.
65 */
66typedef struct {
67 errno_t rc;
68 bool set_color_called;
69 ddev_srv_t *srv;
70 ddev_info_t info;
71} test_response_t;
72
73/** ddev_open(), ddev_close() work for valid display device service */
74PCUT_TEST(open_close)
75{
76 errno_t rc;
77 service_id_t sid;
78 ddev_t *ddev = NULL;
79 test_response_t resp;
80 loc_srv_t *srv;
81
82 async_set_fallback_port_handler(test_ddev_conn, &resp);
83
84 // FIXME This causes this test to be non-reentrant!
85 rc = loc_server_register(test_ddev_server, &srv);
86 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
87
88 rc = loc_service_register(srv, test_ddev_svc, &sid);
89 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
90
91 rc = ddev_open(test_ddev_svc, &ddev);
92 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
93 PCUT_ASSERT_NOT_NULL(ddev);
94
95 ddev_close(ddev);
96 rc = loc_service_unregister(srv, sid);
97 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
98 loc_server_unregister(srv);
99}
100
101/** ddev_get_gc with server returning failure */
102PCUT_TEST(dev_get_gc_failure)
103{
104 errno_t rc;
105 service_id_t sid;
106 ddev_t *ddev = NULL;
107 test_response_t resp;
108 gfx_context_t *gc;
109 loc_srv_t *srv;
110
111 async_set_fallback_port_handler(test_ddev_conn, &resp);
112
113 // FIXME This causes this test to be non-reentrant!
114 rc = loc_server_register(test_ddev_server, &srv);
115 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
116
117 rc = loc_service_register(srv, test_ddev_svc, &sid);
118 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
119
120 rc = ddev_open(test_ddev_svc, &ddev);
121 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
122 PCUT_ASSERT_NOT_NULL(ddev);
123
124 gc = NULL;
125 resp.rc = ENOMEM;
126 rc = ddev_get_gc(ddev, &gc);
127 PCUT_ASSERT_ERRNO_VAL(ENOMEM, rc);
128 PCUT_ASSERT_NULL(gc);
129
130 ddev_close(ddev);
131 rc = loc_service_unregister(srv, sid);
132 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
133 loc_server_unregister(srv);
134}
135
136/** ddev_get_gc with server returning success */
137PCUT_TEST(dev_get_gc_success)
138{
139 errno_t rc;
140 service_id_t sid;
141 ddev_t *ddev = NULL;
142 test_response_t resp;
143 gfx_context_t *gc;
144 gfx_color_t *color;
145 loc_srv_t *srv;
146
147 async_set_fallback_port_handler(test_ddev_conn, &resp);
148
149 // FIXME This causes this test to be non-reentrant!
150 rc = loc_server_register(test_ddev_server, &srv);
151 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
152
153 rc = loc_service_register(srv, test_ddev_svc, &sid);
154 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
155
156 rc = ddev_open(test_ddev_svc, &ddev);
157 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
158 PCUT_ASSERT_NOT_NULL(ddev);
159
160 resp.rc = EOK;
161 gc = NULL;
162 rc = ddev_get_gc(ddev, &gc);
163 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
164 PCUT_ASSERT_NOT_NULL(gc);
165
166 rc = gfx_color_new_rgb_i16(0, 0, 0, &color);
167 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
168
169 resp.set_color_called = false;
170 rc = gfx_set_color(gc, color);
171 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
172 PCUT_ASSERT_TRUE(resp.set_color_called);
173
174 gfx_color_delete(color);
175
176 ddev_close(ddev);
177 rc = loc_service_unregister(srv, sid);
178 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
179 loc_server_unregister(srv);
180}
181
182/** ddev_get_info with server returning failure */
183PCUT_TEST(dev_get_info_failure)
184{
185 errno_t rc;
186 service_id_t sid;
187 ddev_t *ddev = NULL;
188 test_response_t resp;
189 ddev_info_t info;
190 loc_srv_t *srv;
191
192 async_set_fallback_port_handler(test_ddev_conn, &resp);
193
194 // FIXME This causes this test to be non-reentrant!
195 rc = loc_server_register(test_ddev_server, &srv);
196 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
197
198 rc = loc_service_register(srv, test_ddev_svc, &sid);
199 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
200
201 rc = ddev_open(test_ddev_svc, &ddev);
202 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
203 PCUT_ASSERT_NOT_NULL(ddev);
204
205 resp.rc = ENOMEM;
206 rc = ddev_get_info(ddev, &info);
207 PCUT_ASSERT_ERRNO_VAL(resp.rc, rc);
208
209 ddev_close(ddev);
210 rc = loc_service_unregister(srv, sid);
211 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
212 loc_server_unregister(srv);
213}
214
215/** ddev_get_info with server returning success */
216PCUT_TEST(dev_get_info_success)
217{
218 errno_t rc;
219 service_id_t sid;
220 ddev_t *ddev = NULL;
221 test_response_t resp;
222 ddev_info_t info;
223 loc_srv_t *srv;
224
225 async_set_fallback_port_handler(test_ddev_conn, &resp);
226
227 // FIXME This causes this test to be non-reentrant!
228 rc = loc_server_register(test_ddev_server, &srv);
229 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
230
231 rc = loc_service_register(srv, test_ddev_svc, &sid);
232 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
233
234 rc = ddev_open(test_ddev_svc, &ddev);
235 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
236 PCUT_ASSERT_NOT_NULL(ddev);
237
238 resp.rc = EOK;
239
240 ddev_info_init(&resp.info);
241 resp.info.rect.p0.x = 1;
242 resp.info.rect.p0.y = 2;
243 resp.info.rect.p1.x = 3;
244 resp.info.rect.p1.y = 4;
245
246 rc = ddev_get_info(ddev, &info);
247 PCUT_ASSERT_ERRNO_VAL(resp.rc, rc);
248
249 PCUT_ASSERT_INT_EQUALS(resp.info.rect.p0.x, info.rect.p0.x);
250 PCUT_ASSERT_INT_EQUALS(resp.info.rect.p0.y, info.rect.p0.y);
251 PCUT_ASSERT_INT_EQUALS(resp.info.rect.p1.x, info.rect.p1.x);
252 PCUT_ASSERT_INT_EQUALS(resp.info.rect.p1.y, info.rect.p1.y);
253
254 ddev_close(ddev);
255 rc = loc_service_unregister(srv, sid);
256 PCUT_ASSERT_ERRNO_VAL(EOK, rc);
257 loc_server_unregister(srv);
258}
259
260/** Test display device connection.
261 *
262 * This is very similar to connection handler in the display server.
263 * XXX This should be folded into display_srv, if possible
264 */
265static void test_ddev_conn(ipc_call_t *icall, void *arg)
266{
267 test_response_t *resp = (test_response_t *) arg;
268 ddev_srv_t srv;
269 sysarg_t svc_id;
270 gfx_context_t *gc;
271 errno_t rc;
272
273 svc_id = ipc_get_arg2(icall);
274
275 if (svc_id != 0) {
276 /* Set up protocol structure */
277 ddev_srv_initialize(&srv);
278 srv.ops = &test_ddev_ops;
279 srv.arg = arg;
280 resp->srv = &srv;
281
282 /* Handle connection */
283 ddev_conn(icall, &srv);
284
285 resp->srv = NULL;
286 } else {
287 if (resp->rc != EOK) {
288 async_answer_0(icall, resp->rc);
289 return;
290 }
291
292 rc = gfx_context_new(&test_gc_ops, arg, &gc);
293 if (rc != EOK) {
294 async_answer_0(icall, ENOMEM);
295 return;
296 }
297
298 /* GC connection */
299 gc_conn(icall, gc);
300 }
301}
302
303static errno_t test_get_gc(void *arg, sysarg_t *arg2, sysarg_t *arg3)
304{
305 *arg2 = 0;
306 *arg3 = 42;
307 return EOK;
308}
309
310static errno_t test_get_info(void *arg, ddev_info_t *info)
311{
312 test_response_t *resp = (test_response_t *) arg;
313
314 if (resp->rc != EOK)
315 return resp->rc;
316
317 *info = resp->info;
318 return EOK;
319}
320
321static errno_t test_gc_set_color(void *arg, gfx_color_t *color)
322{
323 test_response_t *resp = (test_response_t *) arg;
324
325 resp->set_color_called = true;
326 return resp->rc;
327}
328
329PCUT_EXPORT(ddev);
Note: See TracBrowser for help on using the repository browser.