/* ** 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_nowake(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()) { // call the implementation's glGetString(GL_EXTENSIONS) const char* exts = (const char *)gEGLImpl.hooks[version]->gl.glGetString(GL_EXTENSIONS); // 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. if (exts) { gl_extensions = exts; if (gl_extensions.find("GL_EXT_debug_marker") == std::string::npos) { gl_extensions.insert(0, "GL_EXT_debug_marker "); } // 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 // ----------------------------------------------------------------------------