1 /*
2  * Copyright (C) 2018 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 #if defined(__ANDROID__)
18 
19 #include <cutils/properties.h>
20 #include "Loader.h"
21 #include "egl_angle_platform.h"
22 
23 #pragma GCC diagnostic push
24 #pragma GCC diagnostic ignored "-Wunused-parameter"
25 #include <EGL/Platform.h>
26 #pragma GCC diagnostic pop
27 
28 #include <android/dlext.h>
29 #include <dlfcn.h>
30 #include <graphicsenv/GraphicsEnv.h>
31 #include <time.h>
32 #include <log/log.h>
33 
34 namespace angle {
35 
36 static GetDisplayPlatformFunc angleGetDisplayPlatform = nullptr;
37 static ResetDisplayPlatformFunc angleResetDisplayPlatform = nullptr;
38 
39 static time_t startTime = time(nullptr);
40 
getTraceCategoryEnabledFlag(PlatformMethods *,const char *)41 static const unsigned char* getTraceCategoryEnabledFlag(PlatformMethods* /*platform*/,
42                                                         const char* /*categoryName*/) {
43     // Returning ptr to 'g' (non-zero) to ALWAYS enable tracing initially.
44     // This ptr is what will be passed into "category_group_enabled" of addTraceEvent
45     static const unsigned char traceEnabled = 'g';
46     return &traceEnabled;
47 }
48 
monotonicallyIncreasingTime(PlatformMethods *)49 static double monotonicallyIncreasingTime(PlatformMethods* /*platform*/) {
50     return difftime(time(nullptr), startTime);
51 }
52 
logError(PlatformMethods *,const char * errorMessage)53 static void logError(PlatformMethods* /*platform*/, const char* errorMessage) {
54     ALOGE("ANGLE Error:%s", errorMessage);
55 }
56 
logWarning(PlatformMethods *,const char * warningMessage)57 static void logWarning(PlatformMethods* /*platform*/, const char* warningMessage) {
58     ALOGW("ANGLE Warn:%s", warningMessage);
59 }
60 
logInfo(PlatformMethods *,const char * infoMessage)61 static void logInfo(PlatformMethods* /*platform*/, const char* infoMessage) {
62     ALOGD("ANGLE Info:%s", infoMessage);
63 }
64 
addTraceEvent(PlatformMethods *,char phase,const unsigned char *,const char * name,unsigned long long,double,int,const char **,const unsigned char *,const unsigned long long *,unsigned char)65 static TraceEventHandle addTraceEvent(
66         PlatformMethods* /**platform*/, char phase, const unsigned char* /*category_group_enabled*/,
67         const char* name, unsigned long long /*id*/, double /*timestamp*/, int /*num_args*/,
68         const char** /*arg_names*/, const unsigned char* /*arg_types*/,
69         const unsigned long long* /*arg_values*/, unsigned char /*flags*/) {
70     switch (phase) {
71         case 'B': {
72             ATRACE_BEGIN(name);
73             break;
74         }
75         case 'E': {
76             ATRACE_END();
77             break;
78         }
79         case 'I': {
80             ATRACE_NAME(name);
81             break;
82         }
83         default:
84             // Could handle other event types here
85             break;
86     }
87     // Return any non-zero handle to avoid assert in ANGLE
88     TraceEventHandle result = 1.0;
89     return result;
90 }
91 
assignAnglePlatformMethods(PlatformMethods * platformMethods)92 static void assignAnglePlatformMethods(PlatformMethods* platformMethods) {
93     platformMethods->addTraceEvent = addTraceEvent;
94     platformMethods->getTraceCategoryEnabledFlag = getTraceCategoryEnabledFlag;
95     platformMethods->monotonicallyIncreasingTime = monotonicallyIncreasingTime;
96     platformMethods->logError = logError;
97     platformMethods->logWarning = logWarning;
98     platformMethods->logInfo = logInfo;
99 }
100 
101 // Initialize function ptrs for ANGLE PlatformMethods struct, used for systrace
initializeAnglePlatform(EGLDisplay dpy)102 bool initializeAnglePlatform(EGLDisplay dpy) {
103     // Since we're inside libEGL, use dlsym to lookup fptr for ANGLEGetDisplayPlatform
104     android_namespace_t* ns = android::GraphicsEnv::getInstance().getAngleNamespace();
105     const android_dlextinfo dlextinfo = {
106             .flags = ANDROID_DLEXT_USE_NAMESPACE,
107             .library_namespace = ns,
108     };
109     void* so = android_dlopen_ext("libGLESv2_angle.so", RTLD_LOCAL | RTLD_NOW, &dlextinfo);
110     angleGetDisplayPlatform =
111             reinterpret_cast<GetDisplayPlatformFunc>(dlsym(so, "ANGLEGetDisplayPlatform"));
112 
113     if (!angleGetDisplayPlatform) {
114         ALOGE("dlsym lookup of ANGLEGetDisplayPlatform in libEGL_angle failed!");
115         return false;
116     }
117 
118     angleResetDisplayPlatform =
119             reinterpret_cast<ResetDisplayPlatformFunc>(
120                     eglGetProcAddress("ANGLEResetDisplayPlatform"));
121 
122     PlatformMethods* platformMethods = nullptr;
123     if (!((angleGetDisplayPlatform)(dpy, g_PlatformMethodNames,
124                                                               g_NumPlatformMethods, nullptr,
125                                                               &platformMethods))) {
126         ALOGE("ANGLEGetDisplayPlatform call failed!");
127         return false;
128     }
129     if (platformMethods) {
130         assignAnglePlatformMethods(platformMethods);
131     } else {
132         ALOGE("In initializeAnglePlatform() platformMethods struct ptr is NULL. Not assigning "
133               "tracing function ptrs!");
134     }
135     return true;
136 }
137 
resetAnglePlatform(EGLDisplay dpy)138 void resetAnglePlatform(EGLDisplay dpy) {
139     if (angleResetDisplayPlatform) {
140         angleResetDisplayPlatform(dpy);
141     }
142 }
143 
144 }; // namespace angle
145 
146 #endif // __ANDROID__
147