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