Platform.cc 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288
  1. /* <!-- copyright */
  2. /*
  3. * aria2 - The high speed download utility
  4. *
  5. * Copyright (C) 2006 Tatsuhiro Tsujikawa
  6. *
  7. * This program is free software; you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License as published by
  9. * the Free Software Foundation; either version 2 of the License, or
  10. * (at your option) any later version.
  11. *
  12. * This program is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. * GNU General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU General Public License
  18. * along with this program; if not, write to the Free Software
  19. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  20. *
  21. * In addition, as a special exception, the copyright holders give
  22. * permission to link the code of portions of this program with the
  23. * OpenSSL library under certain conditions as described in each
  24. * individual source file, and distribute linked combinations
  25. * including the two.
  26. * You must obey the GNU General Public License in all respects
  27. * for all of the code used other than OpenSSL. If you modify
  28. * file(s) with this exception, you may extend this exception to your
  29. * version of the file(s), but you are not obligated to do so. If you
  30. * do not wish to do so, delete this exception statement from your
  31. * version. If you delete this exception statement from all source
  32. * files in the program, then also delete it here.
  33. */
  34. /* copyright --> */
  35. #include "Platform.h"
  36. #include <stdlib.h> /* _fmode */
  37. #include <fcntl.h> /* _O_BINARY */
  38. #include <locale.h> // For setlocale, LC_*
  39. #include <iostream>
  40. #ifdef HAVE_OPENSSL
  41. #include <openssl/err.h>
  42. #include <openssl/ssl.h>
  43. #endif // HAVE_OPENSSL
  44. #ifdef HAVE_LIBGCRYPT
  45. #include <gcrypt.h>
  46. #endif // HAVE_LIBGCRYPT
  47. #ifdef HAVE_LIBGNUTLS
  48. #include <gnutls/gnutls.h>
  49. #endif // HAVE_LIBGNUTLS
  50. #ifdef ENABLE_ASYNC_DNS
  51. #include <ares.h>
  52. #endif // ENABLE_ASYNC_DNS
  53. #ifdef HAVE_LIBSSH2
  54. #include <libssh2.h>
  55. #endif // HAVE_LIBSSH2
  56. #include "a2netcompat.h"
  57. #include "DlAbortEx.h"
  58. #include "message.h"
  59. #include "fmt.h"
  60. #include "console.h"
  61. #include "OptionParser.h"
  62. #include "prefs.h"
  63. #ifdef HAVE_LIBGMP
  64. #include "a2gmp.h"
  65. #endif // HAVE_LIBGMP
  66. #include "LogFactory.h"
  67. #include "util.h"
  68. namespace aria2 {
  69. #ifdef HAVE_LIBGNUTLS
  70. namespace {
  71. void gnutls_log_callback(int level, const char* str)
  72. {
  73. using namespace aria2;
  74. // GnuTLS adds a newline. Drop it.
  75. std::string msg(str);
  76. msg.resize(msg.size() - 1);
  77. A2_LOG_DEBUG(fmt("GnuTLS: <%d> %s", level, msg.c_str()));
  78. }
  79. }
  80. #endif // HAVE_LIBGNUTLS
  81. bool Platform::initialized_ = false;
  82. Platform::Platform() { setUp(); }
  83. Platform::~Platform() { tearDown(); }
  84. #ifdef __MINGW32__
  85. namespace {
  86. bool gainPrivilege(LPCTSTR privName)
  87. {
  88. LUID luid;
  89. TOKEN_PRIVILEGES tp;
  90. if (!LookupPrivilegeValue(nullptr, privName, &luid)) {
  91. auto errNum = GetLastError();
  92. A2_LOG_WARN(fmt("Lookup for privilege name %s failed. cause: %s", privName,
  93. util::formatLastError(errNum).c_str()));
  94. return false;
  95. }
  96. tp.PrivilegeCount = 1;
  97. tp.Privileges[0].Luid = luid;
  98. tp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
  99. HANDLE token;
  100. if (!OpenProcessToken(GetCurrentProcess(),
  101. TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY, &token)) {
  102. auto errNum = GetLastError();
  103. A2_LOG_WARN(fmt("Getting process token failed. cause: %s",
  104. util::formatLastError(errNum).c_str()));
  105. return false;
  106. }
  107. auto tokenCloser = defer(token, CloseHandle);
  108. if (!AdjustTokenPrivileges(token, FALSE, &tp, 0, NULL, NULL)) {
  109. auto errNum = GetLastError();
  110. A2_LOG_WARN(fmt("Gaining privilege %s failed. cause: %s", privName,
  111. util::formatLastError(errNum).c_str()));
  112. return false;
  113. }
  114. // Check privilege was really gained
  115. DWORD bufsize = 0;
  116. GetTokenInformation(token, TokenPrivileges, nullptr, 0, &bufsize);
  117. if (bufsize == 0) {
  118. A2_LOG_WARN("Checking privilege failed.");
  119. return false;
  120. }
  121. auto buf = make_unique<char[]>(bufsize);
  122. if (!GetTokenInformation(token, TokenPrivileges, buf.get(), bufsize,
  123. &bufsize)) {
  124. auto errNum = GetLastError();
  125. A2_LOG_WARN(fmt("Checking privilege failed. cause: %s",
  126. util::formatLastError(errNum).c_str()));
  127. return false;
  128. }
  129. auto privs = reinterpret_cast<TOKEN_PRIVILEGES*>(buf.get());
  130. for (size_t i = 0; i < privs->PrivilegeCount; ++i) {
  131. auto& priv = privs->Privileges[i];
  132. if (memcmp(&priv.Luid, &luid, sizeof(luid)) != 0) {
  133. continue;
  134. }
  135. if (priv.Attributes == SE_PRIVILEGE_ENABLED) {
  136. return true;
  137. }
  138. break;
  139. }
  140. A2_LOG_WARN(fmt("Gaining privilege %s failed.", privName));
  141. return false;
  142. }
  143. } // namespace
  144. #endif // __MINGW32__
  145. bool Platform::setUp()
  146. {
  147. if (initialized_) {
  148. return false;
  149. }
  150. initialized_ = true;
  151. #ifdef HAVE_LIBGMP
  152. global::initGmp();
  153. #endif // HAVE_LIBGMP
  154. #ifdef ENABLE_NLS
  155. setlocale(LC_CTYPE, "");
  156. setlocale(LC_MESSAGES, "");
  157. bindtextdomain(PACKAGE, LOCALEDIR);
  158. textdomain(PACKAGE);
  159. #endif // ENABLE_NLS
  160. #ifdef HAVE_OPENSSL
  161. // for SSL initialization
  162. SSL_load_error_strings();
  163. SSL_library_init();
  164. // Need this to "decrypt" p12 files.
  165. OpenSSL_add_all_algorithms();
  166. #endif // HAVE_OPENSSL
  167. #ifdef HAVE_LIBGCRYPT
  168. if (!gcry_check_version("1.2.4")) {
  169. throw DL_ABORT_EX("gcry_check_version() failed.");
  170. }
  171. gcry_control(GCRYCTL_DISABLE_SECMEM, 0);
  172. gcry_control(GCRYCTL_INITIALIZATION_FINISHED, 0);
  173. #endif // HAVE_LIBGCRYPT
  174. #ifdef HAVE_LIBGNUTLS
  175. {
  176. int r = gnutls_global_init();
  177. if (r != GNUTLS_E_SUCCESS) {
  178. throw DL_ABORT_EX(
  179. fmt("gnutls_global_init() failed, cause:%s", gnutls_strerror(r)));
  180. }
  181. gnutls_global_set_log_function(gnutls_log_callback);
  182. gnutls_global_set_log_level(0);
  183. }
  184. #endif // HAVE_LIBGNUTLS
  185. #ifdef CARES_HAVE_ARES_LIBRARY_INIT
  186. int aresErrorCode;
  187. if ((aresErrorCode = ares_library_init(ARES_LIB_INIT_ALL)) != 0) {
  188. global::cerr()->printf("ares_library_init() failed:%s\n",
  189. ares_strerror(aresErrorCode));
  190. }
  191. #endif // CARES_HAVE_ARES_LIBRARY_INIT
  192. #ifdef HAVE_LIBSSH2
  193. {
  194. auto rv = libssh2_init(0);
  195. if (rv != 0) {
  196. throw DL_ABORT_EX(fmt("libssh2_init() failed, code: %d", rv));
  197. }
  198. }
  199. #endif // HAVE_LIBSSH2
  200. #ifdef HAVE_WINSOCK2_H
  201. WSADATA wsaData;
  202. memset(reinterpret_cast<char*>(&wsaData), 0, sizeof(wsaData));
  203. if (WSAStartup(MAKEWORD(1, 1), &wsaData)) {
  204. throw DL_ABORT_EX(MSG_WINSOCK_INIT_FAILD);
  205. }
  206. #endif // HAVE_WINSOCK2_H
  207. #ifdef __MINGW32__
  208. (void)_setmode(_fileno(stdin), _O_BINARY);
  209. (void)_setmode(_fileno(stdout), _O_BINARY);
  210. (void)_setmode(_fileno(stderr), _O_BINARY);
  211. // Windows build: --file-allocation=falloc uses SetFileValidData
  212. // which requires SE_MANAGE_VOLUME_NAME privilege. SetFileValidData
  213. // has security implications (see
  214. // https://msdn.microsoft.com/en-us/library/windows/desktop/aa365544%28v=vs.85%29.aspx).
  215. if (!gainPrivilege(SE_MANAGE_VOLUME_NAME)) {
  216. A2_LOG_WARN("--file-allocation=falloc will not work properly.");
  217. }
  218. #endif // __MINGW32__
  219. return true;
  220. }
  221. bool Platform::tearDown()
  222. {
  223. if (!initialized_) {
  224. return false;
  225. }
  226. initialized_ = false;
  227. #ifdef HAVE_LIBGNUTLS
  228. gnutls_global_deinit();
  229. #endif // HAVE_LIBGNUTLS
  230. #ifdef CARES_HAVE_ARES_LIBRARY_CLEANUP
  231. ares_library_cleanup();
  232. #endif // CARES_HAVE_ARES_LIBRARY_CLEANUP
  233. #ifdef HAVE_LIBSSH2
  234. libssh2_exit();
  235. #endif // HAVE_LIBSSH2
  236. #ifdef HAVE_WINSOCK2_H
  237. WSACleanup();
  238. #endif // HAVE_WINSOCK2_H
  239. // Deletes statically allocated resources. This is done to
  240. // distinguish memory leak from them. This is handy to use
  241. // valgrind.
  242. OptionParser::deleteInstance();
  243. option::deletePrefResource();
  244. return true;
  245. }
  246. bool Platform::isInitialized() { return initialized_; }
  247. } // namespace aria2