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
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
|
// Copyright (c) 2012 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.
#ifndef NET_DNS_DNS_CONFIG_SERVICE_WIN_H_
#define NET_DNS_DNS_CONFIG_SERVICE_WIN_H_
// The sole purpose of dns_config_service_win.h is for unittests so we just
// include these headers here.
#include <winsock2.h>
#include <iphlpapi.h>
#include <string>
#include <vector>
#include "base/memory/ref_counted.h"
#include "base/memory/scoped_ptr.h"
#include "base/string16.h"
#include "net/base/net_export.h"
#include "net/dns/dns_config_service.h"
// The general effort of DnsConfigServiceWin is to configure |nameservers| and
// |search| in DnsConfig. The settings are stored in the Windows registry, but
// to simplify the task we use the IP Helper API wherever possible. That API
// yields the complete and ordered |nameservers|, but to determine |search| we
// need to use the registry. On Windows 7, WMI does return the correct |search|
// but on earlier versions it is insufficient.
//
// Experimental evaluation of Windows behavior suggests that domain parsing is
// naive. Domain suffixes in |search| are not validated until they are appended
// to the resolved name. We attempt to replicate this behavior.
namespace net {
namespace internal {
// Registry key paths.
const wchar_t* const kTcpipPath =
L"SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters";
const wchar_t* const kTcpip6Path =
L"SYSTEM\\CurrentControlSet\\Services\\Tcpip6\\Parameters";
const wchar_t* const kDnscachePath =
L"SYSTEM\\CurrentControlSet\\Services\\Dnscache\\Parameters";
const wchar_t* const kPolicyPath =
L"SOFTWARE\\Policies\\Microsoft\\Windows NT\\DNSClient";
// Returns the path to the HOSTS file.
FilePath GetHostsPath();
// Parses |value| as search list (comma-delimited list of domain names) from
// a registry key and stores it in |out|. Returns true on success. Empty
// entries (e.g., "chromium.org,,org") terminate the list. Non-ascii hostnames
// are converted to punycode.
bool NET_EXPORT_PRIVATE ParseSearchList(const string16& value,
std::vector<std::string>* out);
// All relevant settings read from registry and IP Helper. This isolates our
// logic from system calls and is exposed for unit tests. Keep it an aggregate
// struct for easy initialization.
struct NET_EXPORT_PRIVATE DnsSystemSettings {
// The |set| flag distinguishes between empty and unset values.
struct RegString {
bool set;
string16 value;
};
struct RegDword {
bool set;
DWORD value;
};
struct DevolutionSetting {
// UseDomainNameDevolution
RegDword enabled;
// DomainNameDevolutionLevel
RegDword level;
};
// Filled in by GetAdapterAddresses. Note that the alternative
// GetNetworkParams does not include IPv6 addresses.
scoped_ptr_malloc<IP_ADAPTER_ADDRESSES> addresses;
// SOFTWARE\Policies\Microsoft\Windows NT\DNSClient\SearchList
RegString policy_search_list;
// SYSTEM\CurrentControlSet\Tcpip\Parameters\SearchList
RegString tcpip_search_list;
// SYSTEM\CurrentControlSet\Tcpip\Parameters\Domain
RegString tcpip_domain;
// SOFTWARE\Policies\Microsoft\System\DNSClient\PrimaryDnsSuffix
RegString primary_dns_suffix;
// SOFTWARE\Policies\Microsoft\Windows NT\DNSClient
DevolutionSetting policy_devolution;
// SYSTEM\CurrentControlSet\Dnscache\Parameters
DevolutionSetting dnscache_devolution;
// SYSTEM\CurrentControlSet\Tcpip\Parameters
DevolutionSetting tcpip_devolution;
// SOFTWARE\Policies\Microsoft\Windows NT\DNSClient\AppendToMultiLabelName
RegDword append_to_multi_label_name;
};
enum ConfigParseWinResult {
CONFIG_PARSE_WIN_OK = 0,
CONFIG_PARSE_WIN_READ_IPHELPER,
CONFIG_PARSE_WIN_READ_POLICY_SEARCHLIST,
CONFIG_PARSE_WIN_READ_TCPIP_SEARCHLIST,
CONFIG_PARSE_WIN_READ_DOMAIN,
CONFIG_PARSE_WIN_READ_POLICY_DEVOLUTION,
CONFIG_PARSE_WIN_READ_DNSCACHE_DEVOLUTION,
CONFIG_PARSE_WIN_READ_TCPIP_DEVOLUTION,
CONFIG_PARSE_WIN_READ_APPEND_MULTILABEL,
CONFIG_PARSE_WIN_READ_PRIMARY_SUFFIX,
CONFIG_PARSE_WIN_BAD_ADDRESS,
CONFIG_PARSE_WIN_NO_NAMESERVERS,
CONFIG_PARSE_WIN_MAX // Bounding values for enumeration.
};
// Fills in |dns_config| from |settings|. Exposed for tests.
ConfigParseWinResult NET_EXPORT_PRIVATE ConvertSettingsToDnsConfig(
const DnsSystemSettings& settings,
DnsConfig* dns_config);
// Use DnsConfigService::CreateSystemService to use it outside of tests.
class NET_EXPORT_PRIVATE DnsConfigServiceWin : public DnsConfigService {
public:
DnsConfigServiceWin();
virtual ~DnsConfigServiceWin();
private:
class Watcher;
class ConfigReader;
class HostsReader;
// DnsConfigService:
virtual void ReadNow() OVERRIDE;
virtual bool StartWatching() OVERRIDE;
void OnConfigChanged(bool succeeded);
void OnHostsChanged(bool succeeded);
scoped_ptr<Watcher> watcher_;
scoped_refptr<ConfigReader> config_reader_;
scoped_refptr<HostsReader> hosts_reader_;
DISALLOW_COPY_AND_ASSIGN(DnsConfigServiceWin);
};
} // namespace internal
} // namespace net
#endif // NET_DNS_DNS_CONFIG_SERVICE_WIN_H_
|