1 #include "../common/tdb_private.h"
3 /* Speed up the tests, but do the actual sync tests. */
4 static unsigned int sync_counts
= 0;
5 static inline int fake_fsync(int fd
)
10 #define fsync fake_fsync
13 static inline int fake_msync(void *addr
, size_t length
, int flags
)
18 #define msync fake_msync
22 static inline int fake_fdatasync(int fd
)
27 #define fdatasync fake_fdatasync
30 #include "../common/io.c"
31 #include "../common/tdb.c"
32 #include "../common/lock.c"
33 #include "../common/freelist.c"
34 #include "../common/traverse.c"
35 #include "../common/transaction.c"
36 #include "../common/error.c"
37 #include "../common/open.c"
38 #include "../common/check.c"
39 #include "../common/hash.c"
40 #include "tap-interface.h"
44 static void write_record(struct tdb_context
*tdb
, size_t extra_len
,
48 key
.dsize
= strlen("hi");
49 key
.dptr
= (void *)"hi";
51 data
->dsize
+= extra_len
;
52 tdb_transaction_start(tdb
);
53 tdb_store(tdb
, key
, *data
, TDB_REPLACE
);
54 tdb_transaction_commit(tdb
);
57 int main(int argc
, char *argv
[])
59 struct tdb_context
*tdb
;
62 struct tdb_record rec
;
65 /* Do *not* suppress sync for this test; we do it ourselves. */
66 unsetenv("TDB_NO_FSYNC");
69 tdb
= tdb_open_ex("run-transaction-expand.tdb",
70 1024, TDB_CLEAR_IF_FIRST
,
71 O_CREAT
|O_TRUNC
|O_RDWR
, 0600, &taplogctx
, NULL
);
75 data
.dptr
= calloc(1000, getpagesize());
77 /* Simulate a slowly growing record. */
78 for (i
= 0; i
< 1000; i
++)
79 write_record(tdb
, getpagesize(), &data
);
81 tdb_ofs_read(tdb
, TDB_RECOVERY_HEAD
, &off
);
82 tdb_read(tdb
, off
, &rec
, sizeof(rec
), DOCONV());
83 diag("TDB size = %zu, recovery = %llu-%llu",
84 (size_t)tdb
->map_size
, (unsigned long long)off
, (unsigned long long)(off
+ sizeof(rec
) + rec
.rec_len
));
86 /* We should only be about 5 times larger than largest record. */
87 ok1(tdb
->map_size
< 6 * i
* getpagesize());
90 tdb
= tdb_open_ex("run-transaction-expand.tdb",
91 1024, TDB_CLEAR_IF_FIRST
,
92 O_CREAT
|O_TRUNC
|O_RDWR
, 0600, &taplogctx
, NULL
);
97 /* Simulate a slowly growing record, repacking to keep
98 * recovery area at end. */
99 for (i
= 0; i
< 1000; i
++) {
100 write_record(tdb
, getpagesize(), &data
);
105 tdb_ofs_read(tdb
, TDB_RECOVERY_HEAD
, &off
);
106 tdb_read(tdb
, off
, &rec
, sizeof(rec
), DOCONV());
107 diag("TDB size = %zu, recovery = %llu-%llu",
108 (size_t)tdb
->map_size
, (unsigned long long)off
, (unsigned long long)(off
+ sizeof(rec
) + rec
.rec_len
));
110 /* We should only be about 4 times larger than largest record. */
111 ok1(tdb
->map_size
< 5 * i
* getpagesize());
113 /* We should have synchronized multiple times. */
118 return exit_status();