1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106
|
// Copyright 2011 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifdef UNSAFE_BUFFERS_BUILD
// TODO(crbug.com/40285824): Remove this and convert code to safer constructs.
#pragma allow_unsafe_buffers
#endif
#include "chrome/browser/net/service_providers_win.h"
#include <winsock2.h>
#include <Ws2spi.h>
#include <memory>
#include "base/containers/heap_array.h"
#include "base/notreached.h"
#include "base/values.h"
WinsockLayeredServiceProvider::WinsockLayeredServiceProvider() = default;
WinsockLayeredServiceProvider::WinsockLayeredServiceProvider(
const WinsockLayeredServiceProvider& other) = default;
WinsockLayeredServiceProvider::~WinsockLayeredServiceProvider() = default;
void GetWinsockNamespaceProviders(
WinsockNamespaceProviderList* namespace_list) {
// Find out how just how much memory is needed. If we get the expected error,
// the memory needed is written to size.
DWORD size = 0;
if (WSAEnumNameSpaceProviders(&size, nullptr) != SOCKET_ERROR ||
GetLastError() != WSAEFAULT) {
NOTREACHED();
}
auto namespace_provider_bytes = base::HeapArray<uint8_t>::WithSize(size);
WSANAMESPACE_INFO* namespace_providers =
reinterpret_cast<WSANAMESPACE_INFO*>(namespace_provider_bytes.data());
int num_namespace_providers = WSAEnumNameSpaceProviders(&size,
namespace_providers);
if (num_namespace_providers == SOCKET_ERROR) {
NOTREACHED();
}
for (int i = 0; i < num_namespace_providers; ++i) {
WinsockNamespaceProvider provider;
provider.name = namespace_providers[i].lpszIdentifier;
provider.active = TRUE == namespace_providers[i].fActive;
provider.version = namespace_providers[i].dwVersion;
provider.type = namespace_providers[i].dwNameSpace;
namespace_list->push_back(provider);
}
}
void GetWinsockLayeredServiceProviders(
WinsockLayeredServiceProviderList* service_list) {
// Find out how just how much memory is needed. If we get the expected error,
// the memory needed is written to size.
DWORD size = 0;
int error;
if (SOCKET_ERROR != WSCEnumProtocols(nullptr, nullptr, &size, &error) ||
error != WSAENOBUFS) {
NOTREACHED();
}
auto service_provider_bytes = base::HeapArray<uint8_t>::WithSize(size);
WSAPROTOCOL_INFOW* service_providers =
reinterpret_cast<WSAPROTOCOL_INFOW*>(service_provider_bytes.data());
int num_service_providers =
WSCEnumProtocols(nullptr, service_providers, &size, &error);
if (num_service_providers == SOCKET_ERROR) {
NOTREACHED();
}
for (int i = 0; i < num_service_providers; ++i) {
WinsockLayeredServiceProvider service_provider;
service_provider.name = service_providers[i].szProtocol;
service_provider.version = service_providers[i].iVersion;
service_provider.socket_type = service_providers[i].iSocketType;
service_provider.socket_protocol = service_providers[i].iProtocol;
service_provider.chain_length = service_providers[i].ProtocolChain.ChainLen;
// TODO(mmenke): Add categories under Vista and later.
// http://msdn.microsoft.com/en-us/library/ms742239%28v=VS.85%29.aspx
wchar_t path[MAX_PATH];
int path_length = std::size(path);
if (0 == WSCGetProviderPath(&service_providers[i].ProviderId, path,
&path_length, &error)) {
service_provider.path = path;
}
service_list->push_back(service_provider);
}
return;
}
|