xref: /qemu/tests/qtest/virtio-9p-test.c (revision d41a9462ea15f0e3ef789d496032536c33098275)
12d888c09SAndreas Färber /*
22d888c09SAndreas Färber  * QTest testcase for VirtIO 9P
32d888c09SAndreas Färber  *
42d888c09SAndreas Färber  * Copyright (c) 2014 SUSE LINUX Products GmbH
52d888c09SAndreas Färber  *
62d888c09SAndreas Färber  * This work is licensed under the terms of the GNU GPL, version 2 or later.
72d888c09SAndreas Färber  * See the COPYING file in the top-level directory.
82d888c09SAndreas Färber  */
92d888c09SAndreas Färber 
106f569084SChristian Schoenebeck /*
116f569084SChristian Schoenebeck  * Not so fast! You might want to read the 9p developer docs first:
126f569084SChristian Schoenebeck  * https://wiki.qemu.org/Documentation/9p
136f569084SChristian Schoenebeck  */
146f569084SChristian Schoenebeck 
15fbc04127SPeter Maydell #include "qemu/osdep.h"
160b8fa32fSMarkus Armbruster #include "qemu/module.h"
17684f9120SChristian Schoenebeck #include "libqos/virtio-9p-client.h"
1865b70fc7SGreg Kurz 
19569f3b63SChristian Schoenebeck #define twalk(...) v9fs_twalk((TWalkOpt) __VA_ARGS__)
20bee8fda2SChristian Schoenebeck #define tversion(...) v9fs_tversion((TVersionOpt) __VA_ARGS__)
2174a160abSChristian Schoenebeck #define tattach(...) v9fs_tattach((TAttachOpt) __VA_ARGS__)
222af5be47SChristian Schoenebeck #define tgetattr(...) v9fs_tgetattr((TGetAttrOpt) __VA_ARGS__)
231ebacc40SChristian Schoenebeck #define treaddir(...) v9fs_treaddir((TReadDirOpt) __VA_ARGS__)
243878ce4cSChristian Schoenebeck #define tlopen(...) v9fs_tlopen((TLOpenOpt) __VA_ARGS__)
25ac9e4e61SChristian Schoenebeck #define twrite(...) v9fs_twrite((TWriteOpt) __VA_ARGS__)
26d89146fdSChristian Schoenebeck #define tflush(...) v9fs_tflush((TFlushOpt) __VA_ARGS__)
27e1168010SChristian Schoenebeck #define tmkdir(...) v9fs_tmkdir((TMkdirOpt) __VA_ARGS__)
28bd4660d4SChristian Schoenebeck #define tlcreate(...) v9fs_tlcreate((TlcreateOpt) __VA_ARGS__)
299beabfa5SChristian Schoenebeck #define tsymlink(...) v9fs_tsymlink((TsymlinkOpt) __VA_ARGS__)
30*d41a9462SChristian Schoenebeck #define tlink(...) v9fs_tlink((TlinkOpt) __VA_ARGS__)
31653daf38SChristian Schoenebeck 
32dfbe8b43SEmanuele Giuseppe Esposito static void pci_config(void *obj, void *data, QGuestAllocator *t_alloc)
33557a4cc0SGreg Kurz {
34dfbe8b43SEmanuele Giuseppe Esposito     QVirtio9P *v9p = obj;
35684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
36dfbe8b43SEmanuele Giuseppe Esposito     size_t tag_len = qvirtio_config_readw(v9p->vdev, 0);
3765ceee0aSChristian Schoenebeck     g_autofree char *tag = NULL;
38557a4cc0SGreg Kurz     int i;
39557a4cc0SGreg Kurz 
40dfbe8b43SEmanuele Giuseppe Esposito     g_assert_cmpint(tag_len, ==, strlen(MOUNT_TAG));
41557a4cc0SGreg Kurz 
42557a4cc0SGreg Kurz     tag = g_malloc(tag_len);
43557a4cc0SGreg Kurz     for (i = 0; i < tag_len; i++) {
44dfbe8b43SEmanuele Giuseppe Esposito         tag[i] = qvirtio_config_readb(v9p->vdev, i + 2);
45557a4cc0SGreg Kurz     }
46dfbe8b43SEmanuele Giuseppe Esposito     g_assert_cmpmem(tag, tag_len, MOUNT_TAG, tag_len);
471211d81bSGreg Kurz }
48557a4cc0SGreg Kurz 
49a6821b82SChristian Schoenebeck static inline bool is_same_qid(v9fs_qid a, v9fs_qid b)
50a6821b82SChristian Schoenebeck {
51a6821b82SChristian Schoenebeck     /* don't compare QID version for checking for file ID equalness */
52a6821b82SChristian Schoenebeck     return a[0] == b[0] && memcmp(&a[5], &b[5], 8) == 0;
53a6821b82SChristian Schoenebeck }
54a6821b82SChristian Schoenebeck 
551c450e6eSGreg Kurz static void fs_version(void *obj, void *data, QGuestAllocator *t_alloc)
561c450e6eSGreg Kurz {
57684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
58bee8fda2SChristian Schoenebeck     tversion({ .client = obj });
591c450e6eSGreg Kurz }
601c450e6eSGreg Kurz 
613fe4baf4SGreg Kurz static void fs_attach(void *obj, void *data, QGuestAllocator *t_alloc)
623fe4baf4SGreg Kurz {
63684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
6474a160abSChristian Schoenebeck     tattach({ .client = obj });
653fe4baf4SGreg Kurz }
663fe4baf4SGreg Kurz 
67dfbe8b43SEmanuele Giuseppe Esposito static void fs_walk(void *obj, void *data, QGuestAllocator *t_alloc)
6804b88c84SGreg Kurz {
69dfbe8b43SEmanuele Giuseppe Esposito     QVirtio9P *v9p = obj;
70684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
712893ddd5SGreg Kurz     char *wnames[P9_MAXWELEM];
7204b88c84SGreg Kurz     uint16_t nwqid;
7365ceee0aSChristian Schoenebeck     g_autofree v9fs_qid *wqid = NULL;
7404b88c84SGreg Kurz     int i;
7504b88c84SGreg Kurz 
7604b88c84SGreg Kurz     for (i = 0; i < P9_MAXWELEM; i++) {
772893ddd5SGreg Kurz         wnames[i] = g_strdup_printf(QTEST_V9FS_SYNTH_WALK_FILE, i);
7804b88c84SGreg Kurz     }
7904b88c84SGreg Kurz 
8074a160abSChristian Schoenebeck     tattach({ .client = v9p });
813f3e9232SChristian Schoenebeck     twalk({
82569f3b63SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1,
833f3e9232SChristian Schoenebeck         .nwname = P9_MAXWELEM, .wnames = wnames,
843f3e9232SChristian Schoenebeck         .rwalk = { .nwqid = &nwqid, .wqid = &wqid }
853f3e9232SChristian Schoenebeck     });
8604b88c84SGreg Kurz 
8704b88c84SGreg Kurz     g_assert_cmpint(nwqid, ==, P9_MAXWELEM);
8804b88c84SGreg Kurz 
8904b88c84SGreg Kurz     for (i = 0; i < P9_MAXWELEM; i++) {
9004b88c84SGreg Kurz         g_free(wnames[i]);
9104b88c84SGreg Kurz     }
9204b88c84SGreg Kurz }
9304b88c84SGreg Kurz 
944829469fSChristian Schoenebeck static bool fs_dirents_contain_name(struct V9fsDirent *e, const char* name)
954829469fSChristian Schoenebeck {
964829469fSChristian Schoenebeck     for (; e; e = e->next) {
974829469fSChristian Schoenebeck         if (!strcmp(e->name, name)) {
984829469fSChristian Schoenebeck             return true;
994829469fSChristian Schoenebeck         }
1004829469fSChristian Schoenebeck     }
1014829469fSChristian Schoenebeck     return false;
1024829469fSChristian Schoenebeck }
1034829469fSChristian Schoenebeck 
10446488b62SChristian Schoenebeck /* basic readdir test where reply fits into a single response message */
1054829469fSChristian Schoenebeck static void fs_readdir(void *obj, void *data, QGuestAllocator *t_alloc)
1064829469fSChristian Schoenebeck {
1074829469fSChristian Schoenebeck     QVirtio9P *v9p = obj;
108684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
109569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_READDIR_DIR) };
1104829469fSChristian Schoenebeck     uint16_t nqid;
1114829469fSChristian Schoenebeck     v9fs_qid qid;
1124829469fSChristian Schoenebeck     uint32_t count, nentries;
1134829469fSChristian Schoenebeck     struct V9fsDirent *entries = NULL;
1144829469fSChristian Schoenebeck 
11574a160abSChristian Schoenebeck     tattach({ .client = v9p });
1163f3e9232SChristian Schoenebeck     twalk({
117569f3b63SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1,
1183f3e9232SChristian Schoenebeck         .nwname = 1, .wnames = wnames, .rwalk.nwqid = &nqid
1193f3e9232SChristian Schoenebeck     });
1204829469fSChristian Schoenebeck     g_assert_cmpint(nqid, ==, 1);
1214829469fSChristian Schoenebeck 
1220e4c4ff0SChristian Schoenebeck     tlopen({
1230e4c4ff0SChristian Schoenebeck         .client = v9p, .fid = 1, .flags = O_DIRECTORY, .rlopen.qid = &qid
1240e4c4ff0SChristian Schoenebeck     });
1254829469fSChristian Schoenebeck 
1264829469fSChristian Schoenebeck     /*
1274829469fSChristian Schoenebeck      * submit count = msize - 11, because 11 is the header size of Rreaddir
1284829469fSChristian Schoenebeck      */
129a9a53769SChristian Schoenebeck     treaddir({
1301ebacc40SChristian Schoenebeck         .client = v9p, .fid = 1, .offset = 0, .count = P9_MAX_SIZE - 11,
131a9a53769SChristian Schoenebeck         .rreaddir = {
132a9a53769SChristian Schoenebeck             .count = &count, .nentries = &nentries, .entries = &entries
133a9a53769SChristian Schoenebeck         }
134a9a53769SChristian Schoenebeck     });
1354829469fSChristian Schoenebeck 
1364829469fSChristian Schoenebeck     /*
1374829469fSChristian Schoenebeck      * Assuming msize (P9_MAX_SIZE) is large enough so we can retrieve all
1384829469fSChristian Schoenebeck      * dir entries with only one readdir request.
1394829469fSChristian Schoenebeck      */
1404829469fSChristian Schoenebeck     g_assert_cmpint(
1414829469fSChristian Schoenebeck         nentries, ==,
1424829469fSChristian Schoenebeck         QTEST_V9FS_SYNTH_READDIR_NFILES + 2 /* "." and ".." */
1434829469fSChristian Schoenebeck     );
1444829469fSChristian Schoenebeck 
1454829469fSChristian Schoenebeck     /*
1464829469fSChristian Schoenebeck      * Check all file names exist in returned entries, ignore their order
1474829469fSChristian Schoenebeck      * though.
1484829469fSChristian Schoenebeck      */
1494829469fSChristian Schoenebeck     g_assert_cmpint(fs_dirents_contain_name(entries, "."), ==, true);
1504829469fSChristian Schoenebeck     g_assert_cmpint(fs_dirents_contain_name(entries, ".."), ==, true);
1514829469fSChristian Schoenebeck     for (int i = 0; i < QTEST_V9FS_SYNTH_READDIR_NFILES; ++i) {
15265ceee0aSChristian Schoenebeck         g_autofree char *name =
15365ceee0aSChristian Schoenebeck             g_strdup_printf(QTEST_V9FS_SYNTH_READDIR_FILE, i);
1544829469fSChristian Schoenebeck         g_assert_cmpint(fs_dirents_contain_name(entries, name), ==, true);
1554829469fSChristian Schoenebeck     }
1564829469fSChristian Schoenebeck 
1574829469fSChristian Schoenebeck     v9fs_free_dirents(entries);
1584829469fSChristian Schoenebeck     g_free(wnames[0]);
1594829469fSChristian Schoenebeck }
1604829469fSChristian Schoenebeck 
16146488b62SChristian Schoenebeck /* readdir test where overall request is split over several messages */
1621d98613dSGreg Kurz static void do_readdir_split(QVirtio9P *v9p, uint32_t count)
16346488b62SChristian Schoenebeck {
164569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_READDIR_DIR) };
16546488b62SChristian Schoenebeck     uint16_t nqid;
16646488b62SChristian Schoenebeck     v9fs_qid qid;
16746488b62SChristian Schoenebeck     uint32_t nentries, npartialentries;
16846488b62SChristian Schoenebeck     struct V9fsDirent *entries, *tail, *partialentries;
16946488b62SChristian Schoenebeck     int fid;
17046488b62SChristian Schoenebeck     uint64_t offset;
17146488b62SChristian Schoenebeck 
17274a160abSChristian Schoenebeck     tattach({ .client = v9p });
17346488b62SChristian Schoenebeck 
17446488b62SChristian Schoenebeck     fid = 1;
17546488b62SChristian Schoenebeck     offset = 0;
17646488b62SChristian Schoenebeck     entries = NULL;
17746488b62SChristian Schoenebeck     nentries = 0;
17846488b62SChristian Schoenebeck     tail = NULL;
17946488b62SChristian Schoenebeck 
1803f3e9232SChristian Schoenebeck     twalk({
181569f3b63SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = fid,
1823f3e9232SChristian Schoenebeck         .nwname = 1, .wnames = wnames, .rwalk.nwqid = &nqid
1833f3e9232SChristian Schoenebeck     });
18446488b62SChristian Schoenebeck     g_assert_cmpint(nqid, ==, 1);
18546488b62SChristian Schoenebeck 
1860e4c4ff0SChristian Schoenebeck     tlopen({
1870e4c4ff0SChristian Schoenebeck         .client = v9p, .fid = fid, .flags = O_DIRECTORY, .rlopen.qid = &qid
1880e4c4ff0SChristian Schoenebeck     });
18946488b62SChristian Schoenebeck 
19046488b62SChristian Schoenebeck     /*
19146488b62SChristian Schoenebeck      * send as many Treaddir requests as required to get all directory
19246488b62SChristian Schoenebeck      * entries
19346488b62SChristian Schoenebeck      */
19446488b62SChristian Schoenebeck     while (true) {
19546488b62SChristian Schoenebeck         npartialentries = 0;
19646488b62SChristian Schoenebeck         partialentries = NULL;
19746488b62SChristian Schoenebeck 
198a9a53769SChristian Schoenebeck         treaddir({
1991ebacc40SChristian Schoenebeck             .client = v9p, .fid = fid, .offset = offset, .count = count,
200a9a53769SChristian Schoenebeck             .rreaddir = {
201a9a53769SChristian Schoenebeck                 .count = &count, .nentries = &npartialentries,
202a9a53769SChristian Schoenebeck                 .entries = &partialentries
203a9a53769SChristian Schoenebeck             }
204a9a53769SChristian Schoenebeck         });
20546488b62SChristian Schoenebeck         if (npartialentries > 0 && partialentries) {
20646488b62SChristian Schoenebeck             if (!entries) {
20746488b62SChristian Schoenebeck                 entries = partialentries;
20846488b62SChristian Schoenebeck                 nentries = npartialentries;
20946488b62SChristian Schoenebeck                 tail = partialentries;
21046488b62SChristian Schoenebeck             } else {
21146488b62SChristian Schoenebeck                 tail->next = partialentries;
21246488b62SChristian Schoenebeck                 nentries += npartialentries;
21346488b62SChristian Schoenebeck             }
21446488b62SChristian Schoenebeck             while (tail->next) {
21546488b62SChristian Schoenebeck                 tail = tail->next;
21646488b62SChristian Schoenebeck             }
21746488b62SChristian Schoenebeck             offset = tail->offset;
21846488b62SChristian Schoenebeck         } else {
21946488b62SChristian Schoenebeck             break;
22046488b62SChristian Schoenebeck         }
22146488b62SChristian Schoenebeck     }
22246488b62SChristian Schoenebeck 
22346488b62SChristian Schoenebeck     g_assert_cmpint(
22446488b62SChristian Schoenebeck         nentries, ==,
22546488b62SChristian Schoenebeck         QTEST_V9FS_SYNTH_READDIR_NFILES + 2 /* "." and ".." */
22646488b62SChristian Schoenebeck     );
22746488b62SChristian Schoenebeck 
22846488b62SChristian Schoenebeck     /*
22946488b62SChristian Schoenebeck      * Check all file names exist in returned entries, ignore their order
23046488b62SChristian Schoenebeck      * though.
23146488b62SChristian Schoenebeck      */
23246488b62SChristian Schoenebeck     g_assert_cmpint(fs_dirents_contain_name(entries, "."), ==, true);
23346488b62SChristian Schoenebeck     g_assert_cmpint(fs_dirents_contain_name(entries, ".."), ==, true);
23446488b62SChristian Schoenebeck     for (int i = 0; i < QTEST_V9FS_SYNTH_READDIR_NFILES; ++i) {
23546488b62SChristian Schoenebeck         char *name = g_strdup_printf(QTEST_V9FS_SYNTH_READDIR_FILE, i);
23646488b62SChristian Schoenebeck         g_assert_cmpint(fs_dirents_contain_name(entries, name), ==, true);
23746488b62SChristian Schoenebeck         g_free(name);
23846488b62SChristian Schoenebeck     }
23946488b62SChristian Schoenebeck 
24046488b62SChristian Schoenebeck     v9fs_free_dirents(entries);
24146488b62SChristian Schoenebeck 
24246488b62SChristian Schoenebeck     g_free(wnames[0]);
24346488b62SChristian Schoenebeck }
24446488b62SChristian Schoenebeck 
245dfbe8b43SEmanuele Giuseppe Esposito static void fs_walk_no_slash(void *obj, void *data, QGuestAllocator *t_alloc)
246ba0d1037SGreg Kurz {
247dfbe8b43SEmanuele Giuseppe Esposito     QVirtio9P *v9p = obj;
248684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
249569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup(" /") };
250ba0d1037SGreg Kurz 
25174a160abSChristian Schoenebeck     tattach({ .client = v9p });
2523f3e9232SChristian Schoenebeck     twalk({
253569f3b63SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames,
2543f3e9232SChristian Schoenebeck         .expectErr = ENOENT
2553f3e9232SChristian Schoenebeck     });
256ba0d1037SGreg Kurz 
257ba0d1037SGreg Kurz     g_free(wnames[0]);
258ba0d1037SGreg Kurz }
259ba0d1037SGreg Kurz 
2609472a689SChristian Schoenebeck static void fs_walk_nonexistent(void *obj, void *data, QGuestAllocator *t_alloc)
2619472a689SChristian Schoenebeck {
2629472a689SChristian Schoenebeck     QVirtio9P *v9p = obj;
263684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
2649472a689SChristian Schoenebeck 
26574a160abSChristian Schoenebeck     tattach({ .client = v9p });
26615fbff48SChristian Schoenebeck     /*
26715fbff48SChristian Schoenebeck      * The 9p2000 protocol spec says: "If the first element cannot be walked
26815fbff48SChristian Schoenebeck      * for any reason, Rerror is returned."
26915fbff48SChristian Schoenebeck      */
270569f3b63SChristian Schoenebeck     twalk({ .client = v9p, .path = "non-existent", .expectErr = ENOENT });
2719472a689SChristian Schoenebeck }
2729472a689SChristian Schoenebeck 
27315fbff48SChristian Schoenebeck static void fs_walk_2nd_nonexistent(void *obj, void *data,
27415fbff48SChristian Schoenebeck                                     QGuestAllocator *t_alloc)
27515fbff48SChristian Schoenebeck {
27615fbff48SChristian Schoenebeck     QVirtio9P *v9p = obj;
277684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
2780e43495dSChristian Schoenebeck     v9fs_qid root_qid;
27915fbff48SChristian Schoenebeck     uint16_t nwqid;
28028c73670SChristian Schoenebeck     uint32_t fid;
28115fbff48SChristian Schoenebeck     g_autofree v9fs_qid *wqid = NULL;
28215fbff48SChristian Schoenebeck     g_autofree char *path = g_strdup_printf(
28315fbff48SChristian Schoenebeck         QTEST_V9FS_SYNTH_WALK_FILE "/non-existent", 0
28415fbff48SChristian Schoenebeck     );
28515fbff48SChristian Schoenebeck 
28674a160abSChristian Schoenebeck     tattach({ .client = v9p, .rattach.qid = &root_qid });
287569f3b63SChristian Schoenebeck     fid = twalk({
288569f3b63SChristian Schoenebeck         .client = v9p, .path = path,
2893f3e9232SChristian Schoenebeck         .rwalk = { .nwqid = &nwqid, .wqid = &wqid }
290569f3b63SChristian Schoenebeck     }).newfid;
29115fbff48SChristian Schoenebeck     /*
29215fbff48SChristian Schoenebeck      * The 9p2000 protocol spec says: "nwqid is therefore either nwname or the
29315fbff48SChristian Schoenebeck      * index of the first elementwise walk that failed."
29415fbff48SChristian Schoenebeck      */
29515fbff48SChristian Schoenebeck     assert(nwqid == 1);
2960e43495dSChristian Schoenebeck 
2970e43495dSChristian Schoenebeck     /* returned QID wqid[0] is file ID of 1st subdir */
2980e43495dSChristian Schoenebeck     g_assert(wqid && wqid[0] && !is_same_qid(root_qid, wqid[0]));
2990e43495dSChristian Schoenebeck 
3000e43495dSChristian Schoenebeck     /* expect fid being unaffected by walk above */
30128c73670SChristian Schoenebeck     tgetattr({
3022af5be47SChristian Schoenebeck         .client = v9p, .fid = fid, .request_mask = P9_GETATTR_BASIC,
30328c73670SChristian Schoenebeck         .expectErr = ENOENT
30428c73670SChristian Schoenebeck     });
30515fbff48SChristian Schoenebeck }
30615fbff48SChristian Schoenebeck 
307c1668948SChristian Schoenebeck static void fs_walk_none(void *obj, void *data, QGuestAllocator *t_alloc)
308c1668948SChristian Schoenebeck {
309c1668948SChristian Schoenebeck     QVirtio9P *v9p = obj;
310684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
311c1668948SChristian Schoenebeck     v9fs_qid root_qid;
312c1668948SChristian Schoenebeck     g_autofree v9fs_qid *wqid = NULL;
313a6821b82SChristian Schoenebeck     struct v9fs_attr attr;
314c1668948SChristian Schoenebeck 
315bee8fda2SChristian Schoenebeck     tversion({ .client = v9p });
3161125ddf6SChristian Schoenebeck     tattach({
3171125ddf6SChristian Schoenebeck         .client = v9p, .fid = 0, .n_uname = getuid(),
3181125ddf6SChristian Schoenebeck         .rattach.qid = &root_qid
3191125ddf6SChristian Schoenebeck     });
320c1668948SChristian Schoenebeck 
3213f3e9232SChristian Schoenebeck     twalk({
322569f3b63SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1, .nwname = 0, .wnames = NULL,
3233f3e9232SChristian Schoenebeck         .rwalk.wqid = &wqid
3243f3e9232SChristian Schoenebeck     });
325c1668948SChristian Schoenebeck 
326c1668948SChristian Schoenebeck     /* special case: no QID is returned if nwname=0 was sent */
327c1668948SChristian Schoenebeck     g_assert(wqid == NULL);
328a6821b82SChristian Schoenebeck 
32928c73670SChristian Schoenebeck     tgetattr({
3302af5be47SChristian Schoenebeck         .client = v9p, .fid = 1, .request_mask = P9_GETATTR_BASIC,
33128c73670SChristian Schoenebeck         .rgetattr.attr = &attr
33228c73670SChristian Schoenebeck     });
333a6821b82SChristian Schoenebeck 
334a6821b82SChristian Schoenebeck     g_assert(is_same_qid(root_qid, attr.qid));
335c1668948SChristian Schoenebeck }
336c1668948SChristian Schoenebeck 
337dfbe8b43SEmanuele Giuseppe Esposito static void fs_walk_dotdot(void *obj, void *data, QGuestAllocator *t_alloc)
338a37c0702SGreg Kurz {
339dfbe8b43SEmanuele Giuseppe Esposito     QVirtio9P *v9p = obj;
340684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
341569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup("..") };
34265ceee0aSChristian Schoenebeck     v9fs_qid root_qid;
34365ceee0aSChristian Schoenebeck     g_autofree v9fs_qid *wqid = NULL;
344a37c0702SGreg Kurz 
345bee8fda2SChristian Schoenebeck     tversion({ .client = v9p });
3461125ddf6SChristian Schoenebeck     tattach({
3471125ddf6SChristian Schoenebeck         .client = v9p, .fid = 0, .n_uname = getuid(),
3481125ddf6SChristian Schoenebeck         .rattach.qid = &root_qid
3491125ddf6SChristian Schoenebeck     });
350a37c0702SGreg Kurz 
3513f3e9232SChristian Schoenebeck     twalk({
352569f3b63SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames,
3533f3e9232SChristian Schoenebeck         .rwalk.wqid = &wqid /* We now we'll get one qid */
3543f3e9232SChristian Schoenebeck     });
355a37c0702SGreg Kurz 
356a37c0702SGreg Kurz     g_assert_cmpmem(&root_qid, 13, wqid[0], 13);
357a37c0702SGreg Kurz 
358a37c0702SGreg Kurz     g_free(wnames[0]);
359a37c0702SGreg Kurz }
360a37c0702SGreg Kurz 
361dfbe8b43SEmanuele Giuseppe Esposito static void fs_lopen(void *obj, void *data, QGuestAllocator *t_alloc)
36282469aaeSGreg Kurz {
363dfbe8b43SEmanuele Giuseppe Esposito     QVirtio9P *v9p = obj;
364684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
365569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_LOPEN_FILE) };
36682469aaeSGreg Kurz 
36774a160abSChristian Schoenebeck     tattach({ .client = v9p });
3683f3e9232SChristian Schoenebeck     twalk({
3693f3e9232SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames
3703f3e9232SChristian Schoenebeck     });
37182469aaeSGreg Kurz 
3720e4c4ff0SChristian Schoenebeck     tlopen({ .client = v9p, .fid = 1, .flags = O_WRONLY });
37382469aaeSGreg Kurz 
37482469aaeSGreg Kurz     g_free(wnames[0]);
37582469aaeSGreg Kurz }
37682469aaeSGreg Kurz 
377dfbe8b43SEmanuele Giuseppe Esposito static void fs_write(void *obj, void *data, QGuestAllocator *t_alloc)
378354b86f8SGreg Kurz {
379dfbe8b43SEmanuele Giuseppe Esposito     QVirtio9P *v9p = obj;
380684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
381354b86f8SGreg Kurz     static const uint32_t write_count = P9_MAX_SIZE / 2;
382569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_WRITE_FILE) };
38365ceee0aSChristian Schoenebeck     g_autofree char *buf = g_malloc0(write_count);
384354b86f8SGreg Kurz     uint32_t count;
385354b86f8SGreg Kurz 
38674a160abSChristian Schoenebeck     tattach({ .client = v9p });
3873f3e9232SChristian Schoenebeck     twalk({
3883f3e9232SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames
3893f3e9232SChristian Schoenebeck     });
390354b86f8SGreg Kurz 
3910e4c4ff0SChristian Schoenebeck     tlopen({ .client = v9p, .fid = 1, .flags = O_WRONLY });
392354b86f8SGreg Kurz 
393bb286ff8SChristian Schoenebeck     count = twrite({
394ac9e4e61SChristian Schoenebeck         .client = v9p, .fid = 1, .offset = 0, .count = write_count,
395bb286ff8SChristian Schoenebeck         .data = buf
396bb286ff8SChristian Schoenebeck     }).count;
397354b86f8SGreg Kurz     g_assert_cmpint(count, ==, write_count);
398354b86f8SGreg Kurz 
399354b86f8SGreg Kurz     g_free(wnames[0]);
400354b86f8SGreg Kurz }
401354b86f8SGreg Kurz 
402dfbe8b43SEmanuele Giuseppe Esposito static void fs_flush_success(void *obj, void *data, QGuestAllocator *t_alloc)
403357e2f7fSGreg Kurz {
404dfbe8b43SEmanuele Giuseppe Esposito     QVirtio9P *v9p = obj;
405684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
406569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_FLUSH_FILE) };
407357e2f7fSGreg Kurz     P9Req *req, *flush_req;
408357e2f7fSGreg Kurz     uint32_t reply_len;
409357e2f7fSGreg Kurz     uint8_t should_block;
410357e2f7fSGreg Kurz 
41174a160abSChristian Schoenebeck     tattach({ .client = v9p });
4123f3e9232SChristian Schoenebeck     twalk({
4133f3e9232SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames
4143f3e9232SChristian Schoenebeck     });
415357e2f7fSGreg Kurz 
4160e4c4ff0SChristian Schoenebeck     tlopen({ .client = v9p, .fid = 1, .flags = O_WRONLY });
417357e2f7fSGreg Kurz 
418357e2f7fSGreg Kurz     /* This will cause the 9p server to try to write data to the backend,
419357e2f7fSGreg Kurz      * until the write request gets cancelled.
420357e2f7fSGreg Kurz      */
421357e2f7fSGreg Kurz     should_block = 1;
422ac9e4e61SChristian Schoenebeck     req = twrite({
423ac9e4e61SChristian Schoenebeck         .client = v9p, .fid = 1, .offset = 0,
424ac9e4e61SChristian Schoenebeck         .count = sizeof(should_block), .data = &should_block,
425ac9e4e61SChristian Schoenebeck         .requestOnly = true
426ac9e4e61SChristian Schoenebeck     }).req;
427357e2f7fSGreg Kurz 
428d89146fdSChristian Schoenebeck     flush_req = tflush({
429d89146fdSChristian Schoenebeck         .client = v9p, .oldtag = req->tag, .tag = 1, .requestOnly = true
430d89146fdSChristian Schoenebeck     }).req;
431357e2f7fSGreg Kurz 
432357e2f7fSGreg Kurz     /* The write request is supposed to be flushed: the server should just
433357e2f7fSGreg Kurz      * mark the write request as used and reply to the flush request.
434357e2f7fSGreg Kurz      */
435357e2f7fSGreg Kurz     v9fs_req_wait_for_reply(req, &reply_len);
436357e2f7fSGreg Kurz     g_assert_cmpint(reply_len, ==, 0);
437357e2f7fSGreg Kurz     v9fs_req_free(req);
438357e2f7fSGreg Kurz     v9fs_rflush(flush_req);
439357e2f7fSGreg Kurz 
440357e2f7fSGreg Kurz     g_free(wnames[0]);
441357e2f7fSGreg Kurz }
442357e2f7fSGreg Kurz 
443dfbe8b43SEmanuele Giuseppe Esposito static void fs_flush_ignored(void *obj, void *data, QGuestAllocator *t_alloc)
444357e2f7fSGreg Kurz {
445dfbe8b43SEmanuele Giuseppe Esposito     QVirtio9P *v9p = obj;
446684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
447569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_FLUSH_FILE) };
448357e2f7fSGreg Kurz     P9Req *req, *flush_req;
449357e2f7fSGreg Kurz     uint32_t count;
450357e2f7fSGreg Kurz     uint8_t should_block;
451357e2f7fSGreg Kurz 
45274a160abSChristian Schoenebeck     tattach({ .client = v9p });
4533f3e9232SChristian Schoenebeck     twalk({
4543f3e9232SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames
4553f3e9232SChristian Schoenebeck     });
456357e2f7fSGreg Kurz 
4570e4c4ff0SChristian Schoenebeck     tlopen({ .client = v9p, .fid = 1, .flags = O_WRONLY });
458357e2f7fSGreg Kurz 
459357e2f7fSGreg Kurz     /* This will cause the write request to complete right away, before it
460357e2f7fSGreg Kurz      * could be actually cancelled.
461357e2f7fSGreg Kurz      */
462357e2f7fSGreg Kurz     should_block = 0;
463ac9e4e61SChristian Schoenebeck     req = twrite({
464ac9e4e61SChristian Schoenebeck         .client = v9p, .fid = 1, .offset = 0,
465ac9e4e61SChristian Schoenebeck         .count = sizeof(should_block), .data = &should_block,
466ac9e4e61SChristian Schoenebeck         .requestOnly = true
467ac9e4e61SChristian Schoenebeck     }).req;
468357e2f7fSGreg Kurz 
469d89146fdSChristian Schoenebeck     flush_req = tflush({
470d89146fdSChristian Schoenebeck         .client = v9p, .oldtag = req->tag, .tag = 1, .requestOnly = true
471d89146fdSChristian Schoenebeck     }).req;
472357e2f7fSGreg Kurz 
473357e2f7fSGreg Kurz     /* The write request is supposed to complete. The server should
474357e2f7fSGreg Kurz      * reply to the write request and the flush request.
475357e2f7fSGreg Kurz      */
476357e2f7fSGreg Kurz     v9fs_req_wait_for_reply(req, NULL);
477357e2f7fSGreg Kurz     v9fs_rwrite(req, &count);
478357e2f7fSGreg Kurz     g_assert_cmpint(count, ==, sizeof(should_block));
479357e2f7fSGreg Kurz     v9fs_rflush(flush_req);
480357e2f7fSGreg Kurz 
481357e2f7fSGreg Kurz     g_free(wnames[0]);
482357e2f7fSGreg Kurz }
483357e2f7fSGreg Kurz 
484b37d62d6SChristian Schoenebeck static void do_unlinkat(QVirtio9P *v9p, const char *atpath, const char *rpath,
485b37d62d6SChristian Schoenebeck                         uint32_t flags)
486b37d62d6SChristian Schoenebeck {
48765ceee0aSChristian Schoenebeck     g_autofree char *name = g_strdup(rpath);
488b37d62d6SChristian Schoenebeck     uint32_t fid;
489b37d62d6SChristian Schoenebeck     P9Req *req;
490b37d62d6SChristian Schoenebeck 
491569f3b63SChristian Schoenebeck     fid = twalk({ .client = v9p, .path = atpath }).newfid;
492b37d62d6SChristian Schoenebeck 
493b37d62d6SChristian Schoenebeck     req = v9fs_tunlinkat(v9p, fid, name, flags, 0);
494b37d62d6SChristian Schoenebeck     v9fs_req_wait_for_reply(req, NULL);
495b37d62d6SChristian Schoenebeck     v9fs_runlinkat(req);
496b37d62d6SChristian Schoenebeck }
497b37d62d6SChristian Schoenebeck 
49846488b62SChristian Schoenebeck static void fs_readdir_split_128(void *obj, void *data,
49946488b62SChristian Schoenebeck                                  QGuestAllocator *t_alloc)
50046488b62SChristian Schoenebeck {
501684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
5021d98613dSGreg Kurz     do_readdir_split(obj, 128);
50346488b62SChristian Schoenebeck }
50446488b62SChristian Schoenebeck 
50546488b62SChristian Schoenebeck static void fs_readdir_split_256(void *obj, void *data,
50646488b62SChristian Schoenebeck                                  QGuestAllocator *t_alloc)
50746488b62SChristian Schoenebeck {
508684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
5091d98613dSGreg Kurz     do_readdir_split(obj, 256);
51046488b62SChristian Schoenebeck }
51146488b62SChristian Schoenebeck 
51246488b62SChristian Schoenebeck static void fs_readdir_split_512(void *obj, void *data,
51346488b62SChristian Schoenebeck                                  QGuestAllocator *t_alloc)
51446488b62SChristian Schoenebeck {
515684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
5161d98613dSGreg Kurz     do_readdir_split(obj, 512);
51746488b62SChristian Schoenebeck }
51846488b62SChristian Schoenebeck 
519653daf38SChristian Schoenebeck 
520653daf38SChristian Schoenebeck /* tests using the 9pfs 'local' fs driver */
521653daf38SChristian Schoenebeck 
522653daf38SChristian Schoenebeck static void fs_create_dir(void *obj, void *data, QGuestAllocator *t_alloc)
523653daf38SChristian Schoenebeck {
524653daf38SChristian Schoenebeck     QVirtio9P *v9p = obj;
525684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
526653daf38SChristian Schoenebeck     struct stat st;
52765ceee0aSChristian Schoenebeck     g_autofree char *root_path = virtio_9p_test_path("");
52865ceee0aSChristian Schoenebeck     g_autofree char *new_dir = virtio_9p_test_path("01");
529653daf38SChristian Schoenebeck 
530653daf38SChristian Schoenebeck     g_assert(root_path != NULL);
531653daf38SChristian Schoenebeck 
53274a160abSChristian Schoenebeck     tattach({ .client = v9p });
533e1168010SChristian Schoenebeck     tmkdir({ .client = v9p, .atPath = "/", .name = "01" });
534653daf38SChristian Schoenebeck 
535653daf38SChristian Schoenebeck     /* check if created directory really exists now ... */
536653daf38SChristian Schoenebeck     g_assert(stat(new_dir, &st) == 0);
537653daf38SChristian Schoenebeck     /* ... and is actually a directory */
538653daf38SChristian Schoenebeck     g_assert((st.st_mode & S_IFMT) == S_IFDIR);
539653daf38SChristian Schoenebeck }
540653daf38SChristian Schoenebeck 
541b37d62d6SChristian Schoenebeck static void fs_unlinkat_dir(void *obj, void *data, QGuestAllocator *t_alloc)
542b37d62d6SChristian Schoenebeck {
543b37d62d6SChristian Schoenebeck     QVirtio9P *v9p = obj;
544684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
545b37d62d6SChristian Schoenebeck     struct stat st;
54665ceee0aSChristian Schoenebeck     g_autofree char *root_path = virtio_9p_test_path("");
54765ceee0aSChristian Schoenebeck     g_autofree char *new_dir = virtio_9p_test_path("02");
548b37d62d6SChristian Schoenebeck 
549b37d62d6SChristian Schoenebeck     g_assert(root_path != NULL);
550b37d62d6SChristian Schoenebeck 
55174a160abSChristian Schoenebeck     tattach({ .client = v9p });
552e1168010SChristian Schoenebeck     tmkdir({ .client = v9p, .atPath = "/", .name = "02" });
553b37d62d6SChristian Schoenebeck 
554b37d62d6SChristian Schoenebeck     /* check if created directory really exists now ... */
555b37d62d6SChristian Schoenebeck     g_assert(stat(new_dir, &st) == 0);
556b37d62d6SChristian Schoenebeck     /* ... and is actually a directory */
557b37d62d6SChristian Schoenebeck     g_assert((st.st_mode & S_IFMT) == S_IFDIR);
558b37d62d6SChristian Schoenebeck 
559d3671fd9SWill Cohen     do_unlinkat(v9p, "/", "02", P9_DOTL_AT_REMOVEDIR);
560b37d62d6SChristian Schoenebeck     /* directory should be gone now */
561b37d62d6SChristian Schoenebeck     g_assert(stat(new_dir, &st) != 0);
562b37d62d6SChristian Schoenebeck }
563b37d62d6SChristian Schoenebeck 
564b09dbfddSChristian Schoenebeck static void fs_create_file(void *obj, void *data, QGuestAllocator *t_alloc)
565b09dbfddSChristian Schoenebeck {
566b09dbfddSChristian Schoenebeck     QVirtio9P *v9p = obj;
567684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
568b09dbfddSChristian Schoenebeck     struct stat st;
56965ceee0aSChristian Schoenebeck     g_autofree char *new_file = virtio_9p_test_path("03/1st_file");
570b09dbfddSChristian Schoenebeck 
57174a160abSChristian Schoenebeck     tattach({ .client = v9p });
572e1168010SChristian Schoenebeck     tmkdir({ .client = v9p, .atPath = "/", .name = "03" });
573bd4660d4SChristian Schoenebeck     tlcreate({ .client = v9p, .atPath = "03", .name = "1st_file" });
574b09dbfddSChristian Schoenebeck 
575b09dbfddSChristian Schoenebeck     /* check if created file exists now ... */
576b09dbfddSChristian Schoenebeck     g_assert(stat(new_file, &st) == 0);
577b09dbfddSChristian Schoenebeck     /* ... and is a regular file */
578b09dbfddSChristian Schoenebeck     g_assert((st.st_mode & S_IFMT) == S_IFREG);
579b09dbfddSChristian Schoenebeck }
580b09dbfddSChristian Schoenebeck 
581472c18b8SChristian Schoenebeck static void fs_unlinkat_file(void *obj, void *data, QGuestAllocator *t_alloc)
582472c18b8SChristian Schoenebeck {
583472c18b8SChristian Schoenebeck     QVirtio9P *v9p = obj;
584684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
585472c18b8SChristian Schoenebeck     struct stat st;
58665ceee0aSChristian Schoenebeck     g_autofree char *new_file = virtio_9p_test_path("04/doa_file");
587472c18b8SChristian Schoenebeck 
58874a160abSChristian Schoenebeck     tattach({ .client = v9p });
589e1168010SChristian Schoenebeck     tmkdir({ .client = v9p, .atPath = "/", .name = "04" });
590bd4660d4SChristian Schoenebeck     tlcreate({ .client = v9p, .atPath = "04", .name = "doa_file" });
591472c18b8SChristian Schoenebeck 
592472c18b8SChristian Schoenebeck     /* check if created file exists now ... */
593472c18b8SChristian Schoenebeck     g_assert(stat(new_file, &st) == 0);
594472c18b8SChristian Schoenebeck     /* ... and is a regular file */
595472c18b8SChristian Schoenebeck     g_assert((st.st_mode & S_IFMT) == S_IFREG);
596472c18b8SChristian Schoenebeck 
597472c18b8SChristian Schoenebeck     do_unlinkat(v9p, "04", "doa_file", 0);
598472c18b8SChristian Schoenebeck     /* file should be gone now */
599472c18b8SChristian Schoenebeck     g_assert(stat(new_file, &st) != 0);
600472c18b8SChristian Schoenebeck }
601472c18b8SChristian Schoenebeck 
60259ff563dSChristian Schoenebeck static void fs_symlink_file(void *obj, void *data, QGuestAllocator *t_alloc)
60359ff563dSChristian Schoenebeck {
60459ff563dSChristian Schoenebeck     QVirtio9P *v9p = obj;
605684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
60659ff563dSChristian Schoenebeck     struct stat st;
60765ceee0aSChristian Schoenebeck     g_autofree char *real_file = virtio_9p_test_path("05/real_file");
60865ceee0aSChristian Schoenebeck     g_autofree char *symlink_file = virtio_9p_test_path("05/symlink_file");
60959ff563dSChristian Schoenebeck 
61074a160abSChristian Schoenebeck     tattach({ .client = v9p });
611e1168010SChristian Schoenebeck     tmkdir({ .client = v9p, .atPath = "/", .name = "05" });
612bd4660d4SChristian Schoenebeck     tlcreate({ .client = v9p, .atPath = "05", .name = "real_file" });
61359ff563dSChristian Schoenebeck     g_assert(stat(real_file, &st) == 0);
61459ff563dSChristian Schoenebeck     g_assert((st.st_mode & S_IFMT) == S_IFREG);
61559ff563dSChristian Schoenebeck 
6169beabfa5SChristian Schoenebeck     tsymlink({
6179beabfa5SChristian Schoenebeck         .client = v9p, .atPath = "05", .name = "symlink_file",
6189beabfa5SChristian Schoenebeck         .symtgt = "real_file"
6199beabfa5SChristian Schoenebeck     });
62059ff563dSChristian Schoenebeck 
62159ff563dSChristian Schoenebeck     /* check if created link exists now */
62259ff563dSChristian Schoenebeck     g_assert(stat(symlink_file, &st) == 0);
62359ff563dSChristian Schoenebeck }
62459ff563dSChristian Schoenebeck 
6255b28ab8bSChristian Schoenebeck static void fs_unlinkat_symlink(void *obj, void *data,
6265b28ab8bSChristian Schoenebeck                                 QGuestAllocator *t_alloc)
6275b28ab8bSChristian Schoenebeck {
6285b28ab8bSChristian Schoenebeck     QVirtio9P *v9p = obj;
629684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
6305b28ab8bSChristian Schoenebeck     struct stat st;
63165ceee0aSChristian Schoenebeck     g_autofree char *real_file = virtio_9p_test_path("06/real_file");
63265ceee0aSChristian Schoenebeck     g_autofree char *symlink_file = virtio_9p_test_path("06/symlink_file");
6335b28ab8bSChristian Schoenebeck 
63474a160abSChristian Schoenebeck     tattach({ .client = v9p });
635e1168010SChristian Schoenebeck     tmkdir({ .client = v9p, .atPath = "/", .name = "06" });
636bd4660d4SChristian Schoenebeck     tlcreate({ .client = v9p, .atPath = "06", .name = "real_file" });
6375b28ab8bSChristian Schoenebeck     g_assert(stat(real_file, &st) == 0);
6385b28ab8bSChristian Schoenebeck     g_assert((st.st_mode & S_IFMT) == S_IFREG);
6395b28ab8bSChristian Schoenebeck 
6409beabfa5SChristian Schoenebeck     tsymlink({
6419beabfa5SChristian Schoenebeck         .client = v9p, .atPath = "06", .name = "symlink_file",
6429beabfa5SChristian Schoenebeck         .symtgt = "real_file"
6439beabfa5SChristian Schoenebeck     });
6445b28ab8bSChristian Schoenebeck     g_assert(stat(symlink_file, &st) == 0);
6455b28ab8bSChristian Schoenebeck 
6465b28ab8bSChristian Schoenebeck     do_unlinkat(v9p, "06", "symlink_file", 0);
6475b28ab8bSChristian Schoenebeck     /* symlink should be gone now */
6485b28ab8bSChristian Schoenebeck     g_assert(stat(symlink_file, &st) != 0);
6495b28ab8bSChristian Schoenebeck }
6505b28ab8bSChristian Schoenebeck 
65164e3d403SChristian Schoenebeck static void fs_hardlink_file(void *obj, void *data, QGuestAllocator *t_alloc)
65264e3d403SChristian Schoenebeck {
65364e3d403SChristian Schoenebeck     QVirtio9P *v9p = obj;
654684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
65564e3d403SChristian Schoenebeck     struct stat st_real, st_link;
65665ceee0aSChristian Schoenebeck     g_autofree char *real_file = virtio_9p_test_path("07/real_file");
65765ceee0aSChristian Schoenebeck     g_autofree char *hardlink_file = virtio_9p_test_path("07/hardlink_file");
65864e3d403SChristian Schoenebeck 
65974a160abSChristian Schoenebeck     tattach({ .client = v9p });
660e1168010SChristian Schoenebeck     tmkdir({ .client = v9p, .atPath = "/", .name = "07" });
661bd4660d4SChristian Schoenebeck     tlcreate({ .client = v9p, .atPath = "07", .name = "real_file" });
66264e3d403SChristian Schoenebeck     g_assert(stat(real_file, &st_real) == 0);
66364e3d403SChristian Schoenebeck     g_assert((st_real.st_mode & S_IFMT) == S_IFREG);
66464e3d403SChristian Schoenebeck 
665*d41a9462SChristian Schoenebeck     tlink({
666*d41a9462SChristian Schoenebeck         .client = v9p, .atPath = "07", .name = "hardlink_file",
667*d41a9462SChristian Schoenebeck         .toPath = "07/real_file"
668*d41a9462SChristian Schoenebeck     });
66964e3d403SChristian Schoenebeck 
67064e3d403SChristian Schoenebeck     /* check if link exists now ... */
67164e3d403SChristian Schoenebeck     g_assert(stat(hardlink_file, &st_link) == 0);
67264e3d403SChristian Schoenebeck     /* ... and it's a hard link, right? */
67364e3d403SChristian Schoenebeck     g_assert((st_link.st_mode & S_IFMT) == S_IFREG);
67464e3d403SChristian Schoenebeck     g_assert(st_link.st_dev == st_real.st_dev);
67564e3d403SChristian Schoenebeck     g_assert(st_link.st_ino == st_real.st_ino);
67664e3d403SChristian Schoenebeck }
67764e3d403SChristian Schoenebeck 
6784d0746e2SChristian Schoenebeck static void fs_unlinkat_hardlink(void *obj, void *data,
6794d0746e2SChristian Schoenebeck                                  QGuestAllocator *t_alloc)
6804d0746e2SChristian Schoenebeck {
6814d0746e2SChristian Schoenebeck     QVirtio9P *v9p = obj;
682684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
6834d0746e2SChristian Schoenebeck     struct stat st_real, st_link;
68465ceee0aSChristian Schoenebeck     g_autofree char *real_file = virtio_9p_test_path("08/real_file");
68565ceee0aSChristian Schoenebeck     g_autofree char *hardlink_file = virtio_9p_test_path("08/hardlink_file");
6864d0746e2SChristian Schoenebeck 
68774a160abSChristian Schoenebeck     tattach({ .client = v9p });
688e1168010SChristian Schoenebeck     tmkdir({ .client = v9p, .atPath = "/", .name = "08" });
689bd4660d4SChristian Schoenebeck     tlcreate({ .client = v9p, .atPath = "08", .name = "real_file" });
6904d0746e2SChristian Schoenebeck     g_assert(stat(real_file, &st_real) == 0);
6914d0746e2SChristian Schoenebeck     g_assert((st_real.st_mode & S_IFMT) == S_IFREG);
6924d0746e2SChristian Schoenebeck 
693*d41a9462SChristian Schoenebeck     tlink({
694*d41a9462SChristian Schoenebeck         .client = v9p, .atPath = "08", .name = "hardlink_file",
695*d41a9462SChristian Schoenebeck         .toPath = "08/real_file"
696*d41a9462SChristian Schoenebeck     });
6974d0746e2SChristian Schoenebeck     g_assert(stat(hardlink_file, &st_link) == 0);
6984d0746e2SChristian Schoenebeck 
6994d0746e2SChristian Schoenebeck     do_unlinkat(v9p, "08", "hardlink_file", 0);
7004d0746e2SChristian Schoenebeck     /* symlink should be gone now */
7014d0746e2SChristian Schoenebeck     g_assert(stat(hardlink_file, &st_link) != 0);
7024d0746e2SChristian Schoenebeck     /* and old file should still exist */
7034d0746e2SChristian Schoenebeck     g_assert(stat(real_file, &st_real) == 0);
7044d0746e2SChristian Schoenebeck }
7054d0746e2SChristian Schoenebeck 
7063a565c64SChristian Schoenebeck static void *assign_9p_local_driver(GString *cmd_line, void *arg)
7073a565c64SChristian Schoenebeck {
7083a565c64SChristian Schoenebeck     virtio_9p_assign_local_driver(cmd_line, "security_model=mapped-xattr");
7093a565c64SChristian Schoenebeck     return arg;
7103a565c64SChristian Schoenebeck }
7113a565c64SChristian Schoenebeck 
712dfbe8b43SEmanuele Giuseppe Esposito static void register_virtio_9p_test(void)
7131211d81bSGreg Kurz {
7143a565c64SChristian Schoenebeck 
7153a565c64SChristian Schoenebeck     QOSGraphTestOptions opts = {
7163a565c64SChristian Schoenebeck     };
7173a565c64SChristian Schoenebeck 
7183a565c64SChristian Schoenebeck     /* 9pfs test cases using the 'synth' filesystem driver */
7193a565c64SChristian Schoenebeck     qos_add_test("synth/config", "virtio-9p", pci_config, &opts);
7203a565c64SChristian Schoenebeck     qos_add_test("synth/version/basic", "virtio-9p", fs_version,  &opts);
7213a565c64SChristian Schoenebeck     qos_add_test("synth/attach/basic", "virtio-9p", fs_attach,  &opts);
7223a565c64SChristian Schoenebeck     qos_add_test("synth/walk/basic", "virtio-9p", fs_walk,  &opts);
723eefd2394SChristian Schoenebeck     qos_add_test("synth/walk/no_slash", "virtio-9p", fs_walk_no_slash,
7243a565c64SChristian Schoenebeck                   &opts);
725c1668948SChristian Schoenebeck     qos_add_test("synth/walk/none", "virtio-9p", fs_walk_none, &opts);
726eefd2394SChristian Schoenebeck     qos_add_test("synth/walk/dotdot_from_root", "virtio-9p",
7273a565c64SChristian Schoenebeck                  fs_walk_dotdot,  &opts);
7289472a689SChristian Schoenebeck     qos_add_test("synth/walk/non_existent", "virtio-9p", fs_walk_nonexistent,
7299472a689SChristian Schoenebeck                   &opts);
73015fbff48SChristian Schoenebeck     qos_add_test("synth/walk/2nd_non_existent", "virtio-9p",
73115fbff48SChristian Schoenebeck                  fs_walk_2nd_nonexistent, &opts);
7323a565c64SChristian Schoenebeck     qos_add_test("synth/lopen/basic", "virtio-9p", fs_lopen,  &opts);
7333a565c64SChristian Schoenebeck     qos_add_test("synth/write/basic", "virtio-9p", fs_write,  &opts);
734eefd2394SChristian Schoenebeck     qos_add_test("synth/flush/success", "virtio-9p", fs_flush_success,
7353a565c64SChristian Schoenebeck                   &opts);
736eefd2394SChristian Schoenebeck     qos_add_test("synth/flush/ignored", "virtio-9p", fs_flush_ignored,
7373a565c64SChristian Schoenebeck                   &opts);
7383a565c64SChristian Schoenebeck     qos_add_test("synth/readdir/basic", "virtio-9p", fs_readdir,  &opts);
739eefd2394SChristian Schoenebeck     qos_add_test("synth/readdir/split_512", "virtio-9p",
7403a565c64SChristian Schoenebeck                  fs_readdir_split_512,  &opts);
741eefd2394SChristian Schoenebeck     qos_add_test("synth/readdir/split_256", "virtio-9p",
7423a565c64SChristian Schoenebeck                  fs_readdir_split_256,  &opts);
743eefd2394SChristian Schoenebeck     qos_add_test("synth/readdir/split_128", "virtio-9p",
7443a565c64SChristian Schoenebeck                  fs_readdir_split_128,  &opts);
7453a565c64SChristian Schoenebeck 
7463a565c64SChristian Schoenebeck 
7473a565c64SChristian Schoenebeck     /* 9pfs test cases using the 'local' filesystem driver */
748558f5c42SGreg Kurz 
749558f5c42SGreg Kurz     /*
750558f5c42SGreg Kurz      * XXX: Until we are sure that these tests can run everywhere,
751558f5c42SGreg Kurz      * keep them as "slow" so that they aren't run with "make check".
752558f5c42SGreg Kurz      */
753558f5c42SGreg Kurz     if (!g_test_slow()) {
754558f5c42SGreg Kurz         return;
755558f5c42SGreg Kurz     }
756558f5c42SGreg Kurz 
7573a565c64SChristian Schoenebeck     opts.before = assign_9p_local_driver;
7583a565c64SChristian Schoenebeck     qos_add_test("local/config", "virtio-9p", pci_config,  &opts);
759653daf38SChristian Schoenebeck     qos_add_test("local/create_dir", "virtio-9p", fs_create_dir, &opts);
760b37d62d6SChristian Schoenebeck     qos_add_test("local/unlinkat_dir", "virtio-9p", fs_unlinkat_dir, &opts);
761b09dbfddSChristian Schoenebeck     qos_add_test("local/create_file", "virtio-9p", fs_create_file, &opts);
762472c18b8SChristian Schoenebeck     qos_add_test("local/unlinkat_file", "virtio-9p", fs_unlinkat_file, &opts);
76359ff563dSChristian Schoenebeck     qos_add_test("local/symlink_file", "virtio-9p", fs_symlink_file, &opts);
7645b28ab8bSChristian Schoenebeck     qos_add_test("local/unlinkat_symlink", "virtio-9p", fs_unlinkat_symlink,
7655b28ab8bSChristian Schoenebeck                  &opts);
76664e3d403SChristian Schoenebeck     qos_add_test("local/hardlink_file", "virtio-9p", fs_hardlink_file, &opts);
7674d0746e2SChristian Schoenebeck     qos_add_test("local/unlinkat_hardlink", "virtio-9p", fs_unlinkat_hardlink,
7684d0746e2SChristian Schoenebeck                  &opts);
7691211d81bSGreg Kurz }
7701211d81bSGreg Kurz 
771dfbe8b43SEmanuele Giuseppe Esposito libqos_init(register_virtio_9p_test);
772136b7af2SChristian Schoenebeck 
773136b7af2SChristian Schoenebeck static void __attribute__((constructor)) construct_9p_test(void)
774136b7af2SChristian Schoenebeck {
775136b7af2SChristian Schoenebeck     /* make sure test dir for the 'local' tests exists */
776136b7af2SChristian Schoenebeck     virtio_9p_create_local_test_dir();
777136b7af2SChristian Schoenebeck }
778136b7af2SChristian Schoenebeck 
779136b7af2SChristian Schoenebeck static void __attribute__((destructor)) destruct_9p_test(void)
780136b7af2SChristian Schoenebeck {
781136b7af2SChristian Schoenebeck     /* remove previously created test dir when test suite completed */
782136b7af2SChristian Schoenebeck     virtio_9p_remove_local_test_dir();
783136b7af2SChristian Schoenebeck }
784