/* SPDX-License-Identifier: GPL-3.0-or-later */ /* * dir_iterator.c * * Copyright (C) 2019 David Oberhollenzer */ #include "config.h" #include "io/dir_iterator.h" #include "sqfs/error.h" #include "util/test.h" #include "compat.h" static int compare_entries(const void *a, const void *b) { const sqfs_dir_entry_t *const *lhs = a; const sqfs_dir_entry_t *const *rhs = b; return strcmp((*lhs)->name, (*rhs)->name); } int main(int argc, char **argv) { dir_iterator_t *dir, *suba, *subb, *subc, *sub; sqfs_dir_entry_t *dent, *ent[6]; size_t i; int ret; (void)argc; (void)argv; /* scan the top level hierarchy */ dir = dir_iterator_create(TEST_PATH); TEST_NOT_NULL(dir); ret = dir->next(dir, &ent[0]); TEST_NOT_NULL(ent[0]); TEST_EQUAL_I(ret, 0); ret = dir->next(dir, &ent[1]); TEST_NOT_NULL(ent[1]); TEST_EQUAL_I(ret, 0); ret = dir->next(dir, &ent[2]); TEST_NOT_NULL(ent[2]); TEST_EQUAL_I(ret, 0); ret = dir->next(dir, &ent[3]); TEST_NOT_NULL(ent[3]); TEST_EQUAL_I(ret, 0); ret = dir->next(dir, &ent[4]); TEST_NOT_NULL(ent[4]); TEST_EQUAL_I(ret, 0); ret = dir->next(dir, &ent[5]); TEST_NULL(ent[5]); TEST_ASSERT(ret > 0); dir = sqfs_drop(dir); qsort(ent, 5, sizeof(ent[0]), compare_entries); TEST_STR_EQUAL(ent[0]->name, "."); TEST_ASSERT(S_ISDIR(ent[0]->mode)); TEST_STR_EQUAL(ent[1]->name, ".."); TEST_ASSERT(S_ISDIR(ent[1]->mode)); TEST_STR_EQUAL(ent[2]->name, "dira"); TEST_ASSERT(S_ISDIR(ent[2]->mode)); TEST_STR_EQUAL(ent[3]->name, "dirb"); TEST_ASSERT(S_ISDIR(ent[3]->mode)); TEST_STR_EQUAL(ent[4]->name, "dirc"); TEST_ASSERT(S_ISDIR(ent[4]->mode)); for (i = 0; i < 5; ++i) free(ent[i]); /* scan first sub hierarchy */ dir = dir_iterator_create(TEST_PATH "/dira"); TEST_NOT_NULL(dir); ret = dir->next(dir, &ent[0]); TEST_NOT_NULL(ent[0]); TEST_EQUAL_I(ret, 0); ret = dir->next(dir, &ent[1]); TEST_NOT_NULL(ent[1]); TEST_EQUAL_I(ret, 0); ret = dir->next(dir, &ent[2]); TEST_NOT_NULL(ent[2]); TEST_EQUAL_I(ret, 0); ret = dir->next(dir, &ent[3]); TEST_NOT_NULL(ent[3]); TEST_EQUAL_I(ret, 0); ret = dir->next(dir, &ent[4]); TEST_NOT_NULL(ent[4]); TEST_EQUAL_I(ret, 0); ret = dir->next(dir, &ent[5]); TEST_NULL(ent[5]); TEST_ASSERT(ret > 0); dir = sqfs_drop(dir); qsort(ent, 5, sizeof(ent[0]), compare_entries); TEST_STR_EQUAL(ent[0]->name, "."); TEST_ASSERT(S_ISDIR(ent[0]->mode)); TEST_STR_EQUAL(ent[1]->name, ".."); TEST_ASSERT(S_ISDIR(ent[1]->mode)); TEST_STR_EQUAL(ent[2]->name, "file_a0"); TEST_ASSERT(S_ISREG(ent[2]->mode)); TEST_STR_EQUAL(ent[3]->name, "file_a1"); TEST_ASSERT(S_ISREG(ent[3]->mode)); TEST_STR_EQUAL(ent[4]->name, "file_a2"); TEST_ASSERT(S_ISREG(ent[4]->mode)); for (i = 0; i < 5; ++i) free(ent[i]); /* scan second sub hierarchy */ dir = dir_iterator_create(TEST_PATH "/dirb"); TEST_NOT_NULL(dir); ret = dir->next(dir, &ent[0]); TEST_NOT_NULL(ent[0]); TEST_EQUAL_I(ret, 0); ret = dir->next(dir, &ent[1]); TEST_NOT_NULL(ent[1]); TEST_EQUAL_I(ret, 0); ret = dir->next(dir, &ent[2]); TEST_NOT_NULL(ent[2]); TEST_EQUAL_I(ret, 0); ret = dir->next(dir, &ent[3]); TEST_NOT_NULL(ent[3]); TEST_EQUAL_I(ret, 0); ret = dir->next(dir, &ent[4]); TEST_NOT_NULL(ent[4]); TEST_EQUAL_I(ret, 0); ret = dir->next(dir, &ent[5]); TEST_NOT_NULL(ent[5]); TEST_EQUAL_I(ret, 0); ret = dir->next(dir, &dent); TEST_NULL(dent); TEST_ASSERT(ret > 0); dir = sqfs_drop(dir); qsort(ent, 6, sizeof(ent[0]), compare_entries); TEST_STR_EQUAL(ent[0]->name, "."); TEST_ASSERT(S_ISDIR(ent[0]->mode)); TEST_STR_EQUAL(ent[1]->name, ".."); TEST_ASSERT(S_ISDIR(ent[1]->mode)); TEST_STR_EQUAL(ent[2]->name, "dirx"); TEST_ASSERT(S_ISDIR(ent[2]->mode)); TEST_STR_EQUAL(ent[3]->name, "file_b0"); TEST_ASSERT(S_ISREG(ent[3]->mode)); TEST_STR_EQUAL(ent[4]->name, "file_b1"); TEST_ASSERT(S_ISREG(ent[4]->mode)); TEST_STR_EQUAL(ent[5]->name, "file_b2"); TEST_ASSERT(S_ISREG(ent[5]->mode)); for (i = 0; i < 6; ++i) free(ent[i]); /* scan first sub hierarchy */ dir = dir_iterator_create(TEST_PATH "/dirc"); TEST_NOT_NULL(dir); ret = dir->next(dir, &ent[0]); TEST_NOT_NULL(ent[0]); TEST_EQUAL_I(ret, 0); ret = dir->next(dir, &ent[1]); TEST_NOT_NULL(ent[1]); TEST_EQUAL_I(ret, 0); ret = dir->next(dir, &ent[2]); TEST_NOT_NULL(ent[2]); TEST_EQUAL_I(ret, 0); ret = dir->next(dir, &ent[3]); TEST_NOT_NULL(ent[3]); TEST_EQUAL_I(ret, 0); ret = dir->next(dir, &ent[4]); TEST_NOT_NULL(ent[4]); TEST_EQUAL_I(ret, 0); ret = dir->next(dir, &ent[5]); TEST_NULL(ent[5]); TEST_ASSERT(ret > 0); dir = sqfs_drop(dir); qsort(ent, 5, sizeof(ent[0]), compare_entries); TEST_STR_EQUAL(ent[0]->name, "."); TEST_ASSERT(S_ISDIR(ent[0]->mode)); TEST_STR_EQUAL(ent[1]->name, ".."); TEST_ASSERT(S_ISDIR(ent[1]->mode)); TEST_STR_EQUAL(ent[2]->name, "file_c0"); TEST_ASSERT(S_ISREG(ent[2]->mode)); TEST_STR_EQUAL(ent[3]->name, "file_c1"); TEST_ASSERT(S_ISREG(ent[3]->mode)); TEST_STR_EQUAL(ent[4]->name, "file_c2"); TEST_ASSERT(S_ISREG(ent[4]->mode)); for (i = 0; i < 5; ++i) free(ent[i]); /* test sub directory iterators */ suba = NULL; subb = NULL; subc = NULL; dir = dir_iterator_create(TEST_PATH); TEST_NOT_NULL(dir); for (i = 0; i < 5; ++i) { ret = dir->next(dir, &dent); TEST_NOT_NULL(dent); TEST_EQUAL_I(ret, 0); if (!strcmp(dent->name, "dira")) { TEST_NULL(suba); ret = dir->open_subdir(dir, &suba); TEST_NOT_NULL(suba); TEST_EQUAL_I(ret, 0); } else if (!strcmp(dent->name, "dirb")) { TEST_NULL(subb); ret = dir->open_subdir(dir, &subb); TEST_NOT_NULL(subb); TEST_EQUAL_I(ret, 0); } else if (!strcmp(dent->name, "dirc")) { TEST_NULL(subc); ret = dir->open_subdir(dir, &subc); TEST_NOT_NULL(subc); TEST_EQUAL_I(ret, 0); } free(dent); } ret = dir->next(dir, &dent); TEST_NULL(dent); TEST_ASSERT(ret > 0); dir = sqfs_drop(dir); TEST_NOT_NULL(suba); TEST_NOT_NULL(subb); TEST_NOT_NULL(subc); /* sub iterator a */ for (i = 0; i < 5; ++i) { ret = suba->next(suba, &ent[i]); TEST_NOT_NULL(ent[0]); TEST_EQUAL_I(ret, 0); if (S_ISREG(ent[i]->mode)) { ret = suba->open_subdir(suba, &sub); TEST_NULL(sub); TEST_EQUAL_I(ret, SQFS_ERROR_NOT_DIR); } } ret = suba->next(suba, &dent); TEST_NULL(dent); TEST_ASSERT(ret > 0); suba = sqfs_drop(suba); qsort(ent, 5, sizeof(ent[0]), compare_entries); TEST_STR_EQUAL(ent[0]->name, "."); TEST_ASSERT(S_ISDIR(ent[0]->mode)); TEST_STR_EQUAL(ent[1]->name, ".."); TEST_ASSERT(S_ISDIR(ent[1]->mode)); TEST_STR_EQUAL(ent[2]->name, "file_a0"); TEST_ASSERT(S_ISREG(ent[2]->mode)); TEST_STR_EQUAL(ent[3]->name, "file_a1"); TEST_ASSERT(S_ISREG(ent[3]->mode)); TEST_STR_EQUAL(ent[4]->name, "file_a2"); TEST_ASSERT(S_ISREG(ent[4]->mode)); for (i = 0; i < 5; ++i) free(ent[i]); /* sub iterator b */ for (i = 0; i < 6; ++i) { ret = subb->next(subb, &ent[i]); TEST_NOT_NULL(ent[0]); TEST_EQUAL_I(ret, 0); if (S_ISREG(ent[i]->mode)) { ret = subb->open_subdir(subb, &sub); TEST_NULL(sub); TEST_EQUAL_I(ret, SQFS_ERROR_NOT_DIR); } } ret = subb->next(subb, &dent); TEST_NULL(dent); TEST_ASSERT(ret > 0); subb = sqfs_drop(subb); qsort(ent, 6, sizeof(ent[0]), compare_entries); TEST_STR_EQUAL(ent[0]->name, "."); TEST_ASSERT(S_ISDIR(ent[0]->mode)); TEST_STR_EQUAL(ent[1]->name, ".."); TEST_ASSERT(S_ISDIR(ent[1]->mode)); TEST_STR_EQUAL(ent[2]->name, "dirx"); TEST_ASSERT(S_ISDIR(ent[2]->mode)); TEST_STR_EQUAL(ent[3]->name, "file_b0"); TEST_ASSERT(S_ISREG(ent[3]->mode)); TEST_STR_EQUAL(ent[4]->name, "file_b1"); TEST_ASSERT(S_ISREG(ent[4]->mode)); TEST_STR_EQUAL(ent[5]->name, "file_b2"); TEST_ASSERT(S_ISREG(ent[5]->mode)); for (i = 0; i < 6; ++i) free(ent[i]); /* sub iterator c */ for (i = 0; i < 5; ++i) { ret = subc->next(subc, &ent[i]); TEST_NOT_NULL(ent[0]); TEST_EQUAL_I(ret, 0); if (S_ISREG(ent[i]->mode)) { ret = subc->open_subdir(subc, &sub); TEST_NULL(sub); TEST_EQUAL_I(ret, SQFS_ERROR_NOT_DIR); } } ret = subc->next(subc, &dent); TEST_NULL(dent); TEST_ASSERT(ret > 0); subc = sqfs_drop(subc); qsort(ent, 5, sizeof(ent[0]), compare_entries); TEST_STR_EQUAL(ent[0]->name, "."); TEST_ASSERT(S_ISDIR(ent[0]->mode)); TEST_STR_EQUAL(ent[1]->name, ".."); TEST_ASSERT(S_ISDIR(ent[1]->mode)); TEST_STR_EQUAL(ent[2]->name, "file_c0"); TEST_ASSERT(S_ISREG(ent[2]->mode)); TEST_STR_EQUAL(ent[3]->name, "file_c1"); TEST_ASSERT(S_ISREG(ent[3]->mode)); TEST_STR_EQUAL(ent[4]->name, "file_c2"); TEST_ASSERT(S_ISREG(ent[4]->mode)); for (i = 0; i < 5; ++i) free(ent[i]); return EXIT_SUCCESS; }