speed.c 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247
  1. /* Simple speed test for TDB */
  2. #include <err.h>
  3. #include <time.h>
  4. #include <sys/types.h>
  5. #include <sys/stat.h>
  6. #include <unistd.h>
  7. #include <sys/time.h>
  8. #include <fcntl.h>
  9. #include <stdio.h>
  10. #include <stdlib.h>
  11. #include <string.h>
  12. #include <stdbool.h>
  13. #include <ccan/tdb/tdb.h>
  14. /* Nanoseconds per operation */
  15. static size_t normalize(const struct timeval *start,
  16. const struct timeval *stop,
  17. unsigned int num)
  18. {
  19. struct timeval diff;
  20. timersub(stop, start, &diff);
  21. /* Floating point is more accurate here. */
  22. return (double)(diff.tv_sec * 1000000 + diff.tv_usec)
  23. / num * 1000;
  24. }
  25. static size_t file_size(void)
  26. {
  27. struct stat st;
  28. if (stat("/tmp/speed.tdb", &st) != 0)
  29. return -1;
  30. return st.st_size;
  31. }
  32. static int count_record(struct tdb_context *tdb,
  33. TDB_DATA key, TDB_DATA data, void *p)
  34. {
  35. int *total = p;
  36. *total += *(int *)data.dptr;
  37. return 0;
  38. }
  39. int main(int argc, char *argv[])
  40. {
  41. unsigned int i, j, num = 1000, stage = 0, stopat = -1;
  42. int flags = TDB_DEFAULT;
  43. TDB_DATA key, data;
  44. struct tdb_context *tdb;
  45. struct timeval start, stop;
  46. bool transaction = false;
  47. if (argv[1] && strcmp(argv[1], "--internal") == 0) {
  48. flags = TDB_INTERNAL;
  49. argc--;
  50. argv++;
  51. }
  52. if (argv[1] && strcmp(argv[1], "--transaction") == 0) {
  53. transaction = true;
  54. argc--;
  55. argv++;
  56. }
  57. tdb = tdb_open("/tmp/speed.tdb", 100003, flags, O_RDWR|O_CREAT|O_TRUNC,
  58. 0600);
  59. if (!tdb)
  60. err(1, "Opening /tmp/speed.tdb");
  61. key.dptr = (void *)&i;
  62. key.dsize = sizeof(i);
  63. data = key;
  64. if (argv[1]) {
  65. num = atoi(argv[1]);
  66. argv++;
  67. argc--;
  68. }
  69. if (argv[1]) {
  70. stopat = atoi(argv[1]);
  71. argv++;
  72. argc--;
  73. }
  74. if (transaction && tdb_transaction_start(tdb))
  75. errx(1, "starting transaction: %s", tdb_errorstr(tdb));
  76. /* Add 1000 records. */
  77. printf("Adding %u records: ", num); fflush(stdout);
  78. gettimeofday(&start, NULL);
  79. for (i = 0; i < num; i++)
  80. if (tdb_store(tdb, key, data, TDB_INSERT) != 0)
  81. errx(1, "Inserting key %u in tdb: %s",
  82. i, tdb_errorstr(tdb));
  83. gettimeofday(&stop, NULL);
  84. if (transaction && tdb_transaction_commit(tdb))
  85. errx(1, "committing transaction: %s", tdb_errorstr(tdb));
  86. printf(" %zu ns (%zu bytes)\n",
  87. normalize(&start, &stop, num), file_size());
  88. if (++stage == stopat)
  89. exit(0);
  90. if (transaction && tdb_transaction_start(tdb))
  91. errx(1, "starting transaction: %s", tdb_errorstr(tdb));
  92. /* Finding 1000 records. */
  93. printf("Finding %u records: ", num); fflush(stdout);
  94. gettimeofday(&start, NULL);
  95. for (i = 0; i < num; i++) {
  96. int *dptr;
  97. dptr = (int *)tdb_fetch(tdb, key).dptr;
  98. if (!dptr || *dptr != i)
  99. errx(1, "Fetching key %u in tdb gave %u",
  100. i, dptr ? *dptr : -1);
  101. }
  102. gettimeofday(&stop, NULL);
  103. if (transaction && tdb_transaction_commit(tdb))
  104. errx(1, "committing transaction: %s", tdb_errorstr(tdb));
  105. printf(" %zu ns (%zu bytes)\n",
  106. normalize(&start, &stop, num), file_size());
  107. if (++stage == stopat)
  108. exit(0);
  109. if (transaction && tdb_transaction_start(tdb))
  110. errx(1, "starting transaction: %s", tdb_errorstr(tdb));
  111. /* Missing 1000 records. */
  112. printf("Missing %u records: ", num); fflush(stdout);
  113. gettimeofday(&start, NULL);
  114. for (i = num; i < num*2; i++) {
  115. int *dptr;
  116. dptr = (int *)tdb_fetch(tdb, key).dptr;
  117. if (dptr)
  118. errx(1, "Fetching key %u in tdb gave %u", i, *dptr);
  119. }
  120. gettimeofday(&stop, NULL);
  121. if (transaction && tdb_transaction_commit(tdb))
  122. errx(1, "committing transaction: %s", tdb_errorstr(tdb));
  123. printf(" %zu ns (%zu bytes)\n",
  124. normalize(&start, &stop, num), file_size());
  125. if (++stage == stopat)
  126. exit(0);
  127. if (transaction && tdb_transaction_start(tdb))
  128. errx(1, "starting transaction: %s", tdb_errorstr(tdb));
  129. /* Traverse 1000 records. */
  130. printf("Traversing %u records: ", num); fflush(stdout);
  131. i = 0;
  132. gettimeofday(&start, NULL);
  133. if (tdb_traverse(tdb, count_record, &i) != num)
  134. errx(1, "Traverse returned wrong number of records");
  135. if (i != (num - 1) * (num / 2))
  136. errx(1, "Traverse tallied to %u", i);
  137. gettimeofday(&stop, NULL);
  138. if (transaction && tdb_transaction_commit(tdb))
  139. errx(1, "committing transaction: %s", tdb_errorstr(tdb));
  140. printf(" %zu ns (%zu bytes)\n",
  141. normalize(&start, &stop, num), file_size());
  142. if (++stage == stopat)
  143. exit(0);
  144. if (transaction && tdb_transaction_start(tdb))
  145. errx(1, "starting transaction: %s", tdb_errorstr(tdb));
  146. /* Delete 1000 records (not in order). */
  147. printf("Deleting %u records: ", num); fflush(stdout);
  148. gettimeofday(&start, NULL);
  149. for (j = 0; j < num; j++) {
  150. i = (j + 100003) % num;
  151. if (tdb_delete(tdb, key) != 0)
  152. errx(1, "Deleting key %u in tdb: %s",
  153. i, tdb_errorstr(tdb));
  154. }
  155. gettimeofday(&stop, NULL);
  156. if (transaction && tdb_transaction_commit(tdb))
  157. errx(1, "committing transaction: %s", tdb_errorstr(tdb));
  158. printf(" %zu ns (%zu bytes)\n",
  159. normalize(&start, &stop, num), file_size());
  160. if (++stage == stopat)
  161. exit(0);
  162. if (transaction && tdb_transaction_start(tdb))
  163. errx(1, "starting transaction: %s", tdb_errorstr(tdb));
  164. /* Re-add 1000 records (not in order). */
  165. printf("Re-adding %u records: ", num); fflush(stdout);
  166. gettimeofday(&start, NULL);
  167. for (j = 0; j < num; j++) {
  168. i = (j + 100003) % num;
  169. if (tdb_store(tdb, key, data, TDB_INSERT) != 0)
  170. errx(1, "Inserting key %u in tdb: %s",
  171. i, tdb_errorstr(tdb));
  172. }
  173. gettimeofday(&stop, NULL);
  174. if (transaction && tdb_transaction_commit(tdb))
  175. errx(1, "committing transaction: %s", tdb_errorstr(tdb));
  176. printf(" %zu ns (%zu bytes)\n",
  177. normalize(&start, &stop, num), file_size());
  178. if (++stage == stopat)
  179. exit(0);
  180. if (transaction && tdb_transaction_start(tdb))
  181. errx(1, "starting transaction: %s", tdb_errorstr(tdb));
  182. /* Append 1000 records. */
  183. printf("Appending %u records: ", num); fflush(stdout);
  184. gettimeofday(&start, NULL);
  185. for (i = 0; i < num; i++)
  186. if (tdb_append(tdb, key, data) != 0)
  187. errx(1, "Appending key %u in tdb: %s",
  188. i, tdb_errorstr(tdb));
  189. gettimeofday(&stop, NULL);
  190. if (transaction && tdb_transaction_commit(tdb))
  191. errx(1, "committing transaction: %s", tdb_errorstr(tdb));
  192. printf(" %zu ns (%zu bytes)\n",
  193. normalize(&start, &stop, num), file_size());
  194. if (++stage == stopat)
  195. exit(0);
  196. if (transaction && tdb_transaction_start(tdb))
  197. errx(1, "starting transaction: %s", tdb_errorstr(tdb));
  198. /* Churn 1000 records: not in order! */
  199. printf("Churning %u records: ", num); fflush(stdout);
  200. gettimeofday(&start, NULL);
  201. for (j = 0; j < num; j++) {
  202. i = (j + 1000019) % num;
  203. if (tdb_delete(tdb, key) != 0)
  204. errx(1, "Deleting key %u in tdb: %s",
  205. i, tdb_errorstr(tdb));
  206. i += num;
  207. if (tdb_store(tdb, key, data, TDB_INSERT) != 0)
  208. errx(1, "Inserting key %u in tdb: %s",
  209. i, tdb_errorstr(tdb));
  210. }
  211. gettimeofday(&stop, NULL);
  212. if (transaction && tdb_transaction_commit(tdb))
  213. errx(1, "committing transaction: %s", tdb_errorstr(tdb));
  214. printf(" %zu ns (%zu bytes)\n",
  215. normalize(&start, &stop, num), file_size());
  216. return 0;
  217. }