Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 1 | // Copyright (c) 2013 The Chromium OS Authors. All rights reserved. |
| 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
| 4 | |
| 5 | #include "shill/socket_info_reader.h" |
| 6 | |
| 7 | #include <algorithm> |
| 8 | #include <limits> |
| 9 | |
Ben Chan | a0ddf46 | 2014-02-06 11:32:42 -0800 | [diff] [blame] | 10 | #include <base/strings/string_number_conversions.h> |
| 11 | #include <base/strings/string_split.h> |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 12 | |
| 13 | #include "shill/file_reader.h" |
| 14 | #include "shill/logging.h" |
| 15 | |
| 16 | using base::FilePath; |
| 17 | using std::string; |
| 18 | using std::vector; |
| 19 | |
| 20 | namespace shill { |
| 21 | |
Rebecca Silberstein | c9c31d8 | 2014-10-21 15:01:00 -0700 | [diff] [blame] | 22 | namespace Logging { |
| 23 | static auto kModuleLogScope = ScopeLogger::kLink; |
| 24 | static string ObjectID(SocketInfoReader *s) { return "(socket_info_reader)"; } |
| 25 | } |
| 26 | |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 27 | namespace { |
| 28 | |
Ben Chan | 320d04a | 2013-04-09 13:10:53 -0700 | [diff] [blame] | 29 | const char kTcpv4SocketInfoFilePath[] = "/proc/net/tcp"; |
| 30 | const char kTcpv6SocketInfoFilePath[] = "/proc/net/tcp6"; |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 31 | |
| 32 | } // namespace |
| 33 | |
| 34 | SocketInfoReader::SocketInfoReader() {} |
| 35 | |
| 36 | SocketInfoReader::~SocketInfoReader() {} |
| 37 | |
Ben Chan | 320d04a | 2013-04-09 13:10:53 -0700 | [diff] [blame] | 38 | FilePath SocketInfoReader::GetTcpv4SocketInfoFilePath() const { |
| 39 | return FilePath(kTcpv4SocketInfoFilePath); |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 40 | } |
| 41 | |
Ben Chan | 320d04a | 2013-04-09 13:10:53 -0700 | [diff] [blame] | 42 | FilePath SocketInfoReader::GetTcpv6SocketInfoFilePath() const { |
| 43 | return FilePath(kTcpv6SocketInfoFilePath); |
| 44 | } |
| 45 | |
| 46 | bool SocketInfoReader::LoadTcpSocketInfo(vector<SocketInfo> *info_list) { |
| 47 | info_list->clear(); |
| 48 | bool v4_loaded = AppendSocketInfo(GetTcpv4SocketInfoFilePath(), info_list); |
| 49 | bool v6_loaded = AppendSocketInfo(GetTcpv6SocketInfoFilePath(), info_list); |
| 50 | // Return true if we can load either /proc/net/tcp or /proc/net/tcp6 |
| 51 | // successfully. |
| 52 | return v4_loaded || v6_loaded; |
| 53 | } |
| 54 | |
| 55 | bool SocketInfoReader::AppendSocketInfo(const FilePath &info_file_path, |
| 56 | vector<SocketInfo> *info_list) { |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 57 | FileReader file_reader; |
| 58 | if (!file_reader.Open(info_file_path)) { |
Rebecca Silberstein | c9c31d8 | 2014-10-21 15:01:00 -0700 | [diff] [blame] | 59 | SLOG(this, 2) << __func__ << ": Failed to open '" |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 60 | << info_file_path.value() << "'."; |
| 61 | return false; |
| 62 | } |
| 63 | |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 64 | string line; |
| 65 | while (file_reader.ReadLine(&line)) { |
| 66 | SocketInfo socket_info; |
| 67 | if (ParseSocketInfo(line, &socket_info)) |
| 68 | info_list->push_back(socket_info); |
| 69 | } |
| 70 | return true; |
| 71 | } |
| 72 | |
| 73 | bool SocketInfoReader::ParseSocketInfo(const string &input, |
| 74 | SocketInfo *socket_info) { |
| 75 | vector<string> tokens; |
| 76 | base::SplitStringAlongWhitespace(input, &tokens); |
| 77 | if (tokens.size() < 10) { |
| 78 | return false; |
| 79 | } |
| 80 | |
| 81 | IPAddress ip_address(IPAddress::kFamilyUnknown); |
Ben Chan | 7fab897 | 2014-08-10 17:14:46 -0700 | [diff] [blame] | 82 | uint16_t port = 0; |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 83 | |
| 84 | if (!ParseIPAddressAndPort(tokens[1], &ip_address, &port)) { |
| 85 | return false; |
| 86 | } |
| 87 | socket_info->set_local_ip_address(ip_address); |
| 88 | socket_info->set_local_port(port); |
| 89 | |
| 90 | if (!ParseIPAddressAndPort(tokens[2], &ip_address, &port)) { |
| 91 | return false; |
| 92 | } |
| 93 | socket_info->set_remote_ip_address(ip_address); |
| 94 | socket_info->set_remote_port(port); |
| 95 | |
| 96 | SocketInfo::ConnectionState connection_state = |
| 97 | SocketInfo::kConnectionStateUnknown; |
| 98 | if (!ParseConnectionState(tokens[3], &connection_state)) { |
| 99 | return false; |
| 100 | } |
| 101 | socket_info->set_connection_state(connection_state); |
| 102 | |
Ben Chan | 7fab897 | 2014-08-10 17:14:46 -0700 | [diff] [blame] | 103 | uint64_t transmit_queue_value = 0, receive_queue_value = 0; |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 104 | if (!ParseTransimitAndReceiveQueueValues( |
| 105 | tokens[4], &transmit_queue_value, &receive_queue_value)) { |
| 106 | return false; |
| 107 | } |
| 108 | socket_info->set_transmit_queue_value(transmit_queue_value); |
| 109 | socket_info->set_receive_queue_value(receive_queue_value); |
| 110 | |
| 111 | SocketInfo::TimerState timer_state = SocketInfo::kTimerStateUnknown; |
| 112 | if (!ParseTimerState(tokens[5], &timer_state)) { |
| 113 | return false; |
| 114 | } |
| 115 | socket_info->set_timer_state(timer_state); |
| 116 | |
| 117 | return true; |
| 118 | } |
| 119 | |
| 120 | bool SocketInfoReader::ParseIPAddressAndPort( |
Ben Chan | 7fab897 | 2014-08-10 17:14:46 -0700 | [diff] [blame] | 121 | const string &input, IPAddress *ip_address, uint16_t *port) { |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 122 | vector<string> tokens; |
| 123 | |
| 124 | base::SplitString(input, ':', &tokens); |
| 125 | if (tokens.size() != 2 || |
| 126 | !ParseIPAddress(tokens[0], ip_address) || |
| 127 | !ParsePort(tokens[1], port)) { |
| 128 | return false; |
| 129 | } |
| 130 | |
| 131 | return true; |
| 132 | } |
| 133 | |
| 134 | bool SocketInfoReader::ParseIPAddress(const string &input, |
| 135 | IPAddress *ip_address) { |
Ben Chan | 320d04a | 2013-04-09 13:10:53 -0700 | [diff] [blame] | 136 | ByteString byte_string = ByteString::CreateFromHexString(input); |
| 137 | if (byte_string.IsEmpty()) |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 138 | return false; |
| 139 | |
| 140 | IPAddress::Family family; |
Ben Chan | 320d04a | 2013-04-09 13:10:53 -0700 | [diff] [blame] | 141 | if (byte_string.GetLength() == |
| 142 | IPAddress::GetAddressLength(IPAddress::kFamilyIPv4)) { |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 143 | family = IPAddress::kFamilyIPv4; |
Ben Chan | 320d04a | 2013-04-09 13:10:53 -0700 | [diff] [blame] | 144 | } else if (byte_string.GetLength() == |
| 145 | IPAddress::GetAddressLength(IPAddress::kFamilyIPv6)) { |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 146 | family = IPAddress::kFamilyIPv6; |
Ben Chan | 320d04a | 2013-04-09 13:10:53 -0700 | [diff] [blame] | 147 | } else { |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 148 | return false; |
Ben Chan | 320d04a | 2013-04-09 13:10:53 -0700 | [diff] [blame] | 149 | } |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 150 | |
Ben Chan | 320d04a | 2013-04-09 13:10:53 -0700 | [diff] [blame] | 151 | // Linux kernel prints out IP addresses in network order via |
| 152 | // /proc/net/tcp{,6}. |
| 153 | byte_string.ConvertFromNetToCPUUInt32Array(); |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 154 | |
Ben Chan | 320d04a | 2013-04-09 13:10:53 -0700 | [diff] [blame] | 155 | *ip_address = IPAddress(family, byte_string); |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 156 | return true; |
| 157 | } |
| 158 | |
Ben Chan | 7fab897 | 2014-08-10 17:14:46 -0700 | [diff] [blame] | 159 | bool SocketInfoReader::ParsePort(const string &input, uint16_t *port) { |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 160 | int result = 0; |
| 161 | |
| 162 | if (input.size() != 4 || !base::HexStringToInt(input, &result) || |
Ben Chan | 7fab897 | 2014-08-10 17:14:46 -0700 | [diff] [blame] | 163 | result < 0 || result > std::numeric_limits<uint16_t>::max()) { |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 164 | return false; |
| 165 | } |
| 166 | |
| 167 | *port = result; |
| 168 | return true; |
| 169 | } |
| 170 | |
| 171 | bool SocketInfoReader::ParseTransimitAndReceiveQueueValues( |
| 172 | const string &input, |
Ben Chan | 7fab897 | 2014-08-10 17:14:46 -0700 | [diff] [blame] | 173 | uint64_t *transmit_queue_value, uint64_t *receive_queue_value) { |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 174 | vector<string> tokens; |
Ben Chan | 7fab897 | 2014-08-10 17:14:46 -0700 | [diff] [blame] | 175 | int64_t signed_transmit_queue_value = 0, signed_receive_queue_value = 0; |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 176 | |
| 177 | base::SplitString(input, ':', &tokens); |
| 178 | if (tokens.size() != 2 || |
| 179 | !base::HexStringToInt64(tokens[0], &signed_transmit_queue_value) || |
| 180 | !base::HexStringToInt64(tokens[1], &signed_receive_queue_value)) { |
| 181 | return false; |
| 182 | } |
| 183 | |
Ben Chan | 7fab897 | 2014-08-10 17:14:46 -0700 | [diff] [blame] | 184 | *transmit_queue_value = static_cast<uint64_t>(signed_transmit_queue_value); |
| 185 | *receive_queue_value = static_cast<uint64_t>(signed_receive_queue_value); |
Ben Chan | 3e4bf16 | 2013-04-03 18:14:51 -0700 | [diff] [blame] | 186 | return true; |
| 187 | } |
| 188 | |
| 189 | bool SocketInfoReader::ParseConnectionState( |
| 190 | const string &input, SocketInfo::ConnectionState *connection_state) { |
| 191 | int result = 0; |
| 192 | |
| 193 | if (input.size() != 2 || !base::HexStringToInt(input, &result)) { |
| 194 | return false; |
| 195 | } |
| 196 | |
| 197 | if (result > 0 && result < SocketInfo::kConnectionStateMax) { |
| 198 | *connection_state = static_cast<SocketInfo::ConnectionState>(result); |
| 199 | } else { |
| 200 | *connection_state = SocketInfo::kConnectionStateUnknown; |
| 201 | } |
| 202 | return true; |
| 203 | } |
| 204 | |
| 205 | bool SocketInfoReader::ParseTimerState( |
| 206 | const string &input, SocketInfo::TimerState *timer_state) { |
| 207 | vector<string> tokens; |
| 208 | int result = 0; |
| 209 | |
| 210 | base::SplitString(input, ':', &tokens); |
| 211 | if (tokens.size() != 2 || tokens[0].size() != 2 || |
| 212 | !base::HexStringToInt(tokens[0], &result)) { |
| 213 | return false; |
| 214 | } |
| 215 | |
| 216 | if (result < SocketInfo::kTimerStateMax) { |
| 217 | *timer_state = static_cast<SocketInfo::TimerState>(result); |
| 218 | } else { |
| 219 | *timer_state = SocketInfo::kTimerStateUnknown; |
| 220 | } |
| 221 | return true; |
| 222 | } |
| 223 | |
| 224 | } // namespace shill |