scrypt_cpuperf.c 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186
  1. /*-
  2. * Copyright 2009 Colin Percival
  3. * All rights reserved.
  4. *
  5. * Redistribution and use in source and binary forms, with or without
  6. * modification, are permitted provided that the following conditions
  7. * are met:
  8. * 1. Redistributions of source code must retain the above copyright
  9. * notice, this list of conditions and the following disclaimer.
  10. * 2. Redistributions in binary form must reproduce the above copyright
  11. * notice, this list of conditions and the following disclaimer in the
  12. * documentation and/or other materials provided with the distribution.
  13. *
  14. * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
  15. * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  16. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  17. * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
  18. * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  19. * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
  20. * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  21. * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
  22. * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
  23. * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  24. * SUCH DAMAGE.
  25. *
  26. * This file was originally written by Colin Percival as part of the Tarsnap
  27. * online backup system.
  28. */
  29. #include "scrypt_platform.h"
  30. #include <sys/time.h>
  31. #include <stdint.h>
  32. #include <stdio.h>
  33. #include <time.h>
  34. #include "scrypt.h"
  35. #include "scrypt_cpuperf.h"
  36. #ifdef HAVE_CLOCK_GETTIME
  37. static clock_t clocktouse;
  38. static int
  39. getclockres(double * resd)
  40. {
  41. struct timespec res;
  42. /*
  43. * Try clocks in order of preference until we find one which works.
  44. * (We assume that if clock_getres works, clock_gettime will, too.)
  45. * The use of if/else/if/else/if/else rather than if/elif/elif/else
  46. * is ugly but legal, and allows us to #ifdef things appropriately.
  47. */
  48. #ifdef CLOCK_VIRTUAL
  49. if (clock_getres(CLOCK_VIRTUAL, &res) == 0)
  50. clocktouse = CLOCK_VIRTUAL;
  51. else
  52. #endif
  53. #ifdef CLOCK_MONOTONIC
  54. if (clock_getres(CLOCK_MONOTONIC, &res) == 0)
  55. clocktouse = CLOCK_MONOTONIC;
  56. else
  57. #endif
  58. if (clock_getres(CLOCK_REALTIME, &res) == 0)
  59. clocktouse = CLOCK_REALTIME;
  60. else
  61. return (-1);
  62. /* Convert clock resolution to a double. */
  63. *resd = res.tv_sec + res.tv_nsec * 0.000000001;
  64. return (0);
  65. }
  66. static int
  67. getclocktime(struct timespec * ts)
  68. {
  69. if (clock_gettime(clocktouse, ts))
  70. return (-1);
  71. return (0);
  72. }
  73. #else
  74. static int
  75. getclockres(double * resd)
  76. {
  77. *resd = 1.0 / CLOCKS_PER_SEC;
  78. return (0);
  79. }
  80. static int
  81. getclocktime(struct timespec * ts)
  82. {
  83. struct timeval tv;
  84. if (gettimeofday(&tv, NULL))
  85. return (-1);
  86. ts->tv_sec = tv.tv_sec;
  87. ts->tv_nsec = tv.tv_usec * 1000;
  88. return (0);
  89. }
  90. #endif
  91. static int
  92. getclockdiff(struct timespec * st, double * diffd)
  93. {
  94. struct timespec en;
  95. if (getclocktime(&en))
  96. return (1);
  97. *diffd = (en.tv_nsec - st->tv_nsec) * 0.000000001 +
  98. (en.tv_sec - st->tv_sec);
  99. return (0);
  100. }
  101. /**
  102. * scrypt_cpuperf(opps):
  103. * Estimate the number of salsa20/8 cores which can be executed per second,
  104. * and return the value via opps.
  105. */
  106. int
  107. scrypt_cpuperf(double * opps)
  108. {
  109. struct timespec st;
  110. double resd, diffd;
  111. uint64_t i = 0;
  112. /* Get the clock resolution. */
  113. if (getclockres(&resd))
  114. return (2);
  115. #ifdef DEBUG
  116. fprintf(stderr, "Clock resolution is %f\n", resd);
  117. #endif
  118. /* Loop until the clock ticks. */
  119. if (getclocktime(&st))
  120. return (2);
  121. do {
  122. /* Do an scrypt. */
  123. if (scrypt(NULL, 0, NULL, 0, 16, 1, 1, NULL, 0))
  124. return (3);
  125. /* Has the clock ticked? */
  126. if (getclockdiff(&st, &diffd))
  127. return (2);
  128. if (diffd > 0)
  129. break;
  130. } while (1);
  131. /* Could how many scryps we can do before the next tick. */
  132. if (getclocktime(&st))
  133. return (2);
  134. do {
  135. /* Do an scrypt. */
  136. if (scrypt(NULL, 0, NULL, 0, 128, 1, 1, NULL, 0))
  137. return (3);
  138. /* We invoked the salsa20/8 core 512 times. */
  139. i += 512;
  140. /* Check if we have looped for long enough. */
  141. if (getclockdiff(&st, &diffd))
  142. return (2);
  143. if (diffd > resd)
  144. break;
  145. } while (1);
  146. #ifdef DEBUG
  147. fprintf(stderr, "%ju salsa20/8 cores performed in %f seconds\n",
  148. (uintmax_t)i, diffd);
  149. #endif
  150. /* We can do approximately i salsa20/8 cores per diffd seconds. */
  151. *opps = i / diffd;
  152. return (0);
  153. }