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