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__) 23*4719a2d5SChristian 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 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 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 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 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 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 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 */ 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 */ 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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); 739462db8fbSChristian Schoenebeck } 740462db8fbSChristian Schoenebeck 741981eb106SDaniel Henrique Barboza static void cleanup_9p_local_driver(void *data) 742981eb106SDaniel Henrique Barboza { 743981eb106SDaniel Henrique Barboza /* remove previously created test dir when test is completed */ 744981eb106SDaniel Henrique Barboza virtio_9p_remove_local_test_dir(); 745981eb106SDaniel Henrique Barboza } 746981eb106SDaniel Henrique Barboza 7473a565c64SChristian Schoenebeck static void *assign_9p_local_driver(GString *cmd_line, void *arg) 7483a565c64SChristian Schoenebeck { 749981eb106SDaniel Henrique Barboza /* make sure test dir for the 'local' tests exists */ 750981eb106SDaniel Henrique Barboza virtio_9p_create_local_test_dir(); 751981eb106SDaniel Henrique Barboza 7523a565c64SChristian Schoenebeck virtio_9p_assign_local_driver(cmd_line, "security_model=mapped-xattr"); 753981eb106SDaniel Henrique Barboza 754981eb106SDaniel Henrique Barboza g_test_queue_destroy(cleanup_9p_local_driver, NULL); 7553a565c64SChristian Schoenebeck return arg; 7563a565c64SChristian Schoenebeck } 7573a565c64SChristian Schoenebeck 758dfbe8b43SEmanuele Giuseppe Esposito static void register_virtio_9p_test(void) 7591211d81bSGreg Kurz { 7603a565c64SChristian Schoenebeck 7613a565c64SChristian Schoenebeck QOSGraphTestOptions opts = { 7623a565c64SChristian Schoenebeck }; 7633a565c64SChristian Schoenebeck 7643a565c64SChristian Schoenebeck /* 9pfs test cases using the 'synth' filesystem driver */ 7653a565c64SChristian Schoenebeck qos_add_test("synth/config", "virtio-9p", pci_config, &opts); 7663a565c64SChristian Schoenebeck qos_add_test("synth/version/basic", "virtio-9p", fs_version, &opts); 7673a565c64SChristian Schoenebeck qos_add_test("synth/attach/basic", "virtio-9p", fs_attach, &opts); 7683a565c64SChristian Schoenebeck qos_add_test("synth/walk/basic", "virtio-9p", fs_walk, &opts); 769eefd2394SChristian Schoenebeck qos_add_test("synth/walk/no_slash", "virtio-9p", fs_walk_no_slash, 7703a565c64SChristian Schoenebeck &opts); 771c1668948SChristian Schoenebeck qos_add_test("synth/walk/none", "virtio-9p", fs_walk_none, &opts); 772eefd2394SChristian Schoenebeck qos_add_test("synth/walk/dotdot_from_root", "virtio-9p", 7733a565c64SChristian Schoenebeck fs_walk_dotdot, &opts); 7749472a689SChristian Schoenebeck qos_add_test("synth/walk/non_existent", "virtio-9p", fs_walk_nonexistent, 7759472a689SChristian Schoenebeck &opts); 77615fbff48SChristian Schoenebeck qos_add_test("synth/walk/2nd_non_existent", "virtio-9p", 77715fbff48SChristian Schoenebeck fs_walk_2nd_nonexistent, &opts); 7783a565c64SChristian Schoenebeck qos_add_test("synth/lopen/basic", "virtio-9p", fs_lopen, &opts); 7793a565c64SChristian Schoenebeck qos_add_test("synth/write/basic", "virtio-9p", fs_write, &opts); 780eefd2394SChristian Schoenebeck qos_add_test("synth/flush/success", "virtio-9p", fs_flush_success, 7813a565c64SChristian Schoenebeck &opts); 782eefd2394SChristian Schoenebeck qos_add_test("synth/flush/ignored", "virtio-9p", fs_flush_ignored, 7833a565c64SChristian Schoenebeck &opts); 7843a565c64SChristian Schoenebeck qos_add_test("synth/readdir/basic", "virtio-9p", fs_readdir, &opts); 785eefd2394SChristian Schoenebeck qos_add_test("synth/readdir/split_512", "virtio-9p", 7863a565c64SChristian Schoenebeck fs_readdir_split_512, &opts); 787eefd2394SChristian Schoenebeck qos_add_test("synth/readdir/split_256", "virtio-9p", 7883a565c64SChristian Schoenebeck fs_readdir_split_256, &opts); 789eefd2394SChristian Schoenebeck qos_add_test("synth/readdir/split_128", "virtio-9p", 7903a565c64SChristian Schoenebeck fs_readdir_split_128, &opts); 7913a565c64SChristian Schoenebeck 7923a565c64SChristian Schoenebeck 7933a565c64SChristian Schoenebeck /* 9pfs test cases using the 'local' filesystem driver */ 7943a565c64SChristian Schoenebeck opts.before = assign_9p_local_driver; 7953a565c64SChristian Schoenebeck qos_add_test("local/config", "virtio-9p", pci_config, &opts); 796653daf38SChristian Schoenebeck qos_add_test("local/create_dir", "virtio-9p", fs_create_dir, &opts); 797b37d62d6SChristian Schoenebeck qos_add_test("local/unlinkat_dir", "virtio-9p", fs_unlinkat_dir, &opts); 798b09dbfddSChristian Schoenebeck qos_add_test("local/create_file", "virtio-9p", fs_create_file, &opts); 799472c18b8SChristian Schoenebeck qos_add_test("local/unlinkat_file", "virtio-9p", fs_unlinkat_file, &opts); 80059ff563dSChristian Schoenebeck qos_add_test("local/symlink_file", "virtio-9p", fs_symlink_file, &opts); 8015b28ab8bSChristian Schoenebeck qos_add_test("local/unlinkat_symlink", "virtio-9p", fs_unlinkat_symlink, 8025b28ab8bSChristian Schoenebeck &opts); 80364e3d403SChristian Schoenebeck qos_add_test("local/hardlink_file", "virtio-9p", fs_hardlink_file, &opts); 8044d0746e2SChristian Schoenebeck qos_add_test("local/unlinkat_hardlink", "virtio-9p", fs_unlinkat_hardlink, 8054d0746e2SChristian Schoenebeck &opts); 806462db8fbSChristian Schoenebeck qos_add_test("local/use_after_unlink", "virtio-9p", fs_use_after_unlink, 807462db8fbSChristian Schoenebeck &opts); 8081211d81bSGreg Kurz } 8091211d81bSGreg Kurz 810dfbe8b43SEmanuele Giuseppe Esposito libqos_init(register_virtio_9p_test); 811