blob: 90d3fa204674fa461971696bc38044b584f8ca61 [file] [log] [blame]
/*
* Copyright (C) 2015 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "../dl_loader.h"
#include "../get_gles_proc_address.h"
#include "../log.h"
#include <string>
#include <unordered_map>
#if defined(__LP64__)
#define SYSTEM_LIB_PATH "/system/lib64/"
#else
#define SYSTEM_LIB_PATH "/system/lib/"
#endif
namespace {
void* ResolveSymbol(const char* name, bool bypassLocal) {
using namespace gapic;
typedef void* (*GPAPROC)(const char *name);
if (bypassLocal) {
static DlLoader libegl(SYSTEM_LIB_PATH "libEGL.so");
if (void* proc = libegl.lookup(name)) {
GAPID_INFO("GetGlesProcAddress(%s, %d) -> 0x%x (from libEGL dlsym)", name, bypassLocal, proc);
return proc;
}
static DlLoader libglesv2(SYSTEM_LIB_PATH "libGLESv2.so");
if (void* proc = libglesv2.lookup(name)) {
GAPID_INFO("GetGlesProcAddress(%s, %d) -> 0x%x (from libGLESv2 dlsym)", name, bypassLocal, proc);
return proc;
}
static DlLoader libglesv1(SYSTEM_LIB_PATH "libGLESv1_CM.so");
if (void* proc = libglesv1.lookup(name)) {
GAPID_INFO("GetGlesProcAddress(%s, %d) -> 0x%x (from libGLESv1_CM dlsym)", name, bypassLocal, proc);
return proc;
}
if (GPAPROC gpa = reinterpret_cast<GPAPROC>(libegl.lookup("eglGetProcAddress"))) {
if (void* proc = gpa(name)) {
static DlLoader local(nullptr);
void* local_proc = local.lookup(name);
if (local_proc == proc) {
GAPID_WARNING(
"libEGL eglGetProcAddress returned a local address %p for %s, this will be ignored", proc, name);
} else {
GAPID_INFO("GetGlesProcAddress(%s, %d) -> 0x%x (via libEGL eglGetProcAddress)", name, bypassLocal, proc);
return proc;
}
}
}
} else {
static DlLoader local(nullptr);
if (GPAPROC gpa = reinterpret_cast<GPAPROC>(local.lookup("eglGetProcAddress"))) {
if (void* proc = gpa(name)) {
GAPID_INFO("GetGlesProcAddress(%s, %d) -> 0x%x (via local eglGetProcAddress)", name, bypassLocal, proc);
return proc;
}
}
if (void* proc = local.lookup(name)) {
GAPID_INFO("GetGlesProcAddress(%s, %d) -> 0x%x (from local dlsym)", name, bypassLocal, proc);
return proc;
}
}
GAPID_INFO("GetGlesProcAddress(%s, %d) -> not found", name, bypassLocal);
return nullptr;
}
void* getGlesProcAddress(const char* name, bool bypassLocal) {
static std::unordered_map<std::string, void*> cache;
const std::string cacheKey = std::string(name) + (bypassLocal ? "/direct" : "/local");
auto it = cache.find(cacheKey);
if (it != cache.end()) {
GAPID_INFO("GetGlesProcAddress(%s, %d) -> 0x%x (from cache)", name, bypassLocal, it->second);
return it->second;
}
void* proc = ResolveSymbol(name, bypassLocal);
cache[cacheKey] = proc;
return proc;
}
} // anonymous namespace
namespace gapic {
GetGlesProcAddressFunc* GetGlesProcAddress = getGlesProcAddress;
} // namespace gapic