Vous ne pouvez pas sélectionner plus de 25 sujets Les noms de sujets doivent commencer par une lettre ou un nombre, peuvent contenir des tirets ('-') et peuvent comporter jusqu'à 35 caractères.
 
 
 
 
 
 

157 lignes
4.9 KiB

  1. /* Copyright (c) 2016, Google Inc.
  2. *
  3. * Permission to use, copy, modify, and/or distribute this software for any
  4. * purpose with or without fee is hereby granted, provided that the above
  5. * copyright notice and this permission notice appear in all copies.
  6. *
  7. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
  8. * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
  9. * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
  10. * SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
  11. * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
  12. * OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
  13. * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */
  14. #include <string>
  15. #include <math.h>
  16. #include <stdio.h>
  17. #include <string.h>
  18. #include <openssl/crypto.h>
  19. #include <openssl/newhope.h>
  20. #include <openssl/rand.h>
  21. #include "internal.h"
  22. static const unsigned kNumTests = 1000;
  23. static bool TestNoise(void) {
  24. printf("noise distribution:\n");
  25. size_t buckets[1 + 2 * PARAM_K];
  26. memset(buckets, 0, sizeof(buckets));
  27. for (size_t i = 0; i < kNumTests; i++) {
  28. NEWHOPE_POLY s;
  29. NEWHOPE_POLY_noise(&s);
  30. for (int j = 0; j < PARAM_N; j++) {
  31. uint16_t value = (s.coeffs[j] + PARAM_K) % PARAM_Q;
  32. buckets[value]++;
  33. }
  34. }
  35. int64_t sum = 0, square_sum = 0;
  36. for (int64_t i = 0; i < 1 + 2 * PARAM_K; i++) {
  37. sum += (i - PARAM_K) * (int64_t) buckets[i];
  38. square_sum += (i - PARAM_K) * (i - PARAM_K) * (int64_t) buckets[i];
  39. }
  40. double mean = double(sum) / double(PARAM_N * kNumTests);
  41. double expected_variance = 0.5 * 0.5 * double(PARAM_K * 2);
  42. double variance = double(square_sum) / double(PARAM_N * kNumTests) - mean * mean;
  43. for (size_t i = 0; i < 1 + 2 * PARAM_K; i++) {
  44. std::string dots;
  45. for (size_t j = 0; j < 79 * buckets[i] / buckets[PARAM_K]; j++) {
  46. dots += "+";
  47. }
  48. printf("%+zd\t%zd\t%s\n", i - PARAM_K, buckets[i], dots.c_str());
  49. }
  50. printf("mean: got %f, want %f\n", mean, 0.0);
  51. printf("variance: got %f, want %f\n", variance, expected_variance);
  52. printf("\n");
  53. if (mean < -0.5 || 0.5 < mean) {
  54. fprintf(stderr, "mean out of range: %f\n", mean);
  55. return false;
  56. }
  57. if (variance < expected_variance - 1.0 || expected_variance + 1.0 < variance) {
  58. fprintf(stderr, "variance out of range: got %f, want %f\n", variance,
  59. expected_variance);
  60. return false;
  61. }
  62. return true;
  63. }
  64. static int Hamming32(const uint8_t key[NEWHOPE_KEY_LENGTH]) {
  65. static int kHamming[256] = {
  66. 0, 1, 1, 2, 1, 2, 2, 3, 1, 2, 2, 3, 2, 3, 3, 4,
  67. 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5,
  68. 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5,
  69. 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6,
  70. 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5,
  71. 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6,
  72. 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6,
  73. 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7,
  74. 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5,
  75. 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6,
  76. 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6,
  77. 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7,
  78. 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6,
  79. 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7,
  80. 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7,
  81. 4, 5, 5, 6, 5, 6, 6, 7, 5, 6, 6, 7, 6, 7, 7, 8,
  82. };
  83. int r = 0;
  84. for(int i = 0; i < NEWHOPE_KEY_LENGTH; i++) {
  85. r += kHamming[key[i]];
  86. }
  87. return r;
  88. }
  89. static bool TestKeys(void) {
  90. printf("keys (prior to whitening):\n");
  91. uint8_t key[NEWHOPE_KEY_LENGTH];
  92. uint8_t offermsg[NEWHOPE_OFFERMSG_LENGTH];
  93. bssl::UniquePtr<NEWHOPE_POLY> sk(NEWHOPE_POLY_new()), pk(NEWHOPE_POLY_new()),
  94. sp(NEWHOPE_POLY_new()), ep(NEWHOPE_POLY_new()), epp(NEWHOPE_POLY_new()),
  95. a(NEWHOPE_POLY_new()), bp(NEWHOPE_POLY_new()), rec(NEWHOPE_POLY_new());
  96. int ones = 0;
  97. for (size_t i = 0; i < kNumTests; i++) {
  98. NEWHOPE_offer(offermsg, sk.get());
  99. NEWHOPE_offer_frommsg(pk.get(), a.get(), offermsg);
  100. NEWHOPE_POLY_noise_ntt(sp.get());
  101. NEWHOPE_POLY_noise_ntt(ep.get());
  102. NEWHOPE_POLY_noise(epp.get()); /* intentionally not NTT */
  103. uint8_t rand[32];
  104. RAND_bytes(rand, 32);
  105. NEWHOPE_accept_computation(key, bp.get(), rec.get(),
  106. sp.get(), ep.get(), epp.get(), rand,
  107. pk.get(), a.get());
  108. ones += Hamming32(key);
  109. }
  110. int bits = NEWHOPE_KEY_LENGTH * 8 * kNumTests;
  111. int diff = bits - 2 * ones;
  112. double fraction = (double) abs(diff) / bits;
  113. printf("ones: %d\n", ones);
  114. printf("zeroes: %d\n", (bits - ones));
  115. printf("diff: got %d (%f), want 0\n", diff, fraction);
  116. printf("\n");
  117. if (fraction > 0.01) {
  118. fprintf(stderr, "key bias is too high (%f)\n", fraction);
  119. return false;
  120. }
  121. return true;
  122. }
  123. int main(void) {
  124. if (!TestKeys() ||
  125. !TestNoise()) {
  126. return 1;
  127. }
  128. printf("PASS\n");
  129. return 0;
  130. }