@@ -1378,6 +1378,39 @@ static void fs_hardlink_file(void *obj, void *data, QGuestAllocator *t_alloc)
g_free(real_file);
}
+static void fs_unlinkat_hardlink(void *obj, void *data,
+ QGuestAllocator *t_alloc)
+{
+ QVirtio9P *v9p = obj;
+ P9Req *req;
+ uint32_t dfid, fid;
+ struct stat st_real, st_link;
+ char *real_file = virtio_9p_test_path("08/real_file");
+ char *hardlink_file = virtio_9p_test_path("08/hardlink_file");
+
+ fs_attach(v9p, NULL, t_alloc);
+ fs_mkdir(v9p, data, t_alloc, "/", "08");
+ fid = fs_lcreate(v9p, data, t_alloc, "08", "real_file");
+ g_assert(stat(real_file, &st_real) == 0);
+ g_assert((st_real.st_mode & S_IFMT) == S_IFREG);
+
+ dfid = fs_walk_fid(v9p, data, t_alloc, "08");
+
+ req = v9fs_tlink(v9p, dfid, fid, "hardlink_file", 0);
+ v9fs_req_wait_for_reply(req, NULL);
+ v9fs_rlink(req);
+ g_assert(stat(hardlink_file, &st_link) == 0);
+
+ fs_unlinkat(v9p, data, t_alloc, "08", "hardlink_file", 0);
+ /* symlink should be gone now */
+ g_assert(stat(hardlink_file, &st_link) != 0);
+ /* and old file should still exist */
+ g_assert(stat(real_file, &st_real) == 0);
+
+ g_free(hardlink_file);
+ g_free(real_file);
+}
+
static void *assign_9p_local_driver(GString *cmd_line, void *arg)
{
virtio_9p_assign_local_driver(cmd_line, "security_model=mapped-xattr");
@@ -1424,6 +1457,7 @@ static void register_virtio_9p_test(void)
qos_add_test("local/symlink_file", "virtio-9p", fs_symlink_file, &opts);
qos_add_test("local/unlinkat_symlink", "virtio-9p", fs_unlinkat_symlink, &opts);
qos_add_test("local/hardlink_file", "virtio-9p", fs_hardlink_file, &opts);
+ qos_add_test("local/unlinkat_hardlink", "virtio-9p", fs_unlinkat_hardlink, &opts);
}
libqos_init(register_virtio_9p_test);
This test case uses a Tunlinkat request to remove a previously hard linked file by using the 9pfs 'local' fs driver. Signed-off-by: Christian Schoenebeck <qemu_oss@crudebyte.com> --- tests/qtest/virtio-9p-test.c | 34 ++++++++++++++++++++++++++++++++++ 1 file changed, 34 insertions(+)