bench.h 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277
  1. /*
  2. * $Id$
  3. */
  4. #ifndef _BENCH_H
  5. #define _BENCH_H
  6. #ifdef WIN32
  7. #include <windows.h>
  8. typedef unsigned char bool_t;
  9. #endif
  10. #include <assert.h>
  11. #include <ctype.h>
  12. #include <stdio.h>
  13. #ifndef WIN32
  14. #include <unistd.h>
  15. #endif
  16. #include <stdlib.h>
  17. #include <fcntl.h>
  18. #include <signal.h>
  19. #include <errno.h>
  20. #ifndef WIN32
  21. #include <strings.h>
  22. #endif
  23. #include <sys/types.h>
  24. #ifndef WIN32
  25. #include <sys/mman.h>
  26. #endif
  27. #include <sys/stat.h>
  28. #ifndef WIN32
  29. #include <sys/wait.h>
  30. #include <time.h>
  31. #include <sys/time.h>
  32. #include <sys/socket.h>
  33. #include <sys/un.h>
  34. #include <sys/resource.h>
  35. #define PORTMAP
  36. #include <rpc/rpc.h>
  37. #endif
  38. #include <rpc/types.h>
  39. #include <stdarg.h>
  40. void TRACE(char* format, ...);
  41. #ifndef HAVE_uint
  42. typedef unsigned int uint;
  43. #endif
  44. #ifndef HAVE_uint64
  45. #ifdef HAVE_uint64_t
  46. typedef uint64_t uint64;
  47. #else /* HAVE_uint64_t */
  48. typedef unsigned long long uint64;
  49. #endif /* HAVE_uint64_t */
  50. #endif /* HAVE_uint64 */
  51. #ifndef HAVE_int64
  52. #ifdef HAVE_int64_t
  53. typedef int64_t int64;
  54. #else /* HAVE_int64_t */
  55. typedef long long int64;
  56. #endif /* HAVE_int64_t */
  57. #endif /* HAVE_int64 */
  58. #define NO_PORTMAPPER /* needs to be up here, lib_*.h look at it */
  59. #include "stats.h"
  60. #include "timing.h"
  61. #include "lib_tcp.h"
  62. #include "lib_udp.h"
  63. #include "lib_unix.h"
  64. #ifdef DEBUG
  65. # define debug(x) fprintf x
  66. #else
  67. # define debug(x)
  68. #endif
  69. #ifdef NO_PORTMAPPER
  70. #define TCP_SELECT -31233
  71. #define TCP_XACT -31234
  72. #define TCP_CONTROL -31235
  73. #define TCP_DATA -31236
  74. #define TCP_CONNECT -31237
  75. #define UDP_XACT -31238
  76. #define UDP_DATA -31239
  77. #else
  78. #define TCP_SELECT (u_long)404038 /* XXX - unregistered */
  79. #define TCP_XACT (u_long)404039 /* XXX - unregistered */
  80. #define TCP_CONTROL (u_long)404040 /* XXX - unregistered */
  81. #define TCP_DATA (u_long)404041 /* XXX - unregistered */
  82. #define TCP_CONNECT (u_long)404042 /* XXX - unregistered */
  83. #define UDP_XACT (u_long)404032 /* XXX - unregistered */
  84. #define UDP_DATA (u_long)404033 /* XXX - unregistered */
  85. #define VERS (u_long)1
  86. #endif
  87. #define UNIX_CONTROL "/tmp/lmbench.ctl"
  88. #define UNIX_DATA "/tmp/lmbench.data"
  89. #define UNIX_LAT "/tmp/lmbench.lat"
  90. /*
  91. * socket send/recv buffer optimizations
  92. */
  93. #define SOCKOPT_READ 0x0001
  94. #define SOCKOPT_WRITE 0x0002
  95. #define SOCKOPT_RDWR 0x0003
  96. #define SOCKOPT_PID 0x0004
  97. #define SOCKOPT_REUSE 0x0008
  98. #define SOCKOPT_NONE 0
  99. #ifndef SOCKBUF
  100. #define SOCKBUF (1024*1024)
  101. #endif
  102. #ifndef XFERSIZE
  103. #define XFERSIZE (64*1024) /* all bandwidth I/O should use this */
  104. #endif
  105. #if defined(SYS5) || defined(WIN32)
  106. #define bzero(b, len) memset(b, 0, len)
  107. #define bcopy(s, d, l) memcpy(d, s, l)
  108. #define rindex(s, c) strrchr(s, c)
  109. #endif
  110. #define gettime usecs_spent
  111. #define streq !strcmp
  112. #define ulong unsigned long
  113. #ifdef USE_RAND
  114. #define srand48 srand
  115. #define drand48() ((double)rand() / (double)RAND_MAX)
  116. #endif
  117. #ifdef USE_RANDOM
  118. #define srand48 srand
  119. #define drand48() ((double)rand() / (double)RAND_MAX)
  120. #endif
  121. #ifdef WIN32
  122. #include <process.h>
  123. #define getpid _getpid
  124. int gettimeofday(struct timeval *tv, struct timezone *tz);
  125. #endif
  126. #ifdef KVM_LMBENCH
  127. //#define gettimeofday(tv, tz) getbenchtimeofday(tv);
  128. int getbenchtimeofday(struct timeval *tv);
  129. #endif
  130. #define SMALLEST_LINE 32 /* smallest cache line size */
  131. #define TIME_OPEN2CLOSE
  132. #define GO_AWAY signal(SIGALRM, exit); alarm(60 * 60);
  133. #define REAL_SHORT 50000
  134. #define SHORT 1000000
  135. #define MEDIUM 2000000
  136. #define LONGER 7500000 /* for networking data transfers */
  137. #define ENOUGH REAL_SHORT
  138. #define TRIES 10
  139. typedef struct {
  140. int N;
  141. uint64 u[TRIES];
  142. uint64 n[TRIES];
  143. } result_t;
  144. void insertinit(result_t *r);
  145. void insertsort(uint64, uint64, result_t *);
  146. void save_median();
  147. void save_minimum();
  148. void save_mean();
  149. void save_variance();
  150. void save_results(result_t *r);
  151. void get_results(result_t *r);
  152. #define BENCHO(loop_body, overhead_body, enough) { \
  153. int __i, __N; \
  154. double __oh; \
  155. result_t __overhead, __r; \
  156. insertinit(&__overhead); insertinit(&__r); \
  157. __N = (enough == 0 || get_enough(enough) <= 100000) ? TRIES : 1;\
  158. if (enough < LONGER) {loop_body;} /* warm the cache */ \
  159. for (__i = 0; __i < __N; ++__i) { \
  160. BENCH1(overhead_body, enough); \
  161. if (gettime() > 0) \
  162. insertsort(gettime(), get_n(), &__overhead); \
  163. BENCH1(loop_body, enough); \
  164. if (gettime() > 0) \
  165. insertsort(gettime(), get_n(), &__r); \
  166. } \
  167. for (__i = 0; __i < __r.N; ++__i) { \
  168. __oh = __overhead.u[__i] / (double)__overhead.n[__i]; \
  169. if (__r.u[__i] > (uint64)((double)__r.n[__i] * __oh)) \
  170. __r.u[__i] -= (uint64)((double)__r.n[__i] * __oh); \
  171. else \
  172. __r.u[__i] = 0; \
  173. } \
  174. save_results(&__r); \
  175. }
  176. #define BENCH(loop_body, enough) { \
  177. long __i, __N; \
  178. result_t __r; \
  179. insertinit(&__r); \
  180. __N = (enough == 0 || get_enough(enough) <= 100000) ? TRIES : 1;\
  181. if (enough < LONGER) {loop_body;} /* warm the cache */ \
  182. for (__i = 0; __i < __N; ++__i) { \
  183. BENCH1(loop_body, enough); \
  184. if (gettime() > 0) \
  185. insertsort(gettime(), get_n(), &__r); \
  186. } \
  187. save_results(&__r); \
  188. }
  189. #define BENCH1(loop_body, enough) { \
  190. double __usecs; \
  191. BENCH_INNER(loop_body, enough); \
  192. __usecs = gettime(); \
  193. __usecs -= t_overhead() + get_n() * l_overhead(); \
  194. settime(__usecs >= 0. ? (uint64)__usecs : 0); \
  195. }
  196. #define BENCH_INNER(loop_body, enough) { \
  197. static u_long __iterations = 1; \
  198. int __enough = get_enough(enough); \
  199. u_long __n; \
  200. double __result = 0.; \
  201. \
  202. TRACE("BENCH_INNER(%s, %s): enter: __iterations=%lu\n", #loop_body, #enough, (unsigned long)__iterations); \
  203. while(__result < 0.95 * __enough) { \
  204. TRACE("BENCH_INNER(%s, %s): start: __iterations=%lu\n", #loop_body, #enough, (unsigned long)__iterations); \
  205. start(0); \
  206. for (__n = __iterations; __n > 0; __n--) { \
  207. loop_body; \
  208. } \
  209. __result = stop(0,0); \
  210. TRACE("BENCH_INNER(%s, %s): stop: __result=%f, __result / __enough = %f\n", #loop_body, #enough, __result, __result / (double)__enough); \
  211. if (__result < 0.99 * __enough \
  212. || __result > 1.2 * __enough) { \
  213. if (__result > 150.) { \
  214. double tmp = __iterations / __result; \
  215. tmp *= 1.1 * __enough; \
  216. __iterations = (u_long)(tmp + 1); \
  217. } else { \
  218. if (__iterations > (u_long)1<<27) { \
  219. __result = 0.; \
  220. break; \
  221. } \
  222. __iterations <<= 3; \
  223. } \
  224. } \
  225. } /* while */ \
  226. save_n((uint64)__iterations); settime((uint64)__result); \
  227. }
  228. /*
  229. * Generated from msg.x which is included here:
  230. program XACT_PROG {
  231. version XACT_VERS {
  232. char
  233. RPC_XACT(char) = 1;
  234. } = 1;
  235. } = 3970;
  236. * Please do not edit this file.
  237. * It was generated using rpcgen.
  238. */
  239. #define XACT_PROG ((u_long)404040)
  240. #define XACT_VERS ((u_long)1)
  241. #define RPC_XACT ((u_long)1)
  242. #define RPC_EXIT ((u_long)2)
  243. extern char *rpc_xact_1();
  244. extern char *client_rpc_xact_1();
  245. #endif /* _BENCH_H */