xref: /openbmc/qemu/tests/qtest/virtio-9p-test.c (revision 43e0d9fb)
11e8a1faeSThomas Huth /*
21e8a1faeSThomas Huth  * QTest testcase for VirtIO 9P
31e8a1faeSThomas Huth  *
41e8a1faeSThomas Huth  * Copyright (c) 2014 SUSE LINUX Products GmbH
51e8a1faeSThomas Huth  *
61e8a1faeSThomas Huth  * This work is licensed under the terms of the GNU GPL, version 2 or later.
71e8a1faeSThomas Huth  * See the COPYING file in the top-level directory.
81e8a1faeSThomas Huth  */
91e8a1faeSThomas Huth 
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 
151e8a1faeSThomas Huth #include "qemu/osdep.h"
161e8a1faeSThomas Huth #include "qemu/module.h"
17684f9120SChristian Schoenebeck #include "libqos/virtio-9p-client.h"
181e8a1faeSThomas Huth 
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__)
30d41a9462SChristian Schoenebeck #define tlink(...) v9fs_tlink((TlinkOpt) __VA_ARGS__)
31*43e0d9fbSChristian Schoenebeck #define tunlinkat(...) v9fs_tunlinkat((TunlinkatOpt) __VA_ARGS__)
32653daf38SChristian Schoenebeck 
pci_config(void * obj,void * data,QGuestAllocator * t_alloc)331e8a1faeSThomas Huth static void pci_config(void *obj, void *data, QGuestAllocator *t_alloc)
341e8a1faeSThomas Huth {
351e8a1faeSThomas Huth     QVirtio9P *v9p = obj;
36684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
371e8a1faeSThomas Huth     size_t tag_len = qvirtio_config_readw(v9p->vdev, 0);
3865ceee0aSChristian Schoenebeck     g_autofree char *tag = NULL;
391e8a1faeSThomas Huth     int i;
401e8a1faeSThomas Huth 
411e8a1faeSThomas Huth     g_assert_cmpint(tag_len, ==, strlen(MOUNT_TAG));
421e8a1faeSThomas Huth 
431e8a1faeSThomas Huth     tag = g_malloc(tag_len);
441e8a1faeSThomas Huth     for (i = 0; i < tag_len; i++) {
451e8a1faeSThomas Huth         tag[i] = qvirtio_config_readb(v9p->vdev, i + 2);
461e8a1faeSThomas Huth     }
471e8a1faeSThomas Huth     g_assert_cmpmem(tag, tag_len, MOUNT_TAG, tag_len);
481e8a1faeSThomas Huth }
491e8a1faeSThomas Huth 
is_same_qid(v9fs_qid a,v9fs_qid b)50a6821b82SChristian Schoenebeck static inline bool is_same_qid(v9fs_qid a, v9fs_qid b)
51a6821b82SChristian Schoenebeck {
52a6821b82SChristian Schoenebeck     /* don't compare QID version for checking for file ID equalness */
53a6821b82SChristian Schoenebeck     return a[0] == b[0] && memcmp(&a[5], &b[5], 8) == 0;
54a6821b82SChristian Schoenebeck }
55a6821b82SChristian Schoenebeck 
fs_version(void * obj,void * data,QGuestAllocator * t_alloc)561c450e6eSGreg Kurz static void fs_version(void *obj, void *data, QGuestAllocator *t_alloc)
571c450e6eSGreg Kurz {
58684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
59bee8fda2SChristian Schoenebeck     tversion({ .client = obj });
601c450e6eSGreg Kurz }
611c450e6eSGreg Kurz 
fs_attach(void * obj,void * data,QGuestAllocator * t_alloc)623fe4baf4SGreg Kurz static void fs_attach(void *obj, void *data, QGuestAllocator *t_alloc)
633fe4baf4SGreg Kurz {
64684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
6574a160abSChristian Schoenebeck     tattach({ .client = obj });
663fe4baf4SGreg Kurz }
673fe4baf4SGreg Kurz 
fs_walk(void * obj,void * data,QGuestAllocator * t_alloc)681e8a1faeSThomas Huth static void fs_walk(void *obj, void *data, QGuestAllocator *t_alloc)
691e8a1faeSThomas Huth {
701e8a1faeSThomas Huth     QVirtio9P *v9p = obj;
71684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
721e8a1faeSThomas Huth     char *wnames[P9_MAXWELEM];
731e8a1faeSThomas Huth     uint16_t nwqid;
7465ceee0aSChristian Schoenebeck     g_autofree v9fs_qid *wqid = NULL;
751e8a1faeSThomas Huth     int i;
761e8a1faeSThomas Huth 
771e8a1faeSThomas Huth     for (i = 0; i < P9_MAXWELEM; i++) {
781e8a1faeSThomas Huth         wnames[i] = g_strdup_printf(QTEST_V9FS_SYNTH_WALK_FILE, i);
791e8a1faeSThomas Huth     }
801e8a1faeSThomas Huth 
8174a160abSChristian Schoenebeck     tattach({ .client = v9p });
823f3e9232SChristian Schoenebeck     twalk({
83569f3b63SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1,
843f3e9232SChristian Schoenebeck         .nwname = P9_MAXWELEM, .wnames = wnames,
853f3e9232SChristian Schoenebeck         .rwalk = { .nwqid = &nwqid, .wqid = &wqid }
863f3e9232SChristian Schoenebeck     });
871e8a1faeSThomas Huth 
881e8a1faeSThomas Huth     g_assert_cmpint(nwqid, ==, P9_MAXWELEM);
891e8a1faeSThomas Huth 
901e8a1faeSThomas Huth     for (i = 0; i < P9_MAXWELEM; i++) {
911e8a1faeSThomas Huth         g_free(wnames[i]);
921e8a1faeSThomas Huth     }
931e8a1faeSThomas Huth }
941e8a1faeSThomas Huth 
fs_dirents_contain_name(struct V9fsDirent * e,const char * name)954829469fSChristian Schoenebeck static bool fs_dirents_contain_name(struct V9fsDirent *e, const char* name)
964829469fSChristian Schoenebeck {
974829469fSChristian Schoenebeck     for (; e; e = e->next) {
984829469fSChristian Schoenebeck         if (!strcmp(e->name, name)) {
994829469fSChristian Schoenebeck             return true;
1004829469fSChristian Schoenebeck         }
1014829469fSChristian Schoenebeck     }
1024829469fSChristian Schoenebeck     return false;
1034829469fSChristian Schoenebeck }
1044829469fSChristian Schoenebeck 
10546488b62SChristian Schoenebeck /* basic readdir test where reply fits into a single response message */
fs_readdir(void * obj,void * data,QGuestAllocator * t_alloc)1064829469fSChristian Schoenebeck static void fs_readdir(void *obj, void *data, QGuestAllocator *t_alloc)
1074829469fSChristian Schoenebeck {
1084829469fSChristian Schoenebeck     QVirtio9P *v9p = obj;
109684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
110569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_READDIR_DIR) };
1114829469fSChristian Schoenebeck     uint16_t nqid;
1124829469fSChristian Schoenebeck     v9fs_qid qid;
1134829469fSChristian Schoenebeck     uint32_t count, nentries;
1144829469fSChristian Schoenebeck     struct V9fsDirent *entries = NULL;
1154829469fSChristian Schoenebeck 
11674a160abSChristian Schoenebeck     tattach({ .client = v9p });
1173f3e9232SChristian Schoenebeck     twalk({
118569f3b63SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1,
1193f3e9232SChristian Schoenebeck         .nwname = 1, .wnames = wnames, .rwalk.nwqid = &nqid
1203f3e9232SChristian Schoenebeck     });
1214829469fSChristian Schoenebeck     g_assert_cmpint(nqid, ==, 1);
1224829469fSChristian Schoenebeck 
1230e4c4ff0SChristian Schoenebeck     tlopen({
1240e4c4ff0SChristian Schoenebeck         .client = v9p, .fid = 1, .flags = O_DIRECTORY, .rlopen.qid = &qid
1250e4c4ff0SChristian Schoenebeck     });
1264829469fSChristian Schoenebeck 
1274829469fSChristian Schoenebeck     /*
1284829469fSChristian Schoenebeck      * submit count = msize - 11, because 11 is the header size of Rreaddir
1294829469fSChristian Schoenebeck      */
130a9a53769SChristian Schoenebeck     treaddir({
1311ebacc40SChristian Schoenebeck         .client = v9p, .fid = 1, .offset = 0, .count = P9_MAX_SIZE - 11,
132a9a53769SChristian Schoenebeck         .rreaddir = {
133a9a53769SChristian Schoenebeck             .count = &count, .nentries = &nentries, .entries = &entries
134a9a53769SChristian Schoenebeck         }
135a9a53769SChristian Schoenebeck     });
1364829469fSChristian Schoenebeck 
1374829469fSChristian Schoenebeck     /*
1384829469fSChristian Schoenebeck      * Assuming msize (P9_MAX_SIZE) is large enough so we can retrieve all
1394829469fSChristian Schoenebeck      * dir entries with only one readdir request.
1404829469fSChristian Schoenebeck      */
1414829469fSChristian Schoenebeck     g_assert_cmpint(
1424829469fSChristian Schoenebeck         nentries, ==,
1434829469fSChristian Schoenebeck         QTEST_V9FS_SYNTH_READDIR_NFILES + 2 /* "." and ".." */
1444829469fSChristian Schoenebeck     );
1454829469fSChristian Schoenebeck 
1464829469fSChristian Schoenebeck     /*
1474829469fSChristian Schoenebeck      * Check all file names exist in returned entries, ignore their order
1484829469fSChristian Schoenebeck      * though.
1494829469fSChristian Schoenebeck      */
1504829469fSChristian Schoenebeck     g_assert_cmpint(fs_dirents_contain_name(entries, "."), ==, true);
1514829469fSChristian Schoenebeck     g_assert_cmpint(fs_dirents_contain_name(entries, ".."), ==, true);
1524829469fSChristian Schoenebeck     for (int i = 0; i < QTEST_V9FS_SYNTH_READDIR_NFILES; ++i) {
15365ceee0aSChristian Schoenebeck         g_autofree char *name =
15465ceee0aSChristian Schoenebeck             g_strdup_printf(QTEST_V9FS_SYNTH_READDIR_FILE, i);
1554829469fSChristian Schoenebeck         g_assert_cmpint(fs_dirents_contain_name(entries, name), ==, true);
1564829469fSChristian Schoenebeck     }
1574829469fSChristian Schoenebeck 
1584829469fSChristian Schoenebeck     v9fs_free_dirents(entries);
1594829469fSChristian Schoenebeck     g_free(wnames[0]);
1604829469fSChristian Schoenebeck }
1614829469fSChristian Schoenebeck 
16246488b62SChristian Schoenebeck /* readdir test where overall request is split over several messages */
do_readdir_split(QVirtio9P * v9p,uint32_t count)1631d98613dSGreg Kurz static void do_readdir_split(QVirtio9P *v9p, uint32_t count)
16446488b62SChristian Schoenebeck {
165569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_READDIR_DIR) };
16646488b62SChristian Schoenebeck     uint16_t nqid;
16746488b62SChristian Schoenebeck     v9fs_qid qid;
16846488b62SChristian Schoenebeck     uint32_t nentries, npartialentries;
16946488b62SChristian Schoenebeck     struct V9fsDirent *entries, *tail, *partialentries;
17046488b62SChristian Schoenebeck     int fid;
17146488b62SChristian Schoenebeck     uint64_t offset;
17246488b62SChristian Schoenebeck 
17374a160abSChristian Schoenebeck     tattach({ .client = v9p });
17446488b62SChristian Schoenebeck 
17546488b62SChristian Schoenebeck     fid = 1;
17646488b62SChristian Schoenebeck     offset = 0;
17746488b62SChristian Schoenebeck     entries = NULL;
17846488b62SChristian Schoenebeck     nentries = 0;
17946488b62SChristian Schoenebeck     tail = NULL;
18046488b62SChristian Schoenebeck 
1813f3e9232SChristian Schoenebeck     twalk({
182569f3b63SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = fid,
1833f3e9232SChristian Schoenebeck         .nwname = 1, .wnames = wnames, .rwalk.nwqid = &nqid
1843f3e9232SChristian Schoenebeck     });
18546488b62SChristian Schoenebeck     g_assert_cmpint(nqid, ==, 1);
18646488b62SChristian Schoenebeck 
1870e4c4ff0SChristian Schoenebeck     tlopen({
1880e4c4ff0SChristian Schoenebeck         .client = v9p, .fid = fid, .flags = O_DIRECTORY, .rlopen.qid = &qid
1890e4c4ff0SChristian Schoenebeck     });
19046488b62SChristian Schoenebeck 
19146488b62SChristian Schoenebeck     /*
19246488b62SChristian Schoenebeck      * send as many Treaddir requests as required to get all directory
19346488b62SChristian Schoenebeck      * entries
19446488b62SChristian Schoenebeck      */
19546488b62SChristian Schoenebeck     while (true) {
19646488b62SChristian Schoenebeck         npartialentries = 0;
19746488b62SChristian Schoenebeck         partialentries = NULL;
19846488b62SChristian Schoenebeck 
199a9a53769SChristian Schoenebeck         treaddir({
2001ebacc40SChristian Schoenebeck             .client = v9p, .fid = fid, .offset = offset, .count = count,
201a9a53769SChristian Schoenebeck             .rreaddir = {
202a9a53769SChristian Schoenebeck                 .count = &count, .nentries = &npartialentries,
203a9a53769SChristian Schoenebeck                 .entries = &partialentries
204a9a53769SChristian Schoenebeck             }
205a9a53769SChristian Schoenebeck         });
20646488b62SChristian Schoenebeck         if (npartialentries > 0 && partialentries) {
20746488b62SChristian Schoenebeck             if (!entries) {
20846488b62SChristian Schoenebeck                 entries = partialentries;
20946488b62SChristian Schoenebeck                 nentries = npartialentries;
21046488b62SChristian Schoenebeck                 tail = partialentries;
21146488b62SChristian Schoenebeck             } else {
21246488b62SChristian Schoenebeck                 tail->next = partialentries;
21346488b62SChristian Schoenebeck                 nentries += npartialentries;
21446488b62SChristian Schoenebeck             }
21546488b62SChristian Schoenebeck             while (tail->next) {
21646488b62SChristian Schoenebeck                 tail = tail->next;
21746488b62SChristian Schoenebeck             }
21846488b62SChristian Schoenebeck             offset = tail->offset;
21946488b62SChristian Schoenebeck         } else {
22046488b62SChristian Schoenebeck             break;
22146488b62SChristian Schoenebeck         }
22246488b62SChristian Schoenebeck     }
22346488b62SChristian Schoenebeck 
22446488b62SChristian Schoenebeck     g_assert_cmpint(
22546488b62SChristian Schoenebeck         nentries, ==,
22646488b62SChristian Schoenebeck         QTEST_V9FS_SYNTH_READDIR_NFILES + 2 /* "." and ".." */
22746488b62SChristian Schoenebeck     );
22846488b62SChristian Schoenebeck 
22946488b62SChristian Schoenebeck     /*
23046488b62SChristian Schoenebeck      * Check all file names exist in returned entries, ignore their order
23146488b62SChristian Schoenebeck      * though.
23246488b62SChristian Schoenebeck      */
23346488b62SChristian Schoenebeck     g_assert_cmpint(fs_dirents_contain_name(entries, "."), ==, true);
23446488b62SChristian Schoenebeck     g_assert_cmpint(fs_dirents_contain_name(entries, ".."), ==, true);
23546488b62SChristian Schoenebeck     for (int i = 0; i < QTEST_V9FS_SYNTH_READDIR_NFILES; ++i) {
23646488b62SChristian Schoenebeck         char *name = g_strdup_printf(QTEST_V9FS_SYNTH_READDIR_FILE, i);
23746488b62SChristian Schoenebeck         g_assert_cmpint(fs_dirents_contain_name(entries, name), ==, true);
23846488b62SChristian Schoenebeck         g_free(name);
23946488b62SChristian Schoenebeck     }
24046488b62SChristian Schoenebeck 
24146488b62SChristian Schoenebeck     v9fs_free_dirents(entries);
24246488b62SChristian Schoenebeck 
24346488b62SChristian Schoenebeck     g_free(wnames[0]);
24446488b62SChristian Schoenebeck }
24546488b62SChristian Schoenebeck 
fs_walk_no_slash(void * obj,void * data,QGuestAllocator * t_alloc)2461e8a1faeSThomas Huth static void fs_walk_no_slash(void *obj, void *data, QGuestAllocator *t_alloc)
2471e8a1faeSThomas Huth {
2481e8a1faeSThomas Huth     QVirtio9P *v9p = obj;
249684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
250569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup(" /") };
2511e8a1faeSThomas Huth 
25274a160abSChristian Schoenebeck     tattach({ .client = v9p });
2533f3e9232SChristian Schoenebeck     twalk({
254569f3b63SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames,
2553f3e9232SChristian Schoenebeck         .expectErr = ENOENT
2563f3e9232SChristian Schoenebeck     });
2571e8a1faeSThomas Huth 
2581e8a1faeSThomas Huth     g_free(wnames[0]);
2591e8a1faeSThomas Huth }
2601e8a1faeSThomas Huth 
fs_walk_nonexistent(void * obj,void * data,QGuestAllocator * t_alloc)2619472a689SChristian Schoenebeck static void fs_walk_nonexistent(void *obj, void *data, QGuestAllocator *t_alloc)
2629472a689SChristian Schoenebeck {
2639472a689SChristian Schoenebeck     QVirtio9P *v9p = obj;
264684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
2659472a689SChristian Schoenebeck 
26674a160abSChristian Schoenebeck     tattach({ .client = v9p });
26715fbff48SChristian Schoenebeck     /*
26815fbff48SChristian Schoenebeck      * The 9p2000 protocol spec says: "If the first element cannot be walked
26915fbff48SChristian Schoenebeck      * for any reason, Rerror is returned."
27015fbff48SChristian Schoenebeck      */
271569f3b63SChristian Schoenebeck     twalk({ .client = v9p, .path = "non-existent", .expectErr = ENOENT });
2729472a689SChristian Schoenebeck }
2739472a689SChristian Schoenebeck 
fs_walk_2nd_nonexistent(void * obj,void * data,QGuestAllocator * t_alloc)27415fbff48SChristian Schoenebeck static void fs_walk_2nd_nonexistent(void *obj, void *data,
27515fbff48SChristian Schoenebeck                                     QGuestAllocator *t_alloc)
27615fbff48SChristian Schoenebeck {
27715fbff48SChristian Schoenebeck     QVirtio9P *v9p = obj;
278684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
2790e43495dSChristian Schoenebeck     v9fs_qid root_qid;
28015fbff48SChristian Schoenebeck     uint16_t nwqid;
28128c73670SChristian Schoenebeck     uint32_t fid;
28215fbff48SChristian Schoenebeck     g_autofree v9fs_qid *wqid = NULL;
28315fbff48SChristian Schoenebeck     g_autofree char *path = g_strdup_printf(
28415fbff48SChristian Schoenebeck         QTEST_V9FS_SYNTH_WALK_FILE "/non-existent", 0
28515fbff48SChristian Schoenebeck     );
28615fbff48SChristian Schoenebeck 
28774a160abSChristian Schoenebeck     tattach({ .client = v9p, .rattach.qid = &root_qid });
288569f3b63SChristian Schoenebeck     fid = twalk({
289569f3b63SChristian Schoenebeck         .client = v9p, .path = path,
2903f3e9232SChristian Schoenebeck         .rwalk = { .nwqid = &nwqid, .wqid = &wqid }
291569f3b63SChristian Schoenebeck     }).newfid;
29215fbff48SChristian Schoenebeck     /*
29315fbff48SChristian Schoenebeck      * The 9p2000 protocol spec says: "nwqid is therefore either nwname or the
29415fbff48SChristian Schoenebeck      * index of the first elementwise walk that failed."
29515fbff48SChristian Schoenebeck      */
29615fbff48SChristian Schoenebeck     assert(nwqid == 1);
2970e43495dSChristian Schoenebeck 
2980e43495dSChristian Schoenebeck     /* returned QID wqid[0] is file ID of 1st subdir */
2990e43495dSChristian Schoenebeck     g_assert(wqid && wqid[0] && !is_same_qid(root_qid, wqid[0]));
3000e43495dSChristian Schoenebeck 
3010e43495dSChristian Schoenebeck     /* expect fid being unaffected by walk above */
30228c73670SChristian Schoenebeck     tgetattr({
3032af5be47SChristian Schoenebeck         .client = v9p, .fid = fid, .request_mask = P9_GETATTR_BASIC,
30428c73670SChristian Schoenebeck         .expectErr = ENOENT
30528c73670SChristian Schoenebeck     });
30615fbff48SChristian Schoenebeck }
30715fbff48SChristian Schoenebeck 
fs_walk_none(void * obj,void * data,QGuestAllocator * t_alloc)308c1668948SChristian Schoenebeck static void fs_walk_none(void *obj, void *data, QGuestAllocator *t_alloc)
309c1668948SChristian Schoenebeck {
310c1668948SChristian Schoenebeck     QVirtio9P *v9p = obj;
311684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
312c1668948SChristian Schoenebeck     v9fs_qid root_qid;
313c1668948SChristian Schoenebeck     g_autofree v9fs_qid *wqid = NULL;
314a6821b82SChristian Schoenebeck     struct v9fs_attr attr;
315c1668948SChristian Schoenebeck 
316bee8fda2SChristian Schoenebeck     tversion({ .client = v9p });
3171125ddf6SChristian Schoenebeck     tattach({
3181125ddf6SChristian Schoenebeck         .client = v9p, .fid = 0, .n_uname = getuid(),
3191125ddf6SChristian Schoenebeck         .rattach.qid = &root_qid
3201125ddf6SChristian Schoenebeck     });
321c1668948SChristian Schoenebeck 
3223f3e9232SChristian Schoenebeck     twalk({
323569f3b63SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1, .nwname = 0, .wnames = NULL,
3243f3e9232SChristian Schoenebeck         .rwalk.wqid = &wqid
3253f3e9232SChristian Schoenebeck     });
326c1668948SChristian Schoenebeck 
327c1668948SChristian Schoenebeck     /* special case: no QID is returned if nwname=0 was sent */
328c1668948SChristian Schoenebeck     g_assert(wqid == NULL);
329a6821b82SChristian Schoenebeck 
33028c73670SChristian Schoenebeck     tgetattr({
3312af5be47SChristian Schoenebeck         .client = v9p, .fid = 1, .request_mask = P9_GETATTR_BASIC,
33228c73670SChristian Schoenebeck         .rgetattr.attr = &attr
33328c73670SChristian Schoenebeck     });
334a6821b82SChristian Schoenebeck 
335a6821b82SChristian Schoenebeck     g_assert(is_same_qid(root_qid, attr.qid));
336c1668948SChristian Schoenebeck }
337c1668948SChristian Schoenebeck 
fs_walk_dotdot(void * obj,void * data,QGuestAllocator * t_alloc)3381e8a1faeSThomas Huth static void fs_walk_dotdot(void *obj, void *data, QGuestAllocator *t_alloc)
3391e8a1faeSThomas Huth {
3401e8a1faeSThomas Huth     QVirtio9P *v9p = obj;
341684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
342569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup("..") };
34365ceee0aSChristian Schoenebeck     v9fs_qid root_qid;
34465ceee0aSChristian Schoenebeck     g_autofree v9fs_qid *wqid = NULL;
3451e8a1faeSThomas Huth 
346bee8fda2SChristian Schoenebeck     tversion({ .client = v9p });
3471125ddf6SChristian Schoenebeck     tattach({
3481125ddf6SChristian Schoenebeck         .client = v9p, .fid = 0, .n_uname = getuid(),
3491125ddf6SChristian Schoenebeck         .rattach.qid = &root_qid
3501125ddf6SChristian Schoenebeck     });
3511e8a1faeSThomas Huth 
3523f3e9232SChristian Schoenebeck     twalk({
353569f3b63SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames,
3543f3e9232SChristian Schoenebeck         .rwalk.wqid = &wqid /* We now we'll get one qid */
3553f3e9232SChristian Schoenebeck     });
3561e8a1faeSThomas Huth 
3571e8a1faeSThomas Huth     g_assert_cmpmem(&root_qid, 13, wqid[0], 13);
3581e8a1faeSThomas Huth 
3591e8a1faeSThomas Huth     g_free(wnames[0]);
3601e8a1faeSThomas Huth }
3611e8a1faeSThomas Huth 
fs_lopen(void * obj,void * data,QGuestAllocator * t_alloc)3621e8a1faeSThomas Huth static void fs_lopen(void *obj, void *data, QGuestAllocator *t_alloc)
3631e8a1faeSThomas Huth {
3641e8a1faeSThomas Huth     QVirtio9P *v9p = obj;
365684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
366569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_LOPEN_FILE) };
3671e8a1faeSThomas Huth 
36874a160abSChristian Schoenebeck     tattach({ .client = v9p });
3693f3e9232SChristian Schoenebeck     twalk({
3703f3e9232SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames
3713f3e9232SChristian Schoenebeck     });
3721e8a1faeSThomas Huth 
3730e4c4ff0SChristian Schoenebeck     tlopen({ .client = v9p, .fid = 1, .flags = O_WRONLY });
3741e8a1faeSThomas Huth 
3751e8a1faeSThomas Huth     g_free(wnames[0]);
3761e8a1faeSThomas Huth }
3771e8a1faeSThomas Huth 
fs_write(void * obj,void * data,QGuestAllocator * t_alloc)3781e8a1faeSThomas Huth static void fs_write(void *obj, void *data, QGuestAllocator *t_alloc)
3791e8a1faeSThomas Huth {
3801e8a1faeSThomas Huth     QVirtio9P *v9p = obj;
381684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
3821e8a1faeSThomas Huth     static const uint32_t write_count = P9_MAX_SIZE / 2;
383569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_WRITE_FILE) };
38465ceee0aSChristian Schoenebeck     g_autofree char *buf = g_malloc0(write_count);
3851e8a1faeSThomas Huth     uint32_t count;
3861e8a1faeSThomas Huth 
38774a160abSChristian Schoenebeck     tattach({ .client = v9p });
3883f3e9232SChristian Schoenebeck     twalk({
3893f3e9232SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames
3903f3e9232SChristian Schoenebeck     });
3911e8a1faeSThomas Huth 
3920e4c4ff0SChristian Schoenebeck     tlopen({ .client = v9p, .fid = 1, .flags = O_WRONLY });
3931e8a1faeSThomas Huth 
394bb286ff8SChristian Schoenebeck     count = twrite({
395ac9e4e61SChristian Schoenebeck         .client = v9p, .fid = 1, .offset = 0, .count = write_count,
396bb286ff8SChristian Schoenebeck         .data = buf
397bb286ff8SChristian Schoenebeck     }).count;
3981e8a1faeSThomas Huth     g_assert_cmpint(count, ==, write_count);
3991e8a1faeSThomas Huth 
4001e8a1faeSThomas Huth     g_free(wnames[0]);
4011e8a1faeSThomas Huth }
4021e8a1faeSThomas Huth 
fs_flush_success(void * obj,void * data,QGuestAllocator * t_alloc)4031e8a1faeSThomas Huth static void fs_flush_success(void *obj, void *data, QGuestAllocator *t_alloc)
4041e8a1faeSThomas Huth {
4051e8a1faeSThomas Huth     QVirtio9P *v9p = obj;
406684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
407569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_FLUSH_FILE) };
4081e8a1faeSThomas Huth     P9Req *req, *flush_req;
4091e8a1faeSThomas Huth     uint32_t reply_len;
4101e8a1faeSThomas Huth     uint8_t should_block;
4111e8a1faeSThomas Huth 
41274a160abSChristian Schoenebeck     tattach({ .client = v9p });
4133f3e9232SChristian Schoenebeck     twalk({
4143f3e9232SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames
4153f3e9232SChristian Schoenebeck     });
4161e8a1faeSThomas Huth 
4170e4c4ff0SChristian Schoenebeck     tlopen({ .client = v9p, .fid = 1, .flags = O_WRONLY });
4181e8a1faeSThomas Huth 
4191e8a1faeSThomas Huth     /* This will cause the 9p server to try to write data to the backend,
4201e8a1faeSThomas Huth      * until the write request gets cancelled.
4211e8a1faeSThomas Huth      */
4221e8a1faeSThomas Huth     should_block = 1;
423ac9e4e61SChristian Schoenebeck     req = twrite({
424ac9e4e61SChristian Schoenebeck         .client = v9p, .fid = 1, .offset = 0,
425ac9e4e61SChristian Schoenebeck         .count = sizeof(should_block), .data = &should_block,
426ac9e4e61SChristian Schoenebeck         .requestOnly = true
427ac9e4e61SChristian Schoenebeck     }).req;
4281e8a1faeSThomas Huth 
429d89146fdSChristian Schoenebeck     flush_req = tflush({
430d89146fdSChristian Schoenebeck         .client = v9p, .oldtag = req->tag, .tag = 1, .requestOnly = true
431d89146fdSChristian Schoenebeck     }).req;
4321e8a1faeSThomas Huth 
4331e8a1faeSThomas Huth     /* The write request is supposed to be flushed: the server should just
4341e8a1faeSThomas Huth      * mark the write request as used and reply to the flush request.
4351e8a1faeSThomas Huth      */
4361e8a1faeSThomas Huth     v9fs_req_wait_for_reply(req, &reply_len);
4371e8a1faeSThomas Huth     g_assert_cmpint(reply_len, ==, 0);
4381e8a1faeSThomas Huth     v9fs_req_free(req);
4391e8a1faeSThomas Huth     v9fs_rflush(flush_req);
4401e8a1faeSThomas Huth 
4411e8a1faeSThomas Huth     g_free(wnames[0]);
4421e8a1faeSThomas Huth }
4431e8a1faeSThomas Huth 
fs_flush_ignored(void * obj,void * data,QGuestAllocator * t_alloc)4441e8a1faeSThomas Huth static void fs_flush_ignored(void *obj, void *data, QGuestAllocator *t_alloc)
4451e8a1faeSThomas Huth {
4461e8a1faeSThomas Huth     QVirtio9P *v9p = obj;
447684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
448569f3b63SChristian Schoenebeck     char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_FLUSH_FILE) };
4491e8a1faeSThomas Huth     P9Req *req, *flush_req;
4501e8a1faeSThomas Huth     uint32_t count;
4511e8a1faeSThomas Huth     uint8_t should_block;
4521e8a1faeSThomas Huth 
45374a160abSChristian Schoenebeck     tattach({ .client = v9p });
4543f3e9232SChristian Schoenebeck     twalk({
4553f3e9232SChristian Schoenebeck         .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames
4563f3e9232SChristian Schoenebeck     });
4571e8a1faeSThomas Huth 
4580e4c4ff0SChristian Schoenebeck     tlopen({ .client = v9p, .fid = 1, .flags = O_WRONLY });
4591e8a1faeSThomas Huth 
4601e8a1faeSThomas Huth     /* This will cause the write request to complete right away, before it
4611e8a1faeSThomas Huth      * could be actually cancelled.
4621e8a1faeSThomas Huth      */
4631e8a1faeSThomas Huth     should_block = 0;
464ac9e4e61SChristian Schoenebeck     req = twrite({
465ac9e4e61SChristian Schoenebeck         .client = v9p, .fid = 1, .offset = 0,
466ac9e4e61SChristian Schoenebeck         .count = sizeof(should_block), .data = &should_block,
467ac9e4e61SChristian Schoenebeck         .requestOnly = true
468ac9e4e61SChristian Schoenebeck     }).req;
4691e8a1faeSThomas Huth 
470d89146fdSChristian Schoenebeck     flush_req = tflush({
471d89146fdSChristian Schoenebeck         .client = v9p, .oldtag = req->tag, .tag = 1, .requestOnly = true
472d89146fdSChristian Schoenebeck     }).req;
4731e8a1faeSThomas Huth 
4741e8a1faeSThomas Huth     /* The write request is supposed to complete. The server should
4751e8a1faeSThomas Huth      * reply to the write request and the flush request.
4761e8a1faeSThomas Huth      */
4771e8a1faeSThomas Huth     v9fs_req_wait_for_reply(req, NULL);
4781e8a1faeSThomas Huth     v9fs_rwrite(req, &count);
4791e8a1faeSThomas Huth     g_assert_cmpint(count, ==, sizeof(should_block));
4801e8a1faeSThomas Huth     v9fs_rflush(flush_req);
4811e8a1faeSThomas Huth 
4821e8a1faeSThomas Huth     g_free(wnames[0]);
4831e8a1faeSThomas Huth }
4841e8a1faeSThomas Huth 
fs_readdir_split_128(void * obj,void * data,QGuestAllocator * t_alloc)48546488b62SChristian Schoenebeck static void fs_readdir_split_128(void *obj, void *data,
48646488b62SChristian Schoenebeck                                  QGuestAllocator *t_alloc)
48746488b62SChristian Schoenebeck {
488684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
4891d98613dSGreg Kurz     do_readdir_split(obj, 128);
49046488b62SChristian Schoenebeck }
49146488b62SChristian Schoenebeck 
fs_readdir_split_256(void * obj,void * data,QGuestAllocator * t_alloc)49246488b62SChristian Schoenebeck static void fs_readdir_split_256(void *obj, void *data,
49346488b62SChristian Schoenebeck                                  QGuestAllocator *t_alloc)
49446488b62SChristian Schoenebeck {
495684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
4961d98613dSGreg Kurz     do_readdir_split(obj, 256);
49746488b62SChristian Schoenebeck }
49846488b62SChristian Schoenebeck 
fs_readdir_split_512(void * obj,void * data,QGuestAllocator * t_alloc)49946488b62SChristian Schoenebeck static void fs_readdir_split_512(void *obj, void *data,
50046488b62SChristian Schoenebeck                                  QGuestAllocator *t_alloc)
50146488b62SChristian Schoenebeck {
502684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
5031d98613dSGreg Kurz     do_readdir_split(obj, 512);
50446488b62SChristian Schoenebeck }
50546488b62SChristian Schoenebeck 
506653daf38SChristian Schoenebeck 
507653daf38SChristian Schoenebeck /* tests using the 9pfs 'local' fs driver */
508653daf38SChristian Schoenebeck 
fs_create_dir(void * obj,void * data,QGuestAllocator * t_alloc)509653daf38SChristian Schoenebeck static void fs_create_dir(void *obj, void *data, QGuestAllocator *t_alloc)
510653daf38SChristian Schoenebeck {
511653daf38SChristian Schoenebeck     QVirtio9P *v9p = obj;
512684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
513653daf38SChristian Schoenebeck     struct stat st;
51465ceee0aSChristian Schoenebeck     g_autofree char *root_path = virtio_9p_test_path("");
51565ceee0aSChristian Schoenebeck     g_autofree char *new_dir = virtio_9p_test_path("01");
516653daf38SChristian Schoenebeck 
517653daf38SChristian Schoenebeck     g_assert(root_path != NULL);
518653daf38SChristian Schoenebeck 
51974a160abSChristian Schoenebeck     tattach({ .client = v9p });
520e1168010SChristian Schoenebeck     tmkdir({ .client = v9p, .atPath = "/", .name = "01" });
521653daf38SChristian Schoenebeck 
522653daf38SChristian Schoenebeck     /* check if created directory really exists now ... */
523653daf38SChristian Schoenebeck     g_assert(stat(new_dir, &st) == 0);
524653daf38SChristian Schoenebeck     /* ... and is actually a directory */
525653daf38SChristian Schoenebeck     g_assert((st.st_mode & S_IFMT) == S_IFDIR);
526653daf38SChristian Schoenebeck }
527653daf38SChristian Schoenebeck 
fs_unlinkat_dir(void * obj,void * data,QGuestAllocator * t_alloc)528b37d62d6SChristian Schoenebeck static void fs_unlinkat_dir(void *obj, void *data, QGuestAllocator *t_alloc)
529b37d62d6SChristian Schoenebeck {
530b37d62d6SChristian Schoenebeck     QVirtio9P *v9p = obj;
531684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
532b37d62d6SChristian Schoenebeck     struct stat st;
53365ceee0aSChristian Schoenebeck     g_autofree char *root_path = virtio_9p_test_path("");
53465ceee0aSChristian Schoenebeck     g_autofree char *new_dir = virtio_9p_test_path("02");
535b37d62d6SChristian Schoenebeck 
536b37d62d6SChristian Schoenebeck     g_assert(root_path != NULL);
537b37d62d6SChristian Schoenebeck 
53874a160abSChristian Schoenebeck     tattach({ .client = v9p });
539e1168010SChristian Schoenebeck     tmkdir({ .client = v9p, .atPath = "/", .name = "02" });
540b37d62d6SChristian Schoenebeck 
541b37d62d6SChristian Schoenebeck     /* check if created directory really exists now ... */
542b37d62d6SChristian Schoenebeck     g_assert(stat(new_dir, &st) == 0);
543b37d62d6SChristian Schoenebeck     /* ... and is actually a directory */
544b37d62d6SChristian Schoenebeck     g_assert((st.st_mode & S_IFMT) == S_IFDIR);
545b37d62d6SChristian Schoenebeck 
546*43e0d9fbSChristian Schoenebeck     tunlinkat({
547*43e0d9fbSChristian Schoenebeck         .client = v9p, .atPath = "/", .name = "02",
548*43e0d9fbSChristian Schoenebeck         .flags = P9_DOTL_AT_REMOVEDIR
549*43e0d9fbSChristian Schoenebeck     });
550b37d62d6SChristian Schoenebeck     /* directory should be gone now */
551b37d62d6SChristian Schoenebeck     g_assert(stat(new_dir, &st) != 0);
552b37d62d6SChristian Schoenebeck }
553b37d62d6SChristian Schoenebeck 
fs_create_file(void * obj,void * data,QGuestAllocator * t_alloc)554b09dbfddSChristian Schoenebeck static void fs_create_file(void *obj, void *data, QGuestAllocator *t_alloc)
555b09dbfddSChristian Schoenebeck {
556b09dbfddSChristian Schoenebeck     QVirtio9P *v9p = obj;
557684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
558b09dbfddSChristian Schoenebeck     struct stat st;
55965ceee0aSChristian Schoenebeck     g_autofree char *new_file = virtio_9p_test_path("03/1st_file");
560b09dbfddSChristian Schoenebeck 
56174a160abSChristian Schoenebeck     tattach({ .client = v9p });
562e1168010SChristian Schoenebeck     tmkdir({ .client = v9p, .atPath = "/", .name = "03" });
563bd4660d4SChristian Schoenebeck     tlcreate({ .client = v9p, .atPath = "03", .name = "1st_file" });
564b09dbfddSChristian Schoenebeck 
565b09dbfddSChristian Schoenebeck     /* check if created file exists now ... */
566b09dbfddSChristian Schoenebeck     g_assert(stat(new_file, &st) == 0);
567b09dbfddSChristian Schoenebeck     /* ... and is a regular file */
568b09dbfddSChristian Schoenebeck     g_assert((st.st_mode & S_IFMT) == S_IFREG);
569b09dbfddSChristian Schoenebeck }
570b09dbfddSChristian Schoenebeck 
fs_unlinkat_file(void * obj,void * data,QGuestAllocator * t_alloc)571472c18b8SChristian Schoenebeck static void fs_unlinkat_file(void *obj, void *data, QGuestAllocator *t_alloc)
572472c18b8SChristian Schoenebeck {
573472c18b8SChristian Schoenebeck     QVirtio9P *v9p = obj;
574684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
575472c18b8SChristian Schoenebeck     struct stat st;
57665ceee0aSChristian Schoenebeck     g_autofree char *new_file = virtio_9p_test_path("04/doa_file");
577472c18b8SChristian Schoenebeck 
57874a160abSChristian Schoenebeck     tattach({ .client = v9p });
579e1168010SChristian Schoenebeck     tmkdir({ .client = v9p, .atPath = "/", .name = "04" });
580bd4660d4SChristian Schoenebeck     tlcreate({ .client = v9p, .atPath = "04", .name = "doa_file" });
581472c18b8SChristian Schoenebeck 
582472c18b8SChristian Schoenebeck     /* check if created file exists now ... */
583472c18b8SChristian Schoenebeck     g_assert(stat(new_file, &st) == 0);
584472c18b8SChristian Schoenebeck     /* ... and is a regular file */
585472c18b8SChristian Schoenebeck     g_assert((st.st_mode & S_IFMT) == S_IFREG);
586472c18b8SChristian Schoenebeck 
587*43e0d9fbSChristian Schoenebeck     tunlinkat({ .client = v9p, .atPath = "04", .name = "doa_file" });
588472c18b8SChristian Schoenebeck     /* file should be gone now */
589472c18b8SChristian Schoenebeck     g_assert(stat(new_file, &st) != 0);
590472c18b8SChristian Schoenebeck }
591472c18b8SChristian Schoenebeck 
fs_symlink_file(void * obj,void * data,QGuestAllocator * t_alloc)59259ff563dSChristian Schoenebeck static void fs_symlink_file(void *obj, void *data, QGuestAllocator *t_alloc)
59359ff563dSChristian Schoenebeck {
59459ff563dSChristian Schoenebeck     QVirtio9P *v9p = obj;
595684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
59659ff563dSChristian Schoenebeck     struct stat st;
59765ceee0aSChristian Schoenebeck     g_autofree char *real_file = virtio_9p_test_path("05/real_file");
59865ceee0aSChristian Schoenebeck     g_autofree char *symlink_file = virtio_9p_test_path("05/symlink_file");
59959ff563dSChristian Schoenebeck 
60074a160abSChristian Schoenebeck     tattach({ .client = v9p });
601e1168010SChristian Schoenebeck     tmkdir({ .client = v9p, .atPath = "/", .name = "05" });
602bd4660d4SChristian Schoenebeck     tlcreate({ .client = v9p, .atPath = "05", .name = "real_file" });
60359ff563dSChristian Schoenebeck     g_assert(stat(real_file, &st) == 0);
60459ff563dSChristian Schoenebeck     g_assert((st.st_mode & S_IFMT) == S_IFREG);
60559ff563dSChristian Schoenebeck 
6069beabfa5SChristian Schoenebeck     tsymlink({
6079beabfa5SChristian Schoenebeck         .client = v9p, .atPath = "05", .name = "symlink_file",
6089beabfa5SChristian Schoenebeck         .symtgt = "real_file"
6099beabfa5SChristian Schoenebeck     });
61059ff563dSChristian Schoenebeck 
61159ff563dSChristian Schoenebeck     /* check if created link exists now */
61259ff563dSChristian Schoenebeck     g_assert(stat(symlink_file, &st) == 0);
61359ff563dSChristian Schoenebeck }
61459ff563dSChristian Schoenebeck 
fs_unlinkat_symlink(void * obj,void * data,QGuestAllocator * t_alloc)6155b28ab8bSChristian Schoenebeck static void fs_unlinkat_symlink(void *obj, void *data,
6165b28ab8bSChristian Schoenebeck                                 QGuestAllocator *t_alloc)
6175b28ab8bSChristian Schoenebeck {
6185b28ab8bSChristian Schoenebeck     QVirtio9P *v9p = obj;
619684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
6205b28ab8bSChristian Schoenebeck     struct stat st;
62165ceee0aSChristian Schoenebeck     g_autofree char *real_file = virtio_9p_test_path("06/real_file");
62265ceee0aSChristian Schoenebeck     g_autofree char *symlink_file = virtio_9p_test_path("06/symlink_file");
6235b28ab8bSChristian Schoenebeck 
62474a160abSChristian Schoenebeck     tattach({ .client = v9p });
625e1168010SChristian Schoenebeck     tmkdir({ .client = v9p, .atPath = "/", .name = "06" });
626bd4660d4SChristian Schoenebeck     tlcreate({ .client = v9p, .atPath = "06", .name = "real_file" });
6275b28ab8bSChristian Schoenebeck     g_assert(stat(real_file, &st) == 0);
6285b28ab8bSChristian Schoenebeck     g_assert((st.st_mode & S_IFMT) == S_IFREG);
6295b28ab8bSChristian Schoenebeck 
6309beabfa5SChristian Schoenebeck     tsymlink({
6319beabfa5SChristian Schoenebeck         .client = v9p, .atPath = "06", .name = "symlink_file",
6329beabfa5SChristian Schoenebeck         .symtgt = "real_file"
6339beabfa5SChristian Schoenebeck     });
6345b28ab8bSChristian Schoenebeck     g_assert(stat(symlink_file, &st) == 0);
6355b28ab8bSChristian Schoenebeck 
636*43e0d9fbSChristian Schoenebeck     tunlinkat({ .client = v9p, .atPath = "06", .name = "symlink_file" });
6375b28ab8bSChristian Schoenebeck     /* symlink should be gone now */
6385b28ab8bSChristian Schoenebeck     g_assert(stat(symlink_file, &st) != 0);
6395b28ab8bSChristian Schoenebeck }
6405b28ab8bSChristian Schoenebeck 
fs_hardlink_file(void * obj,void * data,QGuestAllocator * t_alloc)64164e3d403SChristian Schoenebeck static void fs_hardlink_file(void *obj, void *data, QGuestAllocator *t_alloc)
64264e3d403SChristian Schoenebeck {
64364e3d403SChristian Schoenebeck     QVirtio9P *v9p = obj;
644684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
64564e3d403SChristian Schoenebeck     struct stat st_real, st_link;
64665ceee0aSChristian Schoenebeck     g_autofree char *real_file = virtio_9p_test_path("07/real_file");
64765ceee0aSChristian Schoenebeck     g_autofree char *hardlink_file = virtio_9p_test_path("07/hardlink_file");
64864e3d403SChristian Schoenebeck 
64974a160abSChristian Schoenebeck     tattach({ .client = v9p });
650e1168010SChristian Schoenebeck     tmkdir({ .client = v9p, .atPath = "/", .name = "07" });
651bd4660d4SChristian Schoenebeck     tlcreate({ .client = v9p, .atPath = "07", .name = "real_file" });
65264e3d403SChristian Schoenebeck     g_assert(stat(real_file, &st_real) == 0);
65364e3d403SChristian Schoenebeck     g_assert((st_real.st_mode & S_IFMT) == S_IFREG);
65464e3d403SChristian Schoenebeck 
655d41a9462SChristian Schoenebeck     tlink({
656d41a9462SChristian Schoenebeck         .client = v9p, .atPath = "07", .name = "hardlink_file",
657d41a9462SChristian Schoenebeck         .toPath = "07/real_file"
658d41a9462SChristian Schoenebeck     });
65964e3d403SChristian Schoenebeck 
66064e3d403SChristian Schoenebeck     /* check if link exists now ... */
66164e3d403SChristian Schoenebeck     g_assert(stat(hardlink_file, &st_link) == 0);
66264e3d403SChristian Schoenebeck     /* ... and it's a hard link, right? */
66364e3d403SChristian Schoenebeck     g_assert((st_link.st_mode & S_IFMT) == S_IFREG);
66464e3d403SChristian Schoenebeck     g_assert(st_link.st_dev == st_real.st_dev);
66564e3d403SChristian Schoenebeck     g_assert(st_link.st_ino == st_real.st_ino);
66664e3d403SChristian Schoenebeck }
66764e3d403SChristian Schoenebeck 
fs_unlinkat_hardlink(void * obj,void * data,QGuestAllocator * t_alloc)6684d0746e2SChristian Schoenebeck static void fs_unlinkat_hardlink(void *obj, void *data,
6694d0746e2SChristian Schoenebeck                                  QGuestAllocator *t_alloc)
6704d0746e2SChristian Schoenebeck {
6714d0746e2SChristian Schoenebeck     QVirtio9P *v9p = obj;
672684f9120SChristian Schoenebeck     v9fs_set_allocator(t_alloc);
6734d0746e2SChristian Schoenebeck     struct stat st_real, st_link;
67465ceee0aSChristian Schoenebeck     g_autofree char *real_file = virtio_9p_test_path("08/real_file");
67565ceee0aSChristian Schoenebeck     g_autofree char *hardlink_file = virtio_9p_test_path("08/hardlink_file");
6764d0746e2SChristian Schoenebeck 
67774a160abSChristian Schoenebeck     tattach({ .client = v9p });
678e1168010SChristian Schoenebeck     tmkdir({ .client = v9p, .atPath = "/", .name = "08" });
679bd4660d4SChristian Schoenebeck     tlcreate({ .client = v9p, .atPath = "08", .name = "real_file" });
6804d0746e2SChristian Schoenebeck     g_assert(stat(real_file, &st_real) == 0);
6814d0746e2SChristian Schoenebeck     g_assert((st_real.st_mode & S_IFMT) == S_IFREG);
6824d0746e2SChristian Schoenebeck 
683d41a9462SChristian Schoenebeck     tlink({
684d41a9462SChristian Schoenebeck         .client = v9p, .atPath = "08", .name = "hardlink_file",
685d41a9462SChristian Schoenebeck         .toPath = "08/real_file"
686d41a9462SChristian Schoenebeck     });
6874d0746e2SChristian Schoenebeck     g_assert(stat(hardlink_file, &st_link) == 0);
6884d0746e2SChristian Schoenebeck 
689*43e0d9fbSChristian Schoenebeck     tunlinkat({ .client = v9p, .atPath = "08", .name = "hardlink_file" });
6904d0746e2SChristian Schoenebeck     /* symlink should be gone now */
6914d0746e2SChristian Schoenebeck     g_assert(stat(hardlink_file, &st_link) != 0);
6924d0746e2SChristian Schoenebeck     /* and old file should still exist */
6934d0746e2SChristian Schoenebeck     g_assert(stat(real_file, &st_real) == 0);
6944d0746e2SChristian Schoenebeck }
6954d0746e2SChristian Schoenebeck 
assign_9p_local_driver(GString * cmd_line,void * arg)6963a565c64SChristian Schoenebeck static void *assign_9p_local_driver(GString *cmd_line, void *arg)
6973a565c64SChristian Schoenebeck {
6983a565c64SChristian Schoenebeck     virtio_9p_assign_local_driver(cmd_line, "security_model=mapped-xattr");
6993a565c64SChristian Schoenebeck     return arg;
7003a565c64SChristian Schoenebeck }
7013a565c64SChristian Schoenebeck 
register_virtio_9p_test(void)7021e8a1faeSThomas Huth static void register_virtio_9p_test(void)
7031e8a1faeSThomas Huth {
7043a565c64SChristian Schoenebeck 
7053a565c64SChristian Schoenebeck     QOSGraphTestOptions opts = {
7063a565c64SChristian Schoenebeck     };
7073a565c64SChristian Schoenebeck 
7083a565c64SChristian Schoenebeck     /* 9pfs test cases using the 'synth' filesystem driver */
7093a565c64SChristian Schoenebeck     qos_add_test("synth/config", "virtio-9p", pci_config, &opts);
7103a565c64SChristian Schoenebeck     qos_add_test("synth/version/basic", "virtio-9p", fs_version,  &opts);
7113a565c64SChristian Schoenebeck     qos_add_test("synth/attach/basic", "virtio-9p", fs_attach,  &opts);
7123a565c64SChristian Schoenebeck     qos_add_test("synth/walk/basic", "virtio-9p", fs_walk,  &opts);
713eefd2394SChristian Schoenebeck     qos_add_test("synth/walk/no_slash", "virtio-9p", fs_walk_no_slash,
7143a565c64SChristian Schoenebeck                   &opts);
715c1668948SChristian Schoenebeck     qos_add_test("synth/walk/none", "virtio-9p", fs_walk_none, &opts);
716eefd2394SChristian Schoenebeck     qos_add_test("synth/walk/dotdot_from_root", "virtio-9p",
7173a565c64SChristian Schoenebeck                  fs_walk_dotdot,  &opts);
7189472a689SChristian Schoenebeck     qos_add_test("synth/walk/non_existent", "virtio-9p", fs_walk_nonexistent,
7199472a689SChristian Schoenebeck                   &opts);
72015fbff48SChristian Schoenebeck     qos_add_test("synth/walk/2nd_non_existent", "virtio-9p",
72115fbff48SChristian Schoenebeck                  fs_walk_2nd_nonexistent, &opts);
7223a565c64SChristian Schoenebeck     qos_add_test("synth/lopen/basic", "virtio-9p", fs_lopen,  &opts);
7233a565c64SChristian Schoenebeck     qos_add_test("synth/write/basic", "virtio-9p", fs_write,  &opts);
724eefd2394SChristian Schoenebeck     qos_add_test("synth/flush/success", "virtio-9p", fs_flush_success,
7253a565c64SChristian Schoenebeck                   &opts);
726eefd2394SChristian Schoenebeck     qos_add_test("synth/flush/ignored", "virtio-9p", fs_flush_ignored,
7273a565c64SChristian Schoenebeck                   &opts);
7283a565c64SChristian Schoenebeck     qos_add_test("synth/readdir/basic", "virtio-9p", fs_readdir,  &opts);
729eefd2394SChristian Schoenebeck     qos_add_test("synth/readdir/split_512", "virtio-9p",
7303a565c64SChristian Schoenebeck                  fs_readdir_split_512,  &opts);
731eefd2394SChristian Schoenebeck     qos_add_test("synth/readdir/split_256", "virtio-9p",
7323a565c64SChristian Schoenebeck                  fs_readdir_split_256,  &opts);
733eefd2394SChristian Schoenebeck     qos_add_test("synth/readdir/split_128", "virtio-9p",
7343a565c64SChristian Schoenebeck                  fs_readdir_split_128,  &opts);
7353a565c64SChristian Schoenebeck 
7363a565c64SChristian Schoenebeck 
7373a565c64SChristian Schoenebeck     /* 9pfs test cases using the 'local' filesystem driver */
738558f5c42SGreg Kurz 
739558f5c42SGreg Kurz     /*
740558f5c42SGreg Kurz      * XXX: Until we are sure that these tests can run everywhere,
741558f5c42SGreg Kurz      * keep them as "slow" so that they aren't run with "make check".
742558f5c42SGreg Kurz      */
743558f5c42SGreg Kurz     if (!g_test_slow()) {
744558f5c42SGreg Kurz         return;
745558f5c42SGreg Kurz     }
746558f5c42SGreg Kurz 
7473a565c64SChristian Schoenebeck     opts.before = assign_9p_local_driver;
7483a565c64SChristian Schoenebeck     qos_add_test("local/config", "virtio-9p", pci_config,  &opts);
749653daf38SChristian Schoenebeck     qos_add_test("local/create_dir", "virtio-9p", fs_create_dir, &opts);
750b37d62d6SChristian Schoenebeck     qos_add_test("local/unlinkat_dir", "virtio-9p", fs_unlinkat_dir, &opts);
751b09dbfddSChristian Schoenebeck     qos_add_test("local/create_file", "virtio-9p", fs_create_file, &opts);
752472c18b8SChristian Schoenebeck     qos_add_test("local/unlinkat_file", "virtio-9p", fs_unlinkat_file, &opts);
75359ff563dSChristian Schoenebeck     qos_add_test("local/symlink_file", "virtio-9p", fs_symlink_file, &opts);
7545b28ab8bSChristian Schoenebeck     qos_add_test("local/unlinkat_symlink", "virtio-9p", fs_unlinkat_symlink,
7555b28ab8bSChristian Schoenebeck                  &opts);
75664e3d403SChristian Schoenebeck     qos_add_test("local/hardlink_file", "virtio-9p", fs_hardlink_file, &opts);
7574d0746e2SChristian Schoenebeck     qos_add_test("local/unlinkat_hardlink", "virtio-9p", fs_unlinkat_hardlink,
7584d0746e2SChristian Schoenebeck                  &opts);
7591e8a1faeSThomas Huth }
7601e8a1faeSThomas Huth 
7611e8a1faeSThomas Huth libqos_init(register_virtio_9p_test);
762136b7af2SChristian Schoenebeck 
construct_9p_test(void)763136b7af2SChristian Schoenebeck static void __attribute__((constructor)) construct_9p_test(void)
764136b7af2SChristian Schoenebeck {
765136b7af2SChristian Schoenebeck     /* make sure test dir for the 'local' tests exists */
766136b7af2SChristian Schoenebeck     virtio_9p_create_local_test_dir();
767136b7af2SChristian Schoenebeck }
768136b7af2SChristian Schoenebeck 
destruct_9p_test(void)769136b7af2SChristian Schoenebeck static void __attribute__((destructor)) destruct_9p_test(void)
770136b7af2SChristian Schoenebeck {
771136b7af2SChristian Schoenebeck     /* remove previously created test dir when test suite completed */
772136b7af2SChristian Schoenebeck     virtio_9p_remove_local_test_dir();
773136b7af2SChristian Schoenebeck }
774