1 /*
2  * Copyright 2016 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 
17 #include "GpuService.h"
18 
19 #include <binder/Parcel.h>
20 #include <utils/String8.h>
21 #include <vkjson.h>
22 
23 namespace android {
24 
25 // ----------------------------------------------------------------------------
26 
27 class BpGpuService : public BpInterface<IGpuService>
28 {
29 public:
BpGpuService(const sp<IBinder> & impl)30     BpGpuService(const sp<IBinder>& impl) : BpInterface<IGpuService>(impl) {}
31 };
32 
33 IMPLEMENT_META_INTERFACE(GpuService, "android.ui.IGpuService");
34 
onTransact(uint32_t code,const Parcel & data,Parcel * reply,uint32_t flags)35 status_t BnGpuService::onTransact(uint32_t code, const Parcel& data,
36         Parcel* reply, uint32_t flags)
37 {
38     switch (code) {
39     case SHELL_COMMAND_TRANSACTION: {
40         int in = data.readFileDescriptor();
41         int out = data.readFileDescriptor();
42         int err = data.readFileDescriptor();
43         int argc = data.readInt32();
44         Vector<String16> args;
45         for (int i = 0; i < argc && data.dataAvail() > 0; i++) {
46            args.add(data.readString16());
47         }
48         return shellCommand(in, out, err, args);
49     }
50 
51     default:
52         return BBinder::onTransact(code, data, reply, flags);
53     }
54 }
55 
56 // ----------------------------------------------------------------------------
57 
58 namespace {
59     status_t cmd_help(int out);
60     status_t cmd_vkjson(int out, int err);
61 }
62 
63 const char* const GpuService::SERVICE_NAME = "gpu";
64 
GpuService()65 GpuService::GpuService() {}
66 
shellCommand(int,int out,int err,Vector<String16> & args)67 status_t GpuService::shellCommand(int /*in*/, int out, int err,
68         Vector<String16>& args)
69 {
70     ALOGV("GpuService::shellCommand");
71     for (size_t i = 0, n = args.size(); i < n; i++)
72         ALOGV("  arg[%zu]: '%s'", i, String8(args[i]).string());
73 
74     if (args[0] == String16("vkjson"))
75         return cmd_vkjson(out, err);
76     else if (args[0] == String16("help"))
77         return cmd_help(out);
78 
79     return NO_ERROR;
80 }
81 
82 // ----------------------------------------------------------------------------
83 
84 namespace {
85 
cmd_help(int out)86 status_t cmd_help(int out) {
87     FILE* outs = fdopen(out, "w");
88     if (!outs) {
89         ALOGE("vkjson: failed to create out stream: %s (%d)", strerror(errno),
90             errno);
91         return BAD_VALUE;
92     }
93     fprintf(outs,
94         "GPU Service commands:\n"
95         "  vkjson   dump Vulkan device capabilities as JSON\n");
96     fclose(outs);
97     return NO_ERROR;
98 }
99 
vkjsonPrint(FILE * out,FILE * err)100 VkResult vkjsonPrint(FILE* out, FILE* err) {
101     VkResult result;
102 
103     const VkApplicationInfo app_info = {
104         VK_STRUCTURE_TYPE_APPLICATION_INFO, nullptr,
105         "vkjson", 1,    /* app name, version */
106         "", 0,          /* engine name, version */
107         VK_API_VERSION_1_0
108     };
109     const VkInstanceCreateInfo instance_info = {
110         VK_STRUCTURE_TYPE_INSTANCE_CREATE_INFO, nullptr,
111         0,              /* flags */
112         &app_info,
113         0, nullptr,     /* layers */
114         0, nullptr,     /* extensions */
115     };
116     VkInstance instance;
117     result = vkCreateInstance(&instance_info, nullptr, &instance);
118     if (result != VK_SUCCESS) {
119         fprintf(err, "vkCreateInstance failed: %d\n", result);
120         return result;
121     }
122 
123     uint32_t ngpu = 0;
124     result = vkEnumeratePhysicalDevices(instance, &ngpu, nullptr);
125     if (result != VK_SUCCESS) {
126         fprintf(err, "vkEnumeratePhysicalDevices failed: %d\n", result);
127         return result;
128     }
129     std::vector<VkPhysicalDevice> gpus(ngpu, VK_NULL_HANDLE);
130     result = vkEnumeratePhysicalDevices(instance, &ngpu, gpus.data());
131     if (result != VK_SUCCESS) {
132         fprintf(err, "vkEnumeratePhysicalDevices failed: %d\n", result);
133         return result;
134     }
135 
136     for (size_t i = 0, n = gpus.size(); i < n; i++) {
137         auto props = VkJsonGetAllProperties(gpus[i]);
138         std::string json = VkJsonAllPropertiesToJson(props);
139         fwrite(json.data(), 1, json.size(), out);
140         if (i < n - 1)
141             fputc(',', out);
142         fputc('\n', out);
143     }
144 
145     vkDestroyInstance(instance, nullptr);
146 
147     return VK_SUCCESS;
148 }
149 
cmd_vkjson(int out,int err)150 status_t cmd_vkjson(int out, int err) {
151     int errnum;
152     FILE* outs = fdopen(out, "w");
153     if (!outs) {
154         errnum = errno;
155         ALOGE("vkjson: failed to create output stream: %s", strerror(errnum));
156         return -errnum;
157     }
158     FILE* errs = fdopen(err, "w");
159     if (!errs) {
160         errnum = errno;
161         ALOGE("vkjson: failed to create error stream: %s", strerror(errnum));
162         fclose(outs);
163         return -errnum;
164     }
165     fprintf(outs, "[\n");
166     VkResult result = vkjsonPrint(outs, errs);
167     fprintf(outs, "]\n");
168     fclose(errs);
169     fclose(outs);
170     return result >= 0 ? NO_ERROR : UNKNOWN_ERROR;
171 }
172 
173 } // anonymous namespace
174 
175 } // namespace android
176