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__) 20*bee8fda2SChristian Schoenebeck #define tversion(...) v9fs_tversion((TVersionOpt) __VA_ARGS__) 21653daf38SChristian Schoenebeck 22dfbe8b43SEmanuele Giuseppe Esposito static void pci_config(void *obj, void *data, QGuestAllocator *t_alloc) 23557a4cc0SGreg Kurz { 24dfbe8b43SEmanuele Giuseppe Esposito QVirtio9P *v9p = obj; 25684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 26dfbe8b43SEmanuele Giuseppe Esposito size_t tag_len = qvirtio_config_readw(v9p->vdev, 0); 2765ceee0aSChristian Schoenebeck g_autofree char *tag = NULL; 28557a4cc0SGreg Kurz int i; 29557a4cc0SGreg Kurz 30dfbe8b43SEmanuele Giuseppe Esposito g_assert_cmpint(tag_len, ==, strlen(MOUNT_TAG)); 31557a4cc0SGreg Kurz 32557a4cc0SGreg Kurz tag = g_malloc(tag_len); 33557a4cc0SGreg Kurz for (i = 0; i < tag_len; i++) { 34dfbe8b43SEmanuele Giuseppe Esposito tag[i] = qvirtio_config_readb(v9p->vdev, i + 2); 35557a4cc0SGreg Kurz } 36dfbe8b43SEmanuele Giuseppe Esposito g_assert_cmpmem(tag, tag_len, MOUNT_TAG, tag_len); 371211d81bSGreg Kurz } 38557a4cc0SGreg Kurz 39a6821b82SChristian Schoenebeck static inline bool is_same_qid(v9fs_qid a, v9fs_qid b) 40a6821b82SChristian Schoenebeck { 41a6821b82SChristian Schoenebeck /* don't compare QID version for checking for file ID equalness */ 42a6821b82SChristian Schoenebeck return a[0] == b[0] && memcmp(&a[5], &b[5], 8) == 0; 43a6821b82SChristian Schoenebeck } 44a6821b82SChristian Schoenebeck 451c450e6eSGreg Kurz static void fs_version(void *obj, void *data, QGuestAllocator *t_alloc) 461c450e6eSGreg Kurz { 47684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 48*bee8fda2SChristian Schoenebeck tversion({ .client = obj }); 491c450e6eSGreg Kurz } 501c450e6eSGreg Kurz 510e43495dSChristian Schoenebeck static void do_attach_rqid(QVirtio9P *v9p, v9fs_qid *qid) 525c3df1f0SGreg Kurz { 535c3df1f0SGreg Kurz P9Req *req; 545c3df1f0SGreg Kurz 55*bee8fda2SChristian Schoenebeck tversion({ .client = v9p }); 56693b21d2SGreg Kurz req = v9fs_tattach(v9p, 0, getuid(), 0); 57357e2f7fSGreg Kurz v9fs_req_wait_for_reply(req, NULL); 580e43495dSChristian Schoenebeck v9fs_rattach(req, qid); 590e43495dSChristian Schoenebeck } 600e43495dSChristian Schoenebeck 610e43495dSChristian Schoenebeck static void do_attach(QVirtio9P *v9p) 620e43495dSChristian Schoenebeck { 630e43495dSChristian Schoenebeck do_attach_rqid(v9p, NULL); 645c3df1f0SGreg Kurz } 655c3df1f0SGreg Kurz 663fe4baf4SGreg Kurz static void fs_attach(void *obj, void *data, QGuestAllocator *t_alloc) 673fe4baf4SGreg Kurz { 68684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 693fe4baf4SGreg Kurz do_attach(obj); 703fe4baf4SGreg Kurz } 713fe4baf4SGreg Kurz 72dfbe8b43SEmanuele Giuseppe Esposito static void fs_walk(void *obj, void *data, QGuestAllocator *t_alloc) 7304b88c84SGreg Kurz { 74dfbe8b43SEmanuele Giuseppe Esposito QVirtio9P *v9p = obj; 75684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 762893ddd5SGreg Kurz char *wnames[P9_MAXWELEM]; 7704b88c84SGreg Kurz uint16_t nwqid; 7865ceee0aSChristian Schoenebeck g_autofree v9fs_qid *wqid = NULL; 7904b88c84SGreg Kurz int i; 8004b88c84SGreg Kurz 8104b88c84SGreg Kurz for (i = 0; i < P9_MAXWELEM; i++) { 822893ddd5SGreg Kurz wnames[i] = g_strdup_printf(QTEST_V9FS_SYNTH_WALK_FILE, i); 8304b88c84SGreg Kurz } 8404b88c84SGreg Kurz 853fe4baf4SGreg Kurz do_attach(v9p); 863f3e9232SChristian Schoenebeck twalk({ 87569f3b63SChristian Schoenebeck .client = v9p, .fid = 0, .newfid = 1, 883f3e9232SChristian Schoenebeck .nwname = P9_MAXWELEM, .wnames = wnames, 893f3e9232SChristian Schoenebeck .rwalk = { .nwqid = &nwqid, .wqid = &wqid } 903f3e9232SChristian Schoenebeck }); 9104b88c84SGreg Kurz 9204b88c84SGreg Kurz g_assert_cmpint(nwqid, ==, P9_MAXWELEM); 9304b88c84SGreg Kurz 9404b88c84SGreg Kurz for (i = 0; i < P9_MAXWELEM; i++) { 9504b88c84SGreg Kurz g_free(wnames[i]); 9604b88c84SGreg Kurz } 9704b88c84SGreg Kurz } 9804b88c84SGreg Kurz 994829469fSChristian Schoenebeck static bool fs_dirents_contain_name(struct V9fsDirent *e, const char* name) 1004829469fSChristian Schoenebeck { 1014829469fSChristian Schoenebeck for (; e; e = e->next) { 1024829469fSChristian Schoenebeck if (!strcmp(e->name, name)) { 1034829469fSChristian Schoenebeck return true; 1044829469fSChristian Schoenebeck } 1054829469fSChristian Schoenebeck } 1064829469fSChristian Schoenebeck return false; 1074829469fSChristian Schoenebeck } 1084829469fSChristian Schoenebeck 10946488b62SChristian Schoenebeck /* basic readdir test where reply fits into a single response message */ 1104829469fSChristian Schoenebeck static void fs_readdir(void *obj, void *data, QGuestAllocator *t_alloc) 1114829469fSChristian Schoenebeck { 1124829469fSChristian Schoenebeck QVirtio9P *v9p = obj; 113684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 114569f3b63SChristian Schoenebeck char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_READDIR_DIR) }; 1154829469fSChristian Schoenebeck uint16_t nqid; 1164829469fSChristian Schoenebeck v9fs_qid qid; 1174829469fSChristian Schoenebeck uint32_t count, nentries; 1184829469fSChristian Schoenebeck struct V9fsDirent *entries = NULL; 1194829469fSChristian Schoenebeck P9Req *req; 1204829469fSChristian Schoenebeck 1213fe4baf4SGreg Kurz do_attach(v9p); 1223f3e9232SChristian Schoenebeck twalk({ 123569f3b63SChristian Schoenebeck .client = v9p, .fid = 0, .newfid = 1, 1243f3e9232SChristian Schoenebeck .nwname = 1, .wnames = wnames, .rwalk.nwqid = &nqid 1253f3e9232SChristian Schoenebeck }); 1264829469fSChristian Schoenebeck g_assert_cmpint(nqid, ==, 1); 1274829469fSChristian Schoenebeck 1284829469fSChristian Schoenebeck req = v9fs_tlopen(v9p, 1, O_DIRECTORY, 0); 1294829469fSChristian Schoenebeck v9fs_req_wait_for_reply(req, NULL); 1304829469fSChristian Schoenebeck v9fs_rlopen(req, &qid, NULL); 1314829469fSChristian Schoenebeck 1324829469fSChristian Schoenebeck /* 1334829469fSChristian Schoenebeck * submit count = msize - 11, because 11 is the header size of Rreaddir 1344829469fSChristian Schoenebeck */ 1354829469fSChristian Schoenebeck req = v9fs_treaddir(v9p, 1, 0, P9_MAX_SIZE - 11, 0); 1364829469fSChristian Schoenebeck v9fs_req_wait_for_reply(req, NULL); 1374829469fSChristian Schoenebeck v9fs_rreaddir(req, &count, &nentries, &entries); 1384829469fSChristian Schoenebeck 1394829469fSChristian Schoenebeck /* 1404829469fSChristian Schoenebeck * Assuming msize (P9_MAX_SIZE) is large enough so we can retrieve all 1414829469fSChristian Schoenebeck * dir entries with only one readdir request. 1424829469fSChristian Schoenebeck */ 1434829469fSChristian Schoenebeck g_assert_cmpint( 1444829469fSChristian Schoenebeck nentries, ==, 1454829469fSChristian Schoenebeck QTEST_V9FS_SYNTH_READDIR_NFILES + 2 /* "." and ".." */ 1464829469fSChristian Schoenebeck ); 1474829469fSChristian Schoenebeck 1484829469fSChristian Schoenebeck /* 1494829469fSChristian Schoenebeck * Check all file names exist in returned entries, ignore their order 1504829469fSChristian Schoenebeck * though. 1514829469fSChristian Schoenebeck */ 1524829469fSChristian Schoenebeck g_assert_cmpint(fs_dirents_contain_name(entries, "."), ==, true); 1534829469fSChristian Schoenebeck g_assert_cmpint(fs_dirents_contain_name(entries, ".."), ==, true); 1544829469fSChristian Schoenebeck for (int i = 0; i < QTEST_V9FS_SYNTH_READDIR_NFILES; ++i) { 15565ceee0aSChristian Schoenebeck g_autofree char *name = 15665ceee0aSChristian Schoenebeck g_strdup_printf(QTEST_V9FS_SYNTH_READDIR_FILE, i); 1574829469fSChristian Schoenebeck g_assert_cmpint(fs_dirents_contain_name(entries, name), ==, true); 1584829469fSChristian Schoenebeck } 1594829469fSChristian Schoenebeck 1604829469fSChristian Schoenebeck v9fs_free_dirents(entries); 1614829469fSChristian Schoenebeck g_free(wnames[0]); 1624829469fSChristian Schoenebeck } 1634829469fSChristian Schoenebeck 16446488b62SChristian Schoenebeck /* readdir test where overall request is split over several messages */ 1651d98613dSGreg Kurz static void do_readdir_split(QVirtio9P *v9p, uint32_t count) 16646488b62SChristian Schoenebeck { 167569f3b63SChristian Schoenebeck char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_READDIR_DIR) }; 16846488b62SChristian Schoenebeck uint16_t nqid; 16946488b62SChristian Schoenebeck v9fs_qid qid; 17046488b62SChristian Schoenebeck uint32_t nentries, npartialentries; 17146488b62SChristian Schoenebeck struct V9fsDirent *entries, *tail, *partialentries; 17246488b62SChristian Schoenebeck P9Req *req; 17346488b62SChristian Schoenebeck int fid; 17446488b62SChristian Schoenebeck uint64_t offset; 17546488b62SChristian Schoenebeck 1763fe4baf4SGreg Kurz do_attach(v9p); 17746488b62SChristian Schoenebeck 17846488b62SChristian Schoenebeck fid = 1; 17946488b62SChristian Schoenebeck offset = 0; 18046488b62SChristian Schoenebeck entries = NULL; 18146488b62SChristian Schoenebeck nentries = 0; 18246488b62SChristian Schoenebeck tail = NULL; 18346488b62SChristian Schoenebeck 1843f3e9232SChristian Schoenebeck twalk({ 185569f3b63SChristian Schoenebeck .client = v9p, .fid = 0, .newfid = fid, 1863f3e9232SChristian Schoenebeck .nwname = 1, .wnames = wnames, .rwalk.nwqid = &nqid 1873f3e9232SChristian Schoenebeck }); 18846488b62SChristian Schoenebeck g_assert_cmpint(nqid, ==, 1); 18946488b62SChristian Schoenebeck 19046488b62SChristian Schoenebeck req = v9fs_tlopen(v9p, fid, O_DIRECTORY, 0); 19146488b62SChristian Schoenebeck v9fs_req_wait_for_reply(req, NULL); 19246488b62SChristian Schoenebeck v9fs_rlopen(req, &qid, NULL); 19346488b62SChristian Schoenebeck 19446488b62SChristian Schoenebeck /* 19546488b62SChristian Schoenebeck * send as many Treaddir requests as required to get all directory 19646488b62SChristian Schoenebeck * entries 19746488b62SChristian Schoenebeck */ 19846488b62SChristian Schoenebeck while (true) { 19946488b62SChristian Schoenebeck npartialentries = 0; 20046488b62SChristian Schoenebeck partialentries = NULL; 20146488b62SChristian Schoenebeck 20246488b62SChristian Schoenebeck req = v9fs_treaddir(v9p, fid, offset, count, 0); 20346488b62SChristian Schoenebeck v9fs_req_wait_for_reply(req, NULL); 20446488b62SChristian Schoenebeck v9fs_rreaddir(req, &count, &npartialentries, &partialentries); 20546488b62SChristian Schoenebeck if (npartialentries > 0 && partialentries) { 20646488b62SChristian Schoenebeck if (!entries) { 20746488b62SChristian Schoenebeck entries = partialentries; 20846488b62SChristian Schoenebeck nentries = npartialentries; 20946488b62SChristian Schoenebeck tail = partialentries; 21046488b62SChristian Schoenebeck } else { 21146488b62SChristian Schoenebeck tail->next = partialentries; 21246488b62SChristian Schoenebeck nentries += npartialentries; 21346488b62SChristian Schoenebeck } 21446488b62SChristian Schoenebeck while (tail->next) { 21546488b62SChristian Schoenebeck tail = tail->next; 21646488b62SChristian Schoenebeck } 21746488b62SChristian Schoenebeck offset = tail->offset; 21846488b62SChristian Schoenebeck } else { 21946488b62SChristian Schoenebeck break; 22046488b62SChristian Schoenebeck } 22146488b62SChristian Schoenebeck } 22246488b62SChristian Schoenebeck 22346488b62SChristian Schoenebeck g_assert_cmpint( 22446488b62SChristian Schoenebeck nentries, ==, 22546488b62SChristian Schoenebeck QTEST_V9FS_SYNTH_READDIR_NFILES + 2 /* "." and ".." */ 22646488b62SChristian Schoenebeck ); 22746488b62SChristian Schoenebeck 22846488b62SChristian Schoenebeck /* 22946488b62SChristian Schoenebeck * Check all file names exist in returned entries, ignore their order 23046488b62SChristian Schoenebeck * though. 23146488b62SChristian Schoenebeck */ 23246488b62SChristian Schoenebeck g_assert_cmpint(fs_dirents_contain_name(entries, "."), ==, true); 23346488b62SChristian Schoenebeck g_assert_cmpint(fs_dirents_contain_name(entries, ".."), ==, true); 23446488b62SChristian Schoenebeck for (int i = 0; i < QTEST_V9FS_SYNTH_READDIR_NFILES; ++i) { 23546488b62SChristian Schoenebeck char *name = g_strdup_printf(QTEST_V9FS_SYNTH_READDIR_FILE, i); 23646488b62SChristian Schoenebeck g_assert_cmpint(fs_dirents_contain_name(entries, name), ==, true); 23746488b62SChristian Schoenebeck g_free(name); 23846488b62SChristian Schoenebeck } 23946488b62SChristian Schoenebeck 24046488b62SChristian Schoenebeck v9fs_free_dirents(entries); 24146488b62SChristian Schoenebeck 24246488b62SChristian Schoenebeck g_free(wnames[0]); 24346488b62SChristian Schoenebeck } 24446488b62SChristian Schoenebeck 245dfbe8b43SEmanuele Giuseppe Esposito static void fs_walk_no_slash(void *obj, void *data, QGuestAllocator *t_alloc) 246ba0d1037SGreg Kurz { 247dfbe8b43SEmanuele Giuseppe Esposito QVirtio9P *v9p = obj; 248684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 249569f3b63SChristian Schoenebeck char *wnames[] = { g_strdup(" /") }; 250ba0d1037SGreg Kurz 2513fe4baf4SGreg Kurz do_attach(v9p); 2523f3e9232SChristian Schoenebeck twalk({ 253569f3b63SChristian Schoenebeck .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames, 2543f3e9232SChristian Schoenebeck .expectErr = ENOENT 2553f3e9232SChristian Schoenebeck }); 256ba0d1037SGreg Kurz 257ba0d1037SGreg Kurz g_free(wnames[0]); 258ba0d1037SGreg Kurz } 259ba0d1037SGreg Kurz 2609472a689SChristian Schoenebeck static void fs_walk_nonexistent(void *obj, void *data, QGuestAllocator *t_alloc) 2619472a689SChristian Schoenebeck { 2629472a689SChristian Schoenebeck QVirtio9P *v9p = obj; 263684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 2649472a689SChristian Schoenebeck 2659472a689SChristian Schoenebeck do_attach(v9p); 26615fbff48SChristian Schoenebeck /* 26715fbff48SChristian Schoenebeck * The 9p2000 protocol spec says: "If the first element cannot be walked 26815fbff48SChristian Schoenebeck * for any reason, Rerror is returned." 26915fbff48SChristian Schoenebeck */ 270569f3b63SChristian Schoenebeck twalk({ .client = v9p, .path = "non-existent", .expectErr = ENOENT }); 2719472a689SChristian Schoenebeck } 2729472a689SChristian Schoenebeck 27315fbff48SChristian Schoenebeck static void fs_walk_2nd_nonexistent(void *obj, void *data, 27415fbff48SChristian Schoenebeck QGuestAllocator *t_alloc) 27515fbff48SChristian Schoenebeck { 27615fbff48SChristian Schoenebeck QVirtio9P *v9p = obj; 277684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 2780e43495dSChristian Schoenebeck v9fs_qid root_qid; 27915fbff48SChristian Schoenebeck uint16_t nwqid; 2800e43495dSChristian Schoenebeck uint32_t fid, err; 2810e43495dSChristian Schoenebeck P9Req *req; 28215fbff48SChristian Schoenebeck g_autofree v9fs_qid *wqid = NULL; 28315fbff48SChristian Schoenebeck g_autofree char *path = g_strdup_printf( 28415fbff48SChristian Schoenebeck QTEST_V9FS_SYNTH_WALK_FILE "/non-existent", 0 28515fbff48SChristian Schoenebeck ); 28615fbff48SChristian Schoenebeck 2870e43495dSChristian Schoenebeck do_attach_rqid(v9p, &root_qid); 288569f3b63SChristian Schoenebeck fid = twalk({ 289569f3b63SChristian Schoenebeck .client = v9p, .path = path, 2903f3e9232SChristian Schoenebeck .rwalk = { .nwqid = &nwqid, .wqid = &wqid } 291569f3b63SChristian Schoenebeck }).newfid; 29215fbff48SChristian Schoenebeck /* 29315fbff48SChristian Schoenebeck * The 9p2000 protocol spec says: "nwqid is therefore either nwname or the 29415fbff48SChristian Schoenebeck * index of the first elementwise walk that failed." 29515fbff48SChristian Schoenebeck */ 29615fbff48SChristian Schoenebeck assert(nwqid == 1); 2970e43495dSChristian Schoenebeck 2980e43495dSChristian Schoenebeck /* returned QID wqid[0] is file ID of 1st subdir */ 2990e43495dSChristian Schoenebeck g_assert(wqid && wqid[0] && !is_same_qid(root_qid, wqid[0])); 3000e43495dSChristian Schoenebeck 3010e43495dSChristian Schoenebeck /* expect fid being unaffected by walk above */ 3020e43495dSChristian Schoenebeck req = v9fs_tgetattr(v9p, fid, P9_GETATTR_BASIC, 0); 3030e43495dSChristian Schoenebeck v9fs_req_wait_for_reply(req, NULL); 3040e43495dSChristian Schoenebeck v9fs_rlerror(req, &err); 3050e43495dSChristian Schoenebeck 3060e43495dSChristian Schoenebeck g_assert_cmpint(err, ==, ENOENT); 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; 315c1668948SChristian Schoenebeck P9Req *req; 316a6821b82SChristian Schoenebeck struct v9fs_attr attr; 317c1668948SChristian Schoenebeck 318*bee8fda2SChristian Schoenebeck tversion({ .client = v9p }); 319c1668948SChristian Schoenebeck req = v9fs_tattach(v9p, 0, getuid(), 0); 320c1668948SChristian Schoenebeck v9fs_req_wait_for_reply(req, NULL); 321c1668948SChristian Schoenebeck v9fs_rattach(req, &root_qid); 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 331a6821b82SChristian Schoenebeck req = v9fs_tgetattr(v9p, 1, P9_GETATTR_BASIC, 0); 332a6821b82SChristian Schoenebeck v9fs_req_wait_for_reply(req, NULL); 333a6821b82SChristian Schoenebeck v9fs_rgetattr(req, &attr); 334a6821b82SChristian Schoenebeck 335a6821b82SChristian Schoenebeck g_assert(is_same_qid(root_qid, attr.qid)); 336c1668948SChristian Schoenebeck } 337c1668948SChristian Schoenebeck 338dfbe8b43SEmanuele Giuseppe Esposito static void fs_walk_dotdot(void *obj, void *data, QGuestAllocator *t_alloc) 339a37c0702SGreg Kurz { 340dfbe8b43SEmanuele Giuseppe Esposito QVirtio9P *v9p = obj; 341684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 342569f3b63SChristian Schoenebeck char *wnames[] = { g_strdup("..") }; 34365ceee0aSChristian Schoenebeck v9fs_qid root_qid; 34465ceee0aSChristian Schoenebeck g_autofree v9fs_qid *wqid = NULL; 345a37c0702SGreg Kurz P9Req *req; 346a37c0702SGreg Kurz 347*bee8fda2SChristian Schoenebeck tversion({ .client = v9p }); 348693b21d2SGreg Kurz req = v9fs_tattach(v9p, 0, getuid(), 0); 349357e2f7fSGreg Kurz v9fs_req_wait_for_reply(req, NULL); 350a37c0702SGreg Kurz v9fs_rattach(req, &root_qid); 351a37c0702SGreg Kurz 3523f3e9232SChristian Schoenebeck twalk({ 353569f3b63SChristian Schoenebeck .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames, 3543f3e9232SChristian Schoenebeck .rwalk.wqid = &wqid /* We now we'll get one qid */ 3553f3e9232SChristian Schoenebeck }); 356a37c0702SGreg Kurz 357a37c0702SGreg Kurz g_assert_cmpmem(&root_qid, 13, wqid[0], 13); 358a37c0702SGreg Kurz 359a37c0702SGreg Kurz g_free(wnames[0]); 360a37c0702SGreg Kurz } 361a37c0702SGreg Kurz 362dfbe8b43SEmanuele Giuseppe Esposito static void fs_lopen(void *obj, void *data, QGuestAllocator *t_alloc) 36382469aaeSGreg Kurz { 364dfbe8b43SEmanuele Giuseppe Esposito QVirtio9P *v9p = obj; 365684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 366569f3b63SChristian Schoenebeck char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_LOPEN_FILE) }; 36782469aaeSGreg Kurz P9Req *req; 36882469aaeSGreg Kurz 3693fe4baf4SGreg Kurz do_attach(v9p); 3703f3e9232SChristian Schoenebeck twalk({ 3713f3e9232SChristian Schoenebeck .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames 3723f3e9232SChristian Schoenebeck }); 37382469aaeSGreg Kurz 37482469aaeSGreg Kurz req = v9fs_tlopen(v9p, 1, O_WRONLY, 0); 375357e2f7fSGreg Kurz v9fs_req_wait_for_reply(req, NULL); 37682469aaeSGreg Kurz v9fs_rlopen(req, NULL, NULL); 37782469aaeSGreg Kurz 37882469aaeSGreg Kurz g_free(wnames[0]); 37982469aaeSGreg Kurz } 38082469aaeSGreg Kurz 381dfbe8b43SEmanuele Giuseppe Esposito static void fs_write(void *obj, void *data, QGuestAllocator *t_alloc) 382354b86f8SGreg Kurz { 383dfbe8b43SEmanuele Giuseppe Esposito QVirtio9P *v9p = obj; 384684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 385354b86f8SGreg Kurz static const uint32_t write_count = P9_MAX_SIZE / 2; 386569f3b63SChristian Schoenebeck char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_WRITE_FILE) }; 38765ceee0aSChristian Schoenebeck g_autofree char *buf = g_malloc0(write_count); 388354b86f8SGreg Kurz uint32_t count; 389354b86f8SGreg Kurz P9Req *req; 390354b86f8SGreg Kurz 3913fe4baf4SGreg Kurz do_attach(v9p); 3923f3e9232SChristian Schoenebeck twalk({ 3933f3e9232SChristian Schoenebeck .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames 3943f3e9232SChristian Schoenebeck }); 395354b86f8SGreg Kurz 396354b86f8SGreg Kurz req = v9fs_tlopen(v9p, 1, O_WRONLY, 0); 397357e2f7fSGreg Kurz v9fs_req_wait_for_reply(req, NULL); 398354b86f8SGreg Kurz v9fs_rlopen(req, NULL, NULL); 399354b86f8SGreg Kurz 400354b86f8SGreg Kurz req = v9fs_twrite(v9p, 1, 0, write_count, buf, 0); 401357e2f7fSGreg Kurz v9fs_req_wait_for_reply(req, NULL); 402354b86f8SGreg Kurz v9fs_rwrite(req, &count); 403354b86f8SGreg Kurz g_assert_cmpint(count, ==, write_count); 404354b86f8SGreg Kurz 405354b86f8SGreg Kurz g_free(wnames[0]); 406354b86f8SGreg Kurz } 407354b86f8SGreg Kurz 408dfbe8b43SEmanuele Giuseppe Esposito static void fs_flush_success(void *obj, void *data, QGuestAllocator *t_alloc) 409357e2f7fSGreg Kurz { 410dfbe8b43SEmanuele Giuseppe Esposito QVirtio9P *v9p = obj; 411684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 412569f3b63SChristian Schoenebeck char *wnames[] = { g_strdup(QTEST_V9FS_SYNTH_FLUSH_FILE) }; 413357e2f7fSGreg Kurz P9Req *req, *flush_req; 414357e2f7fSGreg Kurz uint32_t reply_len; 415357e2f7fSGreg Kurz uint8_t should_block; 416357e2f7fSGreg Kurz 4173fe4baf4SGreg Kurz do_attach(v9p); 4183f3e9232SChristian Schoenebeck twalk({ 4193f3e9232SChristian Schoenebeck .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames 4203f3e9232SChristian Schoenebeck }); 421357e2f7fSGreg Kurz 422357e2f7fSGreg Kurz req = v9fs_tlopen(v9p, 1, O_WRONLY, 0); 423357e2f7fSGreg Kurz v9fs_req_wait_for_reply(req, NULL); 424357e2f7fSGreg Kurz v9fs_rlopen(req, NULL, NULL); 425357e2f7fSGreg Kurz 426357e2f7fSGreg Kurz /* This will cause the 9p server to try to write data to the backend, 427357e2f7fSGreg Kurz * until the write request gets cancelled. 428357e2f7fSGreg Kurz */ 429357e2f7fSGreg Kurz should_block = 1; 430357e2f7fSGreg Kurz req = v9fs_twrite(v9p, 1, 0, sizeof(should_block), &should_block, 0); 431357e2f7fSGreg Kurz 432357e2f7fSGreg Kurz flush_req = v9fs_tflush(v9p, req->tag, 1); 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 4543fe4baf4SGreg Kurz do_attach(v9p); 4553f3e9232SChristian Schoenebeck twalk({ 4563f3e9232SChristian Schoenebeck .client = v9p, .fid = 0, .newfid = 1, .nwname = 1, .wnames = wnames 4573f3e9232SChristian Schoenebeck }); 458357e2f7fSGreg Kurz 459357e2f7fSGreg Kurz req = v9fs_tlopen(v9p, 1, O_WRONLY, 0); 460357e2f7fSGreg Kurz v9fs_req_wait_for_reply(req, NULL); 461357e2f7fSGreg Kurz v9fs_rlopen(req, NULL, NULL); 462357e2f7fSGreg Kurz 463357e2f7fSGreg Kurz /* This will cause the write request to complete right away, before it 464357e2f7fSGreg Kurz * could be actually cancelled. 465357e2f7fSGreg Kurz */ 466357e2f7fSGreg Kurz should_block = 0; 467357e2f7fSGreg Kurz req = v9fs_twrite(v9p, 1, 0, sizeof(should_block), &should_block, 0); 468357e2f7fSGreg Kurz 469357e2f7fSGreg Kurz flush_req = v9fs_tflush(v9p, req->tag, 1); 470357e2f7fSGreg Kurz 471357e2f7fSGreg Kurz /* The write request is supposed to complete. The server should 472357e2f7fSGreg Kurz * reply to the write request and the flush request. 473357e2f7fSGreg Kurz */ 474357e2f7fSGreg Kurz v9fs_req_wait_for_reply(req, NULL); 475357e2f7fSGreg Kurz v9fs_rwrite(req, &count); 476357e2f7fSGreg Kurz g_assert_cmpint(count, ==, sizeof(should_block)); 477357e2f7fSGreg Kurz v9fs_rflush(flush_req); 478357e2f7fSGreg Kurz 479357e2f7fSGreg Kurz g_free(wnames[0]); 480357e2f7fSGreg Kurz } 481357e2f7fSGreg Kurz 482c1934f63SGreg Kurz static void do_mkdir(QVirtio9P *v9p, const char *path, const char *cname) 483653daf38SChristian Schoenebeck { 48465ceee0aSChristian Schoenebeck g_autofree char *name = g_strdup(cname); 48520018805SChristian Schoenebeck uint32_t fid; 486653daf38SChristian Schoenebeck P9Req *req; 487653daf38SChristian Schoenebeck 488569f3b63SChristian Schoenebeck fid = twalk({ .client = v9p, .path = path }).newfid; 489653daf38SChristian Schoenebeck 490653daf38SChristian Schoenebeck req = v9fs_tmkdir(v9p, fid, name, 0750, 0, 0); 491653daf38SChristian Schoenebeck v9fs_req_wait_for_reply(req, NULL); 492653daf38SChristian Schoenebeck v9fs_rmkdir(req, NULL); 493653daf38SChristian Schoenebeck } 494653daf38SChristian Schoenebeck 495b09dbfddSChristian Schoenebeck /* create a regular file with Tlcreate and return file's fid */ 496b09dbfddSChristian Schoenebeck static uint32_t do_lcreate(QVirtio9P *v9p, const char *path, 497b09dbfddSChristian Schoenebeck const char *cname) 498b09dbfddSChristian Schoenebeck { 49965ceee0aSChristian Schoenebeck g_autofree char *name = g_strdup(cname); 500b09dbfddSChristian Schoenebeck uint32_t fid; 501b09dbfddSChristian Schoenebeck P9Req *req; 502b09dbfddSChristian Schoenebeck 503569f3b63SChristian Schoenebeck fid = twalk({ .client = v9p, .path = path }).newfid; 504b09dbfddSChristian Schoenebeck 505b09dbfddSChristian Schoenebeck req = v9fs_tlcreate(v9p, fid, name, 0, 0750, 0, 0); 506b09dbfddSChristian Schoenebeck v9fs_req_wait_for_reply(req, NULL); 507b09dbfddSChristian Schoenebeck v9fs_rlcreate(req, NULL, NULL); 508b09dbfddSChristian Schoenebeck 509b09dbfddSChristian Schoenebeck return fid; 510b09dbfddSChristian Schoenebeck } 511b09dbfddSChristian Schoenebeck 51259ff563dSChristian Schoenebeck /* create symlink named @a clink in directory @a path pointing to @a to */ 51359ff563dSChristian Schoenebeck static void do_symlink(QVirtio9P *v9p, const char *path, const char *clink, 51459ff563dSChristian Schoenebeck const char *to) 51559ff563dSChristian Schoenebeck { 51665ceee0aSChristian Schoenebeck g_autofree char *name = g_strdup(clink); 51765ceee0aSChristian Schoenebeck g_autofree char *dst = g_strdup(to); 51859ff563dSChristian Schoenebeck uint32_t fid; 51959ff563dSChristian Schoenebeck P9Req *req; 52059ff563dSChristian Schoenebeck 521569f3b63SChristian Schoenebeck fid = twalk({ .client = v9p, .path = path }).newfid; 52259ff563dSChristian Schoenebeck 52359ff563dSChristian Schoenebeck req = v9fs_tsymlink(v9p, fid, name, dst, 0, 0); 52459ff563dSChristian Schoenebeck v9fs_req_wait_for_reply(req, NULL); 52559ff563dSChristian Schoenebeck v9fs_rsymlink(req, NULL); 52659ff563dSChristian Schoenebeck } 52759ff563dSChristian Schoenebeck 52864e3d403SChristian Schoenebeck /* create a hard link named @a clink in directory @a path pointing to @a to */ 52964e3d403SChristian Schoenebeck static void do_hardlink(QVirtio9P *v9p, const char *path, const char *clink, 53064e3d403SChristian Schoenebeck const char *to) 53164e3d403SChristian Schoenebeck { 53264e3d403SChristian Schoenebeck uint32_t dfid, fid; 53364e3d403SChristian Schoenebeck P9Req *req; 53464e3d403SChristian Schoenebeck 535569f3b63SChristian Schoenebeck dfid = twalk({ .client = v9p, .path = path }).newfid; 536569f3b63SChristian Schoenebeck fid = twalk({ .client = v9p, .path = to }).newfid; 53764e3d403SChristian Schoenebeck 53864e3d403SChristian Schoenebeck req = v9fs_tlink(v9p, dfid, fid, clink, 0); 53964e3d403SChristian Schoenebeck v9fs_req_wait_for_reply(req, NULL); 54064e3d403SChristian Schoenebeck v9fs_rlink(req); 54164e3d403SChristian Schoenebeck } 54264e3d403SChristian Schoenebeck 543b37d62d6SChristian Schoenebeck static void do_unlinkat(QVirtio9P *v9p, const char *atpath, const char *rpath, 544b37d62d6SChristian Schoenebeck uint32_t flags) 545b37d62d6SChristian Schoenebeck { 54665ceee0aSChristian Schoenebeck g_autofree char *name = g_strdup(rpath); 547b37d62d6SChristian Schoenebeck uint32_t fid; 548b37d62d6SChristian Schoenebeck P9Req *req; 549b37d62d6SChristian Schoenebeck 550569f3b63SChristian Schoenebeck fid = twalk({ .client = v9p, .path = atpath }).newfid; 551b37d62d6SChristian Schoenebeck 552b37d62d6SChristian Schoenebeck req = v9fs_tunlinkat(v9p, fid, name, flags, 0); 553b37d62d6SChristian Schoenebeck v9fs_req_wait_for_reply(req, NULL); 554b37d62d6SChristian Schoenebeck v9fs_runlinkat(req); 555b37d62d6SChristian Schoenebeck } 556b37d62d6SChristian Schoenebeck 55746488b62SChristian Schoenebeck static void fs_readdir_split_128(void *obj, void *data, 55846488b62SChristian Schoenebeck QGuestAllocator *t_alloc) 55946488b62SChristian Schoenebeck { 560684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 5611d98613dSGreg Kurz do_readdir_split(obj, 128); 56246488b62SChristian Schoenebeck } 56346488b62SChristian Schoenebeck 56446488b62SChristian Schoenebeck static void fs_readdir_split_256(void *obj, void *data, 56546488b62SChristian Schoenebeck QGuestAllocator *t_alloc) 56646488b62SChristian Schoenebeck { 567684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 5681d98613dSGreg Kurz do_readdir_split(obj, 256); 56946488b62SChristian Schoenebeck } 57046488b62SChristian Schoenebeck 57146488b62SChristian Schoenebeck static void fs_readdir_split_512(void *obj, void *data, 57246488b62SChristian Schoenebeck QGuestAllocator *t_alloc) 57346488b62SChristian Schoenebeck { 574684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 5751d98613dSGreg Kurz do_readdir_split(obj, 512); 57646488b62SChristian Schoenebeck } 57746488b62SChristian Schoenebeck 578653daf38SChristian Schoenebeck 579653daf38SChristian Schoenebeck /* tests using the 9pfs 'local' fs driver */ 580653daf38SChristian Schoenebeck 581653daf38SChristian Schoenebeck static void fs_create_dir(void *obj, void *data, QGuestAllocator *t_alloc) 582653daf38SChristian Schoenebeck { 583653daf38SChristian Schoenebeck QVirtio9P *v9p = obj; 584684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 585653daf38SChristian Schoenebeck struct stat st; 58665ceee0aSChristian Schoenebeck g_autofree char *root_path = virtio_9p_test_path(""); 58765ceee0aSChristian Schoenebeck g_autofree char *new_dir = virtio_9p_test_path("01"); 588653daf38SChristian Schoenebeck 589653daf38SChristian Schoenebeck g_assert(root_path != NULL); 590653daf38SChristian Schoenebeck 5913fe4baf4SGreg Kurz do_attach(v9p); 592c1934f63SGreg Kurz do_mkdir(v9p, "/", "01"); 593653daf38SChristian Schoenebeck 594653daf38SChristian Schoenebeck /* check if created directory really exists now ... */ 595653daf38SChristian Schoenebeck g_assert(stat(new_dir, &st) == 0); 596653daf38SChristian Schoenebeck /* ... and is actually a directory */ 597653daf38SChristian Schoenebeck g_assert((st.st_mode & S_IFMT) == S_IFDIR); 598653daf38SChristian Schoenebeck } 599653daf38SChristian Schoenebeck 600b37d62d6SChristian Schoenebeck static void fs_unlinkat_dir(void *obj, void *data, QGuestAllocator *t_alloc) 601b37d62d6SChristian Schoenebeck { 602b37d62d6SChristian Schoenebeck QVirtio9P *v9p = obj; 603684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 604b37d62d6SChristian Schoenebeck struct stat st; 60565ceee0aSChristian Schoenebeck g_autofree char *root_path = virtio_9p_test_path(""); 60665ceee0aSChristian Schoenebeck g_autofree char *new_dir = virtio_9p_test_path("02"); 607b37d62d6SChristian Schoenebeck 608b37d62d6SChristian Schoenebeck g_assert(root_path != NULL); 609b37d62d6SChristian Schoenebeck 610b37d62d6SChristian Schoenebeck do_attach(v9p); 611b37d62d6SChristian Schoenebeck do_mkdir(v9p, "/", "02"); 612b37d62d6SChristian Schoenebeck 613b37d62d6SChristian Schoenebeck /* check if created directory really exists now ... */ 614b37d62d6SChristian Schoenebeck g_assert(stat(new_dir, &st) == 0); 615b37d62d6SChristian Schoenebeck /* ... and is actually a directory */ 616b37d62d6SChristian Schoenebeck g_assert((st.st_mode & S_IFMT) == S_IFDIR); 617b37d62d6SChristian Schoenebeck 618d3671fd9SWill Cohen do_unlinkat(v9p, "/", "02", P9_DOTL_AT_REMOVEDIR); 619b37d62d6SChristian Schoenebeck /* directory should be gone now */ 620b37d62d6SChristian Schoenebeck g_assert(stat(new_dir, &st) != 0); 621b37d62d6SChristian Schoenebeck } 622b37d62d6SChristian Schoenebeck 623b09dbfddSChristian Schoenebeck static void fs_create_file(void *obj, void *data, QGuestAllocator *t_alloc) 624b09dbfddSChristian Schoenebeck { 625b09dbfddSChristian Schoenebeck QVirtio9P *v9p = obj; 626684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 627b09dbfddSChristian Schoenebeck struct stat st; 62865ceee0aSChristian Schoenebeck g_autofree char *new_file = virtio_9p_test_path("03/1st_file"); 629b09dbfddSChristian Schoenebeck 630b09dbfddSChristian Schoenebeck do_attach(v9p); 631b09dbfddSChristian Schoenebeck do_mkdir(v9p, "/", "03"); 632b09dbfddSChristian Schoenebeck do_lcreate(v9p, "03", "1st_file"); 633b09dbfddSChristian Schoenebeck 634b09dbfddSChristian Schoenebeck /* check if created file exists now ... */ 635b09dbfddSChristian Schoenebeck g_assert(stat(new_file, &st) == 0); 636b09dbfddSChristian Schoenebeck /* ... and is a regular file */ 637b09dbfddSChristian Schoenebeck g_assert((st.st_mode & S_IFMT) == S_IFREG); 638b09dbfddSChristian Schoenebeck } 639b09dbfddSChristian Schoenebeck 640472c18b8SChristian Schoenebeck static void fs_unlinkat_file(void *obj, void *data, QGuestAllocator *t_alloc) 641472c18b8SChristian Schoenebeck { 642472c18b8SChristian Schoenebeck QVirtio9P *v9p = obj; 643684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 644472c18b8SChristian Schoenebeck struct stat st; 64565ceee0aSChristian Schoenebeck g_autofree char *new_file = virtio_9p_test_path("04/doa_file"); 646472c18b8SChristian Schoenebeck 647472c18b8SChristian Schoenebeck do_attach(v9p); 648472c18b8SChristian Schoenebeck do_mkdir(v9p, "/", "04"); 649472c18b8SChristian Schoenebeck do_lcreate(v9p, "04", "doa_file"); 650472c18b8SChristian Schoenebeck 651472c18b8SChristian Schoenebeck /* check if created file exists now ... */ 652472c18b8SChristian Schoenebeck g_assert(stat(new_file, &st) == 0); 653472c18b8SChristian Schoenebeck /* ... and is a regular file */ 654472c18b8SChristian Schoenebeck g_assert((st.st_mode & S_IFMT) == S_IFREG); 655472c18b8SChristian Schoenebeck 656472c18b8SChristian Schoenebeck do_unlinkat(v9p, "04", "doa_file", 0); 657472c18b8SChristian Schoenebeck /* file should be gone now */ 658472c18b8SChristian Schoenebeck g_assert(stat(new_file, &st) != 0); 659472c18b8SChristian Schoenebeck } 660472c18b8SChristian Schoenebeck 66159ff563dSChristian Schoenebeck static void fs_symlink_file(void *obj, void *data, QGuestAllocator *t_alloc) 66259ff563dSChristian Schoenebeck { 66359ff563dSChristian Schoenebeck QVirtio9P *v9p = obj; 664684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 66559ff563dSChristian Schoenebeck struct stat st; 66665ceee0aSChristian Schoenebeck g_autofree char *real_file = virtio_9p_test_path("05/real_file"); 66765ceee0aSChristian Schoenebeck g_autofree char *symlink_file = virtio_9p_test_path("05/symlink_file"); 66859ff563dSChristian Schoenebeck 66959ff563dSChristian Schoenebeck do_attach(v9p); 67059ff563dSChristian Schoenebeck do_mkdir(v9p, "/", "05"); 67159ff563dSChristian Schoenebeck do_lcreate(v9p, "05", "real_file"); 67259ff563dSChristian Schoenebeck g_assert(stat(real_file, &st) == 0); 67359ff563dSChristian Schoenebeck g_assert((st.st_mode & S_IFMT) == S_IFREG); 67459ff563dSChristian Schoenebeck 67559ff563dSChristian Schoenebeck do_symlink(v9p, "05", "symlink_file", "real_file"); 67659ff563dSChristian Schoenebeck 67759ff563dSChristian Schoenebeck /* check if created link exists now */ 67859ff563dSChristian Schoenebeck g_assert(stat(symlink_file, &st) == 0); 67959ff563dSChristian Schoenebeck } 68059ff563dSChristian Schoenebeck 6815b28ab8bSChristian Schoenebeck static void fs_unlinkat_symlink(void *obj, void *data, 6825b28ab8bSChristian Schoenebeck QGuestAllocator *t_alloc) 6835b28ab8bSChristian Schoenebeck { 6845b28ab8bSChristian Schoenebeck QVirtio9P *v9p = obj; 685684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 6865b28ab8bSChristian Schoenebeck struct stat st; 68765ceee0aSChristian Schoenebeck g_autofree char *real_file = virtio_9p_test_path("06/real_file"); 68865ceee0aSChristian Schoenebeck g_autofree char *symlink_file = virtio_9p_test_path("06/symlink_file"); 6895b28ab8bSChristian Schoenebeck 6905b28ab8bSChristian Schoenebeck do_attach(v9p); 6915b28ab8bSChristian Schoenebeck do_mkdir(v9p, "/", "06"); 6925b28ab8bSChristian Schoenebeck do_lcreate(v9p, "06", "real_file"); 6935b28ab8bSChristian Schoenebeck g_assert(stat(real_file, &st) == 0); 6945b28ab8bSChristian Schoenebeck g_assert((st.st_mode & S_IFMT) == S_IFREG); 6955b28ab8bSChristian Schoenebeck 6965b28ab8bSChristian Schoenebeck do_symlink(v9p, "06", "symlink_file", "real_file"); 6975b28ab8bSChristian Schoenebeck g_assert(stat(symlink_file, &st) == 0); 6985b28ab8bSChristian Schoenebeck 6995b28ab8bSChristian Schoenebeck do_unlinkat(v9p, "06", "symlink_file", 0); 7005b28ab8bSChristian Schoenebeck /* symlink should be gone now */ 7015b28ab8bSChristian Schoenebeck g_assert(stat(symlink_file, &st) != 0); 7025b28ab8bSChristian Schoenebeck } 7035b28ab8bSChristian Schoenebeck 70464e3d403SChristian Schoenebeck static void fs_hardlink_file(void *obj, void *data, QGuestAllocator *t_alloc) 70564e3d403SChristian Schoenebeck { 70664e3d403SChristian Schoenebeck QVirtio9P *v9p = obj; 707684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 70864e3d403SChristian Schoenebeck struct stat st_real, st_link; 70965ceee0aSChristian Schoenebeck g_autofree char *real_file = virtio_9p_test_path("07/real_file"); 71065ceee0aSChristian Schoenebeck g_autofree char *hardlink_file = virtio_9p_test_path("07/hardlink_file"); 71164e3d403SChristian Schoenebeck 71264e3d403SChristian Schoenebeck do_attach(v9p); 71364e3d403SChristian Schoenebeck do_mkdir(v9p, "/", "07"); 71464e3d403SChristian Schoenebeck do_lcreate(v9p, "07", "real_file"); 71564e3d403SChristian Schoenebeck g_assert(stat(real_file, &st_real) == 0); 71664e3d403SChristian Schoenebeck g_assert((st_real.st_mode & S_IFMT) == S_IFREG); 71764e3d403SChristian Schoenebeck 71864e3d403SChristian Schoenebeck do_hardlink(v9p, "07", "hardlink_file", "07/real_file"); 71964e3d403SChristian Schoenebeck 72064e3d403SChristian Schoenebeck /* check if link exists now ... */ 72164e3d403SChristian Schoenebeck g_assert(stat(hardlink_file, &st_link) == 0); 72264e3d403SChristian Schoenebeck /* ... and it's a hard link, right? */ 72364e3d403SChristian Schoenebeck g_assert((st_link.st_mode & S_IFMT) == S_IFREG); 72464e3d403SChristian Schoenebeck g_assert(st_link.st_dev == st_real.st_dev); 72564e3d403SChristian Schoenebeck g_assert(st_link.st_ino == st_real.st_ino); 72664e3d403SChristian Schoenebeck } 72764e3d403SChristian Schoenebeck 7284d0746e2SChristian Schoenebeck static void fs_unlinkat_hardlink(void *obj, void *data, 7294d0746e2SChristian Schoenebeck QGuestAllocator *t_alloc) 7304d0746e2SChristian Schoenebeck { 7314d0746e2SChristian Schoenebeck QVirtio9P *v9p = obj; 732684f9120SChristian Schoenebeck v9fs_set_allocator(t_alloc); 7334d0746e2SChristian Schoenebeck struct stat st_real, st_link; 73465ceee0aSChristian Schoenebeck g_autofree char *real_file = virtio_9p_test_path("08/real_file"); 73565ceee0aSChristian Schoenebeck g_autofree char *hardlink_file = virtio_9p_test_path("08/hardlink_file"); 7364d0746e2SChristian Schoenebeck 7374d0746e2SChristian Schoenebeck do_attach(v9p); 7384d0746e2SChristian Schoenebeck do_mkdir(v9p, "/", "08"); 7394d0746e2SChristian Schoenebeck do_lcreate(v9p, "08", "real_file"); 7404d0746e2SChristian Schoenebeck g_assert(stat(real_file, &st_real) == 0); 7414d0746e2SChristian Schoenebeck g_assert((st_real.st_mode & S_IFMT) == S_IFREG); 7424d0746e2SChristian Schoenebeck 7434d0746e2SChristian Schoenebeck do_hardlink(v9p, "08", "hardlink_file", "08/real_file"); 7444d0746e2SChristian Schoenebeck g_assert(stat(hardlink_file, &st_link) == 0); 7454d0746e2SChristian Schoenebeck 7464d0746e2SChristian Schoenebeck do_unlinkat(v9p, "08", "hardlink_file", 0); 7474d0746e2SChristian Schoenebeck /* symlink should be gone now */ 7484d0746e2SChristian Schoenebeck g_assert(stat(hardlink_file, &st_link) != 0); 7494d0746e2SChristian Schoenebeck /* and old file should still exist */ 7504d0746e2SChristian Schoenebeck g_assert(stat(real_file, &st_real) == 0); 7514d0746e2SChristian Schoenebeck } 7524d0746e2SChristian Schoenebeck 7533a565c64SChristian Schoenebeck static void *assign_9p_local_driver(GString *cmd_line, void *arg) 7543a565c64SChristian Schoenebeck { 7553a565c64SChristian Schoenebeck virtio_9p_assign_local_driver(cmd_line, "security_model=mapped-xattr"); 7563a565c64SChristian Schoenebeck return arg; 7573a565c64SChristian Schoenebeck } 7583a565c64SChristian Schoenebeck 759dfbe8b43SEmanuele Giuseppe Esposito static void register_virtio_9p_test(void) 7601211d81bSGreg Kurz { 7613a565c64SChristian Schoenebeck 7623a565c64SChristian Schoenebeck QOSGraphTestOptions opts = { 7633a565c64SChristian Schoenebeck }; 7643a565c64SChristian Schoenebeck 7653a565c64SChristian Schoenebeck /* 9pfs test cases using the 'synth' filesystem driver */ 7663a565c64SChristian Schoenebeck qos_add_test("synth/config", "virtio-9p", pci_config, &opts); 7673a565c64SChristian Schoenebeck qos_add_test("synth/version/basic", "virtio-9p", fs_version, &opts); 7683a565c64SChristian Schoenebeck qos_add_test("synth/attach/basic", "virtio-9p", fs_attach, &opts); 7693a565c64SChristian Schoenebeck qos_add_test("synth/walk/basic", "virtio-9p", fs_walk, &opts); 770eefd2394SChristian Schoenebeck qos_add_test("synth/walk/no_slash", "virtio-9p", fs_walk_no_slash, 7713a565c64SChristian Schoenebeck &opts); 772c1668948SChristian Schoenebeck qos_add_test("synth/walk/none", "virtio-9p", fs_walk_none, &opts); 773eefd2394SChristian Schoenebeck qos_add_test("synth/walk/dotdot_from_root", "virtio-9p", 7743a565c64SChristian Schoenebeck fs_walk_dotdot, &opts); 7759472a689SChristian Schoenebeck qos_add_test("synth/walk/non_existent", "virtio-9p", fs_walk_nonexistent, 7769472a689SChristian Schoenebeck &opts); 77715fbff48SChristian Schoenebeck qos_add_test("synth/walk/2nd_non_existent", "virtio-9p", 77815fbff48SChristian Schoenebeck fs_walk_2nd_nonexistent, &opts); 7793a565c64SChristian Schoenebeck qos_add_test("synth/lopen/basic", "virtio-9p", fs_lopen, &opts); 7803a565c64SChristian Schoenebeck qos_add_test("synth/write/basic", "virtio-9p", fs_write, &opts); 781eefd2394SChristian Schoenebeck qos_add_test("synth/flush/success", "virtio-9p", fs_flush_success, 7823a565c64SChristian Schoenebeck &opts); 783eefd2394SChristian Schoenebeck qos_add_test("synth/flush/ignored", "virtio-9p", fs_flush_ignored, 7843a565c64SChristian Schoenebeck &opts); 7853a565c64SChristian Schoenebeck qos_add_test("synth/readdir/basic", "virtio-9p", fs_readdir, &opts); 786eefd2394SChristian Schoenebeck qos_add_test("synth/readdir/split_512", "virtio-9p", 7873a565c64SChristian Schoenebeck fs_readdir_split_512, &opts); 788eefd2394SChristian Schoenebeck qos_add_test("synth/readdir/split_256", "virtio-9p", 7893a565c64SChristian Schoenebeck fs_readdir_split_256, &opts); 790eefd2394SChristian Schoenebeck qos_add_test("synth/readdir/split_128", "virtio-9p", 7913a565c64SChristian Schoenebeck fs_readdir_split_128, &opts); 7923a565c64SChristian Schoenebeck 7933a565c64SChristian Schoenebeck 7943a565c64SChristian Schoenebeck /* 9pfs test cases using the 'local' filesystem driver */ 795558f5c42SGreg Kurz 796558f5c42SGreg Kurz /* 797558f5c42SGreg Kurz * XXX: Until we are sure that these tests can run everywhere, 798558f5c42SGreg Kurz * keep them as "slow" so that they aren't run with "make check". 799558f5c42SGreg Kurz */ 800558f5c42SGreg Kurz if (!g_test_slow()) { 801558f5c42SGreg Kurz return; 802558f5c42SGreg Kurz } 803558f5c42SGreg Kurz 8043a565c64SChristian Schoenebeck opts.before = assign_9p_local_driver; 8053a565c64SChristian Schoenebeck qos_add_test("local/config", "virtio-9p", pci_config, &opts); 806653daf38SChristian Schoenebeck qos_add_test("local/create_dir", "virtio-9p", fs_create_dir, &opts); 807b37d62d6SChristian Schoenebeck qos_add_test("local/unlinkat_dir", "virtio-9p", fs_unlinkat_dir, &opts); 808b09dbfddSChristian Schoenebeck qos_add_test("local/create_file", "virtio-9p", fs_create_file, &opts); 809472c18b8SChristian Schoenebeck qos_add_test("local/unlinkat_file", "virtio-9p", fs_unlinkat_file, &opts); 81059ff563dSChristian Schoenebeck qos_add_test("local/symlink_file", "virtio-9p", fs_symlink_file, &opts); 8115b28ab8bSChristian Schoenebeck qos_add_test("local/unlinkat_symlink", "virtio-9p", fs_unlinkat_symlink, 8125b28ab8bSChristian Schoenebeck &opts); 81364e3d403SChristian Schoenebeck qos_add_test("local/hardlink_file", "virtio-9p", fs_hardlink_file, &opts); 8144d0746e2SChristian Schoenebeck qos_add_test("local/unlinkat_hardlink", "virtio-9p", fs_unlinkat_hardlink, 8154d0746e2SChristian Schoenebeck &opts); 8161211d81bSGreg Kurz } 8171211d81bSGreg Kurz 818dfbe8b43SEmanuele Giuseppe Esposito libqos_init(register_virtio_9p_test); 819136b7af2SChristian Schoenebeck 820136b7af2SChristian Schoenebeck static void __attribute__((constructor)) construct_9p_test(void) 821136b7af2SChristian Schoenebeck { 822136b7af2SChristian Schoenebeck /* make sure test dir for the 'local' tests exists */ 823136b7af2SChristian Schoenebeck virtio_9p_create_local_test_dir(); 824136b7af2SChristian Schoenebeck } 825136b7af2SChristian Schoenebeck 826136b7af2SChristian Schoenebeck static void __attribute__((destructor)) destruct_9p_test(void) 827136b7af2SChristian Schoenebeck { 828136b7af2SChristian Schoenebeck /* remove previously created test dir when test suite completed */ 829136b7af2SChristian Schoenebeck virtio_9p_remove_local_test_dir(); 830136b7af2SChristian Schoenebeck } 831