#include #include #include #include #include #include #include #include static bool verbose = false; #define verbosef(...) do { if (verbose) printf(__VA_ARGS__); } while (0) #define eprintf(fmt, ...) fprintf(stderr, "iochk: "fmt"\n" __VA_OPT__(,) __VA_ARGS__) void check(handle_t h) { const size_t buflen = 4096; const size_t offsets[] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, }; char *buflast = malloc(buflen); char *bufcur = malloc(buflen); if (!buflast || !bufcur) { eprintf("out of memory"); goto end; } long offlast = 0; long retlast = _syscall_read(h, buflast, buflen, offlast); if (retlast < 0) { eprintf("error %d when reading at offset %d", retlast, offlast); goto end; } for (size_t i = 0; i < sizeof(offsets) / sizeof(offsets[0]); i++) { char *tmp; long offcur = offsets[i]; long diff = offcur - offlast; assert(diff >= 0); if (retlast < diff) break; long retcur = _syscall_read(h, bufcur, buflen, offcur); if (retcur < 0) { eprintf("error %d when reading at offset %d", retlast, offcur); break; } if (retcur < retlast + offlast - offcur) { verbosef("warn: unexpected ret %d < %d + %d - %d\n", retcur, retlast, offlast, offcur); } if (memcmp(buflast + diff, bufcur, retlast - diff)) { eprintf("unconsistent read from offsets %d and %d", offlast, offcur); } offlast = offcur; retlast = retcur; tmp = bufcur; bufcur = buflast; buflast = tmp; } // TODO check negative offsets end: free(buflast); free(bufcur); } int main(int argc, char **argv) { int c; while ((c = getopt(argc, argv, "v")) != -1) { switch (c) { case 'v': verbose = true; break; default: return 1; } } if (optind >= argc) { eprintf("no files given"); return 1; } for (; optind < argc; optind++) { const char *path = argv[optind]; verbosef("checking %s...\n", path); handle_t h = camellia_open(path, OPEN_READ); if (h < 0) { eprintf("couldn't open %s", path); continue; } check(h); close(h); } return 0; }