#define NUMPRINTCOLUMNS 32     /* # columns of data to print on each line */
 
+/* Operation flags */
+
+enum opflags { FL_NONE = 0, FL_SKIPPED = 1, FL_CLOSE_OPEN = 2, FL_KEEP_SIZE = 4 };
+
 /*
  *     A log entry is an operation and a bunch of arguments.
  */
 struct log_entry {
        int     operation;
        int     args[3];
+       enum opflags flags;
 };
 
 #define        LOGSIZE 10000
  *
  * When mapped read/writes are disabled, they are simply converted to normal
  * reads and writes. When fallocate/fpunch calls are disabled, they are
- * converted to OP_SKIPPED. Hence OP_SKIPPED needs to have a number higher than
- * the operation selction matrix, as does the OP_CLOSEOPEN which is an
- * operation modifier rather than an operation in itself.
+ * skipped.
  *
  * Because of the "lite" version, we also need to have different "maximum
  * operation" defines to allow the ops to be selected correctly based on the
 #define OP_INSERT_RANGE        9
 #define OP_MAX_FULL            10
 
-/* operation modifiers */
-#define OP_CLOSEOPEN   100
-#define OP_SKIPPED     101
-
 #undef PAGE_SIZE
 #define PAGE_SIZE       getpagesize()
 #undef PAGE_MASK
 
 
 void
