ChipMaster's trial hacks on C++CMS starting with v1.2.1. Not sure I'll follow on with the v2 since it looks to be breaking and mostly frivolous.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

89 lines
2.5 KiB

  1. ///////////////////////////////////////////////////////////////////////////////
  2. //
  3. // Copyright (C) 2008-2010 Artyom Beilis (Tonkikh) <artyomtnk@yahoo.com>
  4. //
  5. // This program is free software: you can redistribute it and/or modify
  6. // it under the terms of the GNU Lesser General Public License as published by
  7. // the Free Software Foundation, either version 3 of the License, or
  8. // (at your option) any later version.
  9. //
  10. // This program is distributed in the hope that it will be useful,
  11. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. // GNU Lesser General Public License for more details.
  14. //
  15. // You should have received a copy of the GNU Lesser General Public License
  16. // along with this program. If not, see <http://www.gnu.org/licenses/>.
  17. //
  18. ///////////////////////////////////////////////////////////////////////////////
  19. #define CPPCMS_SOURCE
  20. #include "urandom.h"
  21. #include "cppcms_error.h"
  22. #ifdef CPPCMS_WIN_NATIVE
  23. #include "windows.h"
  24. #include "wincrypt.h"
  25. #include <sstream>
  26. namespace cppcms {
  27. struct urandom_device::data {
  28. HCRYPTPROV provider;
  29. data() : provider(0) {}
  30. };
  31. urandom_device::urandom_device() :
  32. d(new data())
  33. {
  34. if(CryptAcquireContext(&d->provider,0,0,PROV_RSA_FULL,0))
  35. return;
  36. if(GetLastError() == NTE_BAD_KEYSET) {
  37. if(CryptAcquireContext(&d->provider,0,0,PROV_RSA_FULL,CRYPT_NEWKEYSET))
  38. return;
  39. }
  40. std::ostringstream ss;
  41. ss<<"CryptAcquireContext failed with code 0x"<<std::hex<<GetLastError();
  42. throw cppcms_error(ss.str());
  43. }
  44. urandom_device::~urandom_device()
  45. {
  46. CryptReleaseContext(d->provider,0);
  47. }
  48. void urandom_device::generate(void *ptr,unsigned len)
  49. {
  50. if(CryptGenRandom(d->provider,len,static_cast<BYTE *>(ptr)))
  51. return;
  52. std::ostringstream ss;
  53. ss<<"CryptGenRandom failed with code 0x"<<std::hex<<GetLastError();
  54. throw cppcms_error(ss.str());
  55. }
  56. }
  57. #else
  58. #include <fstream>
  59. namespace cppcms {
  60. struct urandom_device::data {};
  61. urandom_device::urandom_device()
  62. {
  63. }
  64. urandom_device::~urandom_device()
  65. {
  66. }
  67. void urandom_device::generate(void *ptr,unsigned len)
  68. {
  69. std::ifstream u("/dev/urandom");
  70. if(u.good() && !u.read(reinterpret_cast<char *>(ptr),len).fail() && u.gcount()==int(len))
  71. return;
  72. throw cppcms_error("Failed to read /dev/urandom");
  73. }
  74. }
  75. #endif