1 /*
2 * QTest testcase for VirtIO 9P
3 *
4 * Copyright (c) 2014 SUSE LINUX Products GmbH
5 *
6 * This work is licensed under the terms of the GNU GPL, version 2 or later.
7 * See the COPYING file in the top-level directory.
8 */
9
10 /*
11 * Not so fast! You might want to read the 9p developer docs first:
12 * https://wiki.qemu.org/Documentation/9p
13 */
14
15 #include "qemu/osdep.h"
16 #include "qemu/module.h"
17 #include "libqos/virtio-9p-client.h"
18
19 #define twalk(...) v9fs_twalk((TWalkOpt) __VA_ARGS__)
20 #define tversion(...) v9fs_tversion((TVersionOpt) __VA_ARGS__)
21 #define tattach(...) v9fs_tattach((TAttachOpt) __VA_ARGS__)
22 #define tgetattr(...) v9fs_tgetattr((TGetAttrOpt) __VA_ARGS__)
23 #define treaddir(...) v9fs_treaddir((TReadDirOpt) __VA_ARGS__)
24 #define tlopen(...) v9fs_tlopen((TLOpenOpt) __VA_ARGS__)
25 #define twrite(...) v9fs_twrite((TWriteOpt) __VA_ARGS__)
26 #define tflush(...) v9fs_tflush((TFlushOpt) __VA_ARGS__)
27 #define tmkdir(...) v9fs_tmkdir((TMkdirOpt) __VA_ARGS__)
28 #define tlcreate(...) v9fs_tlcreate((TlcreateOpt) __VA_ARGS__)
29 #define tsymlink(...) v9fs_tsymlink((TsymlinkOpt) __VA_ARGS__)
30 #define tlink(...) v9fs_tlink((TlinkOpt) __VA_ARGS__)
31 #define tunlinkat(...) v9fs_tunlinkat((TunlinkatOpt) __VA_ARGS__)
32
pci_config(void * obj,void * data,QGuestAllocator * t_alloc)33 static void pci_config(void *obj, void *data, QGuestAllocator *t_alloc)
34 {
35 QVirtio9P *v9p = obj;
36 v9fs_set_allocator(t_alloc);
37 size_t tag_len = qvirtio_config_readw(v9p->vdev, 0);
38 g_autofree char *tag = NULL;
39 int i;
40
41 g_assert_cmpint(tag_len, ==, strlen(MOUNT_TAG));
42
43 tag = g_malloc(tag_len);
44 for (i = 0; i < tag_len; i++) {
45 tag[i] = qvirtio_config_readb(v9p->vdev, i + 2);
46 }
47 g_assert_cmpmem(tag, tag_len, MOUNT_TAG, tag_len);
48 }
49
is_same_qid(v9fs_qid a,v9fs_qid b)50 static inline bool is_same_qid(v9fs_qid a, v9fs_qid b)
51 {
52 /* don't compare QID version for checking for file ID equalness */
53 return a[0] == b[0] && memcmp(&a[5], &b[5], 8) == 0;
54 }
55
fs_version(void * obj,void * data,QGuestAllocator * t_alloc)56 static void fs_version(void *obj, void *data, QGuestAllocator *t_alloc)
57 {
58 v9fs_set_allocator(t_alloc);
59 tversion({ .client = obj });
60 }
61
fs_attach(void * obj,void * data,QGuestAllocator * t_alloc)62 static void fs_attach(void *obj, void *data, QGuestAllocator *t_alloc)
63 {
64 v9fs_set_allocator(t_alloc);
65 tattach({ .client = obj });
66 }
67
fs_walk(void * obj,void * data,QGuestAllocator * t_alloc)68 static void fs_walk(void *obj, void *data, QGuestAllocator *t_alloc)
69 {
70 QVirtio9P *v9p = obj;
71 v9fs_set_allocator(t_alloc);
72 char *wnames[P9_MAXWELEM];
73 uint16_t nwqid;
74 g_autofree v9fs_qid *wqid = NULL;
75 int i;
76
77 for (i = 0; i < P9_MAXWELEM; i++) {
78 wnames[i] = g_strdup_printf(QTEST_V9FS_SYNTH_WALK_FILE, i);
79 }
80
81 tattach({ .client = v9p });
82 twalk({
83 .client = v9p, .fid = 0, .newfid = 1,
84 .nwname = P9_MAXWELEM, .wnames = wnames,
85 .rwalk = { .nwqid = &nwqid, .wqid = &wqid }
86 });
87
88 g_assert_cmpint(nwqid, ==, P9_MAXWELEM);
89
90 for (i = 0; i < P9_MAXWELEM; i++) {
91 g_free(wnames[i]);
92 }
93 }
94
fs_dirents_contain_name(struct V9fsDirent * e,const char * name)95 static bool fs_dirents_contain_name(struct V9fsDirent *e, const char* name)
96 {
97 for (; e; e = e->next) {
98 if (!strcmp(e->name, name)) {
99 return true;
100 }
101 }
102 return false;
103 }
104
105 /* basic readdir test where reply fits into a single response message */
fs_readdir(void * obj,void * data,QGuestAllocator * t_alloc)106 static void fs_readdir(void *obj, void *data, QGuestAllocator *t_alloc)
107 {
108 QVirtio9P *v9p = obj;
109 v9fs_set_allocator(t_alloc);
110 char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_READDIR_DIR) };
111 uint16_t nqid;
112 v9fs_qid qid;
113 uint32_t count, nentries;
114 struct V9fsDirent *entries = NULL;
115
116 tattach({ .client = v9p });
117 twalk({
118 .client = v9p, .fid = 0, .newfid = 1,
119 .nwname = 1, .wnames = wnames, .rwalk.nwqid = &nqid
120 });
121 g_assert_cmpint(nqid, ==, 1);
122
123 tlopen({
124 .client = v9p, .fid = 1, .flags = O_DIRECTORY, .rlopen.qid = &qid
125 });
126
127 /*
128 * submit count = msize - 11, because 11 is the header size of Rreaddir
129 */
130 treaddir({
131 .client = v9p, .fid = 1, .offset = 0, .count = P9_MAX_SIZE - 11,
132 .rreaddir = {
133 .count = &count, .nentries = &nentries, .entries = &entries
134 }
135 });
136
137 /*
138 * Assuming msize (P9_MAX_SIZE) is large enough so we can retrieve all
139 * dir entries with only one readdir request.
140 */
141 g_assert_cmpint(
142 nentries, ==,
143 QTEST_V9FS_SYNTH_READDIR_NFILES + 2 /* "." and ".." */
144 );
145
146 /*
147 * Check all file names exist in returned entries, ignore their order
148 * though.
149 */
150 g_assert_cmpint(fs_dirents_contain_name(entries, "."), ==, true);
151 g_assert_cmpint(fs_dirents_contain_name(entries, ".."), ==, true);
152 for (int i = 0; i < QTEST_V9FS_SYNTH_READDIR_NFILES; ++i) {
153 g_autofree char *name =
154 g_strdup_printf(QTEST_V9FS_SYNTH_READDIR_FILE, i);
155 g_assert_cmpint(fs_dirents_contain_name(entries, name), ==, true);
156 }
157
158 v9fs_free_dirents(entries);
159 g_free(wnames[0]);
160 }
161
162 /* readdir test where overall request is split over several messages */
do_readdir_split(QVirtio9P * v9p,uint32_t count)163 static void do_readdir_split(QVirtio9P *v9p, uint32_t count)
164 {
165 char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_READDIR_DIR) };
166 uint16_t nqid;
167 v9fs_qid qid;
168 uint32_t nentries, npartialentries;
169 struct V9fsDirent *entries, *tail, *partialentries;
170 int fid;
171 uint64_t offset;
172
173 tattach({ .client = v9p });
174
175 fid = 1;
176 offset = 0;
177 entries = NULL;
178 nentries = 0;
179 tail = NULL;
180
181 twalk({
182 .client = v9p, .fid = 0, .newfid = fid,
183 .nwname = 1, .wnames = wnames, .rwalk.nwqid = &nqid
184 });
185 g_assert_cmpint(nqid, ==, 1);
186
187 tlopen({
188 .client = v9p, .fid = fid, .flags = O_DIRECTORY, .rlopen.qid = &qid
189 });
190
191 /*
192 * send as many Treaddir requests as required to get all directory
193 * entries
194 */
195 while (true) {
196 npartialentries = 0;
197 partialentries = NULL;
198
199 treaddir({
200 .client = v9p, .fid = fid, .offset = offset, .count = count,
201 .rreaddir = {
202 .count = &count, .nentries = &npartialentries,
203 .entries = &partialentries
204 }
205 });
206 if (npartialentries > 0 && partialentries) {
207 if (!entries) {
208 entries = partialentries;
209 nentries = npartialentries;
210 tail = partialentries;
211 } else {
212 tail->next = partialentries;
213 nentries += npartialentries;
214 }
215 while (tail->next) {
216 tail = tail->next;
217 }
218 offset = tail->offset;
219 } else {
220 break;
221 }
222 }
223
224 g_assert_cmpint(
225 nentries, ==,
226 QTEST_V9FS_SYNTH_READDIR_NFILES + 2 /* "." and ".." */
227 );
228
229 /*
230 * Check all file names exist in returned entries, ignore their order
231 * though.
232 */
233 g_assert_cmpint(fs_dirents_contain_name(entries, "."), ==, true);
234 g_assert_cmpint(fs_dirents_contain_name(entries, ".."), ==, true);
235 for (int i = 0; i < QTEST_V9FS_SYNTH_READDIR_NFILES; ++i) {
236 char *name = g_strdup_printf(QTEST_V9FS_SYNTH_READDIR_FILE, i);
237 g_assert_cmpint(fs_dirents_contain_name(entries, name), ==, true);
238 g_free(name);
239 }
240
241 v9fs_free_dirents(entries);
242
243 g_free(wnames[0]);
244 }
245
fs_walk_no_slash(void * obj,void * data,QGuestAllocator * t_alloc)246 static void fs_walk_no_slash(void *obj, void *data, QGuestAllocator *t_alloc)
247 {
248 QVirtio9P *v9p = obj;
249 v9fs_set_allocator(t_alloc);
250 char *wnames[] = { g_strdup(" /") };
251
252 tattach({ .client = v9p });
253 twalk({
254 .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames,
255 .expectErr = ENOENT
256 });
257
258 g_free(wnames[0]);
259 }
260
fs_walk_nonexistent(void * obj,void * data,QGuestAllocator * t_alloc)261 static void fs_walk_nonexistent(void *obj, void *data, QGuestAllocator *t_alloc)
262 {
263 QVirtio9P *v9p = obj;
264 v9fs_set_allocator(t_alloc);
265
266 tattach({ .client = v9p });
267 /*
268 * The 9p2000 protocol spec says: "If the first element cannot be walked
269 * for any reason, Rerror is returned."
270 */
271 twalk({ .client = v9p, .path = "non-existent", .expectErr = ENOENT });
272 }
273
fs_walk_2nd_nonexistent(void * obj,void * data,QGuestAllocator * t_alloc)274 static void fs_walk_2nd_nonexistent(void *obj, void *data,
275 QGuestAllocator *t_alloc)
276 {
277 QVirtio9P *v9p = obj;
278 v9fs_set_allocator(t_alloc);
279 v9fs_qid root_qid;
280 uint16_t nwqid;
281 uint32_t fid;
282 g_autofree v9fs_qid *wqid = NULL;
283 g_autofree char *path = g_strdup_printf(
284 QTEST_V9FS_SYNTH_WALK_FILE "/non-existent", 0
285 );
286
287 tattach({ .client = v9p, .rattach.qid = &root_qid });
288 fid = twalk({
289 .client = v9p, .path = path,
290 .rwalk = { .nwqid = &nwqid, .wqid = &wqid }
291 }).newfid;
292 /*
293 * The 9p2000 protocol spec says: "nwqid is therefore either nwname or the
294 * index of the first elementwise walk that failed."
295 */
296 assert(nwqid == 1);
297
298 /* returned QID wqid[0] is file ID of 1st subdir */
299 g_assert(wqid && wqid[0] && !is_same_qid(root_qid, wqid[0]));
300
301 /* expect fid being unaffected by walk above */
302 tgetattr({
303 .client = v9p, .fid = fid, .request_mask = P9_GETATTR_BASIC,
304 .expectErr = ENOENT
305 });
306 }
307
fs_walk_none(void * obj,void * data,QGuestAllocator * t_alloc)308 static void fs_walk_none(void *obj, void *data, QGuestAllocator *t_alloc)
309 {
310 QVirtio9P *v9p = obj;
311 v9fs_set_allocator(t_alloc);
312 v9fs_qid root_qid;
313 g_autofree v9fs_qid *wqid = NULL;
314 struct v9fs_attr attr;
315
316 tversion({ .client = v9p });
317 tattach({
318 .client = v9p, .fid = 0, .n_uname = getuid(),
319 .rattach.qid = &root_qid
320 });
321
322 twalk({
323 .client = v9p, .fid = 0, .newfid = 1, .nwname = 0, .wnames = NULL,
324 .rwalk.wqid = &wqid
325 });
326
327 /* special case: no QID is returned if nwname=0 was sent */
328 g_assert(wqid == NULL);
329
330 tgetattr({
331 .client = v9p, .fid = 1, .request_mask = P9_GETATTR_BASIC,
332 .rgetattr.attr = &attr
333 });
334
335 g_assert(is_same_qid(root_qid, attr.qid));
336 }
337
fs_walk_dotdot(void * obj,void * data,QGuestAllocator * t_alloc)338 static void fs_walk_dotdot(void *obj, void *data, QGuestAllocator *t_alloc)
339 {
340 QVirtio9P *v9p = obj;
341 v9fs_set_allocator(t_alloc);
342 char *wnames[] = { g_strdup("..") };
343 v9fs_qid root_qid;
344 g_autofree v9fs_qid *wqid = NULL;
345
346 tversion({ .client = v9p });
347 tattach({
348 .client = v9p, .fid = 0, .n_uname = getuid(),
349 .rattach.qid = &root_qid
350 });
351
352 twalk({
353 .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames,
354 .rwalk.wqid = &wqid /* We now we'll get one qid */
355 });
356
357 g_assert_cmpmem(&root_qid, 13, wqid[0], 13);
358
359 g_free(wnames[0]);
360 }
361
fs_lopen(void * obj,void * data,QGuestAllocator * t_alloc)362 static void fs_lopen(void *obj, void *data, QGuestAllocator *t_alloc)
363 {
364 QVirtio9P *v9p = obj;
365 v9fs_set_allocator(t_alloc);
366 char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_LOPEN_FILE) };
367
368 tattach({ .client = v9p });
369 twalk({
370 .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames
371 });
372
373 tlopen({ .client = v9p, .fid = 1, .flags = O_WRONLY });
374
375 g_free(wnames[0]);
376 }
377
fs_write(void * obj,void * data,QGuestAllocator * t_alloc)378 static void fs_write(void *obj, void *data, QGuestAllocator *t_alloc)
379 {
380 QVirtio9P *v9p = obj;
381 v9fs_set_allocator(t_alloc);
382 static const uint32_t write_count = P9_MAX_SIZE / 2;
383 char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_WRITE_FILE) };
384 g_autofree char *buf = g_malloc0(write_count);
385 uint32_t count;
386
387 tattach({ .client = v9p });
388 twalk({
389 .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames
390 });
391
392 tlopen({ .client = v9p, .fid = 1, .flags = O_WRONLY });
393
394 count = twrite({
395 .client = v9p, .fid = 1, .offset = 0, .count = write_count,
396 .data = buf
397 }).count;
398 g_assert_cmpint(count, ==, write_count);
399
400 g_free(wnames[0]);
401 }
402
fs_flush_success(void * obj,void * data,QGuestAllocator * t_alloc)403 static void fs_flush_success(void *obj, void *data, QGuestAllocator *t_alloc)
404 {
405 QVirtio9P *v9p = obj;
406 v9fs_set_allocator(t_alloc);
407 char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_FLUSH_FILE) };
408 P9Req *req, *flush_req;
409 uint32_t reply_len;
410 uint8_t should_block;
411
412 tattach({ .client = v9p });
413 twalk({
414 .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames
415 });
416
417 tlopen({ .client = v9p, .fid = 1, .flags = O_WRONLY });
418
419 /* This will cause the 9p server to try to write data to the backend,
420 * until the write request gets cancelled.
421 */
422 should_block = 1;
423 req = twrite({
424 .client = v9p, .fid = 1, .offset = 0,
425 .count = sizeof(should_block), .data = &should_block,
426 .requestOnly = true
427 }).req;
428
429 flush_req = tflush({
430 .client = v9p, .oldtag = req->tag, .tag = 1, .requestOnly = true
431 }).req;
432
433 /* The write request is supposed to be flushed: the server should just
434 * mark the write request as used and reply to the flush request.
435 */
436 v9fs_req_wait_for_reply(req, &reply_len);
437 g_assert_cmpint(reply_len, ==, 0);
438 v9fs_req_free(req);
439 v9fs_rflush(flush_req);
440
441 g_free(wnames[0]);
442 }
443
fs_flush_ignored(void * obj,void * data,QGuestAllocator * t_alloc)444 static void fs_flush_ignored(void *obj, void *data, QGuestAllocator *t_alloc)
445 {
446 QVirtio9P *v9p = obj;
447 v9fs_set_allocator(t_alloc);
448 char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_FLUSH_FILE) };
449 P9Req *req, *flush_req;
450 uint32_t count;
451 uint8_t should_block;
452
453 tattach({ .client = v9p });
454 twalk({
455 .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames
456 });
457
458 tlopen({ .client = v9p, .fid = 1, .flags = O_WRONLY });
459
460 /* This will cause the write request to complete right away, before it
461 * could be actually cancelled.
462 */
463 should_block = 0;
464 req = twrite({
465 .client = v9p, .fid = 1, .offset = 0,
466 .count = sizeof(should_block), .data = &should_block,
467 .requestOnly = true
468 }).req;
469
470 flush_req = tflush({
471 .client = v9p, .oldtag = req->tag, .tag = 1, .requestOnly = true
472 }).req;
473
474 /* The write request is supposed to complete. The server should
475 * reply to the write request and the flush request.
476 */
477 v9fs_req_wait_for_reply(req, NULL);
478 v9fs_rwrite(req, &count);
479 g_assert_cmpint(count, ==, sizeof(should_block));
480 v9fs_rflush(flush_req);
481
482 g_free(wnames[0]);
483 }
484
fs_readdir_split_128(void * obj,void * data,QGuestAllocator * t_alloc)485 static void fs_readdir_split_128(void *obj, void *data,
486 QGuestAllocator *t_alloc)
487 {
488 v9fs_set_allocator(t_alloc);
489 do_readdir_split(obj, 128);
490 }
491
fs_readdir_split_256(void * obj,void * data,QGuestAllocator * t_alloc)492 static void fs_readdir_split_256(void *obj, void *data,
493 QGuestAllocator *t_alloc)
494 {
495 v9fs_set_allocator(t_alloc);
496 do_readdir_split(obj, 256);
497 }
498
fs_readdir_split_512(void * obj,void * data,QGuestAllocator * t_alloc)499 static void fs_readdir_split_512(void *obj, void *data,
500 QGuestAllocator *t_alloc)
501 {
502 v9fs_set_allocator(t_alloc);
503 do_readdir_split(obj, 512);
504 }
505
506
507 /* tests using the 9pfs 'local' fs driver */
508
fs_create_dir(void * obj,void * data,QGuestAllocator * t_alloc)509 static void fs_create_dir(void *obj, void *data, QGuestAllocator *t_alloc)
510 {
511 QVirtio9P *v9p = obj;
512 v9fs_set_allocator(t_alloc);
513 struct stat st;
514 g_autofree char *root_path = virtio_9p_test_path("");
515 g_autofree char *new_dir = virtio_9p_test_path("01");
516
517 g_assert(root_path != NULL);
518
519 tattach({ .client = v9p });
520 tmkdir({ .client = v9p, .atPath = "/", .name = "01" });
521
522 /* check if created directory really exists now ... */
523 g_assert(stat(new_dir, &st) == 0);
524 /* ... and is actually a directory */
525 g_assert((st.st_mode & S_IFMT) == S_IFDIR);
526 }
527
fs_unlinkat_dir(void * obj,void * data,QGuestAllocator * t_alloc)528 static void fs_unlinkat_dir(void *obj, void *data, QGuestAllocator *t_alloc)
529 {
530 QVirtio9P *v9p = obj;
531 v9fs_set_allocator(t_alloc);
532 struct stat st;
533 g_autofree char *root_path = virtio_9p_test_path("");
534 g_autofree char *new_dir = virtio_9p_test_path("02");
535
536 g_assert(root_path != NULL);
537
538 tattach({ .client = v9p });
539 tmkdir({ .client = v9p, .atPath = "/", .name = "02" });
540
541 /* check if created directory really exists now ... */
542 g_assert(stat(new_dir, &st) == 0);
543 /* ... and is actually a directory */
544 g_assert((st.st_mode & S_IFMT) == S_IFDIR);
545
546 tunlinkat({
547 .client = v9p, .atPath = "/", .name = "02",
548 .flags = P9_DOTL_AT_REMOVEDIR
549 });
550 /* directory should be gone now */
551 g_assert(stat(new_dir, &st) != 0);
552 }
553
fs_create_file(void * obj,void * data,QGuestAllocator * t_alloc)554 static void fs_create_file(void *obj, void *data, QGuestAllocator *t_alloc)
555 {
556 QVirtio9P *v9p = obj;
557 v9fs_set_allocator(t_alloc);
558 struct stat st;
559 g_autofree char *new_file = virtio_9p_test_path("03/1st_file");
560
561 tattach({ .client = v9p });
562 tmkdir({ .client = v9p, .atPath = "/", .name = "03" });
563 tlcreate({ .client = v9p, .atPath = "03", .name = "1st_file" });
564
565 /* check if created file exists now ... */
566 g_assert(stat(new_file, &st) == 0);
567 /* ... and is a regular file */
568 g_assert((st.st_mode & S_IFMT) == S_IFREG);
569 }
570
fs_unlinkat_file(void * obj,void * data,QGuestAllocator * t_alloc)571 static void fs_unlinkat_file(void *obj, void *data, QGuestAllocator *t_alloc)
572 {
573 QVirtio9P *v9p = obj;
574 v9fs_set_allocator(t_alloc);
575 struct stat st;
576 g_autofree char *new_file = virtio_9p_test_path("04/doa_file");
577
578 tattach({ .client = v9p });
579 tmkdir({ .client = v9p, .atPath = "/", .name = "04" });
580 tlcreate({ .client = v9p, .atPath = "04", .name = "doa_file" });
581
582 /* check if created file exists now ... */
583 g_assert(stat(new_file, &st) == 0);
584 /* ... and is a regular file */
585 g_assert((st.st_mode & S_IFMT) == S_IFREG);
586
587 tunlinkat({ .client = v9p, .atPath = "04", .name = "doa_file" });
588 /* file should be gone now */
589 g_assert(stat(new_file, &st) != 0);
590 }
591
fs_symlink_file(void * obj,void * data,QGuestAllocator * t_alloc)592 static void fs_symlink_file(void *obj, void *data, QGuestAllocator *t_alloc)
593 {
594 QVirtio9P *v9p = obj;
595 v9fs_set_allocator(t_alloc);
596 struct stat st;
597 g_autofree char *real_file = virtio_9p_test_path("05/real_file");
598 g_autofree char *symlink_file = virtio_9p_test_path("05/symlink_file");
599
600 tattach({ .client = v9p });
601 tmkdir({ .client = v9p, .atPath = "/", .name = "05" });
602 tlcreate({ .client = v9p, .atPath = "05", .name = "real_file" });
603 g_assert(stat(real_file, &st) == 0);
604 g_assert((st.st_mode & S_IFMT) == S_IFREG);
605
606 tsymlink({
607 .client = v9p, .atPath = "05", .name = "symlink_file",
608 .symtgt = "real_file"
609 });
610
611 /* check if created link exists now */
612 g_assert(stat(symlink_file, &st) == 0);
613 }
614
fs_unlinkat_symlink(void * obj,void * data,QGuestAllocator * t_alloc)615 static void fs_unlinkat_symlink(void *obj, void *data,
616 QGuestAllocator *t_alloc)
617 {
618 QVirtio9P *v9p = obj;
619 v9fs_set_allocator(t_alloc);
620 struct stat st;
621 g_autofree char *real_file = virtio_9p_test_path("06/real_file");
622 g_autofree char *symlink_file = virtio_9p_test_path("06/symlink_file");
623
624 tattach({ .client = v9p });
625 tmkdir({ .client = v9p, .atPath = "/", .name = "06" });
626 tlcreate({ .client = v9p, .atPath = "06", .name = "real_file" });
627 g_assert(stat(real_file, &st) == 0);
628 g_assert((st.st_mode & S_IFMT) == S_IFREG);
629
630 tsymlink({
631 .client = v9p, .atPath = "06", .name = "symlink_file",
632 .symtgt = "real_file"
633 });
634 g_assert(stat(symlink_file, &st) == 0);
635
636 tunlinkat({ .client = v9p, .atPath = "06", .name = "symlink_file" });
637 /* symlink should be gone now */
638 g_assert(stat(symlink_file, &st) != 0);
639 }
640
fs_hardlink_file(void * obj,void * data,QGuestAllocator * t_alloc)641 static void fs_hardlink_file(void *obj, void *data, QGuestAllocator *t_alloc)
642 {
643 QVirtio9P *v9p = obj;
644 v9fs_set_allocator(t_alloc);
645 struct stat st_real, st_link;
646 g_autofree char *real_file = virtio_9p_test_path("07/real_file");
647 g_autofree char *hardlink_file = virtio_9p_test_path("07/hardlink_file");
648
649 tattach({ .client = v9p });
650 tmkdir({ .client = v9p, .atPath = "/", .name = "07" });
651 tlcreate({ .client = v9p, .atPath = "07", .name = "real_file" });
652 g_assert(stat(real_file, &st_real) == 0);
653 g_assert((st_real.st_mode & S_IFMT) == S_IFREG);
654
655 tlink({
656 .client = v9p, .atPath = "07", .name = "hardlink_file",
657 .toPath = "07/real_file"
658 });
659
660 /* check if link exists now ... */
661 g_assert(stat(hardlink_file, &st_link) == 0);
662 /* ... and it's a hard link, right? */
663 g_assert((st_link.st_mode & S_IFMT) == S_IFREG);
664 g_assert(st_link.st_dev == st_real.st_dev);
665 g_assert(st_link.st_ino == st_real.st_ino);
666 }
667
fs_unlinkat_hardlink(void * obj,void * data,QGuestAllocator * t_alloc)668 static void fs_unlinkat_hardlink(void *obj, void *data,
669 QGuestAllocator *t_alloc)
670 {
671 QVirtio9P *v9p = obj;
672 v9fs_set_allocator(t_alloc);
673 struct stat st_real, st_link;
674 g_autofree char *real_file = virtio_9p_test_path("08/real_file");
675 g_autofree char *hardlink_file = virtio_9p_test_path("08/hardlink_file");
676
677 tattach({ .client = v9p });
678 tmkdir({ .client = v9p, .atPath = "/", .name = "08" });
679 tlcreate({ .client = v9p, .atPath = "08", .name = "real_file" });
680 g_assert(stat(real_file, &st_real) == 0);
681 g_assert((st_real.st_mode & S_IFMT) == S_IFREG);
682
683 tlink({
684 .client = v9p, .atPath = "08", .name = "hardlink_file",
685 .toPath = "08/real_file"
686 });
687 g_assert(stat(hardlink_file, &st_link) == 0);
688
689 tunlinkat({ .client = v9p, .atPath = "08", .name = "hardlink_file" });
690 /* symlink should be gone now */
691 g_assert(stat(hardlink_file, &st_link) != 0);
692 /* and old file should still exist */
693 g_assert(stat(real_file, &st_real) == 0);
694 }
695
fs_use_after_unlink(void * obj,void * data,QGuestAllocator * t_alloc)696 static void fs_use_after_unlink(void *obj, void *data,
697 QGuestAllocator *t_alloc)
698 {
699 QVirtio9P *v9p = obj;
700 v9fs_set_allocator(t_alloc);
701 static const uint32_t write_count = P9_MAX_SIZE / 2;
702 g_autofree char *real_file = virtio_9p_test_path("09/doa_file");
703 g_autofree char *buf = g_malloc0(write_count);
704 struct stat st_file;
705 struct v9fs_attr attr;
706 uint32_t fid_file;
707 uint32_t count;
708
709 tattach({ .client = v9p });
710
711 /* create a file "09/doa_file" and make sure it exists and is regular */
712 tmkdir({ .client = v9p, .atPath = "/", .name = "09" });
713 tlcreate({ .client = v9p, .atPath = "09", .name = "doa_file" });
714 g_assert(stat(real_file, &st_file) == 0);
715 g_assert((st_file.st_mode & S_IFMT) == S_IFREG);
716
717 /* request a FID for that regular file that we can work with next */
718 fid_file = twalk({
719 .client = v9p, .fid = 0, .path = "09/doa_file"
720 }).newfid;
721 g_assert(fid_file != 0);
722
723 /* now first open the file in write mode before ... */
724 tlopen({ .client = v9p, .fid = fid_file, .flags = O_WRONLY });
725 /* ... removing the file from file system */
726 tunlinkat({ .client = v9p, .atPath = "09", .name = "doa_file" });
727
728 /* file is removed, but we still have it open, so this should succeed */
729 tgetattr({
730 .client = v9p, .fid = fid_file, .request_mask = P9_GETATTR_BASIC,
731 .rgetattr.attr = &attr
732 });
733 count = twrite({
734 .client = v9p, .fid = fid_file, .offset = 0, .count = write_count,
735 .data = buf
736 }).count;
737 g_assert_cmpint(count, ==, write_count);
738 }
739
cleanup_9p_local_driver(void * data)740 static void cleanup_9p_local_driver(void *data)
741 {
742 /* remove previously created test dir when test is completed */
743 virtio_9p_remove_local_test_dir();
744 }
745
assign_9p_local_driver(GString * cmd_line,void * arg)746 static void *assign_9p_local_driver(GString *cmd_line, void *arg)
747 {
748 /* make sure test dir for the 'local' tests exists */
749 virtio_9p_create_local_test_dir();
750
751 virtio_9p_assign_local_driver(cmd_line, "security_model=mapped-xattr");
752
753 g_test_queue_destroy(cleanup_9p_local_driver, NULL);
754 return arg;
755 }
756
register_virtio_9p_test(void)757 static void register_virtio_9p_test(void)
758 {
759
760 QOSGraphTestOptions opts = {
761 };
762
763 /* 9pfs test cases using the 'synth' filesystem driver */
764 qos_add_test("synth/config", "virtio-9p", pci_config, &opts);
765 qos_add_test("synth/version/basic", "virtio-9p", fs_version, &opts);
766 qos_add_test("synth/attach/basic", "virtio-9p", fs_attach, &opts);
767 qos_add_test("synth/walk/basic", "virtio-9p", fs_walk, &opts);
768 qos_add_test("synth/walk/no_slash", "virtio-9p", fs_walk_no_slash,
769 &opts);
770 qos_add_test("synth/walk/none", "virtio-9p", fs_walk_none, &opts);
771 qos_add_test("synth/walk/dotdot_from_root", "virtio-9p",
772 fs_walk_dotdot, &opts);
773 qos_add_test("synth/walk/non_existent", "virtio-9p", fs_walk_nonexistent,
774 &opts);
775 qos_add_test("synth/walk/2nd_non_existent", "virtio-9p",
776 fs_walk_2nd_nonexistent, &opts);
777 qos_add_test("synth/lopen/basic", "virtio-9p", fs_lopen, &opts);
778 qos_add_test("synth/write/basic", "virtio-9p", fs_write, &opts);
779 qos_add_test("synth/flush/success", "virtio-9p", fs_flush_success,
780 &opts);
781 qos_add_test("synth/flush/ignored", "virtio-9p", fs_flush_ignored,
782 &opts);
783 qos_add_test("synth/readdir/basic", "virtio-9p", fs_readdir, &opts);
784 qos_add_test("synth/readdir/split_512", "virtio-9p",
785 fs_readdir_split_512, &opts);
786 qos_add_test("synth/readdir/split_256", "virtio-9p",
787 fs_readdir_split_256, &opts);
788 qos_add_test("synth/readdir/split_128", "virtio-9p",
789 fs_readdir_split_128, &opts);
790
791
792 /* 9pfs test cases using the 'local' filesystem driver */
793 opts.before = assign_9p_local_driver;
794 qos_add_test("local/config", "virtio-9p", pci_config, &opts);
795 qos_add_test("local/create_dir", "virtio-9p", fs_create_dir, &opts);
796 qos_add_test("local/unlinkat_dir", "virtio-9p", fs_unlinkat_dir, &opts);
797 qos_add_test("local/create_file", "virtio-9p", fs_create_file, &opts);
798 qos_add_test("local/unlinkat_file", "virtio-9p", fs_unlinkat_file, &opts);
799 qos_add_test("local/symlink_file", "virtio-9p", fs_symlink_file, &opts);
800 qos_add_test("local/unlinkat_symlink", "virtio-9p", fs_unlinkat_symlink,
801 &opts);
802 qos_add_test("local/hardlink_file", "virtio-9p", fs_hardlink_file, &opts);
803 qos_add_test("local/unlinkat_hardlink", "virtio-9p", fs_unlinkat_hardlink,
804 &opts);
805 qos_add_test("local/use_after_unlink", "virtio-9p", fs_use_after_unlink,
806 &opts);
807 }
808
809 libqos_init(register_virtio_9p_test);
810