-log4(int operation, int arg0, int arg1, int arg2)
+log4(int operation, int arg0, int arg1, enum opflags flags)
 {
        struct log_entry *le;
 
        le = &oplog[logptr];
        le->operation = operation;
        if (closeopen)
-               le->operation = ~ le->operation;
+               flags |= FL_CLOSE_OPEN;
        le->args[0] = arg0;
        le->args[1] = arg1;
-       le->args[2] = arg2;
+       le->args[2] = file_size;
+       le->flags = flags;
        logptr++;
        logcount++;
        if (logptr >= LOGSIZE)
 {
        int     i, count, down;
        struct log_entry        *lp;
-       char *falloc_type[3] = {"PAST_EOF", "EXTENDING", "INTERIOR"};
 
        prt("LOG DUMP (%d total operations):\n", logcount);
        if (logcount < LOGSIZE) {
                opnum = i+1 + (logcount/LOGSIZE)*LOGSIZE;
                prt("%d(%3d mod 256): ", opnum, opnum%256);
                lp = &oplog[i];
-               if ((closeopen = lp->operation < 0))
-                       lp->operation = ~ lp->operation;
-                       
+
+               if (lp->flags & FL_SKIPPED) {
+                       prt("SKIPPED (no operation)");
+                       goto skipped;
+               }
+
                switch (lp->operation) {
                case OP_MAPREAD:
                        prt("MAPREAD  0x%x thru 0x%x\t(0x%x bytes)",
                                prt("\t***WWWW");
                        break;
                case OP_TRUNCATE:
-                       down = lp->args[0] < lp->args[1];
+                       down = lp->args[1] < lp->args[2];
                        prt("TRUNCATE %s\tfrom 0x%x to 0x%x",
-                           down ? "DOWN" : "UP", lp->args[1], lp->args[0]);
-                       if (badoff >= lp->args[!down] &&
-                           badoff < lp->args[!!down])
+                           down ? "DOWN" : "UP", lp->args[2], lp->args[1]);
+                       if (badoff >= lp->args[1 + !down] &&
+                           badoff < lp->args[1 + !!down])
                                prt("\t******WWWW");
                        break;
                case OP_FALLOCATE:
                        /* 0: offset 1: length 2: where alloced */
-                       prt("FALLOC   0x%x thru 0x%x\t(0x%x bytes) %s",
+                       prt("FALLOC   0x%x thru 0x%x\t(0x%x bytes) ",
                                lp->args[0], lp->args[0] + lp->args[1],
-                               lp->args[1], falloc_type[lp->args[2]]);
+                               lp->args[1]);
+                       if (lp->args[0] + lp->args[1] <= lp->args[2])
+                               prt("INTERIOR");
+                       else if (lp->flags & FL_KEEP_SIZE)
+                               prt("PAST_EOF");
+                       else
+                               prt("EXTENDING");
                        if (badoff >= lp->args[0] &&
                            badoff < lp->args[0] + lp->args[1])
                                prt("\t******FFFF");
                                                     lp->args[0] + lp->args[1])
                                prt("\t******IIII");
                        break;
-               case OP_SKIPPED:
-                       prt("SKIPPED (no operation)");
-                       break;
                default:
                        prt("BOGUS LOG ENTRY (operation code = %d)!",
                            lp->operation);
                }
-               if (closeopen)
+           skipped:
+               if (lp->flags & FL_CLOSE_OPEN)
                        prt("\n\t\tCLOSE/OPEN");
                prt("\n");
                i++;
        if (size == 0) {
                if (!quiet && testcalls > simulatedopcount && !o_direct)
                        prt("skipping zero size read\n");
-               log4(OP_SKIPPED, OP_READ, offset, size);
+               log4(OP_READ, offset, size, FL_SKIPPED);
                return;
        }
        if (size + offset > file_size) {
                if (!quiet && testcalls > simulatedopcount)
                        prt("skipping seek/read past end of file\n");
-               log4(OP_SKIPPED, OP_READ, offset, size);
+               log4(OP_READ, offset, size, FL_SKIPPED);
                return;
        }
 
-       log4(OP_READ, offset, size, 0);
+       log4(OP_READ, offset, size, FL_NONE);
 
        if (testcalls <= simulatedopcount)
                return;
        if (size == 0) {
                if (!quiet && testcalls > simulatedopcount)
                        prt("skipping zero size read\n");
-               log4(OP_SKIPPED, OP_MAPREAD, offset, size);
+               log4(OP_MAPREAD, offset, size, FL_SKIPPED);
                return;
        }
        if (size + offset > file_size) {
                if (!quiet && testcalls > simulatedopcount)
                        prt("skipping seek/read past end of file\n");
-               log4(OP_SKIPPED, OP_MAPREAD, offset, size);
+               log4(OP_MAPREAD, offset, size, FL_SKIPPED);
                return;
        }
 
-       log4(OP_MAPREAD, offset, size, 0);
+       log4(OP_MAPREAD, offset, size, FL_NONE);
 
        if (testcalls <= simulatedopcount)
                return;
        if (size == 0) {
                if (!quiet && testcalls > simulatedopcount && !o_direct)
                        prt("skipping zero size write\n");
-               log4(OP_SKIPPED, OP_WRITE, offset, size);
+               log4(OP_WRITE, offset, size, FL_SKIPPED);
                return;
        }
 
-       log4(OP_WRITE, offset, size, file_size);
+       log4(OP_WRITE, offset, size, FL_NONE);
 
        gendata(original_buf, good_buf, offset, size);
        if (file_size < offset + size) {
        if (size == 0) {
                if (!quiet && testcalls > simulatedopcount)
                        prt("skipping zero size write\n");
-               log4(OP_SKIPPED, OP_MAPWRITE, offset, size);
+               log4(OP_MAPWRITE, offset, size, FL_SKIPPED);
                return;
        }
        cur_filesize = file_size;
 
-       log4(OP_MAPWRITE, offset, size, 0);
+       log4(OP_MAPWRITE, offset, size, FL_NONE);
 
        gendata(original_buf, good_buf, offset, size);
        if (file_size < offset + size) {
                        prt("truncating to largest ever: 0x%x\n", size);
        }
 
-       log4(OP_TRUNCATE, size, (unsigned)file_size, 0);
+       log4(OP_TRUNCATE, 0, size, FL_NONE);
 
        if (size > file_size)
                memset(good_buf + file_size, '\0', size - file_size);
        if (length == 0) {
                if (!quiet && testcalls > simulatedopcount)
                        prt("skipping zero length punch hole\n");
-                       log4(OP_SKIPPED, OP_PUNCH_HOLE, offset, length);
+                       log4(OP_PUNCH_HOLE, offset, length, FL_SKIPPED);
                return;
        }
 
        if (file_size <= (loff_t)offset) {
                if (!quiet && testcalls > simulatedopcount)
                        prt("skipping hole punch off the end of the file\n");
-                       log4(OP_SKIPPED, OP_PUNCH_HOLE, offset, length);
+                       log4(OP_PUNCH_HOLE, offset, length, FL_SKIPPED);
                return;
        }
 
        end_offset = offset + length;
 
-       log4(OP_PUNCH_HOLE, offset, length, 0);
+       log4(OP_PUNCH_HOLE, offset, length, FL_NONE);
 
        if (testcalls <= simulatedopcount)
                return;
        if (length == 0) {
                if (!quiet && testcalls > simulatedopcount)
                        prt("skipping zero length zero range\n");
-                       log4(OP_SKIPPED, OP_ZERO_RANGE, offset, length);
+                       log4(OP_ZERO_RANGE, offset, length, FL_SKIPPED |
+                            (keep_size ? FL_KEEP_SIZE : FL_NONE));
                return;
        }
 
         *      1: extending prealloc
         *      2: interior prealloc
         */
-       log4(OP_ZERO_RANGE, offset, length, (end_offset > file_size) ? (keep_size ? 0 : 1) : 2);
+       log4(OP_ZERO_RANGE, offset, length,
+            keep_size ? FL_KEEP_SIZE : FL_NONE);
 
        if (testcalls <= simulatedopcount)
                return;
        if (length == 0) {
                if (!quiet && testcalls > simulatedopcount)
                        prt("skipping zero length collapse range\n");
-               log4(OP_SKIPPED, OP_COLLAPSE_RANGE, offset, length);
+               log4(OP_COLLAPSE_RANGE, offset, length, FL_SKIPPED);
                return;
        }
 
        if ((loff_t)end_offset >= file_size) {
                if (!quiet && testcalls > simulatedopcount)
                        prt("skipping collapse range behind EOF\n");
-               log4(OP_SKIPPED, OP_COLLAPSE_RANGE, offset, length);
+               log4(OP_COLLAPSE_RANGE, offset, length, FL_SKIPPED);
                return;
        }
 
-       log4(OP_COLLAPSE_RANGE, offset, length, 0);
+       log4(OP_COLLAPSE_RANGE, offset, length, FL_NONE);
 
        if (testcalls <= simulatedopcount)
                return;
        if (length == 0) {
                if (!quiet && testcalls > simulatedopcount)
                        prt("skipping zero length insert range\n");
-               log4(OP_SKIPPED, OP_INSERT_RANGE, offset, length);
+               log4(OP_INSERT_RANGE, offset, length, FL_SKIPPED);
                return;
        }
 
        if ((loff_t)offset >= file_size) {
                if (!quiet && testcalls > simulatedopcount)
                        prt("skipping insert range behind EOF\n");
-               log4(OP_SKIPPED, OP_INSERT_RANGE, offset, length);
+               log4(OP_INSERT_RANGE, offset, length, FL_SKIPPED);
                return;
        }
 
-       log4(OP_INSERT_RANGE, offset, length, 0);
+       log4(OP_INSERT_RANGE, offset, length, FL_NONE);
 
        if (testcalls <= simulatedopcount)
                return;
         if (length == 0) {
                 if (!quiet && testcalls > simulatedopcount)
                         prt("skipping zero length fallocate\n");
-                log4(OP_SKIPPED, OP_FALLOCATE, offset, length);
+                log4(OP_FALLOCATE, offset, length, FL_SKIPPED |
+                    (keep_size ? FL_KEEP_SIZE : FL_NONE));
                 return;
         }
 
         *      1: extending prealloc
         *      2: interior prealloc
         */
-       log4(OP_FALLOCATE, offset, length, (end_offset > file_size) ? (keep_size ? 0 : 1) : 2);
+       log4(OP_FALLOCATE, offset, length,
+            keep_size ? FL_KEEP_SIZE : FL_NONE);
 
        if (end_offset > file_size) {
                memset(good_buf + file_size, '\0', end_offset - file_size);
                break;
        case OP_FALLOCATE:
                if (!fallocate_calls) {
-                       log4(OP_SKIPPED, OP_FALLOCATE, offset, size);
+                       log4(OP_FALLOCATE, offset, size, FL_SKIPPED);
                        goto out;
                }
                break;
        case OP_PUNCH_HOLE:
                if (!punch_hole_calls) {
-                       log4(OP_SKIPPED, OP_PUNCH_HOLE, offset, size);
+                       log4(OP_PUNCH_HOLE, offset, size, FL_SKIPPED);
                        goto out;
                }
                break;
        case OP_ZERO_RANGE:
                if (!zero_range_calls) {
-                       log4(OP_SKIPPED, OP_ZERO_RANGE, offset, size);
+                       log4(OP_ZERO_RANGE, offset, size, FL_SKIPPED);
                        goto out;
                }
                break;
        case OP_COLLAPSE_RANGE:
                if (!collapse_range_calls) {
-                       log4(OP_SKIPPED, OP_COLLAPSE_RANGE, offset, size);
+                       log4(OP_COLLAPSE_RANGE, offset, size, FL_SKIPPED);
                        goto out;
                }
                break;
        case OP_INSERT_RANGE:
                if (!insert_range_calls) {
-                       log4(OP_SKIPPED, OP_INSERT_RANGE, offset, size);
+                       log4(OP_INSERT_RANGE, offset, size, FL_SKIPPED);
                        goto out;
                }
                break;
                offset = offset & ~(block_size - 1);
                size = size & ~(block_size - 1);
                if (size == 0) {
-                       log4(OP_SKIPPED, OP_COLLAPSE_RANGE, offset, size);
+                       log4(OP_COLLAPSE_RANGE, offset, size, FL_SKIPPED);
                        goto out;
                }
                do_collapse_range(offset, size);
                offset = offset & ~(block_size - 1);
                size = size & ~(block_size - 1);
                if (size == 0) {
-                       log4(OP_SKIPPED, OP_INSERT_RANGE, offset, size);
+                       log4(OP_INSERT_RANGE, offset, size, FL_SKIPPED);
                        goto out;
                }
                if (file_size + size > maxfilelen) {
-                       log4(OP_SKIPPED, OP_INSERT_RANGE, offset, size);
+                       log4(OP_INSERT_RANGE, offset, size, FL_SKIPPED);
                        goto out;
                }