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