xref: /qemu/tests/qtest/virtio-9p-test.c (revision 1125ddf66f47dc4986d97948253890fdb3c0a6d6)
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__)
22653daf38SChristian Schoenebeck 
23dfbe8b43SEmanuele Giuseppe Esposito static void pci_config(void *obj, void *data, QGuestAllocator *t_alloc)
24557a4cc0SGreg Kurz {
25dfbe8b43SEmanuele Giuseppe Esposito     QVirtio9P *v9p = obj;
26684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
27dfbe8b43SEmanuele Giuseppe Esposito     size_t tag_len = qvirtio_config_readw(v9p->vdev, 0);
2865ceee0aSChristian Schoenebeck     g_autofree char *tag = NULL;
29557a4cc0SGreg Kurz     int i;
30557a4cc0SGreg Kurz 
31dfbe8b43SEmanuele Giuseppe Esposito     g_assert_cmpint(tag_len, ==, strlen(MOUNT_TAG));
32557a4cc0SGreg Kurz 
33557a4cc0SGreg Kurz     tag = g_malloc(tag_len);
34557a4cc0SGreg Kurz     for (i = 0; i < tag_len; i++) {
35dfbe8b43SEmanuele Giuseppe Esposito         tag[i] = qvirtio_config_readb(v9p->vdev, i + 2);
36557a4cc0SGreg Kurz     }
37dfbe8b43SEmanuele Giuseppe Esposito     g_assert_cmpmem(tag, tag_len, MOUNT_TAG, tag_len);
381211d81bSGreg Kurz }
39557a4cc0SGreg Kurz 
40a6821b82SChristian Schoenebeck static inline bool is_same_qid(v9fs_qid a, v9fs_qid b)
41a6821b82SChristian Schoenebeck {
42a6821b82SChristian Schoenebeck     /* don't compare QID version for checking for file ID equalness */
43a6821b82SChristian Schoenebeck     return a[0] == b[0] && memcmp(&a[5], &b[5], 8) == 0;
44a6821b82SChristian Schoenebeck }
45a6821b82SChristian Schoenebeck 
461c450e6eSGreg Kurz static void fs_version(void *obj, void *data, QGuestAllocator *t_alloc)
471c450e6eSGreg Kurz {
48684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
49bee8fda2SChristian Schoenebeck     tversion({ .client = obj });
501c450e6eSGreg Kurz }
511c450e6eSGreg Kurz 
523fe4baf4SGreg Kurz static void fs_attach(void *obj, void *data, QGuestAllocator *t_alloc)
533fe4baf4SGreg Kurz {
54684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
5574a160abSChristian Schoenebeck     tattach({ .client = obj });
563fe4baf4SGreg Kurz }
573fe4baf4SGreg Kurz 
58dfbe8b43SEmanuele Giuseppe Esposito static void fs_walk(void *obj, void *data, QGuestAllocator *t_alloc)
5904b88c84SGreg Kurz {
60dfbe8b43SEmanuele Giuseppe Esposito     QVirtio9P *v9p = obj;
61684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
622893ddd5SGreg Kurz     char *wnames[P9_MAXWELEM];
6304b88c84SGreg Kurz     uint16_t nwqid;
6465ceee0aSChristian Schoenebeck     g_autofree v9fs_qid *wqid = NULL;
6504b88c84SGreg Kurz     int i;
6604b88c84SGreg Kurz 
6704b88c84SGreg Kurz     for (i = 0; i < P9_MAXWELEM; i++) {
682893ddd5SGreg Kurz         wnames[i] = g_strdup_printf(QTEST_V9FS_SYNTH_WALK_FILE, i);
6904b88c84SGreg Kurz     }
7004b88c84SGreg Kurz 
7174a160abSChristian Schoenebeck     tattach({ .client = v9p });
723f3e9232SChristian Schoenebeck     twalk({
73569f3b63SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1,
743f3e9232SChristian Schoenebeck         .nwname = P9_MAXWELEM, .wnames = wnames,
753f3e9232SChristian Schoenebeck         .rwalk = { .nwqid = &nwqid, .wqid = &wqid }
763f3e9232SChristian Schoenebeck     });
7704b88c84SGreg Kurz 
7804b88c84SGreg Kurz     g_assert_cmpint(nwqid, ==, P9_MAXWELEM);
7904b88c84SGreg Kurz 
8004b88c84SGreg Kurz     for (i = 0; i < P9_MAXWELEM; i++) {
8104b88c84SGreg Kurz         g_free(wnames[i]);
8204b88c84SGreg Kurz     }
8304b88c84SGreg Kurz }
8404b88c84SGreg Kurz 
854829469fSChristian Schoenebeck static bool fs_dirents_contain_name(struct V9fsDirent *e, const char* name)
864829469fSChristian Schoenebeck {
874829469fSChristian Schoenebeck     for (; e; e = e->next) {
884829469fSChristian Schoenebeck         if (!strcmp(e->name, name)) {
894829469fSChristian Schoenebeck             return true;
904829469fSChristian Schoenebeck         }
914829469fSChristian Schoenebeck     }
924829469fSChristian Schoenebeck     return false;
934829469fSChristian Schoenebeck }
944829469fSChristian Schoenebeck 
9546488b62SChristian Schoenebeck /* basic readdir test where reply fits into a single response message */
964829469fSChristian Schoenebeck static void fs_readdir(void *obj, void *data, QGuestAllocator *t_alloc)
974829469fSChristian Schoenebeck {
984829469fSChristian Schoenebeck     QVirtio9P *v9p = obj;
99684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
100569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_READDIR_DIR) };
1014829469fSChristian Schoenebeck     uint16_t nqid;
1024829469fSChristian Schoenebeck     v9fs_qid qid;
1034829469fSChristian Schoenebeck     uint32_t count, nentries;
1044829469fSChristian Schoenebeck     struct V9fsDirent *entries = NULL;
1054829469fSChristian Schoenebeck     P9Req *req;
1064829469fSChristian Schoenebeck 
10774a160abSChristian Schoenebeck     tattach({ .client = v9p });
1083f3e9232SChristian Schoenebeck     twalk({
109569f3b63SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1,
1103f3e9232SChristian Schoenebeck         .nwname = 1, .wnames = wnames, .rwalk.nwqid = &nqid
1113f3e9232SChristian Schoenebeck     });
1124829469fSChristian Schoenebeck     g_assert_cmpint(nqid, ==, 1);
1134829469fSChristian Schoenebeck 
1144829469fSChristian Schoenebeck     req = v9fs_tlopen(v9p, 1, O_DIRECTORY, 0);
1154829469fSChristian Schoenebeck     v9fs_req_wait_for_reply(req, NULL);
1164829469fSChristian Schoenebeck     v9fs_rlopen(req, &qid, NULL);
1174829469fSChristian Schoenebeck 
1184829469fSChristian Schoenebeck     /*
1194829469fSChristian Schoenebeck      * submit count = msize - 11, because 11 is the header size of Rreaddir
1204829469fSChristian Schoenebeck      */
1214829469fSChristian Schoenebeck     req = v9fs_treaddir(v9p, 1, 0, P9_MAX_SIZE - 11, 0);
1224829469fSChristian Schoenebeck     v9fs_req_wait_for_reply(req, NULL);
1234829469fSChristian Schoenebeck     v9fs_rreaddir(req, &count, &nentries, &entries);
1244829469fSChristian Schoenebeck 
1254829469fSChristian Schoenebeck     /*
1264829469fSChristian Schoenebeck      * Assuming msize (P9_MAX_SIZE) is large enough so we can retrieve all
1274829469fSChristian Schoenebeck      * dir entries with only one readdir request.
1284829469fSChristian Schoenebeck      */
1294829469fSChristian Schoenebeck     g_assert_cmpint(
1304829469fSChristian Schoenebeck         nentries, ==,
1314829469fSChristian Schoenebeck         QTEST_V9FS_SYNTH_READDIR_NFILES + 2 /* "." and ".." */
1324829469fSChristian Schoenebeck     );
1334829469fSChristian Schoenebeck 
1344829469fSChristian Schoenebeck     /*
1354829469fSChristian Schoenebeck      * Check all file names exist in returned entries, ignore their order
1364829469fSChristian Schoenebeck      * though.
1374829469fSChristian Schoenebeck      */
1384829469fSChristian Schoenebeck     g_assert_cmpint(fs_dirents_contain_name(entries, "."), ==, true);
1394829469fSChristian Schoenebeck     g_assert_cmpint(fs_dirents_contain_name(entries, ".."), ==, true);
1404829469fSChristian Schoenebeck     for (int i = 0; i < QTEST_V9FS_SYNTH_READDIR_NFILES; ++i) {
14165ceee0aSChristian Schoenebeck         g_autofree char *name =
14265ceee0aSChristian Schoenebeck             g_strdup_printf(QTEST_V9FS_SYNTH_READDIR_FILE, i);
1434829469fSChristian Schoenebeck         g_assert_cmpint(fs_dirents_contain_name(entries, name), ==, true);
1444829469fSChristian Schoenebeck     }
1454829469fSChristian Schoenebeck 
1464829469fSChristian Schoenebeck     v9fs_free_dirents(entries);
1474829469fSChristian Schoenebeck     g_free(wnames[0]);
1484829469fSChristian Schoenebeck }
1494829469fSChristian Schoenebeck 
15046488b62SChristian Schoenebeck /* readdir test where overall request is split over several messages */
1511d98613dSGreg Kurz static void do_readdir_split(QVirtio9P *v9p, uint32_t count)
15246488b62SChristian Schoenebeck {
153569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_READDIR_DIR) };
15446488b62SChristian Schoenebeck     uint16_t nqid;
15546488b62SChristian Schoenebeck     v9fs_qid qid;
15646488b62SChristian Schoenebeck     uint32_t nentries, npartialentries;
15746488b62SChristian Schoenebeck     struct V9fsDirent *entries, *tail, *partialentries;
15846488b62SChristian Schoenebeck     P9Req *req;
15946488b62SChristian Schoenebeck     int fid;
16046488b62SChristian Schoenebeck     uint64_t offset;
16146488b62SChristian Schoenebeck 
16274a160abSChristian Schoenebeck     tattach({ .client = v9p });
16346488b62SChristian Schoenebeck 
16446488b62SChristian Schoenebeck     fid = 1;
16546488b62SChristian Schoenebeck     offset = 0;
16646488b62SChristian Schoenebeck     entries = NULL;
16746488b62SChristian Schoenebeck     nentries = 0;
16846488b62SChristian Schoenebeck     tail = NULL;
16946488b62SChristian Schoenebeck 
1703f3e9232SChristian Schoenebeck     twalk({
171569f3b63SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = fid,
1723f3e9232SChristian Schoenebeck         .nwname = 1, .wnames = wnames, .rwalk.nwqid = &nqid
1733f3e9232SChristian Schoenebeck     });
17446488b62SChristian Schoenebeck     g_assert_cmpint(nqid, ==, 1);
17546488b62SChristian Schoenebeck 
17646488b62SChristian Schoenebeck     req = v9fs_tlopen(v9p, fid, O_DIRECTORY, 0);
17746488b62SChristian Schoenebeck     v9fs_req_wait_for_reply(req, NULL);
17846488b62SChristian Schoenebeck     v9fs_rlopen(req, &qid, NULL);
17946488b62SChristian Schoenebeck 
18046488b62SChristian Schoenebeck     /*
18146488b62SChristian Schoenebeck      * send as many Treaddir requests as required to get all directory
18246488b62SChristian Schoenebeck      * entries
18346488b62SChristian Schoenebeck      */
18446488b62SChristian Schoenebeck     while (true) {
18546488b62SChristian Schoenebeck         npartialentries = 0;
18646488b62SChristian Schoenebeck         partialentries = NULL;
18746488b62SChristian Schoenebeck 
18846488b62SChristian Schoenebeck         req = v9fs_treaddir(v9p, fid, offset, count, 0);
18946488b62SChristian Schoenebeck         v9fs_req_wait_for_reply(req, NULL);
19046488b62SChristian Schoenebeck         v9fs_rreaddir(req, &count, &npartialentries, &partialentries);
19146488b62SChristian Schoenebeck         if (npartialentries > 0 && partialentries) {
19246488b62SChristian Schoenebeck             if (!entries) {
19346488b62SChristian Schoenebeck                 entries = partialentries;
19446488b62SChristian Schoenebeck                 nentries = npartialentries;
19546488b62SChristian Schoenebeck                 tail = partialentries;
19646488b62SChristian Schoenebeck             } else {
19746488b62SChristian Schoenebeck                 tail->next = partialentries;
19846488b62SChristian Schoenebeck                 nentries += npartialentries;
19946488b62SChristian Schoenebeck             }
20046488b62SChristian Schoenebeck             while (tail->next) {
20146488b62SChristian Schoenebeck                 tail = tail->next;
20246488b62SChristian Schoenebeck             }
20346488b62SChristian Schoenebeck             offset = tail->offset;
20446488b62SChristian Schoenebeck         } else {
20546488b62SChristian Schoenebeck             break;
20646488b62SChristian Schoenebeck         }
20746488b62SChristian Schoenebeck     }
20846488b62SChristian Schoenebeck 
20946488b62SChristian Schoenebeck     g_assert_cmpint(
21046488b62SChristian Schoenebeck         nentries, ==,
21146488b62SChristian Schoenebeck         QTEST_V9FS_SYNTH_READDIR_NFILES + 2 /* "." and ".." */
21246488b62SChristian Schoenebeck     );
21346488b62SChristian Schoenebeck 
21446488b62SChristian Schoenebeck     /*
21546488b62SChristian Schoenebeck      * Check all file names exist in returned entries, ignore their order
21646488b62SChristian Schoenebeck      * though.
21746488b62SChristian Schoenebeck      */
21846488b62SChristian Schoenebeck     g_assert_cmpint(fs_dirents_contain_name(entries, "."), ==, true);
21946488b62SChristian Schoenebeck     g_assert_cmpint(fs_dirents_contain_name(entries, ".."), ==, true);
22046488b62SChristian Schoenebeck     for (int i = 0; i < QTEST_V9FS_SYNTH_READDIR_NFILES; ++i) {
22146488b62SChristian Schoenebeck         char *name = g_strdup_printf(QTEST_V9FS_SYNTH_READDIR_FILE, i);
22246488b62SChristian Schoenebeck         g_assert_cmpint(fs_dirents_contain_name(entries, name), ==, true);
22346488b62SChristian Schoenebeck         g_free(name);
22446488b62SChristian Schoenebeck     }
22546488b62SChristian Schoenebeck 
22646488b62SChristian Schoenebeck     v9fs_free_dirents(entries);
22746488b62SChristian Schoenebeck 
22846488b62SChristian Schoenebeck     g_free(wnames[0]);
22946488b62SChristian Schoenebeck }
23046488b62SChristian Schoenebeck 
231dfbe8b43SEmanuele Giuseppe Esposito static void fs_walk_no_slash(void *obj, void *data, QGuestAllocator *t_alloc)
232ba0d1037SGreg Kurz {
233dfbe8b43SEmanuele Giuseppe Esposito     QVirtio9P *v9p = obj;
234684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
235569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup(" /") };
236ba0d1037SGreg Kurz 
23774a160abSChristian Schoenebeck     tattach({ .client = v9p });
2383f3e9232SChristian Schoenebeck     twalk({
239569f3b63SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames,
2403f3e9232SChristian Schoenebeck         .expectErr = ENOENT
2413f3e9232SChristian Schoenebeck     });
242ba0d1037SGreg Kurz 
243ba0d1037SGreg Kurz     g_free(wnames[0]);
244ba0d1037SGreg Kurz }
245ba0d1037SGreg Kurz 
2469472a689SChristian Schoenebeck static void fs_walk_nonexistent(void *obj, void *data, QGuestAllocator *t_alloc)
2479472a689SChristian Schoenebeck {
2489472a689SChristian Schoenebeck     QVirtio9P *v9p = obj;
249684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
2509472a689SChristian Schoenebeck 
25174a160abSChristian Schoenebeck     tattach({ .client = v9p });
25215fbff48SChristian Schoenebeck     /*
25315fbff48SChristian Schoenebeck      * The 9p2000 protocol spec says: "If the first element cannot be walked
25415fbff48SChristian Schoenebeck      * for any reason, Rerror is returned."
25515fbff48SChristian Schoenebeck      */
256569f3b63SChristian Schoenebeck     twalk({ .client = v9p, .path = "non-existent", .expectErr = ENOENT });
2579472a689SChristian Schoenebeck }
2589472a689SChristian Schoenebeck 
25915fbff48SChristian Schoenebeck static void fs_walk_2nd_nonexistent(void *obj, void *data,
26015fbff48SChristian Schoenebeck                                     QGuestAllocator *t_alloc)
26115fbff48SChristian Schoenebeck {
26215fbff48SChristian Schoenebeck     QVirtio9P *v9p = obj;
263684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
2640e43495dSChristian Schoenebeck     v9fs_qid root_qid;
26515fbff48SChristian Schoenebeck     uint16_t nwqid;
2660e43495dSChristian Schoenebeck     uint32_t fid, err;
2670e43495dSChristian Schoenebeck     P9Req *req;
26815fbff48SChristian Schoenebeck     g_autofree v9fs_qid *wqid = NULL;
26915fbff48SChristian Schoenebeck     g_autofree char *path = g_strdup_printf(
27015fbff48SChristian Schoenebeck         QTEST_V9FS_SYNTH_WALK_FILE "/non-existent", 0
27115fbff48SChristian Schoenebeck     );
27215fbff48SChristian Schoenebeck 
27374a160abSChristian Schoenebeck     tattach({ .client = v9p, .rattach.qid = &root_qid });
274569f3b63SChristian Schoenebeck     fid = twalk({
275569f3b63SChristian Schoenebeck         .client = v9p, .path = path,
2763f3e9232SChristian Schoenebeck         .rwalk = { .nwqid = &nwqid, .wqid = &wqid }
277569f3b63SChristian Schoenebeck     }).newfid;
27815fbff48SChristian Schoenebeck     /*
27915fbff48SChristian Schoenebeck      * The 9p2000 protocol spec says: "nwqid is therefore either nwname or the
28015fbff48SChristian Schoenebeck      * index of the first elementwise walk that failed."
28115fbff48SChristian Schoenebeck      */
28215fbff48SChristian Schoenebeck     assert(nwqid == 1);
2830e43495dSChristian Schoenebeck 
2840e43495dSChristian Schoenebeck     /* returned QID wqid[0] is file ID of 1st subdir */
2850e43495dSChristian Schoenebeck     g_assert(wqid && wqid[0] && !is_same_qid(root_qid, wqid[0]));
2860e43495dSChristian Schoenebeck 
2870e43495dSChristian Schoenebeck     /* expect fid being unaffected by walk above */
2880e43495dSChristian Schoenebeck     req = v9fs_tgetattr(v9p, fid, P9_GETATTR_BASIC, 0);
2890e43495dSChristian Schoenebeck     v9fs_req_wait_for_reply(req, NULL);
2900e43495dSChristian Schoenebeck     v9fs_rlerror(req, &err);
2910e43495dSChristian Schoenebeck 
2920e43495dSChristian Schoenebeck     g_assert_cmpint(err, ==, ENOENT);
29315fbff48SChristian Schoenebeck }
29415fbff48SChristian Schoenebeck 
295c1668948SChristian Schoenebeck static void fs_walk_none(void *obj, void *data, QGuestAllocator *t_alloc)
296c1668948SChristian Schoenebeck {
297c1668948SChristian Schoenebeck     QVirtio9P *v9p = obj;
298684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
299c1668948SChristian Schoenebeck     v9fs_qid root_qid;
300c1668948SChristian Schoenebeck     g_autofree v9fs_qid *wqid = NULL;
301c1668948SChristian Schoenebeck     P9Req *req;
302a6821b82SChristian Schoenebeck     struct v9fs_attr attr;
303c1668948SChristian Schoenebeck 
304bee8fda2SChristian Schoenebeck     tversion({ .client = v9p });
305*1125ddf6SChristian Schoenebeck     tattach({
306*1125ddf6SChristian Schoenebeck         .client = v9p, .fid = 0, .n_uname = getuid(),
307*1125ddf6SChristian Schoenebeck         .rattach.qid = &root_qid
308*1125ddf6SChristian Schoenebeck     });
309c1668948SChristian Schoenebeck 
3103f3e9232SChristian Schoenebeck     twalk({
311569f3b63SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1, .nwname = 0, .wnames = NULL,
3123f3e9232SChristian Schoenebeck         .rwalk.wqid = &wqid
3133f3e9232SChristian Schoenebeck     });
314c1668948SChristian Schoenebeck 
315c1668948SChristian Schoenebeck     /* special case: no QID is returned if nwname=0 was sent */
316c1668948SChristian Schoenebeck     g_assert(wqid == NULL);
317a6821b82SChristian Schoenebeck 
318a6821b82SChristian Schoenebeck     req = v9fs_tgetattr(v9p, 1, P9_GETATTR_BASIC, 0);
319a6821b82SChristian Schoenebeck     v9fs_req_wait_for_reply(req, NULL);
320a6821b82SChristian Schoenebeck     v9fs_rgetattr(req, &attr);
321a6821b82SChristian Schoenebeck 
322a6821b82SChristian Schoenebeck     g_assert(is_same_qid(root_qid, attr.qid));
323c1668948SChristian Schoenebeck }
324c1668948SChristian Schoenebeck 
325dfbe8b43SEmanuele Giuseppe Esposito static void fs_walk_dotdot(void *obj, void *data, QGuestAllocator *t_alloc)
326a37c0702SGreg Kurz {
327dfbe8b43SEmanuele Giuseppe Esposito     QVirtio9P *v9p = obj;
328684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
329569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup("..") };
33065ceee0aSChristian Schoenebeck     v9fs_qid root_qid;
33165ceee0aSChristian Schoenebeck     g_autofree v9fs_qid *wqid = NULL;
332a37c0702SGreg Kurz 
333bee8fda2SChristian Schoenebeck     tversion({ .client = v9p });
334*1125ddf6SChristian Schoenebeck     tattach({
335*1125ddf6SChristian Schoenebeck         .client = v9p, .fid = 0, .n_uname = getuid(),
336*1125ddf6SChristian Schoenebeck         .rattach.qid = &root_qid
337*1125ddf6SChristian Schoenebeck     });
338a37c0702SGreg Kurz 
3393f3e9232SChristian Schoenebeck     twalk({
340569f3b63SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames,
3413f3e9232SChristian Schoenebeck         .rwalk.wqid = &wqid /* We now we'll get one qid */
3423f3e9232SChristian Schoenebeck     });
343a37c0702SGreg Kurz 
344a37c0702SGreg Kurz     g_assert_cmpmem(&root_qid, 13, wqid[0], 13);
345a37c0702SGreg Kurz 
346a37c0702SGreg Kurz     g_free(wnames[0]);
347a37c0702SGreg Kurz }
348a37c0702SGreg Kurz 
349dfbe8b43SEmanuele Giuseppe Esposito static void fs_lopen(void *obj, void *data, QGuestAllocator *t_alloc)
35082469aaeSGreg Kurz {
351dfbe8b43SEmanuele Giuseppe Esposito     QVirtio9P *v9p = obj;
352684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
353569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_LOPEN_FILE) };
35482469aaeSGreg Kurz     P9Req *req;
35582469aaeSGreg Kurz 
35674a160abSChristian Schoenebeck     tattach({ .client = v9p });
3573f3e9232SChristian Schoenebeck     twalk({
3583f3e9232SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames
3593f3e9232SChristian Schoenebeck     });
36082469aaeSGreg Kurz 
36182469aaeSGreg Kurz     req = v9fs_tlopen(v9p, 1, O_WRONLY, 0);
362357e2f7fSGreg Kurz     v9fs_req_wait_for_reply(req, NULL);
36382469aaeSGreg Kurz     v9fs_rlopen(req, NULL, NULL);
36482469aaeSGreg Kurz 
36582469aaeSGreg Kurz     g_free(wnames[0]);
36682469aaeSGreg Kurz }
36782469aaeSGreg Kurz 
368dfbe8b43SEmanuele Giuseppe Esposito static void fs_write(void *obj, void *data, QGuestAllocator *t_alloc)
369354b86f8SGreg Kurz {
370dfbe8b43SEmanuele Giuseppe Esposito     QVirtio9P *v9p = obj;
371684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
372354b86f8SGreg Kurz     static const uint32_t write_count = P9_MAX_SIZE / 2;
373569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_WRITE_FILE) };
37465ceee0aSChristian Schoenebeck     g_autofree char *buf = g_malloc0(write_count);
375354b86f8SGreg Kurz     uint32_t count;
376354b86f8SGreg Kurz     P9Req *req;
377354b86f8SGreg Kurz 
37874a160abSChristian Schoenebeck     tattach({ .client = v9p });
3793f3e9232SChristian Schoenebeck     twalk({
3803f3e9232SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames
3813f3e9232SChristian Schoenebeck     });
382354b86f8SGreg Kurz 
383354b86f8SGreg Kurz     req = v9fs_tlopen(v9p, 1, O_WRONLY, 0);
384357e2f7fSGreg Kurz     v9fs_req_wait_for_reply(req, NULL);
385354b86f8SGreg Kurz     v9fs_rlopen(req, NULL, NULL);
386354b86f8SGreg Kurz 
387354b86f8SGreg Kurz     req = v9fs_twrite(v9p, 1, 0, write_count, buf, 0);
388357e2f7fSGreg Kurz     v9fs_req_wait_for_reply(req, NULL);
389354b86f8SGreg Kurz     v9fs_rwrite(req, &count);
390354b86f8SGreg Kurz     g_assert_cmpint(count, ==, write_count);
391354b86f8SGreg Kurz 
392354b86f8SGreg Kurz     g_free(wnames[0]);
393354b86f8SGreg Kurz }
394354b86f8SGreg Kurz 
395dfbe8b43SEmanuele Giuseppe Esposito static void fs_flush_success(void *obj, void *data, QGuestAllocator *t_alloc)
396357e2f7fSGreg Kurz {
397dfbe8b43SEmanuele Giuseppe Esposito     QVirtio9P *v9p = obj;
398684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
399569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_FLUSH_FILE) };
400357e2f7fSGreg Kurz     P9Req *req, *flush_req;
401357e2f7fSGreg Kurz     uint32_t reply_len;
402357e2f7fSGreg Kurz     uint8_t should_block;
403357e2f7fSGreg Kurz 
40474a160abSChristian Schoenebeck     tattach({ .client = v9p });
4053f3e9232SChristian Schoenebeck     twalk({
4063f3e9232SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames
4073f3e9232SChristian Schoenebeck     });
408357e2f7fSGreg Kurz 
409357e2f7fSGreg Kurz     req = v9fs_tlopen(v9p, 1, O_WRONLY, 0);
410357e2f7fSGreg Kurz     v9fs_req_wait_for_reply(req, NULL);
411357e2f7fSGreg Kurz     v9fs_rlopen(req, NULL, NULL);
412357e2f7fSGreg Kurz 
413357e2f7fSGreg Kurz     /* This will cause the 9p server to try to write data to the backend,
414357e2f7fSGreg Kurz      * until the write request gets cancelled.
415357e2f7fSGreg Kurz      */
416357e2f7fSGreg Kurz     should_block = 1;
417357e2f7fSGreg Kurz     req = v9fs_twrite(v9p, 1, 0, sizeof(should_block), &should_block, 0);
418357e2f7fSGreg Kurz 
419357e2f7fSGreg Kurz     flush_req = v9fs_tflush(v9p, req->tag, 1);
420357e2f7fSGreg Kurz 
421357e2f7fSGreg Kurz     /* The write request is supposed to be flushed: the server should just
422357e2f7fSGreg Kurz      * mark the write request as used and reply to the flush request.
423357e2f7fSGreg Kurz      */
424357e2f7fSGreg Kurz     v9fs_req_wait_for_reply(req, &reply_len);
425357e2f7fSGreg Kurz     g_assert_cmpint(reply_len, ==, 0);
426357e2f7fSGreg Kurz     v9fs_req_free(req);
427357e2f7fSGreg Kurz     v9fs_rflush(flush_req);
428357e2f7fSGreg Kurz 
429357e2f7fSGreg Kurz     g_free(wnames[0]);
430357e2f7fSGreg Kurz }
431357e2f7fSGreg Kurz 
432dfbe8b43SEmanuele Giuseppe Esposito static void fs_flush_ignored(void *obj, void *data, QGuestAllocator *t_alloc)
433357e2f7fSGreg Kurz {
434dfbe8b43SEmanuele Giuseppe Esposito     QVirtio9P *v9p = obj;
435684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
436569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_FLUSH_FILE) };
437357e2f7fSGreg Kurz     P9Req *req, *flush_req;
438357e2f7fSGreg Kurz     uint32_t count;
439357e2f7fSGreg Kurz     uint8_t should_block;
440357e2f7fSGreg Kurz 
44174a160abSChristian Schoenebeck     tattach({ .client = v9p });
4423f3e9232SChristian Schoenebeck     twalk({
4433f3e9232SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames
4443f3e9232SChristian Schoenebeck     });
445357e2f7fSGreg Kurz 
446357e2f7fSGreg Kurz     req = v9fs_tlopen(v9p, 1, O_WRONLY, 0);
447357e2f7fSGreg Kurz     v9fs_req_wait_for_reply(req, NULL);
448357e2f7fSGreg Kurz     v9fs_rlopen(req, NULL, NULL);
449357e2f7fSGreg Kurz 
450357e2f7fSGreg Kurz     /* This will cause the write request to complete right away, before it
451357e2f7fSGreg Kurz      * could be actually cancelled.
452357e2f7fSGreg Kurz      */
453357e2f7fSGreg Kurz     should_block = 0;
454357e2f7fSGreg Kurz     req = v9fs_twrite(v9p, 1, 0, sizeof(should_block), &should_block, 0);
455357e2f7fSGreg Kurz 
456357e2f7fSGreg Kurz     flush_req = v9fs_tflush(v9p, req->tag, 1);
457357e2f7fSGreg Kurz 
458357e2f7fSGreg Kurz     /* The write request is supposed to complete. The server should
459357e2f7fSGreg Kurz      * reply to the write request and the flush request.
460357e2f7fSGreg Kurz      */
461357e2f7fSGreg Kurz     v9fs_req_wait_for_reply(req, NULL);
462357e2f7fSGreg Kurz     v9fs_rwrite(req, &count);
463357e2f7fSGreg Kurz     g_assert_cmpint(count, ==, sizeof(should_block));
464357e2f7fSGreg Kurz     v9fs_rflush(flush_req);
465357e2f7fSGreg Kurz 
466357e2f7fSGreg Kurz     g_free(wnames[0]);
467357e2f7fSGreg Kurz }
468357e2f7fSGreg Kurz 
469c1934f63SGreg Kurz static void do_mkdir(QVirtio9P *v9p, const char *path, const char *cname)
470653daf38SChristian Schoenebeck {
47165ceee0aSChristian Schoenebeck     g_autofree char *name = g_strdup(cname);
47220018805SChristian Schoenebeck     uint32_t fid;
473653daf38SChristian Schoenebeck     P9Req *req;
474653daf38SChristian Schoenebeck 
475569f3b63SChristian Schoenebeck     fid = twalk({ .client = v9p, .path = path }).newfid;
476653daf38SChristian Schoenebeck 
477653daf38SChristian Schoenebeck     req = v9fs_tmkdir(v9p, fid, name, 0750, 0, 0);
478653daf38SChristian Schoenebeck     v9fs_req_wait_for_reply(req, NULL);
479653daf38SChristian Schoenebeck     v9fs_rmkdir(req, NULL);
480653daf38SChristian Schoenebeck }
481653daf38SChristian Schoenebeck 
482b09dbfddSChristian Schoenebeck /* create a regular file with Tlcreate and return file's fid */
483b09dbfddSChristian Schoenebeck static uint32_t do_lcreate(QVirtio9P *v9p, const char *path,
484b09dbfddSChristian Schoenebeck                            const char *cname)
485b09dbfddSChristian Schoenebeck {
48665ceee0aSChristian Schoenebeck     g_autofree char *name = g_strdup(cname);
487b09dbfddSChristian Schoenebeck     uint32_t fid;
488b09dbfddSChristian Schoenebeck     P9Req *req;
489b09dbfddSChristian Schoenebeck 
490569f3b63SChristian Schoenebeck     fid = twalk({ .client = v9p, .path = path }).newfid;
491b09dbfddSChristian Schoenebeck 
492b09dbfddSChristian Schoenebeck     req = v9fs_tlcreate(v9p, fid, name, 0, 0750, 0, 0);
493b09dbfddSChristian Schoenebeck     v9fs_req_wait_for_reply(req, NULL);
494b09dbfddSChristian Schoenebeck     v9fs_rlcreate(req, NULL, NULL);
495b09dbfddSChristian Schoenebeck 
496b09dbfddSChristian Schoenebeck     return fid;
497b09dbfddSChristian Schoenebeck }
498b09dbfddSChristian Schoenebeck 
49959ff563dSChristian Schoenebeck /* create symlink named @a clink in directory @a path pointing to @a to */
50059ff563dSChristian Schoenebeck static void do_symlink(QVirtio9P *v9p, const char *path, const char *clink,
50159ff563dSChristian Schoenebeck                        const char *to)
50259ff563dSChristian Schoenebeck {
50365ceee0aSChristian Schoenebeck     g_autofree char *name = g_strdup(clink);
50465ceee0aSChristian Schoenebeck     g_autofree char *dst = g_strdup(to);
50559ff563dSChristian Schoenebeck     uint32_t fid;
50659ff563dSChristian Schoenebeck     P9Req *req;
50759ff563dSChristian Schoenebeck 
508569f3b63SChristian Schoenebeck     fid = twalk({ .client = v9p, .path = path }).newfid;
50959ff563dSChristian Schoenebeck 
51059ff563dSChristian Schoenebeck     req = v9fs_tsymlink(v9p, fid, name, dst, 0, 0);
51159ff563dSChristian Schoenebeck     v9fs_req_wait_for_reply(req, NULL);
51259ff563dSChristian Schoenebeck     v9fs_rsymlink(req, NULL);
51359ff563dSChristian Schoenebeck }
51459ff563dSChristian Schoenebeck 
51564e3d403SChristian Schoenebeck /* create a hard link named @a clink in directory @a path pointing to @a to */
51664e3d403SChristian Schoenebeck static void do_hardlink(QVirtio9P *v9p, const char *path, const char *clink,
51764e3d403SChristian Schoenebeck                         const char *to)
51864e3d403SChristian Schoenebeck {
51964e3d403SChristian Schoenebeck     uint32_t dfid, fid;
52064e3d403SChristian Schoenebeck     P9Req *req;
52164e3d403SChristian Schoenebeck 
522569f3b63SChristian Schoenebeck     dfid = twalk({ .client = v9p, .path = path }).newfid;
523569f3b63SChristian Schoenebeck     fid = twalk({ .client = v9p, .path = to }).newfid;
52464e3d403SChristian Schoenebeck 
52564e3d403SChristian Schoenebeck     req = v9fs_tlink(v9p, dfid, fid, clink, 0);
52664e3d403SChristian Schoenebeck     v9fs_req_wait_for_reply(req, NULL);
52764e3d403SChristian Schoenebeck     v9fs_rlink(req);
52864e3d403SChristian Schoenebeck }
52964e3d403SChristian Schoenebeck 
530b37d62d6SChristian Schoenebeck static void do_unlinkat(QVirtio9P *v9p, const char *atpath, const char *rpath,
531b37d62d6SChristian Schoenebeck                         uint32_t flags)
532b37d62d6SChristian Schoenebeck {
53365ceee0aSChristian Schoenebeck     g_autofree char *name = g_strdup(rpath);
534b37d62d6SChristian Schoenebeck     uint32_t fid;
535b37d62d6SChristian Schoenebeck     P9Req *req;
536b37d62d6SChristian Schoenebeck 
537569f3b63SChristian Schoenebeck     fid = twalk({ .client = v9p, .path = atpath }).newfid;
538b37d62d6SChristian Schoenebeck 
539b37d62d6SChristian Schoenebeck     req = v9fs_tunlinkat(v9p, fid, name, flags, 0);
540b37d62d6SChristian Schoenebeck     v9fs_req_wait_for_reply(req, NULL);
541b37d62d6SChristian Schoenebeck     v9fs_runlinkat(req);
542b37d62d6SChristian Schoenebeck }
543b37d62d6SChristian Schoenebeck 
54446488b62SChristian Schoenebeck static void fs_readdir_split_128(void *obj, void *data,
54546488b62SChristian Schoenebeck                                  QGuestAllocator *t_alloc)
54646488b62SChristian Schoenebeck {
547684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
5481d98613dSGreg Kurz     do_readdir_split(obj, 128);
54946488b62SChristian Schoenebeck }
55046488b62SChristian Schoenebeck 
55146488b62SChristian Schoenebeck static void fs_readdir_split_256(void *obj, void *data,
55246488b62SChristian Schoenebeck                                  QGuestAllocator *t_alloc)
55346488b62SChristian Schoenebeck {
554684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
5551d98613dSGreg Kurz     do_readdir_split(obj, 256);
55646488b62SChristian Schoenebeck }
55746488b62SChristian Schoenebeck 
55846488b62SChristian Schoenebeck static void fs_readdir_split_512(void *obj, void *data,
55946488b62SChristian Schoenebeck                                  QGuestAllocator *t_alloc)
56046488b62SChristian Schoenebeck {
561684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
5621d98613dSGreg Kurz     do_readdir_split(obj, 512);
56346488b62SChristian Schoenebeck }
56446488b62SChristian Schoenebeck 
565653daf38SChristian Schoenebeck 
566653daf38SChristian Schoenebeck /* tests using the 9pfs 'local' fs driver */
567653daf38SChristian Schoenebeck 
568653daf38SChristian Schoenebeck static void fs_create_dir(void *obj, void *data, QGuestAllocator *t_alloc)
569653daf38SChristian Schoenebeck {
570653daf38SChristian Schoenebeck     QVirtio9P *v9p = obj;
571684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
572653daf38SChristian Schoenebeck     struct stat st;
57365ceee0aSChristian Schoenebeck     g_autofree char *root_path = virtio_9p_test_path("");
57465ceee0aSChristian Schoenebeck     g_autofree char *new_dir = virtio_9p_test_path("01");
575653daf38SChristian Schoenebeck 
576653daf38SChristian Schoenebeck     g_assert(root_path != NULL);
577653daf38SChristian Schoenebeck 
57874a160abSChristian Schoenebeck     tattach({ .client = v9p });
579c1934f63SGreg Kurz     do_mkdir(v9p, "/", "01");
580653daf38SChristian Schoenebeck 
581653daf38SChristian Schoenebeck     /* check if created directory really exists now ... */
582653daf38SChristian Schoenebeck     g_assert(stat(new_dir, &st) == 0);
583653daf38SChristian Schoenebeck     /* ... and is actually a directory */
584653daf38SChristian Schoenebeck     g_assert((st.st_mode & S_IFMT) == S_IFDIR);
585653daf38SChristian Schoenebeck }
586653daf38SChristian Schoenebeck 
587b37d62d6SChristian Schoenebeck static void fs_unlinkat_dir(void *obj, void *data, QGuestAllocator *t_alloc)
588b37d62d6SChristian Schoenebeck {
589b37d62d6SChristian Schoenebeck     QVirtio9P *v9p = obj;
590684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
591b37d62d6SChristian Schoenebeck     struct stat st;
59265ceee0aSChristian Schoenebeck     g_autofree char *root_path = virtio_9p_test_path("");
59365ceee0aSChristian Schoenebeck     g_autofree char *new_dir = virtio_9p_test_path("02");
594b37d62d6SChristian Schoenebeck 
595b37d62d6SChristian Schoenebeck     g_assert(root_path != NULL);
596b37d62d6SChristian Schoenebeck 
59774a160abSChristian Schoenebeck     tattach({ .client = v9p });
598b37d62d6SChristian Schoenebeck     do_mkdir(v9p, "/", "02");
599b37d62d6SChristian Schoenebeck 
600b37d62d6SChristian Schoenebeck     /* check if created directory really exists now ... */
601b37d62d6SChristian Schoenebeck     g_assert(stat(new_dir, &st) == 0);
602b37d62d6SChristian Schoenebeck     /* ... and is actually a directory */
603b37d62d6SChristian Schoenebeck     g_assert((st.st_mode & S_IFMT) == S_IFDIR);
604b37d62d6SChristian Schoenebeck 
605d3671fd9SWill Cohen     do_unlinkat(v9p, "/", "02", P9_DOTL_AT_REMOVEDIR);
606b37d62d6SChristian Schoenebeck     /* directory should be gone now */
607b37d62d6SChristian Schoenebeck     g_assert(stat(new_dir, &st) != 0);
608b37d62d6SChristian Schoenebeck }
609b37d62d6SChristian Schoenebeck 
610b09dbfddSChristian Schoenebeck static void fs_create_file(void *obj, void *data, QGuestAllocator *t_alloc)
611b09dbfddSChristian Schoenebeck {
612b09dbfddSChristian Schoenebeck     QVirtio9P *v9p = obj;
613684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
614b09dbfddSChristian Schoenebeck     struct stat st;
61565ceee0aSChristian Schoenebeck     g_autofree char *new_file = virtio_9p_test_path("03/1st_file");
616b09dbfddSChristian Schoenebeck 
61774a160abSChristian Schoenebeck     tattach({ .client = v9p });
618b09dbfddSChristian Schoenebeck     do_mkdir(v9p, "/", "03");
619b09dbfddSChristian Schoenebeck     do_lcreate(v9p, "03", "1st_file");
620b09dbfddSChristian Schoenebeck 
621b09dbfddSChristian Schoenebeck     /* check if created file exists now ... */
622b09dbfddSChristian Schoenebeck     g_assert(stat(new_file, &st) == 0);
623b09dbfddSChristian Schoenebeck     /* ... and is a regular file */
624b09dbfddSChristian Schoenebeck     g_assert((st.st_mode & S_IFMT) == S_IFREG);
625b09dbfddSChristian Schoenebeck }
626b09dbfddSChristian Schoenebeck 
627472c18b8SChristian Schoenebeck static void fs_unlinkat_file(void *obj, void *data, QGuestAllocator *t_alloc)
628472c18b8SChristian Schoenebeck {
629472c18b8SChristian Schoenebeck     QVirtio9P *v9p = obj;
630684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
631472c18b8SChristian Schoenebeck     struct stat st;
63265ceee0aSChristian Schoenebeck     g_autofree char *new_file = virtio_9p_test_path("04/doa_file");
633472c18b8SChristian Schoenebeck 
63474a160abSChristian Schoenebeck     tattach({ .client = v9p });
635472c18b8SChristian Schoenebeck     do_mkdir(v9p, "/", "04");
636472c18b8SChristian Schoenebeck     do_lcreate(v9p, "04", "doa_file");
637472c18b8SChristian Schoenebeck 
638472c18b8SChristian Schoenebeck     /* check if created file exists now ... */
639472c18b8SChristian Schoenebeck     g_assert(stat(new_file, &st) == 0);
640472c18b8SChristian Schoenebeck     /* ... and is a regular file */
641472c18b8SChristian Schoenebeck     g_assert((st.st_mode & S_IFMT) == S_IFREG);
642472c18b8SChristian Schoenebeck 
643472c18b8SChristian Schoenebeck     do_unlinkat(v9p, "04", "doa_file", 0);
644472c18b8SChristian Schoenebeck     /* file should be gone now */
645472c18b8SChristian Schoenebeck     g_assert(stat(new_file, &st) != 0);
646472c18b8SChristian Schoenebeck }
647472c18b8SChristian Schoenebeck 
64859ff563dSChristian Schoenebeck static void fs_symlink_file(void *obj, void *data, QGuestAllocator *t_alloc)
64959ff563dSChristian Schoenebeck {
65059ff563dSChristian Schoenebeck     QVirtio9P *v9p = obj;
651684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
65259ff563dSChristian Schoenebeck     struct stat st;
65365ceee0aSChristian Schoenebeck     g_autofree char *real_file = virtio_9p_test_path("05/real_file");
65465ceee0aSChristian Schoenebeck     g_autofree char *symlink_file = virtio_9p_test_path("05/symlink_file");
65559ff563dSChristian Schoenebeck 
65674a160abSChristian Schoenebeck     tattach({ .client = v9p });
65759ff563dSChristian Schoenebeck     do_mkdir(v9p, "/", "05");
65859ff563dSChristian Schoenebeck     do_lcreate(v9p, "05", "real_file");
65959ff563dSChristian Schoenebeck     g_assert(stat(real_file, &st) == 0);
66059ff563dSChristian Schoenebeck     g_assert((st.st_mode & S_IFMT) == S_IFREG);
66159ff563dSChristian Schoenebeck 
66259ff563dSChristian Schoenebeck     do_symlink(v9p, "05", "symlink_file", "real_file");
66359ff563dSChristian Schoenebeck 
66459ff563dSChristian Schoenebeck     /* check if created link exists now */
66559ff563dSChristian Schoenebeck     g_assert(stat(symlink_file, &st) == 0);
66659ff563dSChristian Schoenebeck }
66759ff563dSChristian Schoenebeck 
6685b28ab8bSChristian Schoenebeck static void fs_unlinkat_symlink(void *obj, void *data,
6695b28ab8bSChristian Schoenebeck                                 QGuestAllocator *t_alloc)
6705b28ab8bSChristian Schoenebeck {
6715b28ab8bSChristian Schoenebeck     QVirtio9P *v9p = obj;
672684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
6735b28ab8bSChristian Schoenebeck     struct stat st;
67465ceee0aSChristian Schoenebeck     g_autofree char *real_file = virtio_9p_test_path("06/real_file");
67565ceee0aSChristian Schoenebeck     g_autofree char *symlink_file = virtio_9p_test_path("06/symlink_file");
6765b28ab8bSChristian Schoenebeck 
67774a160abSChristian Schoenebeck     tattach({ .client = v9p });
6785b28ab8bSChristian Schoenebeck     do_mkdir(v9p, "/", "06");
6795b28ab8bSChristian Schoenebeck     do_lcreate(v9p, "06", "real_file");
6805b28ab8bSChristian Schoenebeck     g_assert(stat(real_file, &st) == 0);
6815b28ab8bSChristian Schoenebeck     g_assert((st.st_mode & S_IFMT) == S_IFREG);
6825b28ab8bSChristian Schoenebeck 
6835b28ab8bSChristian Schoenebeck     do_symlink(v9p, "06", "symlink_file", "real_file");
6845b28ab8bSChristian Schoenebeck     g_assert(stat(symlink_file, &st) == 0);
6855b28ab8bSChristian Schoenebeck 
6865b28ab8bSChristian Schoenebeck     do_unlinkat(v9p, "06", "symlink_file", 0);
6875b28ab8bSChristian Schoenebeck     /* symlink should be gone now */
6885b28ab8bSChristian Schoenebeck     g_assert(stat(symlink_file, &st) != 0);
6895b28ab8bSChristian Schoenebeck }
6905b28ab8bSChristian Schoenebeck 
69164e3d403SChristian Schoenebeck static void fs_hardlink_file(void *obj, void *data, QGuestAllocator *t_alloc)
69264e3d403SChristian Schoenebeck {
69364e3d403SChristian Schoenebeck     QVirtio9P *v9p = obj;
694684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
69564e3d403SChristian Schoenebeck     struct stat st_real, st_link;
69665ceee0aSChristian Schoenebeck     g_autofree char *real_file = virtio_9p_test_path("07/real_file");
69765ceee0aSChristian Schoenebeck     g_autofree char *hardlink_file = virtio_9p_test_path("07/hardlink_file");
69864e3d403SChristian Schoenebeck 
69974a160abSChristian Schoenebeck     tattach({ .client = v9p });
70064e3d403SChristian Schoenebeck     do_mkdir(v9p, "/", "07");
70164e3d403SChristian Schoenebeck     do_lcreate(v9p, "07", "real_file");
70264e3d403SChristian Schoenebeck     g_assert(stat(real_file, &st_real) == 0);
70364e3d403SChristian Schoenebeck     g_assert((st_real.st_mode & S_IFMT) == S_IFREG);
70464e3d403SChristian Schoenebeck 
70564e3d403SChristian Schoenebeck     do_hardlink(v9p, "07", "hardlink_file", "07/real_file");
70664e3d403SChristian Schoenebeck 
70764e3d403SChristian Schoenebeck     /* check if link exists now ... */
70864e3d403SChristian Schoenebeck     g_assert(stat(hardlink_file, &st_link) == 0);
70964e3d403SChristian Schoenebeck     /* ... and it's a hard link, right? */
71064e3d403SChristian Schoenebeck     g_assert((st_link.st_mode & S_IFMT) == S_IFREG);
71164e3d403SChristian Schoenebeck     g_assert(st_link.st_dev == st_real.st_dev);
71264e3d403SChristian Schoenebeck     g_assert(st_link.st_ino == st_real.st_ino);
71364e3d403SChristian Schoenebeck }
71464e3d403SChristian Schoenebeck 
7154d0746e2SChristian Schoenebeck static void fs_unlinkat_hardlink(void *obj, void *data,
7164d0746e2SChristian Schoenebeck                                  QGuestAllocator *t_alloc)
7174d0746e2SChristian Schoenebeck {
7184d0746e2SChristian Schoenebeck     QVirtio9P *v9p = obj;
719684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
7204d0746e2SChristian Schoenebeck     struct stat st_real, st_link;
72165ceee0aSChristian Schoenebeck     g_autofree char *real_file = virtio_9p_test_path("08/real_file");
72265ceee0aSChristian Schoenebeck     g_autofree char *hardlink_file = virtio_9p_test_path("08/hardlink_file");
7234d0746e2SChristian Schoenebeck 
72474a160abSChristian Schoenebeck     tattach({ .client = v9p });
7254d0746e2SChristian Schoenebeck     do_mkdir(v9p, "/", "08");
7264d0746e2SChristian Schoenebeck     do_lcreate(v9p, "08", "real_file");
7274d0746e2SChristian Schoenebeck     g_assert(stat(real_file, &st_real) == 0);
7284d0746e2SChristian Schoenebeck     g_assert((st_real.st_mode & S_IFMT) == S_IFREG);
7294d0746e2SChristian Schoenebeck 
7304d0746e2SChristian Schoenebeck     do_hardlink(v9p, "08", "hardlink_file", "08/real_file");
7314d0746e2SChristian Schoenebeck     g_assert(stat(hardlink_file, &st_link) == 0);
7324d0746e2SChristian Schoenebeck 
7334d0746e2SChristian Schoenebeck     do_unlinkat(v9p, "08", "hardlink_file", 0);
7344d0746e2SChristian Schoenebeck     /* symlink should be gone now */
7354d0746e2SChristian Schoenebeck     g_assert(stat(hardlink_file, &st_link) != 0);
7364d0746e2SChristian Schoenebeck     /* and old file should still exist */
7374d0746e2SChristian Schoenebeck     g_assert(stat(real_file, &st_real) == 0);
7384d0746e2SChristian Schoenebeck }
7394d0746e2SChristian Schoenebeck 
7403a565c64SChristian Schoenebeck static void *assign_9p_local_driver(GString *cmd_line, void *arg)
7413a565c64SChristian Schoenebeck {
7423a565c64SChristian Schoenebeck     virtio_9p_assign_local_driver(cmd_line, "security_model=mapped-xattr");
7433a565c64SChristian Schoenebeck     return arg;
7443a565c64SChristian Schoenebeck }
7453a565c64SChristian Schoenebeck 
746dfbe8b43SEmanuele Giuseppe Esposito static void register_virtio_9p_test(void)
7471211d81bSGreg Kurz {
7483a565c64SChristian Schoenebeck 
7493a565c64SChristian Schoenebeck     QOSGraphTestOptions opts = {
7503a565c64SChristian Schoenebeck     };
7513a565c64SChristian Schoenebeck 
7523a565c64SChristian Schoenebeck     /* 9pfs test cases using the 'synth' filesystem driver */
7533a565c64SChristian Schoenebeck     qos_add_test("synth/config", "virtio-9p", pci_config, &opts);
7543a565c64SChristian Schoenebeck     qos_add_test("synth/version/basic", "virtio-9p", fs_version,  &opts);
7553a565c64SChristian Schoenebeck     qos_add_test("synth/attach/basic", "virtio-9p", fs_attach,  &opts);
7563a565c64SChristian Schoenebeck     qos_add_test("synth/walk/basic", "virtio-9p", fs_walk,  &opts);
757eefd2394SChristian Schoenebeck     qos_add_test("synth/walk/no_slash", "virtio-9p", fs_walk_no_slash,
7583a565c64SChristian Schoenebeck                   &opts);
759c1668948SChristian Schoenebeck     qos_add_test("synth/walk/none", "virtio-9p", fs_walk_none, &opts);
760eefd2394SChristian Schoenebeck     qos_add_test("synth/walk/dotdot_from_root", "virtio-9p",
7613a565c64SChristian Schoenebeck                  fs_walk_dotdot,  &opts);
7629472a689SChristian Schoenebeck     qos_add_test("synth/walk/non_existent", "virtio-9p", fs_walk_nonexistent,
7639472a689SChristian Schoenebeck                   &opts);
76415fbff48SChristian Schoenebeck     qos_add_test("synth/walk/2nd_non_existent", "virtio-9p",
76515fbff48SChristian Schoenebeck                  fs_walk_2nd_nonexistent, &opts);
7663a565c64SChristian Schoenebeck     qos_add_test("synth/lopen/basic", "virtio-9p", fs_lopen,  &opts);
7673a565c64SChristian Schoenebeck     qos_add_test("synth/write/basic", "virtio-9p", fs_write,  &opts);
768eefd2394SChristian Schoenebeck     qos_add_test("synth/flush/success", "virtio-9p", fs_flush_success,
7693a565c64SChristian Schoenebeck                   &opts);
770eefd2394SChristian Schoenebeck     qos_add_test("synth/flush/ignored", "virtio-9p", fs_flush_ignored,
7713a565c64SChristian Schoenebeck                   &opts);
7723a565c64SChristian Schoenebeck     qos_add_test("synth/readdir/basic", "virtio-9p", fs_readdir,  &opts);
773eefd2394SChristian Schoenebeck     qos_add_test("synth/readdir/split_512", "virtio-9p",
7743a565c64SChristian Schoenebeck                  fs_readdir_split_512,  &opts);
775eefd2394SChristian Schoenebeck     qos_add_test("synth/readdir/split_256", "virtio-9p",
7763a565c64SChristian Schoenebeck                  fs_readdir_split_256,  &opts);
777eefd2394SChristian Schoenebeck     qos_add_test("synth/readdir/split_128", "virtio-9p",
7783a565c64SChristian Schoenebeck                  fs_readdir_split_128,  &opts);
7793a565c64SChristian Schoenebeck 
7803a565c64SChristian Schoenebeck 
7813a565c64SChristian Schoenebeck     /* 9pfs test cases using the 'local' filesystem driver */
782558f5c42SGreg Kurz 
783558f5c42SGreg Kurz     /*
784558f5c42SGreg Kurz      * XXX: Until we are sure that these tests can run everywhere,
785558f5c42SGreg Kurz      * keep them as "slow" so that they aren't run with "make check".
786558f5c42SGreg Kurz      */
787558f5c42SGreg Kurz     if (!g_test_slow()) {
788558f5c42SGreg Kurz         return;
789558f5c42SGreg Kurz     }
790558f5c42SGreg Kurz 
7913a565c64SChristian Schoenebeck     opts.before = assign_9p_local_driver;
7923a565c64SChristian Schoenebeck     qos_add_test("local/config", "virtio-9p", pci_config,  &opts);
793653daf38SChristian Schoenebeck     qos_add_test("local/create_dir", "virtio-9p", fs_create_dir, &opts);
794b37d62d6SChristian Schoenebeck     qos_add_test("local/unlinkat_dir", "virtio-9p", fs_unlinkat_dir, &opts);
795b09dbfddSChristian Schoenebeck     qos_add_test("local/create_file", "virtio-9p", fs_create_file, &opts);
796472c18b8SChristian Schoenebeck     qos_add_test("local/unlinkat_file", "virtio-9p", fs_unlinkat_file, &opts);
79759ff563dSChristian Schoenebeck     qos_add_test("local/symlink_file", "virtio-9p", fs_symlink_file, &opts);
7985b28ab8bSChristian Schoenebeck     qos_add_test("local/unlinkat_symlink", "virtio-9p", fs_unlinkat_symlink,
7995b28ab8bSChristian Schoenebeck                  &opts);
80064e3d403SChristian Schoenebeck     qos_add_test("local/hardlink_file", "virtio-9p", fs_hardlink_file, &opts);
8014d0746e2SChristian Schoenebeck     qos_add_test("local/unlinkat_hardlink", "virtio-9p", fs_unlinkat_hardlink,
8024d0746e2SChristian Schoenebeck                  &opts);
8031211d81bSGreg Kurz }
8041211d81bSGreg Kurz 
805dfbe8b43SEmanuele Giuseppe Esposito libqos_init(register_virtio_9p_test);
806136b7af2SChristian Schoenebeck 
807136b7af2SChristian Schoenebeck static void __attribute__((constructor)) construct_9p_test(void)
808136b7af2SChristian Schoenebeck {
809136b7af2SChristian Schoenebeck     /* make sure test dir for the 'local' tests exists */
810136b7af2SChristian Schoenebeck     virtio_9p_create_local_test_dir();
811136b7af2SChristian Schoenebeck }
812136b7af2SChristian Schoenebeck 
813136b7af2SChristian Schoenebeck static void __attribute__((destructor)) destruct_9p_test(void)
814136b7af2SChristian Schoenebeck {
815136b7af2SChristian Schoenebeck     /* remove previously created test dir when test suite completed */
816136b7af2SChristian Schoenebeck     virtio_9p_remove_local_test_dir();
817136b7af2SChristian Schoenebeck }
818