/* * GPL HEADER START * * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 only, * as published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License version 2 for more details (a copy is included * in the LICENSE file that accompanied this code). * * You should have received a copy of the GNU General Public License * version 2 along with this program; If not, see * http://www.gnu.org/licenses/gpl-2.0.html * * GPL HEADER END */ /* * Copyright (c) 2004, 2010, Oracle and/or its affiliates. All rights reserved. * Use is subject to license terms. * * Copyright (c) 2011, 2014, Intel Corporation. */ /* * This file is part of Lustre, http://www.lustre.org/ * Lustre is a trademark of Sun Microsystems, Inc. * * lustre/tests/ll_dirstripe_verify.c * * ll_dirstripe_verify : * - to verify if the file has the same lov_user_md setting as the parent dir. * - if dir's offset is set -1, ll_dirstripe_verify * is used to further verify if file1 and file2's obdidx is continuous. */ #include #include #include #include #include #include #include #include #include #include #include #define MAX_LOV_UUID_COUNT 1000 /* Returns bytes read on success and a negative value on failure. * If zero bytes are read it will be treated as failure as such * zero cannot be returned from this function. */ int read_proc_entry(char *proc_path, char *buf, int len) { int rc, fd; memset(buf, 0, len); fd = open(proc_path, O_RDONLY); if (fd < 0) { llapi_error(LLAPI_MSG_ERROR, -errno, "cannot open '%s'", proc_path); return -2; } rc = read(fd, buf, len - 1); if (rc < 0) { llapi_error(LLAPI_MSG_ERROR, -errno, "error reading from '%s'", proc_path); rc = -3; } else if (rc == 0) { llapi_err_noerrno(LLAPI_MSG_ERROR, "read zero bytes from '%s'", proc_path); rc = -4; } else if (buf[rc - 1] == '\n') { buf[rc - 1] = '\0'; /* Remove trailing newline */ } close(fd); return rc; } int compare(struct obd_uuid *puuid, struct lov_user_md *lum_dir, struct lov_user_md *lum_file1, struct lov_user_md *lum_file2) { int stripe_count = 0, min_stripe_count = 0, def_stripe_count = 1; int stripe_size = 0; int stripe_offset = -1; int ost_count; char buf[128]; glob_t path; int i; if (cfs_get_param_paths(&path, "lov/%s/stripecount", puuid->uuid) != 0) return 2; if (read_proc_entry(path.gl_pathv[0], buf, sizeof(buf)) < 0) { cfs_free_param_data(&path); return 5; } cfs_free_param_data(&path); def_stripe_count = (short)atoi(buf); if (cfs_get_param_paths(&path, "lov/%s/numobd", puuid->uuid) != 0) return 2; if (read_proc_entry(path.gl_pathv[0], buf, sizeof(buf)) < 0) { cfs_free_param_data(&path); return 6; } cfs_free_param_data(&path); ost_count = atoi(buf); if (lum_dir == NULL) { stripe_count = def_stripe_count; min_stripe_count = -1; } else { stripe_count = (signed short)lum_dir->lmm_stripe_count; printf("dir stripe %d, ", stripe_count); min_stripe_count = 1; } printf("default stripe %d, ost count %d\n", def_stripe_count, ost_count); if (stripe_count == 0) { min_stripe_count = -1; stripe_count = 1; } stripe_count = (stripe_count > 0 && stripe_count <= ost_count) ? stripe_count : ost_count; min_stripe_count = min_stripe_count > 0 ? stripe_count : ((stripe_count + 1) / 2); if (lum_file1->lmm_stripe_count != stripe_count || lum_file1->lmm_stripe_count < min_stripe_count) { llapi_err_noerrno(LLAPI_MSG_ERROR, "file1 stripe count %d != dir %d\n", lum_file1->lmm_stripe_count, stripe_count); return 7; } if (lum_file1->lmm_stripe_count < stripe_count) llapi_err_noerrno(LLAPI_MSG_WARN, "warning: file1 used fewer stripes" " %d < dir %d (likely due to bug 4900)\n", lum_file1->lmm_stripe_count, stripe_count); if (lum_dir != NULL) stripe_size = (int)lum_dir->lmm_stripe_size; if (stripe_size == 0) { if (cfs_get_param_paths(&path, "lov/%s/stripesize", puuid->uuid) != 0) return 2; if (read_proc_entry(path.gl_pathv[0], buf, sizeof(buf)) < 0) { cfs_free_param_data(&path); return 5; } cfs_free_param_data(&path); stripe_size = atoi(buf); } if (lum_file1->lmm_stripe_size != stripe_size) { llapi_err_noerrno(LLAPI_MSG_ERROR, "file1 stripe size %d != dir %d\n", lum_file1->lmm_stripe_size, stripe_size); return 8; } if (lum_dir != NULL) stripe_offset = (short int)lum_dir->lmm_stripe_offset; if (stripe_offset != -1) { for (i = 0; i < stripe_count; i++) if (lum_file1->lmm_objects[i].l_ost_idx != (stripe_offset + i) % ost_count) { llapi_err_noerrno(LLAPI_MSG_WARN, "warning: file1 non-sequential " "stripe[%d] %d != %d\n", i, lum_file1->lmm_objects[i].l_ost_idx, (stripe_offset + i) % ost_count); } } else if (lum_file2 != NULL) { int next, idx, stripe = stripe_count - 1; next = (lum_file1->lmm_objects[stripe].l_ost_idx + 1) % ost_count; idx = lum_file2->lmm_objects[0].l_ost_idx; if (idx != next) { llapi_err_noerrno(LLAPI_MSG_WARN, "warning: non-sequential " "file1 stripe[%d] %d != file2 stripe[0] %d\n", stripe, lum_file1->lmm_objects[stripe].l_ost_idx, idx); } } return 0; } int main(int argc, char **argv) { struct lov_user_md *lum_dir, *lum_file1 = NULL, *lum_file2 = NULL; struct obd_uuid uuid; int lum_size, rc; DIR *dir; if (argc < 3) { llapi_err_noerrno(LLAPI_MSG_ERROR, "Usage: %s [filename2]\n", argv[0]); return 1; } dir = opendir(argv[1]); if (dir == NULL) { rc = -errno; llapi_error(LLAPI_MSG_ERROR, rc, "error: %s opendir failed", argv[1]); return rc; } lum_size = lov_user_md_size(MAX_LOV_UUID_COUNT, LOV_USER_MAGIC); lum_dir = (struct lov_user_md *)malloc(lum_size); if (lum_dir == NULL) { rc = -ENOMEM; llapi_error(LLAPI_MSG_ERROR, rc, "error: can't allocate %d bytes " "for dir EA", lum_size); goto cleanup; } rc = llapi_file_get_stripe(argv[1], lum_dir); if (rc) { if (rc == -ENODATA) { free(lum_dir); lum_dir = NULL; } else { llapi_error(LLAPI_MSG_ERROR, rc, "error: can't get EA for %s", argv[1]); goto cleanup; } } /* XXX should be llapi_lov_getname() */ rc = llapi_file_get_lov_uuid(argv[1], &uuid); if (rc) { llapi_error(LLAPI_MSG_ERROR, rc, "error: can't get lov name for %s", argv[1]); return rc; } lum_file1 = malloc(lum_size); if (lum_file1 == NULL) { rc = -ENOMEM; llapi_error(LLAPI_MSG_ERROR, rc, "error: can't allocate %d bytes for EA", lum_size); goto cleanup; } rc = llapi_file_get_stripe(argv[2], lum_file1); if (rc) { llapi_error(LLAPI_MSG_ERROR, rc, "error: unable to get EA for %s", argv[2]); goto cleanup; } if (argc == 4) { lum_file2 = (struct lov_user_md *)malloc(lum_size); if (lum_file2 == NULL) { rc = -ENOMEM; llapi_error(LLAPI_MSG_ERROR, rc, "error: can't allocate %d " "bytes for file2 EA", lum_size); goto cleanup; } rc = llapi_file_get_stripe(argv[3], lum_file2); if (rc) { llapi_error(LLAPI_MSG_ERROR, rc, "error: can't get EA for %s", argv[3]); goto cleanup; } } rc = compare(&uuid, lum_dir, lum_file1, lum_file2); cleanup: closedir(dir); if (lum_dir != NULL) free(lum_dir); if (lum_file1 != NULL) free(lum_file1); if (lum_file2 != NULL) free(lum_file2); return rc; }