/* * Copyright (C) 2016 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifndef _RESOLVER_STATS_H_ #define _RESOLVER_STATS_H_ #include <time.h> namespace android { namespace net { struct ResolverStats { // Offsets into the per-server resolver stats as encoded in vector<int32_t> stats of // getResolverInfo() of Netd's binder interface. The stats are based on data reported by // android_net_res_stats_get_info_for_net(), the usability is calculated by applying // android_net_res_stats_get_usable_servers() to this data. enum ResolverStatsOffsets { STATS_SUCCESSES = 0, // # successes counted for this server STATS_ERRORS, // # errors STATS_TIMEOUTS, // # timeouts STATS_INTERNAL_ERRORS, // # internal errors STATS_RTT_AVG, // average round-trip-time STATS_LAST_SAMPLE_TIME, // time in s when the last sample was recorded STATS_USABLE, // whether the server is considered usable STATS_COUNT // total count of integers in the per-server data }; int successes{-1}; int errors{-1}; int timeouts{-1}; int internal_errors{-1}; int rtt_avg{-1}; time_t last_sample_time{0}; bool usable{false}; // Serialize the resolver stats to the end of |out|. void encode(std::vector<int32_t>* out) const; // Read the serialized resolverstats starting at |in[ofs]|. ssize_t decode(const std::vector<int32_t>& in, ssize_t ofs); // Serialize the contents of |stats| and append them to the end of |out|. Multiple arrays // can be written to the same output vector in sequence, however, the corresponding call // to decodeAll() will return the combined contents in one vector. static void encodeAll(const std::vector<ResolverStats>& stats, std::vector<int32_t>* out); // Decodes the serialized ResolverStats from |in| and appends them to stats. static bool decodeAll(const std::vector<int32_t>& in, std::vector<ResolverStats>* stats); }; inline void ResolverStats::encode(std::vector<int32_t>* out) const { size_t ofs = out->size(); out->resize(ofs + STATS_COUNT); int32_t* cur = &(*out)[ofs]; cur[STATS_SUCCESSES] = successes; cur[STATS_ERRORS] = errors; cur[STATS_TIMEOUTS] = timeouts; cur[STATS_INTERNAL_ERRORS] = internal_errors; cur[STATS_RTT_AVG] = rtt_avg; cur[STATS_LAST_SAMPLE_TIME] = last_sample_time; cur[STATS_USABLE] = usable; } // Read the serialized resolverstats starting at |in[ofs]|. inline ssize_t ResolverStats::decode(const std::vector<int32_t>& in, ssize_t ofs) { if (ofs < 0 || static_cast<size_t>(ofs) + STATS_COUNT > in.size()) { return -1; } const int32_t* cur = &in[ofs]; successes = cur[STATS_SUCCESSES]; errors = cur[STATS_ERRORS]; timeouts = cur[STATS_TIMEOUTS]; internal_errors = cur[STATS_INTERNAL_ERRORS]; rtt_avg = cur[STATS_RTT_AVG]; last_sample_time = cur[STATS_LAST_SAMPLE_TIME]; usable = cur[STATS_USABLE]; return ofs + STATS_COUNT; } inline void ResolverStats::encodeAll(const std::vector<ResolverStats>& stats, std::vector<int32_t>* out) { for (const auto& s : stats) { s.encode(out); } } // TODO: Replace with a better representation, e.g. a Parcelable. inline bool ResolverStats::decodeAll(const std::vector<int32_t>& in, std::vector<ResolverStats>* stats) { ssize_t size = in.size(); if (size % STATS_COUNT) { return false; } stats->resize(size / STATS_COUNT); ssize_t ofs = 0; for (auto& s : *stats) { ofs = s.decode(in, ofs); if (ofs < 0) { return false; } } return true; } } // namespace net } // namespace android #endif /* _RESOLVER_STATS_H_ */