1 /* mtest.c - memory-mapped database tester/toy */
3 * Copyright 2011 Howard Chu, Symas Corp.
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted only as authorized by the OpenLDAP
10 * A copy of this license is available in the file LICENSE in the
11 * top-level directory of the distribution or, alternatively, at
12 * <http://www.OpenLDAP.org/license.html>.
14 #define _XOPEN_SOURCE 500 /* srandom(), random() */
20 int main(int argc,char * argv[])
28 MDB_cursor *cursor, *cur2;
35 count = (random()%384) + 64;
36 values = (int *)malloc(count*sizeof(int));
38 for(i = 0;i<count;i++) {
39 values[i] = random()%1024;
42 rc = mdbenv_create(&env);
43 rc = mdbenv_set_mapsize(env, 10485760);
44 rc = mdbenv_open(env, "./testdb", MDB_FIXEDMAP|MDB_NOSYNC, 0664);
45 rc = mdb_txn_begin(env, 0, &txn);
46 rc = mdb_open(txn, NULL, 0, &dbi);
48 key.mv_size = sizeof(int);
50 data.mv_size = sizeof(sval);
53 printf("Adding %d values\n", count);
54 for (i=0;i<count;i++) {
55 sprintf(sval, "%03x %d foo bar", values[i], values[i]);
56 rc = mdb_put(txn, dbi, &key, &data, MDB_NOOVERWRITE);
59 if (j) printf("%d duplicates skipped\n", j);
60 rc = mdb_txn_commit(txn);
61 rc = mdbenv_stat(env, &mst);
63 rc = mdb_txn_begin(env, 1, &txn);
64 rc = mdb_cursor_open(txn, dbi, &cursor);
65 while ((rc = mdb_cursor_get(cursor, &key, &data, MDB_NEXT)) == 0) {
66 printf("key: %p %.*s, data: %p %.*s\n",
67 key.mv_data, (int) key.mv_size, (char *) key.mv_data,
68 data.mv_data, (int) data.mv_size, (char *) data.mv_data);
70 mdb_cursor_close(cursor);
75 for (i= count - 1; i > -1; i-= (random()%5)) {
78 rc = mdb_txn_begin(env, 0, &txn);
79 sprintf(sval, "%03x ", values[i]);
80 rc = mdb_del(txn, dbi, &key, NULL);
85 rc = mdb_txn_commit(txn);
89 printf("Deleted %d values\n", j);
91 rc = mdbenv_stat(env, &mst);
92 rc = mdb_txn_begin(env, 1, &txn);
93 rc = mdb_cursor_open(txn, dbi, &cursor);
94 while ((rc = mdb_cursor_get(cursor, &key, &data, MDB_NEXT)) == 0) {
95 printf("key: %.*s, data: %.*s\n",
96 (int) key.mv_size, (char *) key.mv_data,
97 (int) data.mv_size, (char *) data.mv_data);
100 /* write ops aren't coordinated with cursors,
101 * this stuff all breaks
103 printf("Deleting with cursor\n");
104 rc = mdb_txn_begin(env, 0, &txn);
105 rc = mdb_cursor_open(db, txn, &cur2);
106 for (i=0; i<50; i++) {
107 rc = mdb_cursor_get(cur2, &key, &data, MDB_NEXT);
108 printf("key: %p %.*s, data: %p %.*s\n",
109 key.mv_data, (int) key.mv_size, (char *) key.mv_data,
110 data.mv_data, (int) data.mv_size, (char *) data.mv_data);
111 rc = mdb_del(db, txn, &key, NULL);
114 printf("Restarting cursor in txn\n");
115 rc = mdb_cursor_get(cur2, &key, &data, MDB_FIRST);
116 printf("key: %p %.*s, data: %p %.*s\n",
117 key.mv_data, (int) key.mv_size, (char *) key.mv_data,
118 data.mv_data, (int) data.mv_size, (char *) data.mv_data);
119 for (i=0; i<32; i++) {
120 rc = mdb_cursor_get(cur2, &key, &data, MDB_NEXT);
121 printf("key: %p %.*s, data: %p %.*s\n",
122 key.mv_data, (int) key.mv_size, (char *) key.mv_data,
123 data.mv_data, (int) data.mv_size, (char *) data.mv_data);
125 mdb_cursor_close(cur2);
126 rc = mdb_txn_commit(txn);
127 mdb_cursor_close(cursor);
128 printf("Restarting cursor outside txn\n");
129 rc = mdb_cursor_open(db, NULL, &cursor);
130 rc = mdb_cursor_get(cursor, &key, &data, MDB_FIRST);
131 printf("key: %p %.*s, data: %p %.*s\n",
132 key.mv_data, (int) key.mv_size, (char *) key.mv_data,
133 data.mv_data, (int) data.mv_size, (char *) data.mv_data);
134 for (i=0; i<32; i++) {
135 rc = mdb_cursor_get(cursor, &key, &data, MDB_NEXT);
136 printf("key: %p %.*s, data: %p %.*s\n",
137 key.mv_data, (int) key.mv_size, (char *) key.mv_data,
138 data.mv_data, (int) data.mv_size, (char *) data.mv_data);
141 mdb_cursor_close(cursor);