Home | History | Annotate | Download | only in base
      1 /*
      2  * libjingle
      3  * Copyright 2004--2005, Google Inc.
      4  *
      5  * Redistribution and use in source and binary forms, with or without
      6  * modification, are permitted provided that the following conditions are met:
      7  *
      8  *  1. Redistributions of source code must retain the above copyright notice,
      9  *     this list of conditions and the following disclaimer.
     10  *  2. Redistributions in binary form must reproduce the above copyright notice,
     11  *     this list of conditions and the following disclaimer in the documentation
     12  *     and/or other materials provided with the distribution.
     13  *  3. The name of the author may not be used to endorse or promote products
     14  *     derived from this software without specific prior written permission.
     15  *
     16  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
     17  * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
     18  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
     19  * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
     20  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
     21  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
     22  * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
     23  * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
     24  * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
     25  * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
     26  */
     27 
     28 #include "talk/base/helpers.h"
     29 
     30 #ifdef WIN32
     31 #define WIN32_LEAN_AND_MEAN
     32 #include <windows.h>
     33 #include <ntsecapi.h>
     34 #else
     35 #ifdef SSL_USE_OPENSSL
     36 #include <openssl/rand.h>
     37 #endif
     38 #endif
     39 
     40 #include "talk/base/base64.h"
     41 #include "talk/base/logging.h"
     42 #include "talk/base/scoped_ptr.h"
     43 #include "talk/base/time.h"
     44 
     45 namespace talk_base {
     46 
     47 // Base class for RNG implementations.
     48 class RandomGenerator {
     49  public:
     50   virtual ~RandomGenerator() {}
     51   virtual bool Init(const void* seed, size_t len) = 0;
     52   virtual bool Generate(void* buf, size_t len) = 0;
     53 };
     54 
     55 // The real random generators, using either CryptoAPI or OpenSSL.
     56 // We also support the 'old' generator on Mac/Linux until we have time to
     57 // fully test the OpenSSL one.
     58 #ifdef WIN32
     59 class SecureRandomGenerator : public RandomGenerator {
     60  public:
     61   SecureRandomGenerator() : advapi32_(NULL), rtl_gen_random_(NULL) {}
     62   ~SecureRandomGenerator() {
     63     FreeLibrary(advapi32_);
     64   }
     65 
     66   virtual bool Init(const void* seed, size_t seed_len) {
     67     // We don't do any additional seeding on Win32, we just use the CryptoAPI
     68     // RNG (which is exposed as a hidden function off of ADVAPI32 so that we
     69     // don't need to drag in all of CryptoAPI)
     70     if (rtl_gen_random_) {
     71       return true;
     72     }
     73 
     74     advapi32_ = LoadLibrary(L"advapi32.dll");
     75     if (!advapi32_) {
     76       return false;
     77     }
     78 
     79     rtl_gen_random_ = reinterpret_cast<RtlGenRandomProc>(
     80         GetProcAddress(advapi32_, "SystemFunction036"));
     81     if (!rtl_gen_random_) {
     82       FreeLibrary(advapi32_);
     83       return false;
     84     }
     85 
     86     return true;
     87   }
     88   virtual bool Generate(void* buf, size_t len) {
     89     if (!rtl_gen_random_ && !Init(NULL, 0)) {
     90       return false;
     91     }
     92     return (rtl_gen_random_(buf, len) != FALSE);
     93   }
     94 
     95  private:
     96   typedef BOOL (WINAPI *RtlGenRandomProc)(PVOID, ULONG);
     97   HINSTANCE advapi32_;
     98   RtlGenRandomProc rtl_gen_random_;
     99 };
    100 #else
    101 #ifndef SSL_USE_OPENSSL
    102 // The old RNG.
    103 class SecureRandomGenerator : public RandomGenerator {
    104  public:
    105   SecureRandomGenerator() : seed_(1) {
    106   }
    107   ~SecureRandomGenerator() {
    108   }
    109   virtual bool Init(const void* seed, size_t len) {
    110     uint32 hash = 0;
    111     for (size_t i = 0; i < len; ++i) {
    112       hash = ((hash << 2) + hash) + static_cast<const char*>(seed)[i];
    113     }
    114 
    115     seed_ = Time() ^ hash;
    116     return true;
    117   }
    118   virtual bool Generate(void* buf, size_t len) {
    119     for (size_t i = 0; i < len; ++i) {
    120       static_cast<uint8*>(buf)[i] = static_cast<uint8>(GetRandom());
    121     }
    122     return true;
    123   }
    124 
    125  private:
    126   int GetRandom() {
    127     return ((seed_ = seed_ * 214013L + 2531011L) >> 16) & 0x7fff;
    128   }
    129   int seed_;
    130 };
    131 #else
    132 // The OpenSSL RNG. Need to make sure it doesn't run out of entropy.
    133 class SecureRandomGenerator : public RandomGenerator {
    134  public:
    135   SecureRandomGenerator() : inited_(false) {
    136   }
    137   ~SecureRandomGenerator() {
    138   }
    139   virtual bool Init(const void* seed, size_t len) {
    140     // By default, seed from the system state.
    141     if (!inited_) {
    142       if (RAND_poll() <= 0) {
    143         return false;
    144       }
    145       inited_ = true;
    146     }
    147     // Allow app data to be mixed in, if provided.
    148     if (seed) {
    149       RAND_seed(seed, len);
    150     }
    151     return true;
    152   }
    153   virtual bool Generate(void* buf, size_t len) {
    154     if (!inited_ && !Init(NULL, 0)) {
    155       return false;
    156     }
    157     return (RAND_bytes(reinterpret_cast<unsigned char*>(buf), len) > 0);
    158   }
    159 
    160  private:
    161   bool inited_;
    162 };
    163 #endif  // SSL_USE_OPENSSL
    164 #endif  // WIN32
    165 
    166 // A test random generator, for predictable output.
    167 class TestRandomGenerator : public RandomGenerator {
    168  public:
    169   TestRandomGenerator() : seed_(7) {
    170   }
    171   ~TestRandomGenerator() {
    172   }
    173   virtual bool Init(const void* seed, size_t len) {
    174     return true;
    175   }
    176   virtual bool Generate(void* buf, size_t len) {
    177     for (size_t i = 0; i < len; ++i) {
    178       static_cast<uint8*>(buf)[i] = static_cast<uint8>(GetRandom());
    179     }
    180     return true;
    181   }
    182 
    183  private:
    184   int GetRandom() {
    185     return ((seed_ = seed_ * 214013L + 2531011L) >> 16) & 0x7fff;
    186   }
    187   int seed_;
    188 };
    189 
    190 // TODO: Use Base64::Base64Table instead.
    191 static const char BASE64[64] = {
    192   'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M',
    193   'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z',
    194   'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm',
    195   'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z',
    196   '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '+', '/'
    197 };
    198 
    199 static scoped_ptr<RandomGenerator> g_rng(new SecureRandomGenerator());
    200 
    201 void SetRandomTestMode(bool test) {
    202   if (!test) {
    203     g_rng.reset(new SecureRandomGenerator());
    204   } else {
    205     g_rng.reset(new TestRandomGenerator());
    206   }
    207 }
    208 
    209 bool InitRandom(int seed) {
    210   return InitRandom(reinterpret_cast<const char*>(&seed), sizeof(seed));
    211 }
    212 
    213 bool InitRandom(const char* seed, size_t len) {
    214   if (!g_rng->Init(seed, len)) {
    215     LOG(LS_ERROR) << "Failed to init random generator!";
    216     return false;
    217   }
    218   return true;
    219 }
    220 
    221 std::string CreateRandomString(size_t len) {
    222   std::string str;
    223   CreateRandomString(len, &str);
    224   return str;
    225 }
    226 
    227 bool CreateRandomString(size_t len, std::string* str) {
    228   str->clear();
    229   scoped_array<uint8> bytes(new uint8[len]);
    230   if (!g_rng->Generate(bytes.get(), len)) {
    231     LOG(LS_ERROR) << "Failed to generate random string!";
    232     return false;
    233   }
    234   str->reserve(len);
    235   for (size_t i = 0; i < len; ++i) {
    236     str->push_back(BASE64[bytes[i] & 63]);
    237   }
    238   return true;
    239 }
    240 
    241 uint32 CreateRandomId() {
    242   uint32 id;
    243   if (!g_rng->Generate(&id, sizeof(id))) {
    244     LOG(LS_ERROR) << "Failed to generate random id!";
    245   }
    246   return id;
    247 }
    248 
    249 uint32 CreateRandomNonZeroId() {
    250   uint32 id;
    251   do {
    252     id = CreateRandomId();
    253   } while (id == 0);
    254   return id;
    255 }
    256 
    257 }  // namespace talk_base
    258