blob: 561f9cbc1901c4741ebfc5be217b78838e023dbb [file] [log] [blame]
Yifan Hong443df792017-05-09 18:49:45 -07001/*
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
Steven Moreland0b8e3872019-06-25 08:54:34 -070017#pragma once
Yifan Hong443df792017-05-09 18:49:45 -070018
Yifan Honga6b93f02017-09-13 16:53:37 -070019#include <getopt.h>
Yifan Hong443df792017-05-09 18:49:45 -070020#include <stdint.h>
21
22#include <fstream>
23#include <string>
24#include <vector>
25
26#include <android-base/macros.h>
27#include <android/hidl/manager/1.0/IServiceManager.h>
Devin Moorec03e3aa2020-12-11 15:11:17 -080028#include <binderdebug/BinderDebug.h>
Yifan Hongb2d096a2018-05-01 15:25:23 -070029#include <hidl-util/FqInstance.h>
30#include <vintf/HalManifest.h>
Yifan Hongbdf44f82018-05-25 14:20:00 -070031#include <vintf/VintfObject.h>
Yifan Hong443df792017-05-09 18:49:45 -070032
Yifan Hongded398e2017-09-07 13:54:28 -070033#include "Command.h"
Yifan Hong443df792017-05-09 18:49:45 -070034#include "NullableOStream.h"
35#include "TableEntry.h"
Yifan Hong1bc1e9f2017-08-29 17:28:12 -070036#include "TextTable.h"
Yifan Hong443df792017-05-09 18:49:45 -070037#include "utils.h"
38
39namespace android {
40namespace lshal {
41
42class Lshal;
43
Nirav Atrecce988d2018-05-16 11:14:46 -070044enum class HalType {
45 BINDERIZED_SERVICES = 0,
46 PASSTHROUGH_CLIENTS,
Yifan Hong13ba0a92018-06-25 16:15:56 -070047 PASSTHROUGH_LIBRARIES,
48 VINTF_MANIFEST,
Yifan Hong3212f172018-06-28 12:39:50 -070049 LAZY_HALS,
Yifan Hong30528a22020-08-07 18:24:06 -070050
51 // Not a real HalType. Used to determine all HalTypes.
52 LAST,
Nirav Atrecce988d2018-05-16 11:14:46 -070053};
54
Yifan Hongded398e2017-09-07 13:54:28 -070055class ListCommand : public Command {
Yifan Hong443df792017-05-09 18:49:45 -070056public:
Chih-Hung Hsieh45e31c72018-12-20 15:47:01 -080057 explicit ListCommand(Lshal &lshal) : Command(lshal) {}
Yifan Hong8bf73162017-09-07 18:06:13 -070058 virtual ~ListCommand() = default;
Yifan Honga8bedc62017-09-08 18:00:31 -070059 Status main(const Arg &arg) override;
60 void usage() const override;
Yifan Hong795b6ec2017-09-13 11:25:28 -070061 std::string getSimpleDescription() const override;
62 std::string getName() const override { return GetName(); }
63
64 static std::string GetName();
Yifan Honga6b93f02017-09-13 16:53:37 -070065
66 struct RegisteredOption {
67 // short alternative, e.g. 'v'. If '\0', no short options is available.
68 char shortOption;
69 // long alternative, e.g. 'init-vintf'
70 std::string longOption;
71 // no_argument, required_argument or optional_argument
72 int hasArg;
73 // value written to 'flag' by getopt_long
74 int val;
75 // operation when the argument is present
76 std::function<Status(ListCommand* thiz, const char* arg)> op;
77 // help message
78 std::string help;
79
80 const std::string& getHelpMessageForArgument() const;
81 };
82 // A list of acceptable command line options
83 // key: value returned by getopt_long
84 using RegisteredOptions = std::vector<RegisteredOption>;
85
Yifan Hongf31aa052018-02-02 15:17:51 -080086 static std::string INIT_VINTF_NOTES;
87
Yifan Hongb2a2ecb2017-09-07 15:08:22 -070088protected:
Yifan Honga8bedc62017-09-08 18:00:31 -070089 Status parseArgs(const Arg &arg);
Yifan Hongbdf44f82018-05-25 14:20:00 -070090 // Retrieve first-hand information
Yifan Hong443df792017-05-09 18:49:45 -070091 Status fetch();
Yifan Hongbdf44f82018-05-25 14:20:00 -070092 // Retrieve derived information base on existing table
Yifan Hong93b8bff2017-09-14 16:02:52 -070093 virtual void postprocess();
Yifan Hongca3b6602017-09-07 16:44:27 -070094 Status dump();
Yifan Hong20f4ee82018-06-25 16:21:29 -070095 void putEntry(HalType type, TableEntry &&entry);
Yifan Hong443df792017-05-09 18:49:45 -070096 Status fetchPassthrough(const sp<::android::hidl::manager::V1_0::IServiceManager> &manager);
97 Status fetchBinderized(const sp<::android::hidl::manager::V1_0::IServiceManager> &manager);
98 Status fetchAllLibraries(const sp<::android::hidl::manager::V1_0::IServiceManager> &manager);
Yifan Hong13ba0a92018-06-25 16:15:56 -070099 Status fetchManifestHals();
Yifan Hong3212f172018-06-28 12:39:50 -0700100 Status fetchLazyHals();
Steven Morelandd8e20192017-05-24 11:23:08 -0700101
Yifan Hong22ea7b82017-09-14 18:07:43 -0700102 Status fetchBinderizedEntry(const sp<::android::hidl::manager::V1_0::IServiceManager> &manager,
103 TableEntry *entry);
104
Hridya Valsarajub49b0b12020-02-25 12:27:14 -0800105 // Get relevant information for a PID by parsing files under
106 // /dev/binderfs/binder_logs or /d/binder.
Yifan Hong1243dde2017-09-14 17:49:30 -0700107 // It is a virtual member function so that it can be mocked.
Devin Moorec03e3aa2020-12-11 15:11:17 -0800108 virtual bool getPidInfo(pid_t serverPid, BinderPidInfo *info) const;
Yifan Hong1243dde2017-09-14 17:49:30 -0700109 // Retrieve from mCachedPidInfos and call getPidInfo if necessary.
Devin Moorec03e3aa2020-12-11 15:11:17 -0800110 const BinderPidInfo* getPidInfoCached(pid_t serverPid);
Steven Morelandd8e20192017-05-24 11:23:08 -0700111
Yifan Hongca3b6602017-09-07 16:44:27 -0700112 void dumpTable(const NullableOStream<std::ostream>& out) const;
113 void dumpVintf(const NullableOStream<std::ostream>& out) const;
Yifan Hong1bc1e9f2017-08-29 17:28:12 -0700114 void addLine(TextTable *table, const std::string &interfaceName, const std::string &transport,
115 const std::string &arch, const std::string &threadUsage, const std::string &server,
116 const std::string &serverCmdline, const std::string &address,
117 const std::string &clients, const std::string &clientCmdlines) const;
118 void addLine(TextTable *table, const TableEntry &entry);
Yifan Hong8bf73162017-09-07 18:06:13 -0700119 // Read and return /proc/{pid}/cmdline.
120 virtual std::string parseCmdline(pid_t pid) const;
Yifan Hong443df792017-05-09 18:49:45 -0700121 // Return /proc/{pid}/cmdline if it exists, else empty string.
Yifan Hongf31aa052018-02-02 15:17:51 -0800122 const std::string& getCmdline(pid_t pid);
Yifan Hong443df792017-05-09 18:49:45 -0700123 // Call getCmdline on all pid in pids. If it returns empty string, the process might
124 // have died, and the pid is removed from pids.
125 void removeDeadProcesses(Pids *pids);
Yifan Hongf31aa052018-02-02 15:17:51 -0800126
127 virtual Partition getPartition(pid_t pid);
Yifan Hongb2d096a2018-05-01 15:25:23 -0700128 Partition resolvePartition(Partition processPartition, const FqInstance &fqInstance) const;
Yifan Hongf31aa052018-02-02 15:17:51 -0800129
Yifan Hongbdf44f82018-05-25 14:20:00 -0700130 VintfInfo getVintfInfo(const std::string &fqInstanceName, vintf::TransportArch ta) const;
131 // Allow to mock these functions for testing.
132 virtual std::shared_ptr<const vintf::HalManifest> getDeviceManifest() const;
133 virtual std::shared_ptr<const vintf::CompatibilityMatrix> getDeviceMatrix() const;
134 virtual std::shared_ptr<const vintf::HalManifest> getFrameworkManifest() const;
135 virtual std::shared_ptr<const vintf::CompatibilityMatrix> getFrameworkMatrix() const;
136
Yifan Hong443df792017-05-09 18:49:45 -0700137 void forEachTable(const std::function<void(Table &)> &f);
138 void forEachTable(const std::function<void(const Table &)> &f) const;
Yifan Hongdb730532018-06-25 16:32:01 -0700139 Table* tableForType(HalType type);
140 const Table* tableForType(HalType type) const;
Yifan Hong443df792017-05-09 18:49:45 -0700141
Yifan Hong76ac14a2017-09-08 14:59:04 -0700142 NullableOStream<std::ostream> err() const;
143 NullableOStream<std::ostream> out() const;
144
Yifan Honga6b93f02017-09-13 16:53:37 -0700145 void registerAllOptions();
146
Yifan Hongb2d096a2018-05-01 15:25:23 -0700147 // helper functions to dumpVintf.
148 bool addEntryWithInstance(const TableEntry &entry, vintf::HalManifest *manifest) const;
149 bool addEntryWithoutInstance(const TableEntry &entry, const vintf::HalManifest *manifest) const;
150
Yifan Hong13ba0a92018-06-25 16:15:56 -0700151 // Helper function. Whether to fetch entries corresponding to a given HAL type.
152 bool shouldFetchHalType(const HalType &type) const;
153
154 void initFetchTypes();
Nirav Atrecce988d2018-05-16 11:14:46 -0700155
Yifan Hong3212f172018-06-28 12:39:50 -0700156 // Helper functions ti add HALs that are listed in VINTF manifest to LAZY_HALS table.
157 bool hasHwbinderEntry(const TableEntry& entry) const;
158 bool hasPassthroughEntry(const TableEntry& entry) const;
159
Yifan Hong443df792017-05-09 18:49:45 -0700160 Table mServicesTable{};
161 Table mPassthroughRefTable{};
162 Table mImplementationsTable{};
Yifan Hong13ba0a92018-06-25 16:15:56 -0700163 Table mManifestHalsTable{};
Yifan Hong3212f172018-06-28 12:39:50 -0700164 Table mLazyHalsTable{};
Yifan Hong443df792017-05-09 18:49:45 -0700165
Yifan Hongca3b6602017-09-07 16:44:27 -0700166 std::string mFileOutputPath;
Yifan Hong443df792017-05-09 18:49:45 -0700167 TableEntryCompare mSortColumn = nullptr;
Yifan Hong443df792017-05-09 18:49:45 -0700168
Yifan Hong443df792017-05-09 18:49:45 -0700169 bool mEmitDebugInfo = false;
170
Yifan Hongf31aa052018-02-02 15:17:51 -0800171 // If true, output in VINTF format. Output only entries from the specified partition.
Yifan Hong443df792017-05-09 18:49:45 -0700172 bool mVintf = false;
Yifan Hongf31aa052018-02-02 15:17:51 -0800173 Partition mVintfPartition = Partition::UNKNOWN;
Yifan Hong6da06912017-05-12 16:56:43 -0700174
175 // If true, explanatory text are not emitted.
176 bool mNeat = false;
177
Yifan Hong13ba0a92018-06-25 16:15:56 -0700178 // Type(s) of HAL associations to list.
179 std::vector<HalType> mListTypes{};
180 // Type(s) of HAL associations to fetch.
181 std::set<HalType> mFetchTypes{};
Nirav Atrecce988d2018-05-16 11:14:46 -0700182
Yifan Hong443df792017-05-09 18:49:45 -0700183 // If an entry does not exist, need to ask /proc/{pid}/cmdline to get it.
184 // If an entry exist but is an empty string, process might have died.
185 // If an entry exist and not empty, it contains the cached content of /proc/{pid}/cmdline.
186 std::map<pid_t, std::string> mCmdlines;
187
Yifan Hong1243dde2017-09-14 17:49:30 -0700188 // Cache for getPidInfo.
Devin Moorec03e3aa2020-12-11 15:11:17 -0800189 std::map<pid_t, BinderPidInfo> mCachedPidInfos;
Yifan Hong1243dde2017-09-14 17:49:30 -0700190
Yifan Hongf31aa052018-02-02 15:17:51 -0800191 // Cache for getPartition.
192 std::map<pid_t, Partition> mPartitions;
193
Yifan Honga6b93f02017-09-13 16:53:37 -0700194 RegisteredOptions mOptions;
195 // All selected columns
196 std::vector<TableColumnType> mSelectedColumns;
197 // If true, emit cmdlines instead of PIDs
198 bool mEnableCmdlines = false;
199
Yifan Hong91e655d2017-09-13 15:44:56 -0700200private:
Yifan Hong443df792017-05-09 18:49:45 -0700201 DISALLOW_COPY_AND_ASSIGN(ListCommand);
202};
203
204
205} // namespace lshal
206} // namespace android