/* * This file and its contents are supplied under the terms of the * Common Development and Distribution License ("CDDL"), version 1.0. * You may only use this file in accordance with the terms of version * 1.0 of the CDDL. * * A full copy of the text of the CDDL should have accompanied this * source. A copy of the CDDL is also available via the Internet at * http://www.illumos.org/license/CDDL. */ /* * Copyright (c) 2018 by Delphix. All rights reserved. */ #include <stdio.h> #include <fcntl.h> #include <unistd.h> #include <stdlib.h> #include <string.h> #include <assert.h> #include <sys/param.h> #include <sys/types.h> #include <sys/stat.h> #include <errno.h> static void usage(const char *msg, int exit_value) { (void) fprintf(stderr, "usage: get_diff file redacted_file\n%s\n", msg); exit(exit_value); } /* * This utility compares two files, an original and its redacted counterpart * (in that order). It compares the files 512 bytes at a time, printing out * any ranges (as offset and length) where the redacted file does not match * the original. This output is used to verify that the expected ranges of * a redacted file do not contain the original data. */ int main(int argc, char *argv[]) { off_t diff_off = 0, diff_len = 0, off = 0; int fd1, fd2; char *fname1, *fname2; char buf1[DEV_BSIZE], buf2[DEV_BSIZE]; ssize_t bytes; if (argc != 3) usage("Incorrect number of arguments.", 1); if ((fname1 = argv[1]) == NULL) usage("Filename missing.", 1); if ((fd1 = open(fname1, O_LARGEFILE | O_RDONLY)) < 0) { perror("open1 failed"); exit(1); } if ((fname2 = argv[2]) == NULL) usage("Redacted filename missing.", 1); if ((fd2 = open(fname2, O_LARGEFILE | O_RDONLY)) < 0) { perror("open2 failed"); exit(1); } while ((bytes = pread(fd1, buf1, DEV_BSIZE, off)) > 0) { if (pread(fd2, buf2, DEV_BSIZE, off) < 0) { if (errno == EIO) { /* * A read in a redacted section of a file will * fail with EIO. If we get EIO, continue on * but ensure that a comparison of buf1 and * buf2 will fail, indicating a redacted block. */ buf2[0] = ~buf1[0]; } else { perror("pread failed"); exit(1); } } if (memcmp(buf1, buf2, bytes) == 0) { if (diff_len != 0) { (void) fprintf(stdout, "%lld,%lld\n", (long long)diff_off, (long long)diff_len); assert(off == diff_off + diff_len); diff_len = 0; } diff_off = 0; } else { if (diff_len == 0) diff_off = off; assert(off == diff_off + diff_len); diff_len += bytes; } off += bytes; } if (diff_len != 0 && diff_len != 0) { (void) fprintf(stdout, "%lld,%lld\n", (long long)diff_off, (long long)diff_len); } (void) close(fd1); (void) close(fd2); return (0); }