1 /*
2 * Copyright 2011 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 #include "EGLClientIface.h"
18 #include "HostConnection.h"
19 #include "GL2Encoder.h"
20 #include "GLES/gl.h"
21 #include "GLES/glext.h"
22 #include "ErrorLog.h"
23 #include "gralloc_cb.h"
24 #include "ThreadInfo.h"
25 #include "EGLImage.h"
26
27 //XXX: fix this macro to get the context from fast tls path
28 #define GET_CONTEXT GL2Encoder * ctx = getEGLThreadInfo()->hostConn->gl2Encoder();
29
30 #include "gl2_entry.cpp"
31
32 //The functions table
33 #include "gl2_ftable.h"
34
35
36 static EGLClient_eglInterface * s_egl = NULL;
37 static EGLClient_glesInterface * s_gl = NULL;
38
39 #define DEFINE_AND_VALIDATE_HOST_CONNECTION(ret) \
40 HostConnection *hostCon = HostConnection::get(); \
41 if (!hostCon) { \
42 ALOGE("egl: Failed to get host connection\n"); \
43 return ret; \
44 } \
45 renderControl_encoder_context_t *rcEnc = hostCon->rcEncoder(); \
46 if (!rcEnc) { \
47 ALOGE("egl: Failed to get renderControl encoder context\n"); \
48 return ret; \
49 }
50
51 //GL extensions
glEGLImageTargetTexture2DOES(void * self,GLenum target,GLeglImageOES img)52 void glEGLImageTargetTexture2DOES(void * self, GLenum target, GLeglImageOES img)
53 {
54 (void)self;
55 (void)target;
56
57 DBG("glEGLImageTargetTexture2DOES v2 target=%#x img=%p\n", target, img);
58
59 EGLImage_t *image = (EGLImage_t*)img;
60 GLeglImageOES hostImage = reinterpret_cast<GLeglImageOES>((intptr_t)image->host_egl_image);
61
62 if (image->target == EGL_NATIVE_BUFFER_ANDROID) {
63 //TODO: check error - we don't have a way to set gl error
64 android_native_buffer_t* native_buffer = image->native_buffer;
65
66 if (native_buffer->common.magic != ANDROID_NATIVE_BUFFER_MAGIC) {
67 return;
68 }
69
70 if (native_buffer->common.version != sizeof(android_native_buffer_t)) {
71 return;
72 }
73
74 GET_CONTEXT;
75 DEFINE_AND_VALIDATE_HOST_CONNECTION();
76
77 ctx->override2DTextureTarget(target);
78 ctx->associateEGLImage(target, hostImage);
79 rcEnc->rcBindTexture(rcEnc, ((cb_handle_t *)(native_buffer->handle))->hostHandle);
80 ctx->restore2DTextureTarget(target);
81 }
82 else if (image->target == EGL_GL_TEXTURE_2D_KHR) {
83 GET_CONTEXT;
84 ctx->override2DTextureTarget(target);
85 ctx->associateEGLImage(target, hostImage);
86 ctx->m_glEGLImageTargetTexture2DOES_enc(self, target, hostImage);
87 ctx->restore2DTextureTarget(target);
88 }
89 }
90
glEGLImageTargetRenderbufferStorageOES(void * self,GLenum target,GLeglImageOES image)91 void glEGLImageTargetRenderbufferStorageOES(void *self, GLenum target, GLeglImageOES image)
92 {
93 (void)self;
94 (void)target;
95
96 DBG("glEGLImageTargetRenderbufferStorageOES v2 image=%p\n", image);
97 //TODO: check error - we don't have a way to set gl error
98 android_native_buffer_t* native_buffer = (android_native_buffer_t*)image;
99
100 if (native_buffer->common.magic != ANDROID_NATIVE_BUFFER_MAGIC) {
101 return;
102 }
103
104 if (native_buffer->common.version != sizeof(android_native_buffer_t)) {
105 return;
106 }
107
108 DEFINE_AND_VALIDATE_HOST_CONNECTION();
109 rcEnc->rcBindRenderbuffer(rcEnc, ((cb_handle_t *)(native_buffer->handle))->hostHandle);
110
111 return;
112 }
113
getProcAddress(const char * procname)114 void * getProcAddress(const char * procname)
115 {
116 // search in GL function table
117 for (int i=0; i<gl2_num_funcs; i++) {
118 if (!strcmp(gl2_funcs_by_name[i].name, procname)) {
119 return gl2_funcs_by_name[i].proc;
120 }
121 }
122 return NULL;
123 }
124
finish()125 void finish()
126 {
127 glFinish();
128 }
129
getIntegerv(unsigned int pname,int * param)130 void getIntegerv(unsigned int pname, int* param)
131 {
132 glGetIntegerv((GLenum)pname, (GLint*)param);
133 }
134
my_glGetString(void * self,GLenum name)135 const GLubyte *my_glGetString (void *self, GLenum name)
136 {
137 (void)self;
138
139 //see ref in https://www.khronos.org/opengles/sdk/docs/man
140 //name in glGetString can be one of the following five values
141 switch (name) {
142 case GL_VERSION:
143 case GL_VENDOR:
144 case GL_RENDERER:
145 case GL_SHADING_LANGUAGE_VERSION:
146 case GL_EXTENSIONS:
147 if (s_egl) {
148 return (const GLubyte*)s_egl->getGLString(name);
149 }
150 break;
151 default:
152 GET_CONTEXT;
153 ctx->setError(GL_INVALID_ENUM);
154 break;
155 }
156 return NULL;
157 }
158
init()159 void init()
160 {
161 GET_CONTEXT;
162 ctx->m_glEGLImageTargetTexture2DOES_enc = ctx->glEGLImageTargetTexture2DOES;
163 ctx->glEGLImageTargetTexture2DOES = &glEGLImageTargetTexture2DOES;
164 ctx->glEGLImageTargetRenderbufferStorageOES = &glEGLImageTargetRenderbufferStorageOES;
165 ctx->glGetString = &my_glGetString;
166 }
167 extern "C" {
init_emul_gles(EGLClient_eglInterface * eglIface)168 EGLClient_glesInterface * init_emul_gles(EGLClient_eglInterface *eglIface)
169 {
170 s_egl = eglIface;
171
172 if (!s_gl) {
173 s_gl = new EGLClient_glesInterface();
174 s_gl->getProcAddress = getProcAddress;
175 s_gl->finish = finish;
176 s_gl->init = init;
177 s_gl->getIntegerv = getIntegerv;
178 }
179
180 return s_gl;
181 }
182 } //extern
183