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