// 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