1 /*
2  * Copyright (C) 2017 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 #define LOG_TAG "lshal"
17 #include <android-base/logging.h>
18 
19 #include <map>
20 
21 #include <android-base/hex.h>
22 #include <android-base/strings.h>
23 #include <hidl-hash/Hash.h>
24 #include <vintf/parse_string.h>
25 
26 #include "TableEntry.h"
27 
28 #include "TextTable.h"
29 #include "utils.h"
30 
31 namespace android {
32 namespace lshal {
33 
getArchString(vintf::Arch arch)34 static const std::string &getArchString(vintf::Arch arch) {
35     static const std::string sStr64 = "64";
36     static const std::string sStr32 = "32";
37     static const std::string sStrBoth = "32+64";
38     static const std::string sStrUnknown = "?";
39     switch (arch) {
40         case vintf::Arch::ARCH_64:
41             return sStr64;
42         case vintf::Arch::ARCH_32:
43             return sStr32;
44         case vintf::Arch::ARCH_32_64:
45             return sStrBoth;
46         case vintf::Arch::ARCH_EMPTY: // fall through
47         default:
48             return sStrUnknown;
49     }
50 }
51 
getTitle(TableColumnType type)52 static std::string getTitle(TableColumnType type) {
53     switch (type) {
54         case TableColumnType::INTERFACE_NAME:   return "Interface";
55         case TableColumnType::TRANSPORT:        return "Transport";
56         case TableColumnType::SERVER_PID:       return "Server";
57         case TableColumnType::SERVER_CMD:       return "Server CMD";
58         case TableColumnType::SERVER_ADDR:      return "PTR";
59         case TableColumnType::CLIENT_PIDS:      return "Clients";
60         case TableColumnType::CLIENT_CMDS:      return "Clients CMD";
61         case TableColumnType::ARCH:             return "Arch";
62         case TableColumnType::THREADS:          return "Thread Use";
63         case TableColumnType::RELEASED:         return "R";
64         case TableColumnType::HASH:             return "Hash";
65         case TableColumnType::VINTF:            return "VINTF";
66         case TableColumnType::SERVICE_STATUS:   return "Status";
67         default:
68             LOG(FATAL) << __func__ << "Should not reach here. " << static_cast<int>(type);
69             return "";
70     }
71 }
72 
getField(TableColumnType type) const73 std::string TableEntry::getField(TableColumnType type) const {
74     switch (type) {
75         case TableColumnType::INTERFACE_NAME:
76             return interfaceName;
77         case TableColumnType::TRANSPORT:
78             return vintf::to_string(transport);
79         case TableColumnType::SERVER_PID:
80             return serverPid == NO_PID ? "N/A" : std::to_string(serverPid);
81         case TableColumnType::SERVER_CMD:
82             return serverCmdline;
83         case TableColumnType::SERVER_ADDR:
84             return serverObjectAddress == NO_PTR ? "N/A" : toHexString(serverObjectAddress);
85         case TableColumnType::CLIENT_PIDS:
86             return join(clientPids, " ");
87         case TableColumnType::CLIENT_CMDS:
88             return join(clientCmdlines, ";");
89         case TableColumnType::ARCH:
90             return getArchString(arch);
91         case TableColumnType::THREADS:
92             return getThreadUsage();
93         case TableColumnType::RELEASED:
94             return isReleased();
95         case TableColumnType::HASH:
96             return hash;
97         case TableColumnType::VINTF:
98             return getVintfInfo();
99         case TableColumnType::SERVICE_STATUS:
100             return lshal::to_string(serviceStatus);
101         default:
102             LOG(FATAL) << __func__ << "Should not reach here. " << static_cast<int>(type);
103             return "";
104     }
105 }
106 
isReleased() const107 std::string TableEntry::isReleased() const {
108     static const std::string unreleased = android::base::HexString(Hash::kEmptyHash.data(),
109                                                                    Hash::kEmptyHash.size());
110 
111     if (hash.empty()) {
112         return "?";
113     }
114     if (hash == unreleased) {
115         return "N"; // unknown or unreleased
116     }
117     return "Y"; // released
118 }
119 
getVintfInfo() const120 std::string TableEntry::getVintfInfo() const {
121     static const std::map<VintfInfo, std::string> values{
122             {DEVICE_MANIFEST, "DM"},
123             {DEVICE_MATRIX, "DC"},
124             {FRAMEWORK_MANIFEST, "FM"},
125             {FRAMEWORK_MATRIX, "FC"},
126     };
127     std::vector<std::string> ret;
128     for (const auto& pair : values) {
129         if (vintfInfo & pair.first) {
130             ret.push_back(pair.second);
131         }
132     }
133     auto joined = base::Join(ret, ',');
134     return joined.empty() ? "X" : joined;
135 }
136 
to_string(ServiceStatus s)137 std::string to_string(ServiceStatus s) {
138     switch (s) {
139         case ServiceStatus::ALIVE: return "alive";
140         case ServiceStatus::NON_RESPONSIVE: return "non-responsive";
141         case ServiceStatus::DECLARED: return "declared";
142         case ServiceStatus::UNKNOWN: return "N/A";
143     }
144 
145     LOG(FATAL) << __func__ << "Should not reach here." << static_cast<int>(s);
146     return "";
147 }
148 
createTextTable(bool neat,const std::function<std::string (const std::string &)> & emitDebugInfo) const149 TextTable Table::createTextTable(bool neat,
150     const std::function<std::string(const std::string&)>& emitDebugInfo) const {
151 
152     TextTable textTable;
153     std::vector<std::string> row;
154     if (!neat) {
155         textTable.add(mDescription);
156 
157         row.clear();
158         for (TableColumnType type : mSelectedColumns) {
159             row.push_back(getTitle(type));
160         }
161         textTable.add(std::move(row));
162     }
163 
164     for (const auto& entry : mEntries) {
165         row.clear();
166         for (TableColumnType type : mSelectedColumns) {
167             row.push_back(entry.getField(type));
168         }
169         textTable.add(std::move(row));
170 
171         if (emitDebugInfo) {
172             std::string debugInfo = emitDebugInfo(entry.interfaceName);
173             if (!debugInfo.empty()) textTable.add(debugInfo);
174         }
175     }
176     return textTable;
177 }
178 
createTextTable()179 TextTable MergedTable::createTextTable() {
180     TextTable textTable;
181     for (const Table* table : mTables) {
182         textTable.addAll(table->createTextTable());
183     }
184     return textTable;
185 }
186 
operator ==(const TableEntry & other) const187 bool TableEntry::operator==(const TableEntry& other) const {
188     if (this == &other) {
189         return true;
190     }
191     return interfaceName == other.interfaceName && transport == other.transport &&
192         serverPid == other.serverPid && threadUsage == other.threadUsage &&
193         threadCount == other.threadCount && serverCmdline == other.serverCmdline &&
194         serverObjectAddress == other.serverObjectAddress && clientPids == other.clientPids &&
195         clientCmdlines == other.clientCmdlines && arch == other.arch;
196 }
197 
to_string() const198 std::string TableEntry::to_string() const {
199     using vintf::operator<<;
200     std::stringstream ss;
201     ss << "name=" << interfaceName << ";transport=" << transport << ";thread=" << getThreadUsage()
202        << ";server=" << serverPid
203        << "(" << serverObjectAddress << ";" << serverCmdline << ");clients=["
204        << join(clientPids, ";") << "](" << join(clientCmdlines, ";") << ");arch="
205        << getArchString(arch);
206     return ss.str();
207 
208 }
209 
210 } // namespace lshal
211 } // namespace android
212