blob: b68fd61c1177d361fadfabd97b3823408d31c711 [file] [log] [blame]
/*
** 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_metadata({}),
egl_cta861_3_metadata({}) {
if (win) {
win->incStrong(this);
}
}
egl_surface_t::~egl_surface_t() {
if (win != NULL) {
disconnect();
win->decStrong(this);
}
}
void egl_surface_t::disconnect() {
if (win != NULL && 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;
return EGL_TRUE;
case EGL_SMPTE2086_DISPLAY_PRIMARY_RY_EXT:
egl_smpte2086_metadata.displayPrimaryRed.y = value;
return EGL_TRUE;
case EGL_SMPTE2086_DISPLAY_PRIMARY_GX_EXT:
egl_smpte2086_metadata.displayPrimaryGreen.x = value;
return EGL_TRUE;
case EGL_SMPTE2086_DISPLAY_PRIMARY_GY_EXT:
egl_smpte2086_metadata.displayPrimaryGreen.y = value;
return EGL_TRUE;
case EGL_SMPTE2086_DISPLAY_PRIMARY_BX_EXT:
egl_smpte2086_metadata.displayPrimaryBlue.x = value;
return EGL_TRUE;
case EGL_SMPTE2086_DISPLAY_PRIMARY_BY_EXT:
egl_smpte2086_metadata.displayPrimaryBlue.y = value;
return EGL_TRUE;
case EGL_SMPTE2086_WHITE_POINT_X_EXT:
egl_smpte2086_metadata.whitePoint.x = value;
return EGL_TRUE;
case EGL_SMPTE2086_WHITE_POINT_Y_EXT:
egl_smpte2086_metadata.whitePoint.y = value;
return EGL_TRUE;
case EGL_SMPTE2086_MAX_LUMINANCE_EXT:
egl_smpte2086_metadata.maxLuminance = value;
return EGL_TRUE;
case EGL_SMPTE2086_MIN_LUMINANCE_EXT:
egl_smpte2086_metadata.minLuminance = value;
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;
return EGL_TRUE;
case EGL_CTA861_3_MAX_FRAME_AVERAGE_LEVEL_EXT:
egl_cta861_3_metadata.maxFrameAverageLightLevel = value;
return EGL_TRUE;
}
return EGL_FALSE;
}
const android_smpte2086_metadata egl_surface_t::getSmpte2086Metadata() {
android_smpte2086_metadata metadata;
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 metadata;
}
const android_cta861_3_metadata egl_surface_t::getCta8613Metadata() {
android_cta861_3_metadata metadata;
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 metadata;
}
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(0), draw(0), cnx(cnx), version(version) {
}
void egl_context_t::onLooseCurrent() {
read = NULL;
draw = NULL;
}
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
// ----------------------------------------------------------------------------