12d888c09SAndreas Färber /*
22d888c09SAndreas Färber * QTest testcase for VirtIO 9P
32d888c09SAndreas Färber *
42d888c09SAndreas Färber * Copyright (c) 2014 SUSE LINUX Products GmbH
52d888c09SAndreas Färber *
62d888c09SAndreas Färber * This work is licensed under the terms of the GNU GPL, version 2 or later.
72d888c09SAndreas Färber * See the COPYING file in the top-level directory.
82d888c09SAndreas Färber */
92d888c09SAndreas Färber
106f569084SChristian Schoenebeck /*
116f569084SChristian Schoenebeck * Not so fast! You might want to read the 9p developer docs first:
126f569084SChristian Schoenebeck * https://wiki.qemu.org/Documentation/9p
136f569084SChristian Schoenebeck */
146f569084SChristian Schoenebeck
15fbc04127SPeter Maydell #include "qemu/osdep.h"
160b8fa32fSMarkus Armbruster #include "qemu/module.h"
17684f9120SChristian Schoenebeck #include "libqos/virtio-9p-client.h"
1865b70fc7SGreg Kurz
19569f3b63SChristian Schoenebeck #define twalk(...) v9fs_twalk((TWalkOpt) __VA_ARGS__)
20bee8fda2SChristian Schoenebeck #define tversion(...) v9fs_tversion((TVersionOpt) __VA_ARGS__)
2174a160abSChristian Schoenebeck #define tattach(...) v9fs_tattach((TAttachOpt) __VA_ARGS__)
222af5be47SChristian Schoenebeck #define tgetattr(...) v9fs_tgetattr((TGetAttrOpt) __VA_ARGS__)
234719a2d5SChristian Schoenebeck #define tsetattr(...) v9fs_tsetattr((TSetAttrOpt) __VA_ARGS__)
241ebacc40SChristian Schoenebeck #define treaddir(...) v9fs_treaddir((TReadDirOpt) __VA_ARGS__)
253878ce4cSChristian Schoenebeck #define tlopen(...) v9fs_tlopen((TLOpenOpt) __VA_ARGS__)
26ac9e4e61SChristian Schoenebeck #define twrite(...) v9fs_twrite((TWriteOpt) __VA_ARGS__)
27d89146fdSChristian Schoenebeck #define tflush(...) v9fs_tflush((TFlushOpt) __VA_ARGS__)
28e1168010SChristian Schoenebeck #define tmkdir(...) v9fs_tmkdir((TMkdirOpt) __VA_ARGS__)
29bd4660d4SChristian Schoenebeck #define tlcreate(...) v9fs_tlcreate((TlcreateOpt) __VA_ARGS__)
309beabfa5SChristian Schoenebeck #define tsymlink(...) v9fs_tsymlink((TsymlinkOpt) __VA_ARGS__)
31d41a9462SChristian Schoenebeck #define tlink(...) v9fs_tlink((TlinkOpt) __VA_ARGS__)
3243e0d9fbSChristian Schoenebeck #define tunlinkat(...) v9fs_tunlinkat((TunlinkatOpt) __VA_ARGS__)
33653daf38SChristian Schoenebeck
pci_config(void * obj,void * data,QGuestAllocator * t_alloc)34dfbe8b43SEmanuele Giuseppe Esposito static void pci_config(void *obj, void *data, QGuestAllocator *t_alloc)
35557a4cc0SGreg Kurz {
36dfbe8b43SEmanuele Giuseppe Esposito QVirtio9P *v9p = obj;
37684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
38dfbe8b43SEmanuele Giuseppe Esposito size_t tag_len = qvirtio_config_readw(v9p->vdev, 0);
3965ceee0aSChristian Schoenebeck g_autofree char *tag = NULL;
40557a4cc0SGreg Kurz int i;
41557a4cc0SGreg Kurz
42dfbe8b43SEmanuele Giuseppe Esposito g_assert_cmpint(tag_len, ==, strlen(MOUNT_TAG));
43557a4cc0SGreg Kurz
44557a4cc0SGreg Kurz tag = g_malloc(tag_len);
45557a4cc0SGreg Kurz for (i = 0; i < tag_len; i++) {
46dfbe8b43SEmanuele Giuseppe Esposito tag[i] = qvirtio_config_readb(v9p->vdev, i + 2);
47557a4cc0SGreg Kurz }
48dfbe8b43SEmanuele Giuseppe Esposito g_assert_cmpmem(tag, tag_len, MOUNT_TAG, tag_len);
491211d81bSGreg Kurz }
50557a4cc0SGreg Kurz
is_same_qid(v9fs_qid a,v9fs_qid b)51a6821b82SChristian Schoenebeck static inline bool is_same_qid(v9fs_qid a, v9fs_qid b)
52a6821b82SChristian Schoenebeck {
53a6821b82SChristian Schoenebeck /* don't compare QID version for checking for file ID equalness */
54a6821b82SChristian Schoenebeck return a[0] == b[0] && memcmp(&a[5], &b[5], 8) == 0;
55a6821b82SChristian Schoenebeck }
56a6821b82SChristian Schoenebeck
fs_version(void * obj,void * data,QGuestAllocator * t_alloc)571c450e6eSGreg Kurz static void fs_version(void *obj, void *data, QGuestAllocator *t_alloc)
581c450e6eSGreg Kurz {
59684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
60bee8fda2SChristian Schoenebeck tversion({ .client = obj });
611c450e6eSGreg Kurz }
621c450e6eSGreg Kurz
fs_attach(void * obj,void * data,QGuestAllocator * t_alloc)633fe4baf4SGreg Kurz static void fs_attach(void *obj, void *data, QGuestAllocator *t_alloc)
643fe4baf4SGreg Kurz {
65684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
6674a160abSChristian Schoenebeck tattach({ .client = obj });
673fe4baf4SGreg Kurz }
683fe4baf4SGreg Kurz
fs_walk(void * obj,void * data,QGuestAllocator * t_alloc)69dfbe8b43SEmanuele Giuseppe Esposito static void fs_walk(void *obj, void *data, QGuestAllocator *t_alloc)
7004b88c84SGreg Kurz {
71dfbe8b43SEmanuele Giuseppe Esposito QVirtio9P *v9p = obj;
72684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
732893ddd5SGreg Kurz char *wnames[P9_MAXWELEM];
7404b88c84SGreg Kurz uint16_t nwqid;
7565ceee0aSChristian Schoenebeck g_autofree v9fs_qid *wqid = NULL;
7604b88c84SGreg Kurz int i;
7704b88c84SGreg Kurz
7804b88c84SGreg Kurz for (i = 0; i < P9_MAXWELEM; i++) {
792893ddd5SGreg Kurz wnames[i] = g_strdup_printf(QTEST_V9FS_SYNTH_WALK_FILE, i);
8004b88c84SGreg Kurz }
8104b88c84SGreg Kurz
8274a160abSChristian Schoenebeck tattach({ .client = v9p });
833f3e9232SChristian Schoenebeck twalk({
84569f3b63SChristian Schoenebeck .client = v9p, .fid = 0, .newfid = 1,
853f3e9232SChristian Schoenebeck .nwname = P9_MAXWELEM, .wnames = wnames,
863f3e9232SChristian Schoenebeck .rwalk = { .nwqid = &nwqid, .wqid = &wqid }
873f3e9232SChristian Schoenebeck });
8804b88c84SGreg Kurz
8904b88c84SGreg Kurz g_assert_cmpint(nwqid, ==, P9_MAXWELEM);
9004b88c84SGreg Kurz
9104b88c84SGreg Kurz for (i = 0; i < P9_MAXWELEM; i++) {
9204b88c84SGreg Kurz g_free(wnames[i]);
9304b88c84SGreg Kurz }
9404b88c84SGreg Kurz }
9504b88c84SGreg Kurz
fs_dirents_contain_name(struct V9fsDirent * e,const char * name)964829469fSChristian Schoenebeck static bool fs_dirents_contain_name(struct V9fsDirent *e, const char* name)
974829469fSChristian Schoenebeck {
984829469fSChristian Schoenebeck for (; e; e = e->next) {
994829469fSChristian Schoenebeck if (!strcmp(e->name, name)) {
1004829469fSChristian Schoenebeck return true;
1014829469fSChristian Schoenebeck }
1024829469fSChristian Schoenebeck }
1034829469fSChristian Schoenebeck return false;
1044829469fSChristian Schoenebeck }
1054829469fSChristian Schoenebeck
10646488b62SChristian Schoenebeck /* basic readdir test where reply fits into a single response message */
fs_readdir(void * obj,void * data,QGuestAllocator * t_alloc)1074829469fSChristian Schoenebeck static void fs_readdir(void *obj, void *data, QGuestAllocator *t_alloc)
1084829469fSChristian Schoenebeck {
1094829469fSChristian Schoenebeck QVirtio9P *v9p = obj;
110684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
111569f3b63SChristian Schoenebeck char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_READDIR_DIR) };
1124829469fSChristian Schoenebeck uint16_t nqid;
1134829469fSChristian Schoenebeck v9fs_qid qid;
1144829469fSChristian Schoenebeck uint32_t count, nentries;
1154829469fSChristian Schoenebeck struct V9fsDirent *entries = NULL;
1164829469fSChristian Schoenebeck
11774a160abSChristian Schoenebeck tattach({ .client = v9p });
1183f3e9232SChristian Schoenebeck twalk({
119569f3b63SChristian Schoenebeck .client = v9p, .fid = 0, .newfid = 1,
1203f3e9232SChristian Schoenebeck .nwname = 1, .wnames = wnames, .rwalk.nwqid = &nqid
1213f3e9232SChristian Schoenebeck });
1224829469fSChristian Schoenebeck g_assert_cmpint(nqid, ==, 1);
1234829469fSChristian Schoenebeck
1240e4c4ff0SChristian Schoenebeck tlopen({
1250e4c4ff0SChristian Schoenebeck .client = v9p, .fid = 1, .flags = O_DIRECTORY, .rlopen.qid = &qid
1260e4c4ff0SChristian Schoenebeck });
1274829469fSChristian Schoenebeck
1284829469fSChristian Schoenebeck /*
1294829469fSChristian Schoenebeck * submit count = msize - 11, because 11 is the header size of Rreaddir
1304829469fSChristian Schoenebeck */
131a9a53769SChristian Schoenebeck treaddir({
1321ebacc40SChristian Schoenebeck .client = v9p, .fid = 1, .offset = 0, .count = P9_MAX_SIZE - 11,
133a9a53769SChristian Schoenebeck .rreaddir = {
134a9a53769SChristian Schoenebeck .count = &count, .nentries = &nentries, .entries = &entries
135a9a53769SChristian Schoenebeck }
136a9a53769SChristian Schoenebeck });
1374829469fSChristian Schoenebeck
1384829469fSChristian Schoenebeck /*
1394829469fSChristian Schoenebeck * Assuming msize (P9_MAX_SIZE) is large enough so we can retrieve all
1404829469fSChristian Schoenebeck * dir entries with only one readdir request.
1414829469fSChristian Schoenebeck */
1424829469fSChristian Schoenebeck g_assert_cmpint(
1434829469fSChristian Schoenebeck nentries, ==,
1444829469fSChristian Schoenebeck QTEST_V9FS_SYNTH_READDIR_NFILES + 2 /* "." and ".." */
1454829469fSChristian Schoenebeck );
1464829469fSChristian Schoenebeck
1474829469fSChristian Schoenebeck /*
1484829469fSChristian Schoenebeck * Check all file names exist in returned entries, ignore their order
1494829469fSChristian Schoenebeck * though.
1504829469fSChristian Schoenebeck */
1514829469fSChristian Schoenebeck g_assert_cmpint(fs_dirents_contain_name(entries, "."), ==, true);
1524829469fSChristian Schoenebeck g_assert_cmpint(fs_dirents_contain_name(entries, ".."), ==, true);
1534829469fSChristian Schoenebeck for (int i = 0; i < QTEST_V9FS_SYNTH_READDIR_NFILES; ++i) {
15465ceee0aSChristian Schoenebeck g_autofree char *name =
15565ceee0aSChristian Schoenebeck g_strdup_printf(QTEST_V9FS_SYNTH_READDIR_FILE, i);
1564829469fSChristian Schoenebeck g_assert_cmpint(fs_dirents_contain_name(entries, name), ==, true);
1574829469fSChristian Schoenebeck }
1584829469fSChristian Schoenebeck
1594829469fSChristian Schoenebeck v9fs_free_dirents(entries);
1604829469fSChristian Schoenebeck g_free(wnames[0]);
1614829469fSChristian Schoenebeck }
1624829469fSChristian Schoenebeck
16346488b62SChristian Schoenebeck /* readdir test where overall request is split over several messages */
do_readdir_split(QVirtio9P * v9p,uint32_t count)1641d98613dSGreg Kurz static void do_readdir_split(QVirtio9P *v9p, uint32_t count)
16546488b62SChristian Schoenebeck {
166569f3b63SChristian Schoenebeck char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_READDIR_DIR) };
16746488b62SChristian Schoenebeck uint16_t nqid;
16846488b62SChristian Schoenebeck v9fs_qid qid;
16946488b62SChristian Schoenebeck uint32_t nentries, npartialentries;
17046488b62SChristian Schoenebeck struct V9fsDirent *entries, *tail, *partialentries;
17146488b62SChristian Schoenebeck int fid;
17246488b62SChristian Schoenebeck uint64_t offset;
17346488b62SChristian Schoenebeck
17474a160abSChristian Schoenebeck tattach({ .client = v9p });
17546488b62SChristian Schoenebeck
17646488b62SChristian Schoenebeck fid = 1;
17746488b62SChristian Schoenebeck offset = 0;
17846488b62SChristian Schoenebeck entries = NULL;
17946488b62SChristian Schoenebeck nentries = 0;
18046488b62SChristian Schoenebeck tail = NULL;
18146488b62SChristian Schoenebeck
1823f3e9232SChristian Schoenebeck twalk({
183569f3b63SChristian Schoenebeck .client = v9p, .fid = 0, .newfid = fid,
1843f3e9232SChristian Schoenebeck .nwname = 1, .wnames = wnames, .rwalk.nwqid = &nqid
1853f3e9232SChristian Schoenebeck });
18646488b62SChristian Schoenebeck g_assert_cmpint(nqid, ==, 1);
18746488b62SChristian Schoenebeck
1880e4c4ff0SChristian Schoenebeck tlopen({
1890e4c4ff0SChristian Schoenebeck .client = v9p, .fid = fid, .flags = O_DIRECTORY, .rlopen.qid = &qid
1900e4c4ff0SChristian Schoenebeck });
19146488b62SChristian Schoenebeck
19246488b62SChristian Schoenebeck /*
19346488b62SChristian Schoenebeck * send as many Treaddir requests as required to get all directory
19446488b62SChristian Schoenebeck * entries
19546488b62SChristian Schoenebeck */
19646488b62SChristian Schoenebeck while (true) {
19746488b62SChristian Schoenebeck npartialentries = 0;
19846488b62SChristian Schoenebeck partialentries = NULL;
19946488b62SChristian Schoenebeck
200a9a53769SChristian Schoenebeck treaddir({
2011ebacc40SChristian Schoenebeck .client = v9p, .fid = fid, .offset = offset, .count = count,
202a9a53769SChristian Schoenebeck .rreaddir = {
203a9a53769SChristian Schoenebeck .count = &count, .nentries = &npartialentries,
204a9a53769SChristian Schoenebeck .entries = &partialentries
205a9a53769SChristian Schoenebeck }
206a9a53769SChristian Schoenebeck });
20746488b62SChristian Schoenebeck if (npartialentries > 0 && partialentries) {
20846488b62SChristian Schoenebeck if (!entries) {
20946488b62SChristian Schoenebeck entries = partialentries;
21046488b62SChristian Schoenebeck nentries = npartialentries;
21146488b62SChristian Schoenebeck tail = partialentries;
21246488b62SChristian Schoenebeck } else {
21346488b62SChristian Schoenebeck tail->next = partialentries;
21446488b62SChristian Schoenebeck nentries += npartialentries;
21546488b62SChristian Schoenebeck }
21646488b62SChristian Schoenebeck while (tail->next) {
21746488b62SChristian Schoenebeck tail = tail->next;
21846488b62SChristian Schoenebeck }
21946488b62SChristian Schoenebeck offset = tail->offset;
22046488b62SChristian Schoenebeck } else {
22146488b62SChristian Schoenebeck break;
22246488b62SChristian Schoenebeck }
22346488b62SChristian Schoenebeck }
22446488b62SChristian Schoenebeck
22546488b62SChristian Schoenebeck g_assert_cmpint(
22646488b62SChristian Schoenebeck nentries, ==,
22746488b62SChristian Schoenebeck QTEST_V9FS_SYNTH_READDIR_NFILES + 2 /* "." and ".." */
22846488b62SChristian Schoenebeck );
22946488b62SChristian Schoenebeck
23046488b62SChristian Schoenebeck /*
23146488b62SChristian Schoenebeck * Check all file names exist in returned entries, ignore their order
23246488b62SChristian Schoenebeck * though.
23346488b62SChristian Schoenebeck */
23446488b62SChristian Schoenebeck g_assert_cmpint(fs_dirents_contain_name(entries, "."), ==, true);
23546488b62SChristian Schoenebeck g_assert_cmpint(fs_dirents_contain_name(entries, ".."), ==, true);
23646488b62SChristian Schoenebeck for (int i = 0; i < QTEST_V9FS_SYNTH_READDIR_NFILES; ++i) {
23746488b62SChristian Schoenebeck char *name = g_strdup_printf(QTEST_V9FS_SYNTH_READDIR_FILE, i);
23846488b62SChristian Schoenebeck g_assert_cmpint(fs_dirents_contain_name(entries, name), ==, true);
23946488b62SChristian Schoenebeck g_free(name);
24046488b62SChristian Schoenebeck }
24146488b62SChristian Schoenebeck
24246488b62SChristian Schoenebeck v9fs_free_dirents(entries);
24346488b62SChristian Schoenebeck
24446488b62SChristian Schoenebeck g_free(wnames[0]);
24546488b62SChristian Schoenebeck }
24646488b62SChristian Schoenebeck
fs_walk_no_slash(void * obj,void * data,QGuestAllocator * t_alloc)247dfbe8b43SEmanuele Giuseppe Esposito static void fs_walk_no_slash(void *obj, void *data, QGuestAllocator *t_alloc)
248ba0d1037SGreg Kurz {
249dfbe8b43SEmanuele Giuseppe Esposito QVirtio9P *v9p = obj;
250684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
251569f3b63SChristian Schoenebeck char *wnames[] = { g_strdup(" /") };
252ba0d1037SGreg Kurz
25374a160abSChristian Schoenebeck tattach({ .client = v9p });
2543f3e9232SChristian Schoenebeck twalk({
255569f3b63SChristian Schoenebeck .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames,
2563f3e9232SChristian Schoenebeck .expectErr = ENOENT
2573f3e9232SChristian Schoenebeck });
258ba0d1037SGreg Kurz
259ba0d1037SGreg Kurz g_free(wnames[0]);
260ba0d1037SGreg Kurz }
261ba0d1037SGreg Kurz
fs_walk_nonexistent(void * obj,void * data,QGuestAllocator * t_alloc)2629472a689SChristian Schoenebeck static void fs_walk_nonexistent(void *obj, void *data, QGuestAllocator *t_alloc)
2639472a689SChristian Schoenebeck {
2649472a689SChristian Schoenebeck QVirtio9P *v9p = obj;
265684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
2669472a689SChristian Schoenebeck
26774a160abSChristian Schoenebeck tattach({ .client = v9p });
26815fbff48SChristian Schoenebeck /*
26915fbff48SChristian Schoenebeck * The 9p2000 protocol spec says: "If the first element cannot be walked
27015fbff48SChristian Schoenebeck * for any reason, Rerror is returned."
27115fbff48SChristian Schoenebeck */
272569f3b63SChristian Schoenebeck twalk({ .client = v9p, .path = "non-existent", .expectErr = ENOENT });
2739472a689SChristian Schoenebeck }
2749472a689SChristian Schoenebeck
fs_walk_2nd_nonexistent(void * obj,void * data,QGuestAllocator * t_alloc)27515fbff48SChristian Schoenebeck static void fs_walk_2nd_nonexistent(void *obj, void *data,
27615fbff48SChristian Schoenebeck QGuestAllocator *t_alloc)
27715fbff48SChristian Schoenebeck {
27815fbff48SChristian Schoenebeck QVirtio9P *v9p = obj;
279684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
2800e43495dSChristian Schoenebeck v9fs_qid root_qid;
28115fbff48SChristian Schoenebeck uint16_t nwqid;
28228c73670SChristian Schoenebeck uint32_t fid;
28315fbff48SChristian Schoenebeck g_autofree v9fs_qid *wqid = NULL;
28415fbff48SChristian Schoenebeck g_autofree char *path = g_strdup_printf(
28515fbff48SChristian Schoenebeck QTEST_V9FS_SYNTH_WALK_FILE "/non-existent", 0
28615fbff48SChristian Schoenebeck );
28715fbff48SChristian Schoenebeck
28874a160abSChristian Schoenebeck tattach({ .client = v9p, .rattach.qid = &root_qid });
289569f3b63SChristian Schoenebeck fid = twalk({
290569f3b63SChristian Schoenebeck .client = v9p, .path = path,
2913f3e9232SChristian Schoenebeck .rwalk = { .nwqid = &nwqid, .wqid = &wqid }
292569f3b63SChristian Schoenebeck }).newfid;
29315fbff48SChristian Schoenebeck /*
29415fbff48SChristian Schoenebeck * The 9p2000 protocol spec says: "nwqid is therefore either nwname or the
29515fbff48SChristian Schoenebeck * index of the first elementwise walk that failed."
29615fbff48SChristian Schoenebeck */
29715fbff48SChristian Schoenebeck assert(nwqid == 1);
2980e43495dSChristian Schoenebeck
2990e43495dSChristian Schoenebeck /* returned QID wqid[0] is file ID of 1st subdir */
3000e43495dSChristian Schoenebeck g_assert(wqid && wqid[0] && !is_same_qid(root_qid, wqid[0]));
3010e43495dSChristian Schoenebeck
3020e43495dSChristian Schoenebeck /* expect fid being unaffected by walk above */
30328c73670SChristian Schoenebeck tgetattr({
3042af5be47SChristian Schoenebeck .client = v9p, .fid = fid, .request_mask = P9_GETATTR_BASIC,
30528c73670SChristian Schoenebeck .expectErr = ENOENT
30628c73670SChristian Schoenebeck });
30715fbff48SChristian Schoenebeck }
30815fbff48SChristian Schoenebeck
fs_walk_none(void * obj,void * data,QGuestAllocator * t_alloc)309c1668948SChristian Schoenebeck static void fs_walk_none(void *obj, void *data, QGuestAllocator *t_alloc)
310c1668948SChristian Schoenebeck {
311c1668948SChristian Schoenebeck QVirtio9P *v9p = obj;
312684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
313c1668948SChristian Schoenebeck v9fs_qid root_qid;
314c1668948SChristian Schoenebeck g_autofree v9fs_qid *wqid = NULL;
315a6821b82SChristian Schoenebeck struct v9fs_attr attr;
316c1668948SChristian Schoenebeck
317bee8fda2SChristian Schoenebeck tversion({ .client = v9p });
3181125ddf6SChristian Schoenebeck tattach({
3191125ddf6SChristian Schoenebeck .client = v9p, .fid = 0, .n_uname = getuid(),
3201125ddf6SChristian Schoenebeck .rattach.qid = &root_qid
3211125ddf6SChristian Schoenebeck });
322c1668948SChristian Schoenebeck
3233f3e9232SChristian Schoenebeck twalk({
324569f3b63SChristian Schoenebeck .client = v9p, .fid = 0, .newfid = 1, .nwname = 0, .wnames = NULL,
3253f3e9232SChristian Schoenebeck .rwalk.wqid = &wqid
3263f3e9232SChristian Schoenebeck });
327c1668948SChristian Schoenebeck
328c1668948SChristian Schoenebeck /* special case: no QID is returned if nwname=0 was sent */
329c1668948SChristian Schoenebeck g_assert(wqid == NULL);
330a6821b82SChristian Schoenebeck
33128c73670SChristian Schoenebeck tgetattr({
3322af5be47SChristian Schoenebeck .client = v9p, .fid = 1, .request_mask = P9_GETATTR_BASIC,
33328c73670SChristian Schoenebeck .rgetattr.attr = &attr
33428c73670SChristian Schoenebeck });
335a6821b82SChristian Schoenebeck
336a6821b82SChristian Schoenebeck g_assert(is_same_qid(root_qid, attr.qid));
337c1668948SChristian Schoenebeck }
338c1668948SChristian Schoenebeck
fs_walk_dotdot(void * obj,void * data,QGuestAllocator * t_alloc)339dfbe8b43SEmanuele Giuseppe Esposito static void fs_walk_dotdot(void *obj, void *data, QGuestAllocator *t_alloc)
340a37c0702SGreg Kurz {
341dfbe8b43SEmanuele Giuseppe Esposito QVirtio9P *v9p = obj;
342684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
343569f3b63SChristian Schoenebeck char *wnames[] = { g_strdup("..") };
34465ceee0aSChristian Schoenebeck v9fs_qid root_qid;
34565ceee0aSChristian Schoenebeck g_autofree v9fs_qid *wqid = NULL;
346a37c0702SGreg Kurz
347bee8fda2SChristian Schoenebeck tversion({ .client = v9p });
3481125ddf6SChristian Schoenebeck tattach({
3491125ddf6SChristian Schoenebeck .client = v9p, .fid = 0, .n_uname = getuid(),
3501125ddf6SChristian Schoenebeck .rattach.qid = &root_qid
3511125ddf6SChristian Schoenebeck });
352a37c0702SGreg Kurz
3533f3e9232SChristian Schoenebeck twalk({
354569f3b63SChristian Schoenebeck .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames,
3553f3e9232SChristian Schoenebeck .rwalk.wqid = &wqid /* We now we'll get one qid */
3563f3e9232SChristian Schoenebeck });
357a37c0702SGreg Kurz
358a37c0702SGreg Kurz g_assert_cmpmem(&root_qid, 13, wqid[0], 13);
359a37c0702SGreg Kurz
360a37c0702SGreg Kurz g_free(wnames[0]);
361a37c0702SGreg Kurz }
362a37c0702SGreg Kurz
fs_lopen(void * obj,void * data,QGuestAllocator * t_alloc)363dfbe8b43SEmanuele Giuseppe Esposito static void fs_lopen(void *obj, void *data, QGuestAllocator *t_alloc)
36482469aaeSGreg Kurz {
365dfbe8b43SEmanuele Giuseppe Esposito QVirtio9P *v9p = obj;
366684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
367569f3b63SChristian Schoenebeck char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_LOPEN_FILE) };
36882469aaeSGreg Kurz
36974a160abSChristian Schoenebeck tattach({ .client = v9p });
3703f3e9232SChristian Schoenebeck twalk({
3713f3e9232SChristian Schoenebeck .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames
3723f3e9232SChristian Schoenebeck });
37382469aaeSGreg Kurz
3740e4c4ff0SChristian Schoenebeck tlopen({ .client = v9p, .fid = 1, .flags = O_WRONLY });
37582469aaeSGreg Kurz
37682469aaeSGreg Kurz g_free(wnames[0]);
37782469aaeSGreg Kurz }
37882469aaeSGreg Kurz
fs_write(void * obj,void * data,QGuestAllocator * t_alloc)379dfbe8b43SEmanuele Giuseppe Esposito static void fs_write(void *obj, void *data, QGuestAllocator *t_alloc)
380354b86f8SGreg Kurz {
381dfbe8b43SEmanuele Giuseppe Esposito QVirtio9P *v9p = obj;
382684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
383354b86f8SGreg Kurz static const uint32_t write_count = P9_MAX_SIZE / 2;
384569f3b63SChristian Schoenebeck char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_WRITE_FILE) };
38565ceee0aSChristian Schoenebeck g_autofree char *buf = g_malloc0(write_count);
386354b86f8SGreg Kurz uint32_t count;
387354b86f8SGreg Kurz
38874a160abSChristian Schoenebeck tattach({ .client = v9p });
3893f3e9232SChristian Schoenebeck twalk({
3903f3e9232SChristian Schoenebeck .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames
3913f3e9232SChristian Schoenebeck });
392354b86f8SGreg Kurz
3930e4c4ff0SChristian Schoenebeck tlopen({ .client = v9p, .fid = 1, .flags = O_WRONLY });
394354b86f8SGreg Kurz
395bb286ff8SChristian Schoenebeck count = twrite({
396ac9e4e61SChristian Schoenebeck .client = v9p, .fid = 1, .offset = 0, .count = write_count,
397bb286ff8SChristian Schoenebeck .data = buf
398bb286ff8SChristian Schoenebeck }).count;
399354b86f8SGreg Kurz g_assert_cmpint(count, ==, write_count);
400354b86f8SGreg Kurz
401354b86f8SGreg Kurz g_free(wnames[0]);
402354b86f8SGreg Kurz }
403354b86f8SGreg Kurz
fs_flush_success(void * obj,void * data,QGuestAllocator * t_alloc)404dfbe8b43SEmanuele Giuseppe Esposito static void fs_flush_success(void *obj, void *data, QGuestAllocator *t_alloc)
405357e2f7fSGreg Kurz {
406dfbe8b43SEmanuele Giuseppe Esposito QVirtio9P *v9p = obj;
407684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
408569f3b63SChristian Schoenebeck char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_FLUSH_FILE) };
409357e2f7fSGreg Kurz P9Req *req, *flush_req;
410357e2f7fSGreg Kurz uint32_t reply_len;
411357e2f7fSGreg Kurz uint8_t should_block;
412357e2f7fSGreg Kurz
41374a160abSChristian Schoenebeck tattach({ .client = v9p });
4143f3e9232SChristian Schoenebeck twalk({
4153f3e9232SChristian Schoenebeck .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames
4163f3e9232SChristian Schoenebeck });
417357e2f7fSGreg Kurz
4180e4c4ff0SChristian Schoenebeck tlopen({ .client = v9p, .fid = 1, .flags = O_WRONLY });
419357e2f7fSGreg Kurz
420357e2f7fSGreg Kurz /* This will cause the 9p server to try to write data to the backend,
421357e2f7fSGreg Kurz * until the write request gets cancelled.
422357e2f7fSGreg Kurz */
423357e2f7fSGreg Kurz should_block = 1;
424ac9e4e61SChristian Schoenebeck req = twrite({
425ac9e4e61SChristian Schoenebeck .client = v9p, .fid = 1, .offset = 0,
426ac9e4e61SChristian Schoenebeck .count = sizeof(should_block), .data = &should_block,
427ac9e4e61SChristian Schoenebeck .requestOnly = true
428ac9e4e61SChristian Schoenebeck }).req;
429357e2f7fSGreg Kurz
430d89146fdSChristian Schoenebeck flush_req = tflush({
431d89146fdSChristian Schoenebeck .client = v9p, .oldtag = req->tag, .tag = 1, .requestOnly = true
432d89146fdSChristian Schoenebeck }).req;
433357e2f7fSGreg Kurz
434357e2f7fSGreg Kurz /* The write request is supposed to be flushed: the server should just
435357e2f7fSGreg Kurz * mark the write request as used and reply to the flush request.
436357e2f7fSGreg Kurz */
437357e2f7fSGreg Kurz v9fs_req_wait_for_reply(req, &reply_len);
438357e2f7fSGreg Kurz g_assert_cmpint(reply_len, ==, 0);
439357e2f7fSGreg Kurz v9fs_req_free(req);
440357e2f7fSGreg Kurz v9fs_rflush(flush_req);
441357e2f7fSGreg Kurz
442357e2f7fSGreg Kurz g_free(wnames[0]);
443357e2f7fSGreg Kurz }
444357e2f7fSGreg Kurz
fs_flush_ignored(void * obj,void * data,QGuestAllocator * t_alloc)445dfbe8b43SEmanuele Giuseppe Esposito static void fs_flush_ignored(void *obj, void *data, QGuestAllocator *t_alloc)
446357e2f7fSGreg Kurz {
447dfbe8b43SEmanuele Giuseppe Esposito QVirtio9P *v9p = obj;
448684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
449569f3b63SChristian Schoenebeck char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_FLUSH_FILE) };
450357e2f7fSGreg Kurz P9Req *req, *flush_req;
451357e2f7fSGreg Kurz uint32_t count;
452357e2f7fSGreg Kurz uint8_t should_block;
453357e2f7fSGreg Kurz
45474a160abSChristian Schoenebeck tattach({ .client = v9p });
4553f3e9232SChristian Schoenebeck twalk({
4563f3e9232SChristian Schoenebeck .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames
4573f3e9232SChristian Schoenebeck });
458357e2f7fSGreg Kurz
4590e4c4ff0SChristian Schoenebeck tlopen({ .client = v9p, .fid = 1, .flags = O_WRONLY });
460357e2f7fSGreg Kurz
461357e2f7fSGreg Kurz /* This will cause the write request to complete right away, before it
462357e2f7fSGreg Kurz * could be actually cancelled.
463357e2f7fSGreg Kurz */
464357e2f7fSGreg Kurz should_block = 0;
465ac9e4e61SChristian Schoenebeck req = twrite({
466ac9e4e61SChristian Schoenebeck .client = v9p, .fid = 1, .offset = 0,
467ac9e4e61SChristian Schoenebeck .count = sizeof(should_block), .data = &should_block,
468ac9e4e61SChristian Schoenebeck .requestOnly = true
469ac9e4e61SChristian Schoenebeck }).req;
470357e2f7fSGreg Kurz
471d89146fdSChristian Schoenebeck flush_req = tflush({
472d89146fdSChristian Schoenebeck .client = v9p, .oldtag = req->tag, .tag = 1, .requestOnly = true
473d89146fdSChristian Schoenebeck }).req;
474357e2f7fSGreg Kurz
475357e2f7fSGreg Kurz /* The write request is supposed to complete. The server should
476357e2f7fSGreg Kurz * reply to the write request and the flush request.
477357e2f7fSGreg Kurz */
478357e2f7fSGreg Kurz v9fs_req_wait_for_reply(req, NULL);
479357e2f7fSGreg Kurz v9fs_rwrite(req, &count);
480357e2f7fSGreg Kurz g_assert_cmpint(count, ==, sizeof(should_block));
481357e2f7fSGreg Kurz v9fs_rflush(flush_req);
482357e2f7fSGreg Kurz
483357e2f7fSGreg Kurz g_free(wnames[0]);
484357e2f7fSGreg Kurz }
485357e2f7fSGreg Kurz
fs_readdir_split_128(void * obj,void * data,QGuestAllocator * t_alloc)48646488b62SChristian Schoenebeck static void fs_readdir_split_128(void *obj, void *data,
48746488b62SChristian Schoenebeck QGuestAllocator *t_alloc)
48846488b62SChristian Schoenebeck {
489684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
4901d98613dSGreg Kurz do_readdir_split(obj, 128);
49146488b62SChristian Schoenebeck }
49246488b62SChristian Schoenebeck
fs_readdir_split_256(void * obj,void * data,QGuestAllocator * t_alloc)49346488b62SChristian Schoenebeck static void fs_readdir_split_256(void *obj, void *data,
49446488b62SChristian Schoenebeck QGuestAllocator *t_alloc)
49546488b62SChristian Schoenebeck {
496684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
4971d98613dSGreg Kurz do_readdir_split(obj, 256);
49846488b62SChristian Schoenebeck }
49946488b62SChristian Schoenebeck
fs_readdir_split_512(void * obj,void * data,QGuestAllocator * t_alloc)50046488b62SChristian Schoenebeck static void fs_readdir_split_512(void *obj, void *data,
50146488b62SChristian Schoenebeck QGuestAllocator *t_alloc)
50246488b62SChristian Schoenebeck {
503684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
5041d98613dSGreg Kurz do_readdir_split(obj, 512);
50546488b62SChristian Schoenebeck }
50646488b62SChristian Schoenebeck
507653daf38SChristian Schoenebeck
508653daf38SChristian Schoenebeck /* tests using the 9pfs 'local' fs driver */
509653daf38SChristian Schoenebeck
fs_create_dir(void * obj,void * data,QGuestAllocator * t_alloc)510653daf38SChristian Schoenebeck static void fs_create_dir(void *obj, void *data, QGuestAllocator *t_alloc)
511653daf38SChristian Schoenebeck {
512653daf38SChristian Schoenebeck QVirtio9P *v9p = obj;
513684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
514653daf38SChristian Schoenebeck struct stat st;
51565ceee0aSChristian Schoenebeck g_autofree char *root_path = virtio_9p_test_path("");
51665ceee0aSChristian Schoenebeck g_autofree char *new_dir = virtio_9p_test_path("01");
517653daf38SChristian Schoenebeck
518653daf38SChristian Schoenebeck g_assert(root_path != NULL);
519653daf38SChristian Schoenebeck
52074a160abSChristian Schoenebeck tattach({ .client = v9p });
521e1168010SChristian Schoenebeck tmkdir({ .client = v9p, .atPath = "/", .name = "01" });
522653daf38SChristian Schoenebeck
523653daf38SChristian Schoenebeck /* check if created directory really exists now ... */
524653daf38SChristian Schoenebeck g_assert(stat(new_dir, &st) == 0);
525653daf38SChristian Schoenebeck /* ... and is actually a directory */
526653daf38SChristian Schoenebeck g_assert((st.st_mode & S_IFMT) == S_IFDIR);
527653daf38SChristian Schoenebeck }
528653daf38SChristian Schoenebeck
fs_unlinkat_dir(void * obj,void * data,QGuestAllocator * t_alloc)529b37d62d6SChristian Schoenebeck static void fs_unlinkat_dir(void *obj, void *data, QGuestAllocator *t_alloc)
530b37d62d6SChristian Schoenebeck {
531b37d62d6SChristian Schoenebeck QVirtio9P *v9p = obj;
532684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
533b37d62d6SChristian Schoenebeck struct stat st;
53465ceee0aSChristian Schoenebeck g_autofree char *root_path = virtio_9p_test_path("");
53565ceee0aSChristian Schoenebeck g_autofree char *new_dir = virtio_9p_test_path("02");
536b37d62d6SChristian Schoenebeck
537b37d62d6SChristian Schoenebeck g_assert(root_path != NULL);
538b37d62d6SChristian Schoenebeck
53974a160abSChristian Schoenebeck tattach({ .client = v9p });
540e1168010SChristian Schoenebeck tmkdir({ .client = v9p, .atPath = "/", .name = "02" });
541b37d62d6SChristian Schoenebeck
542b37d62d6SChristian Schoenebeck /* check if created directory really exists now ... */
543b37d62d6SChristian Schoenebeck g_assert(stat(new_dir, &st) == 0);
544b37d62d6SChristian Schoenebeck /* ... and is actually a directory */
545b37d62d6SChristian Schoenebeck g_assert((st.st_mode & S_IFMT) == S_IFDIR);
546b37d62d6SChristian Schoenebeck
54743e0d9fbSChristian Schoenebeck tunlinkat({
54843e0d9fbSChristian Schoenebeck .client = v9p, .atPath = "/", .name = "02",
54943e0d9fbSChristian Schoenebeck .flags = P9_DOTL_AT_REMOVEDIR
55043e0d9fbSChristian Schoenebeck });
551b37d62d6SChristian Schoenebeck /* directory should be gone now */
552b37d62d6SChristian Schoenebeck g_assert(stat(new_dir, &st) != 0);
553b37d62d6SChristian Schoenebeck }
554b37d62d6SChristian Schoenebeck
fs_create_file(void * obj,void * data,QGuestAllocator * t_alloc)555b09dbfddSChristian Schoenebeck static void fs_create_file(void *obj, void *data, QGuestAllocator *t_alloc)
556b09dbfddSChristian Schoenebeck {
557b09dbfddSChristian Schoenebeck QVirtio9P *v9p = obj;
558684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
559b09dbfddSChristian Schoenebeck struct stat st;
56065ceee0aSChristian Schoenebeck g_autofree char *new_file = virtio_9p_test_path("03/1st_file");
561b09dbfddSChristian Schoenebeck
56274a160abSChristian Schoenebeck tattach({ .client = v9p });
563e1168010SChristian Schoenebeck tmkdir({ .client = v9p, .atPath = "/", .name = "03" });
564bd4660d4SChristian Schoenebeck tlcreate({ .client = v9p, .atPath = "03", .name = "1st_file" });
565b09dbfddSChristian Schoenebeck
566b09dbfddSChristian Schoenebeck /* check if created file exists now ... */
567b09dbfddSChristian Schoenebeck g_assert(stat(new_file, &st) == 0);
568b09dbfddSChristian Schoenebeck /* ... and is a regular file */
569b09dbfddSChristian Schoenebeck g_assert((st.st_mode & S_IFMT) == S_IFREG);
570b09dbfddSChristian Schoenebeck }
571b09dbfddSChristian Schoenebeck
fs_unlinkat_file(void * obj,void * data,QGuestAllocator * t_alloc)572472c18b8SChristian Schoenebeck static void fs_unlinkat_file(void *obj, void *data, QGuestAllocator *t_alloc)
573472c18b8SChristian Schoenebeck {
574472c18b8SChristian Schoenebeck QVirtio9P *v9p = obj;
575684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
576472c18b8SChristian Schoenebeck struct stat st;
57765ceee0aSChristian Schoenebeck g_autofree char *new_file = virtio_9p_test_path("04/doa_file");
578472c18b8SChristian Schoenebeck
57974a160abSChristian Schoenebeck tattach({ .client = v9p });
580e1168010SChristian Schoenebeck tmkdir({ .client = v9p, .atPath = "/", .name = "04" });
581bd4660d4SChristian Schoenebeck tlcreate({ .client = v9p, .atPath = "04", .name = "doa_file" });
582472c18b8SChristian Schoenebeck
583472c18b8SChristian Schoenebeck /* check if created file exists now ... */
584472c18b8SChristian Schoenebeck g_assert(stat(new_file, &st) == 0);
585472c18b8SChristian Schoenebeck /* ... and is a regular file */
586472c18b8SChristian Schoenebeck g_assert((st.st_mode & S_IFMT) == S_IFREG);
587472c18b8SChristian Schoenebeck
58843e0d9fbSChristian Schoenebeck tunlinkat({ .client = v9p, .atPath = "04", .name = "doa_file" });
589472c18b8SChristian Schoenebeck /* file should be gone now */
590472c18b8SChristian Schoenebeck g_assert(stat(new_file, &st) != 0);
591472c18b8SChristian Schoenebeck }
592472c18b8SChristian Schoenebeck
fs_symlink_file(void * obj,void * data,QGuestAllocator * t_alloc)59359ff563dSChristian Schoenebeck static void fs_symlink_file(void *obj, void *data, QGuestAllocator *t_alloc)
59459ff563dSChristian Schoenebeck {
59559ff563dSChristian Schoenebeck QVirtio9P *v9p = obj;
596684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
59759ff563dSChristian Schoenebeck struct stat st;
59865ceee0aSChristian Schoenebeck g_autofree char *real_file = virtio_9p_test_path("05/real_file");
59965ceee0aSChristian Schoenebeck g_autofree char *symlink_file = virtio_9p_test_path("05/symlink_file");
60059ff563dSChristian Schoenebeck
60174a160abSChristian Schoenebeck tattach({ .client = v9p });
602e1168010SChristian Schoenebeck tmkdir({ .client = v9p, .atPath = "/", .name = "05" });
603bd4660d4SChristian Schoenebeck tlcreate({ .client = v9p, .atPath = "05", .name = "real_file" });
60459ff563dSChristian Schoenebeck g_assert(stat(real_file, &st) == 0);
60559ff563dSChristian Schoenebeck g_assert((st.st_mode & S_IFMT) == S_IFREG);
60659ff563dSChristian Schoenebeck
6079beabfa5SChristian Schoenebeck tsymlink({
6089beabfa5SChristian Schoenebeck .client = v9p, .atPath = "05", .name = "symlink_file",
6099beabfa5SChristian Schoenebeck .symtgt = "real_file"
6109beabfa5SChristian Schoenebeck });
61159ff563dSChristian Schoenebeck
61259ff563dSChristian Schoenebeck /* check if created link exists now */
61359ff563dSChristian Schoenebeck g_assert(stat(symlink_file, &st) == 0);
61459ff563dSChristian Schoenebeck }
61559ff563dSChristian Schoenebeck
fs_unlinkat_symlink(void * obj,void * data,QGuestAllocator * t_alloc)6165b28ab8bSChristian Schoenebeck static void fs_unlinkat_symlink(void *obj, void *data,
6175b28ab8bSChristian Schoenebeck QGuestAllocator *t_alloc)
6185b28ab8bSChristian Schoenebeck {
6195b28ab8bSChristian Schoenebeck QVirtio9P *v9p = obj;
620684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
6215b28ab8bSChristian Schoenebeck struct stat st;
62265ceee0aSChristian Schoenebeck g_autofree char *real_file = virtio_9p_test_path("06/real_file");
62365ceee0aSChristian Schoenebeck g_autofree char *symlink_file = virtio_9p_test_path("06/symlink_file");
6245b28ab8bSChristian Schoenebeck
62574a160abSChristian Schoenebeck tattach({ .client = v9p });
626e1168010SChristian Schoenebeck tmkdir({ .client = v9p, .atPath = "/", .name = "06" });
627bd4660d4SChristian Schoenebeck tlcreate({ .client = v9p, .atPath = "06", .name = "real_file" });
6285b28ab8bSChristian Schoenebeck g_assert(stat(real_file, &st) == 0);
6295b28ab8bSChristian Schoenebeck g_assert((st.st_mode & S_IFMT) == S_IFREG);
6305b28ab8bSChristian Schoenebeck
6319beabfa5SChristian Schoenebeck tsymlink({
6329beabfa5SChristian Schoenebeck .client = v9p, .atPath = "06", .name = "symlink_file",
6339beabfa5SChristian Schoenebeck .symtgt = "real_file"
6349beabfa5SChristian Schoenebeck });
6355b28ab8bSChristian Schoenebeck g_assert(stat(symlink_file, &st) == 0);
6365b28ab8bSChristian Schoenebeck
63743e0d9fbSChristian Schoenebeck tunlinkat({ .client = v9p, .atPath = "06", .name = "symlink_file" });
6385b28ab8bSChristian Schoenebeck /* symlink should be gone now */
6395b28ab8bSChristian Schoenebeck g_assert(stat(symlink_file, &st) != 0);
6405b28ab8bSChristian Schoenebeck }
6415b28ab8bSChristian Schoenebeck
fs_hardlink_file(void * obj,void * data,QGuestAllocator * t_alloc)64264e3d403SChristian Schoenebeck static void fs_hardlink_file(void *obj, void *data, QGuestAllocator *t_alloc)
64364e3d403SChristian Schoenebeck {
64464e3d403SChristian Schoenebeck QVirtio9P *v9p = obj;
645684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
64664e3d403SChristian Schoenebeck struct stat st_real, st_link;
64765ceee0aSChristian Schoenebeck g_autofree char *real_file = virtio_9p_test_path("07/real_file");
64865ceee0aSChristian Schoenebeck g_autofree char *hardlink_file = virtio_9p_test_path("07/hardlink_file");
64964e3d403SChristian Schoenebeck
65074a160abSChristian Schoenebeck tattach({ .client = v9p });
651e1168010SChristian Schoenebeck tmkdir({ .client = v9p, .atPath = "/", .name = "07" });
652bd4660d4SChristian Schoenebeck tlcreate({ .client = v9p, .atPath = "07", .name = "real_file" });
65364e3d403SChristian Schoenebeck g_assert(stat(real_file, &st_real) == 0);
65464e3d403SChristian Schoenebeck g_assert((st_real.st_mode & S_IFMT) == S_IFREG);
65564e3d403SChristian Schoenebeck
656d41a9462SChristian Schoenebeck tlink({
657d41a9462SChristian Schoenebeck .client = v9p, .atPath = "07", .name = "hardlink_file",
658d41a9462SChristian Schoenebeck .toPath = "07/real_file"
659d41a9462SChristian Schoenebeck });
66064e3d403SChristian Schoenebeck
66164e3d403SChristian Schoenebeck /* check if link exists now ... */
66264e3d403SChristian Schoenebeck g_assert(stat(hardlink_file, &st_link) == 0);
66364e3d403SChristian Schoenebeck /* ... and it's a hard link, right? */
66464e3d403SChristian Schoenebeck g_assert((st_link.st_mode & S_IFMT) == S_IFREG);
66564e3d403SChristian Schoenebeck g_assert(st_link.st_dev == st_real.st_dev);
66664e3d403SChristian Schoenebeck g_assert(st_link.st_ino == st_real.st_ino);
66764e3d403SChristian Schoenebeck }
66864e3d403SChristian Schoenebeck
fs_unlinkat_hardlink(void * obj,void * data,QGuestAllocator * t_alloc)6694d0746e2SChristian Schoenebeck static void fs_unlinkat_hardlink(void *obj, void *data,
6704d0746e2SChristian Schoenebeck QGuestAllocator *t_alloc)
6714d0746e2SChristian Schoenebeck {
6724d0746e2SChristian Schoenebeck QVirtio9P *v9p = obj;
673684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc);
6744d0746e2SChristian Schoenebeck struct stat st_real, st_link;
67565ceee0aSChristian Schoenebeck g_autofree char *real_file = virtio_9p_test_path("08/real_file");
67665ceee0aSChristian Schoenebeck g_autofree char *hardlink_file = virtio_9p_test_path("08/hardlink_file");
6774d0746e2SChristian Schoenebeck
67874a160abSChristian Schoenebeck tattach({ .client = v9p });
679e1168010SChristian Schoenebeck tmkdir({ .client = v9p, .atPath = "/", .name = "08" });
680bd4660d4SChristian Schoenebeck tlcreate({ .client = v9p, .atPath = "08", .name = "real_file" });
6814d0746e2SChristian Schoenebeck g_assert(stat(real_file, &st_real) == 0);
6824d0746e2SChristian Schoenebeck g_assert((st_real.st_mode & S_IFMT) == S_IFREG);
6834d0746e2SChristian Schoenebeck
684d41a9462SChristian Schoenebeck tlink({
685d41a9462SChristian Schoenebeck .client = v9p, .atPath = "08", .name = "hardlink_file",
686d41a9462SChristian Schoenebeck .toPath = "08/real_file"
687d41a9462SChristian Schoenebeck });
6884d0746e2SChristian Schoenebeck g_assert(stat(hardlink_file, &st_link) == 0);
6894d0746e2SChristian Schoenebeck
69043e0d9fbSChristian Schoenebeck tunlinkat({ .client = v9p, .atPath = "08", .name = "hardlink_file" });
6914d0746e2SChristian Schoenebeck /* symlink should be gone now */
6924d0746e2SChristian Schoenebeck g_assert(stat(hardlink_file, &st_link) != 0);
6934d0746e2SChristian Schoenebeck /* and old file should still exist */
6944d0746e2SChristian Schoenebeck g_assert(stat(real_file, &st_real) == 0);
6954d0746e2SChristian Schoenebeck }
6964d0746e2SChristian Schoenebeck
fs_use_after_unlink(void * obj,void * data,QGuestAllocator * t_alloc)697462db8fbSChristian Schoenebeck static void fs_use_after_unlink(void *obj, void *data,
698462db8fbSChristian Schoenebeck QGuestAllocator *t_alloc)
699462db8fbSChristian Schoenebeck {
700462db8fbSChristian Schoenebeck QVirtio9P *v9p = obj;
701462db8fbSChristian Schoenebeck v9fs_set_allocator(t_alloc);
702462db8fbSChristian Schoenebeck static const uint32_t write_count = P9_MAX_SIZE / 2;
703462db8fbSChristian Schoenebeck g_autofree char *real_file = virtio_9p_test_path("09/doa_file");
704462db8fbSChristian Schoenebeck g_autofree char *buf = g_malloc0(write_count);
705462db8fbSChristian Schoenebeck struct stat st_file;
706eaab44ccSChristian Schoenebeck struct v9fs_attr attr;
707462db8fbSChristian Schoenebeck uint32_t fid_file;
708462db8fbSChristian Schoenebeck uint32_t count;
709462db8fbSChristian Schoenebeck
710462db8fbSChristian Schoenebeck tattach({ .client = v9p });
711462db8fbSChristian Schoenebeck
712462db8fbSChristian Schoenebeck /* create a file "09/doa_file" and make sure it exists and is regular */
713462db8fbSChristian Schoenebeck tmkdir({ .client = v9p, .atPath = "/", .name = "09" });
714462db8fbSChristian Schoenebeck tlcreate({ .client = v9p, .atPath = "09", .name = "doa_file" });
715462db8fbSChristian Schoenebeck g_assert(stat(real_file, &st_file) == 0);
716462db8fbSChristian Schoenebeck g_assert((st_file.st_mode & S_IFMT) == S_IFREG);
717462db8fbSChristian Schoenebeck
718462db8fbSChristian Schoenebeck /* request a FID for that regular file that we can work with next */
719462db8fbSChristian Schoenebeck fid_file = twalk({
720462db8fbSChristian Schoenebeck .client = v9p, .fid = 0, .path = "09/doa_file"
721462db8fbSChristian Schoenebeck }).newfid;
722462db8fbSChristian Schoenebeck g_assert(fid_file != 0);
723462db8fbSChristian Schoenebeck
724462db8fbSChristian Schoenebeck /* now first open the file in write mode before ... */
725462db8fbSChristian Schoenebeck tlopen({ .client = v9p, .fid = fid_file, .flags = O_WRONLY });
726462db8fbSChristian Schoenebeck /* ... removing the file from file system */
727462db8fbSChristian Schoenebeck tunlinkat({ .client = v9p, .atPath = "09", .name = "doa_file" });
728462db8fbSChristian Schoenebeck
729462db8fbSChristian Schoenebeck /* file is removed, but we still have it open, so this should succeed */
730eaab44ccSChristian Schoenebeck tgetattr({
731eaab44ccSChristian Schoenebeck .client = v9p, .fid = fid_file, .request_mask = P9_GETATTR_BASIC,
732eaab44ccSChristian Schoenebeck .rgetattr.attr = &attr
733eaab44ccSChristian Schoenebeck });
734462db8fbSChristian Schoenebeck count = twrite({
735462db8fbSChristian Schoenebeck .client = v9p, .fid = fid_file, .offset = 0, .count = write_count,
736462db8fbSChristian Schoenebeck .data = buf
737462db8fbSChristian Schoenebeck }).count;
738462db8fbSChristian Schoenebeck g_assert_cmpint(count, ==, write_count);
739*610dc187SGreg Kurz
740*610dc187SGreg Kurz /* truncate file to (arbitrarily chosen) size 2001 */
741*610dc187SGreg Kurz tsetattr({
742*610dc187SGreg Kurz .client = v9p, .fid = fid_file, .attr = (v9fs_attr) {
743*610dc187SGreg Kurz .valid = P9_SETATTR_SIZE,
744*610dc187SGreg Kurz .size = 2001
745*610dc187SGreg Kurz }
746*610dc187SGreg Kurz });
747*610dc187SGreg Kurz /* truncate apparently succeeded, let's double-check the size */
748*610dc187SGreg Kurz tgetattr({
749*610dc187SGreg Kurz .client = v9p, .fid = fid_file, .request_mask = P9_GETATTR_BASIC,
750*610dc187SGreg Kurz .rgetattr.attr = &attr
751*610dc187SGreg Kurz });
752*610dc187SGreg Kurz g_assert_cmpint(attr.size, ==, 2001);
753462db8fbSChristian Schoenebeck }
754462db8fbSChristian Schoenebeck
cleanup_9p_local_driver(void * data)755981eb106SDaniel Henrique Barboza static void cleanup_9p_local_driver(void *data)
756981eb106SDaniel Henrique Barboza {
757981eb106SDaniel Henrique Barboza /* remove previously created test dir when test is completed */
758981eb106SDaniel Henrique Barboza virtio_9p_remove_local_test_dir();
759981eb106SDaniel Henrique Barboza }
760981eb106SDaniel Henrique Barboza
assign_9p_local_driver(GString * cmd_line,void * arg)7613a565c64SChristian Schoenebeck static void *assign_9p_local_driver(GString *cmd_line, void *arg)
7623a565c64SChristian Schoenebeck {
763981eb106SDaniel Henrique Barboza /* make sure test dir for the 'local' tests exists */
764981eb106SDaniel Henrique Barboza virtio_9p_create_local_test_dir();
765981eb106SDaniel Henrique Barboza
7663a565c64SChristian Schoenebeck virtio_9p_assign_local_driver(cmd_line, "security_model=mapped-xattr");
767981eb106SDaniel Henrique Barboza
768981eb106SDaniel Henrique Barboza g_test_queue_destroy(cleanup_9p_local_driver, NULL);
7693a565c64SChristian Schoenebeck return arg;
7703a565c64SChristian Schoenebeck }
7713a565c64SChristian Schoenebeck
register_virtio_9p_test(void)772dfbe8b43SEmanuele Giuseppe Esposito static void register_virtio_9p_test(void)
7731211d81bSGreg Kurz {
7743a565c64SChristian Schoenebeck
7753a565c64SChristian Schoenebeck QOSGraphTestOptions opts = {
7763a565c64SChristian Schoenebeck };
7773a565c64SChristian Schoenebeck
7783a565c64SChristian Schoenebeck /* 9pfs test cases using the 'synth' filesystem driver */
7793a565c64SChristian Schoenebeck qos_add_test("synth/config", "virtio-9p", pci_config, &opts);
7803a565c64SChristian Schoenebeck qos_add_test("synth/version/basic", "virtio-9p", fs_version, &opts);
7813a565c64SChristian Schoenebeck qos_add_test("synth/attach/basic", "virtio-9p", fs_attach, &opts);
7823a565c64SChristian Schoenebeck qos_add_test("synth/walk/basic", "virtio-9p", fs_walk, &opts);
783eefd2394SChristian Schoenebeck qos_add_test("synth/walk/no_slash", "virtio-9p", fs_walk_no_slash,
7843a565c64SChristian Schoenebeck &opts);
785c1668948SChristian Schoenebeck qos_add_test("synth/walk/none", "virtio-9p", fs_walk_none, &opts);
786eefd2394SChristian Schoenebeck qos_add_test("synth/walk/dotdot_from_root", "virtio-9p",
7873a565c64SChristian Schoenebeck fs_walk_dotdot, &opts);
7889472a689SChristian Schoenebeck qos_add_test("synth/walk/non_existent", "virtio-9p", fs_walk_nonexistent,
7899472a689SChristian Schoenebeck &opts);
79015fbff48SChristian Schoenebeck qos_add_test("synth/walk/2nd_non_existent", "virtio-9p",
79115fbff48SChristian Schoenebeck fs_walk_2nd_nonexistent, &opts);
7923a565c64SChristian Schoenebeck qos_add_test("synth/lopen/basic", "virtio-9p", fs_lopen, &opts);
7933a565c64SChristian Schoenebeck qos_add_test("synth/write/basic", "virtio-9p", fs_write, &opts);
794eefd2394SChristian Schoenebeck qos_add_test("synth/flush/success", "virtio-9p", fs_flush_success,
7953a565c64SChristian Schoenebeck &opts);
796eefd2394SChristian Schoenebeck qos_add_test("synth/flush/ignored", "virtio-9p", fs_flush_ignored,
7973a565c64SChristian Schoenebeck &opts);
7983a565c64SChristian Schoenebeck qos_add_test("synth/readdir/basic", "virtio-9p", fs_readdir, &opts);
799eefd2394SChristian Schoenebeck qos_add_test("synth/readdir/split_512", "virtio-9p",
8003a565c64SChristian Schoenebeck fs_readdir_split_512, &opts);
801eefd2394SChristian Schoenebeck qos_add_test("synth/readdir/split_256", "virtio-9p",
8023a565c64SChristian Schoenebeck fs_readdir_split_256, &opts);
803eefd2394SChristian Schoenebeck qos_add_test("synth/readdir/split_128", "virtio-9p",
8043a565c64SChristian Schoenebeck fs_readdir_split_128, &opts);
8053a565c64SChristian Schoenebeck
8063a565c64SChristian Schoenebeck
8073a565c64SChristian Schoenebeck /* 9pfs test cases using the 'local' filesystem driver */
8083a565c64SChristian Schoenebeck opts.before = assign_9p_local_driver;
8093a565c64SChristian Schoenebeck qos_add_test("local/config", "virtio-9p", pci_config, &opts);
810653daf38SChristian Schoenebeck qos_add_test("local/create_dir", "virtio-9p", fs_create_dir, &opts);
811b37d62d6SChristian Schoenebeck qos_add_test("local/unlinkat_dir", "virtio-9p", fs_unlinkat_dir, &opts);
812b09dbfddSChristian Schoenebeck qos_add_test("local/create_file", "virtio-9p", fs_create_file, &opts);
813472c18b8SChristian Schoenebeck qos_add_test("local/unlinkat_file", "virtio-9p", fs_unlinkat_file, &opts);
81459ff563dSChristian Schoenebeck qos_add_test("local/symlink_file", "virtio-9p", fs_symlink_file, &opts);
8155b28ab8bSChristian Schoenebeck qos_add_test("local/unlinkat_symlink", "virtio-9p", fs_unlinkat_symlink,
8165b28ab8bSChristian Schoenebeck &opts);
81764e3d403SChristian Schoenebeck qos_add_test("local/hardlink_file", "virtio-9p", fs_hardlink_file, &opts);
8184d0746e2SChristian Schoenebeck qos_add_test("local/unlinkat_hardlink", "virtio-9p", fs_unlinkat_hardlink,
8194d0746e2SChristian Schoenebeck &opts);
820462db8fbSChristian Schoenebeck qos_add_test("local/use_after_unlink", "virtio-9p", fs_use_after_unlink,
821462db8fbSChristian Schoenebeck &opts);
8221211d81bSGreg Kurz }
8231211d81bSGreg Kurz
824dfbe8b43SEmanuele Giuseppe Esposito libqos_init(register_virtio_9p_test);
825