|  | /* | 
|  | ** Copyright 2007, 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 "egl_object.h" | 
|  |  | 
|  | #include <sstream> | 
|  |  | 
|  | namespace android { | 
|  |  | 
|  | egl_object_t::egl_object_t(egl_display_t* disp) : display(disp), count(1) { | 
|  | // NOTE: this does an implicit incRef | 
|  | display->addObject(this); | 
|  | } | 
|  |  | 
|  | egl_object_t::~egl_object_t() {} | 
|  |  | 
|  | void egl_object_t::terminate() { | 
|  | // this marks the object as "terminated" | 
|  | display->removeObject(this); | 
|  | if (decRef() == 1) { | 
|  | // shouldn't happen because this is called from LocalRef | 
|  | ALOGE("egl_object_t::terminate() removed the last reference!"); | 
|  | } | 
|  | } | 
|  |  | 
|  | void egl_object_t::destroy() { | 
|  | if (decRef() == 1) { | 
|  | delete this; | 
|  | } | 
|  | } | 
|  |  | 
|  | bool egl_object_t::get(egl_display_t const* display, egl_object_t* object) { | 
|  | // used by LocalRef, this does an incRef() atomically with | 
|  | // checking that the object is valid. | 
|  | return display->getObject(object); | 
|  | } | 
|  |  | 
|  | egl_surface_t::egl_surface_t(egl_display_t* dpy, EGLConfig config, EGLNativeWindowType win, | 
|  | EGLSurface surface, EGLint colorSpace, egl_connection_t const* cnx) | 
|  | : egl_object_t(dpy), | 
|  | surface(surface), | 
|  | config(config), | 
|  | win(win), | 
|  | cnx(cnx), | 
|  | connected(true), | 
|  | colorSpace(colorSpace), | 
|  | egl_smpte2086_dirty(false), | 
|  | egl_cta861_3_dirty(false) { | 
|  | egl_smpte2086_metadata.displayPrimaryRed = {EGL_DONT_CARE, EGL_DONT_CARE}; | 
|  | egl_smpte2086_metadata.displayPrimaryGreen = {EGL_DONT_CARE, EGL_DONT_CARE}; | 
|  | egl_smpte2086_metadata.displayPrimaryBlue = {EGL_DONT_CARE, EGL_DONT_CARE}; | 
|  | egl_smpte2086_metadata.whitePoint = {EGL_DONT_CARE, EGL_DONT_CARE}; | 
|  | egl_smpte2086_metadata.maxLuminance = EGL_DONT_CARE; | 
|  | egl_smpte2086_metadata.minLuminance = EGL_DONT_CARE; | 
|  | egl_cta861_3_metadata.maxFrameAverageLightLevel = EGL_DONT_CARE; | 
|  | egl_cta861_3_metadata.maxContentLightLevel = EGL_DONT_CARE; | 
|  |  | 
|  | if (win) { | 
|  | win->incStrong(this); | 
|  | } | 
|  | } | 
|  |  | 
|  | egl_surface_t::~egl_surface_t() { | 
|  | if (win != nullptr) { | 
|  | disconnect(); | 
|  | win->decStrong(this); | 
|  | } | 
|  | } | 
|  |  | 
|  | void egl_surface_t::disconnect() { | 
|  | if (win != nullptr && connected) { | 
|  | native_window_set_buffers_format(win, 0); | 
|  | if (native_window_api_disconnect(win, NATIVE_WINDOW_API_EGL)) { | 
|  | ALOGW("EGLNativeWindowType %p disconnect failed", win); | 
|  | } | 
|  | connected = false; | 
|  | } | 
|  | } | 
|  |  | 
|  | EGLBoolean egl_surface_t::setSmpte2086Attribute(EGLint attribute, EGLint value) { | 
|  | switch (attribute) { | 
|  | case EGL_SMPTE2086_DISPLAY_PRIMARY_RX_EXT: | 
|  | egl_smpte2086_metadata.displayPrimaryRed.x = value; | 
|  | egl_smpte2086_dirty = true; | 
|  | return EGL_TRUE; | 
|  | case EGL_SMPTE2086_DISPLAY_PRIMARY_RY_EXT: | 
|  | egl_smpte2086_metadata.displayPrimaryRed.y = value; | 
|  | egl_smpte2086_dirty = true; | 
|  | return EGL_TRUE; | 
|  | case EGL_SMPTE2086_DISPLAY_PRIMARY_GX_EXT: | 
|  | egl_smpte2086_metadata.displayPrimaryGreen.x = value; | 
|  | egl_smpte2086_dirty = true; | 
|  | return EGL_TRUE; | 
|  | case EGL_SMPTE2086_DISPLAY_PRIMARY_GY_EXT: | 
|  | egl_smpte2086_metadata.displayPrimaryGreen.y = value; | 
|  | egl_smpte2086_dirty = true; | 
|  | return EGL_TRUE; | 
|  | case EGL_SMPTE2086_DISPLAY_PRIMARY_BX_EXT: | 
|  | egl_smpte2086_metadata.displayPrimaryBlue.x = value; | 
|  | egl_smpte2086_dirty = true; | 
|  | return EGL_TRUE; | 
|  | case EGL_SMPTE2086_DISPLAY_PRIMARY_BY_EXT: | 
|  | egl_smpte2086_metadata.displayPrimaryBlue.y = value; | 
|  | egl_smpte2086_dirty = true; | 
|  | return EGL_TRUE; | 
|  | case EGL_SMPTE2086_WHITE_POINT_X_EXT: | 
|  | egl_smpte2086_metadata.whitePoint.x = value; | 
|  | egl_smpte2086_dirty = true; | 
|  | return EGL_TRUE; | 
|  | case EGL_SMPTE2086_WHITE_POINT_Y_EXT: | 
|  | egl_smpte2086_metadata.whitePoint.y = value; | 
|  | egl_smpte2086_dirty = true; | 
|  | return EGL_TRUE; | 
|  | case EGL_SMPTE2086_MAX_LUMINANCE_EXT: | 
|  | egl_smpte2086_metadata.maxLuminance = value; | 
|  | egl_smpte2086_dirty = true; | 
|  | return EGL_TRUE; | 
|  | case EGL_SMPTE2086_MIN_LUMINANCE_EXT: | 
|  | egl_smpte2086_metadata.minLuminance = value; | 
|  | egl_smpte2086_dirty = true; | 
|  | return EGL_TRUE; | 
|  | } | 
|  | return EGL_FALSE; | 
|  | } | 
|  |  | 
|  | EGLBoolean egl_surface_t::setCta8613Attribute(EGLint attribute, EGLint value) { | 
|  | switch (attribute) { | 
|  | case EGL_CTA861_3_MAX_CONTENT_LIGHT_LEVEL_EXT: | 
|  | egl_cta861_3_metadata.maxContentLightLevel = value; | 
|  | egl_cta861_3_dirty = true; | 
|  | return EGL_TRUE; | 
|  | case EGL_CTA861_3_MAX_FRAME_AVERAGE_LEVEL_EXT: | 
|  | egl_cta861_3_metadata.maxFrameAverageLightLevel = value; | 
|  | egl_cta861_3_dirty = true; | 
|  | return EGL_TRUE; | 
|  | } | 
|  | return EGL_FALSE; | 
|  | } | 
|  |  | 
|  | EGLBoolean egl_surface_t::getSmpte2086Metadata(android_smpte2086_metadata& metadata) const { | 
|  | if (!egl_smpte2086_dirty) return EGL_FALSE; | 
|  | if (egl_smpte2086_metadata.displayPrimaryRed.x == EGL_DONT_CARE || | 
|  | egl_smpte2086_metadata.displayPrimaryRed.y == EGL_DONT_CARE || | 
|  | egl_smpte2086_metadata.displayPrimaryGreen.x == EGL_DONT_CARE || | 
|  | egl_smpte2086_metadata.displayPrimaryGreen.y == EGL_DONT_CARE || | 
|  | egl_smpte2086_metadata.displayPrimaryBlue.x == EGL_DONT_CARE || | 
|  | egl_smpte2086_metadata.displayPrimaryBlue.y == EGL_DONT_CARE || | 
|  | egl_smpte2086_metadata.whitePoint.x == EGL_DONT_CARE || | 
|  | egl_smpte2086_metadata.whitePoint.y == EGL_DONT_CARE || | 
|  | egl_smpte2086_metadata.maxLuminance == EGL_DONT_CARE || | 
|  | egl_smpte2086_metadata.minLuminance == EGL_DONT_CARE) { | 
|  | ALOGW("egl_surface_t: incomplete SMPTE 2086 metadata!"); | 
|  | return EGL_FALSE; | 
|  | } | 
|  |  | 
|  | metadata.displayPrimaryRed.x = static_cast<float>(egl_smpte2086_metadata.displayPrimaryRed.x) / | 
|  | EGL_METADATA_SCALING_EXT; | 
|  | metadata.displayPrimaryRed.y = static_cast<float>(egl_smpte2086_metadata.displayPrimaryRed.y) / | 
|  | EGL_METADATA_SCALING_EXT; | 
|  | metadata.displayPrimaryGreen.x = | 
|  | static_cast<float>(egl_smpte2086_metadata.displayPrimaryGreen.x) / | 
|  | EGL_METADATA_SCALING_EXT; | 
|  | metadata.displayPrimaryGreen.y = | 
|  | static_cast<float>(egl_smpte2086_metadata.displayPrimaryGreen.y) / | 
|  | EGL_METADATA_SCALING_EXT; | 
|  | metadata.displayPrimaryBlue.x = | 
|  | static_cast<float>(egl_smpte2086_metadata.displayPrimaryBlue.x) / | 
|  | EGL_METADATA_SCALING_EXT; | 
|  | metadata.displayPrimaryBlue.y = | 
|  | static_cast<float>(egl_smpte2086_metadata.displayPrimaryBlue.y) / | 
|  | EGL_METADATA_SCALING_EXT; | 
|  | metadata.whitePoint.x = | 
|  | static_cast<float>(egl_smpte2086_metadata.whitePoint.x) / EGL_METADATA_SCALING_EXT; | 
|  | metadata.whitePoint.y = | 
|  | static_cast<float>(egl_smpte2086_metadata.whitePoint.y) / EGL_METADATA_SCALING_EXT; | 
|  | metadata.maxLuminance = | 
|  | static_cast<float>(egl_smpte2086_metadata.maxLuminance) / EGL_METADATA_SCALING_EXT; | 
|  | metadata.minLuminance = | 
|  | static_cast<float>(egl_smpte2086_metadata.minLuminance) / EGL_METADATA_SCALING_EXT; | 
|  |  | 
|  | return EGL_TRUE; | 
|  | } | 
|  |  | 
|  | EGLBoolean egl_surface_t::getCta8613Metadata(android_cta861_3_metadata& metadata) const { | 
|  | if (!egl_cta861_3_dirty) return EGL_FALSE; | 
|  |  | 
|  | if (egl_cta861_3_metadata.maxContentLightLevel == EGL_DONT_CARE || | 
|  | egl_cta861_3_metadata.maxFrameAverageLightLevel == EGL_DONT_CARE) { | 
|  | ALOGW("egl_surface_t: incomplete CTA861.3 metadata!"); | 
|  | return EGL_FALSE; | 
|  | } | 
|  |  | 
|  | metadata.maxContentLightLevel = static_cast<float>(egl_cta861_3_metadata.maxContentLightLevel) / | 
|  | EGL_METADATA_SCALING_EXT; | 
|  | metadata.maxFrameAverageLightLevel = | 
|  | static_cast<float>(egl_cta861_3_metadata.maxFrameAverageLightLevel) / | 
|  | EGL_METADATA_SCALING_EXT; | 
|  |  | 
|  | return EGL_TRUE; | 
|  | } | 
|  |  | 
|  | EGLBoolean egl_surface_t::getColorSpaceAttribute(EGLint attribute, EGLint* value) const { | 
|  | if (attribute == EGL_GL_COLORSPACE_KHR) { | 
|  | *value = colorSpace; | 
|  | return EGL_TRUE; | 
|  | } | 
|  | return EGL_FALSE; | 
|  | } | 
|  |  | 
|  | EGLBoolean egl_surface_t::getSmpte2086Attribute(EGLint attribute, EGLint* value) const { | 
|  | switch (attribute) { | 
|  | case EGL_SMPTE2086_DISPLAY_PRIMARY_RX_EXT: | 
|  | *value = egl_smpte2086_metadata.displayPrimaryRed.x; | 
|  | return EGL_TRUE; | 
|  | break; | 
|  | case EGL_SMPTE2086_DISPLAY_PRIMARY_RY_EXT: | 
|  | *value = egl_smpte2086_metadata.displayPrimaryRed.y; | 
|  | return EGL_TRUE; | 
|  | break; | 
|  | case EGL_SMPTE2086_DISPLAY_PRIMARY_GX_EXT: | 
|  | *value = egl_smpte2086_metadata.displayPrimaryGreen.x; | 
|  | return EGL_TRUE; | 
|  | break; | 
|  | case EGL_SMPTE2086_DISPLAY_PRIMARY_GY_EXT: | 
|  | *value = egl_smpte2086_metadata.displayPrimaryGreen.y; | 
|  | return EGL_TRUE; | 
|  | break; | 
|  | case EGL_SMPTE2086_DISPLAY_PRIMARY_BX_EXT: | 
|  | *value = egl_smpte2086_metadata.displayPrimaryBlue.x; | 
|  | return EGL_TRUE; | 
|  | break; | 
|  | case EGL_SMPTE2086_DISPLAY_PRIMARY_BY_EXT: | 
|  | *value = egl_smpte2086_metadata.displayPrimaryBlue.y; | 
|  | return EGL_TRUE; | 
|  | break; | 
|  | case EGL_SMPTE2086_WHITE_POINT_X_EXT: | 
|  | *value = egl_smpte2086_metadata.whitePoint.x; | 
|  | return EGL_TRUE; | 
|  | break; | 
|  | case EGL_SMPTE2086_WHITE_POINT_Y_EXT: | 
|  | *value = egl_smpte2086_metadata.whitePoint.y; | 
|  | return EGL_TRUE; | 
|  | break; | 
|  | case EGL_SMPTE2086_MAX_LUMINANCE_EXT: | 
|  | *value = egl_smpte2086_metadata.maxLuminance; | 
|  | return EGL_TRUE; | 
|  | break; | 
|  | case EGL_SMPTE2086_MIN_LUMINANCE_EXT: | 
|  | *value = egl_smpte2086_metadata.minLuminance; | 
|  | return EGL_TRUE; | 
|  | break; | 
|  | } | 
|  | return EGL_FALSE; | 
|  | } | 
|  |  | 
|  | EGLBoolean egl_surface_t::getCta8613Attribute(EGLint attribute, EGLint* value) const { | 
|  | switch (attribute) { | 
|  | case EGL_CTA861_3_MAX_CONTENT_LIGHT_LEVEL_EXT: | 
|  | *value = egl_cta861_3_metadata.maxContentLightLevel; | 
|  | return EGL_TRUE; | 
|  | break; | 
|  | case EGL_CTA861_3_MAX_FRAME_AVERAGE_LEVEL_EXT: | 
|  | *value = egl_cta861_3_metadata.maxFrameAverageLightLevel; | 
|  | return EGL_TRUE; | 
|  | break; | 
|  | } | 
|  | return EGL_FALSE; | 
|  | } | 
|  |  | 
|  | void egl_surface_t::terminate() { | 
|  | disconnect(); | 
|  | egl_object_t::terminate(); | 
|  | } | 
|  |  | 
|  | egl_context_t::egl_context_t(EGLDisplay dpy, EGLContext context, EGLConfig config, | 
|  | egl_connection_t const* cnx, int version) | 
|  | : egl_object_t(get_display(dpy)), | 
|  | dpy(dpy), | 
|  | context(context), | 
|  | config(config), | 
|  | read(nullptr), | 
|  | draw(nullptr), | 
|  | cnx(cnx), | 
|  | version(version) {} | 
|  |  | 
|  | void egl_context_t::onLooseCurrent() { | 
|  | read = nullptr; | 
|  | draw = nullptr; | 
|  | } | 
|  |  | 
|  | void egl_context_t::onMakeCurrent(EGLSurface draw, EGLSurface read) { | 
|  | this->read = read; | 
|  | this->draw = draw; | 
|  |  | 
|  | /* | 
|  | * Here we cache the GL_EXTENSIONS string for this context and we | 
|  | * add the extensions always handled by the wrapper | 
|  | */ | 
|  | if (!gl_extensions.empty()) return; | 
|  |  | 
|  | // call the implementation's glGetString(GL_EXTENSIONS) | 
|  | const char* exts = (const char*)gEGLImpl.hooks[version]->gl.glGetString(GL_EXTENSIONS); | 
|  | if (!exts) return; | 
|  |  | 
|  | // If this context is sharing with another context, and the other context was reset | 
|  | // e.g. due to robustness failure, this context might also be reset and glGetString can | 
|  | // return NULL. | 
|  | gl_extensions = exts; | 
|  | if (gl_extensions.find("GL_EXT_debug_marker") == std::string::npos) { | 
|  | gl_extensions.insert(0, "GL_EXT_debug_marker "); | 
|  | // eglGetProcAddress could return function pointers to these | 
|  | // functions while they actually don't work. Fix them now. | 
|  | __eglMustCastToProperFunctionPointerType* f; | 
|  | f = (__eglMustCastToProperFunctionPointerType*)&gEGLImpl.hooks[version] | 
|  | ->gl.glInsertEventMarkerEXT; | 
|  | if (*f != gl_noop) *f = gl_noop; | 
|  | f = (__eglMustCastToProperFunctionPointerType*)&gEGLImpl.hooks[version] | 
|  | ->gl.glPushGroupMarkerEXT; | 
|  | if (*f != gl_noop) *f = gl_noop; | 
|  | f = (__eglMustCastToProperFunctionPointerType*)&gEGLImpl.hooks[version] | 
|  | ->gl.glPopGroupMarkerEXT; | 
|  | if (*f != gl_noop) *f = gl_noop; | 
|  | } | 
|  |  | 
|  | // tokenize the supported extensions for the glGetStringi() wrapper | 
|  | std::stringstream ss; | 
|  | std::string str; | 
|  | ss << gl_extensions; | 
|  | while (ss >> str) { | 
|  | tokenized_gl_extensions.push_back(str); | 
|  | } | 
|  | } | 
|  |  | 
|  | }; // namespace android |