|  | // Copyright (c) 2011 The Chromium Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include <winsock2.h> | 
|  |  | 
|  | #include "net/base/winsock_init.h" | 
|  |  | 
|  | #include "base/lazy_instance.h" | 
|  | #include "base/logging.h" | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | class WinsockInitSingleton { | 
|  | public: | 
|  | WinsockInitSingleton() { | 
|  | WORD winsock_ver = MAKEWORD(2, 2); | 
|  | WSAData wsa_data; | 
|  | bool did_init = (WSAStartup(winsock_ver, &wsa_data) == 0); | 
|  | if (did_init) { | 
|  | DCHECK(wsa_data.wVersion == winsock_ver); | 
|  |  | 
|  | // The first time WSAGetLastError is called, the delay load helper will | 
|  | // resolve the address with GetProcAddress and fixup the import.  If a | 
|  | // third party application hooks system functions without correctly | 
|  | // restoring the error code, it is possible that the error code will be | 
|  | // overwritten during delay load resolution.  The result of the first | 
|  | // call may be incorrect, so make sure the function is bound and future | 
|  | // results will be correct. | 
|  | WSAGetLastError(); | 
|  | } | 
|  | } | 
|  |  | 
|  | ~WinsockInitSingleton() { | 
|  | // Don't call WSACleanup() since the worker pool threads can continue to | 
|  | // call getaddrinfo() after Winsock has shutdown, which can lead to crashes. | 
|  | } | 
|  | }; | 
|  |  | 
|  | static base::LazyInstance<WinsockInitSingleton> g_winsock_init_singleton = | 
|  | LAZY_INSTANCE_INITIALIZER; | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | namespace net { | 
|  |  | 
|  | void EnsureWinsockInit() { | 
|  | g_winsock_init_singleton.Get(); | 
|  | } | 
|  |  | 
|  | }  // namespace net |