• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1# Native Image Development
2
3## When to Use
4
5The native image module is used for associating a surface with an OpenGL external texture. It functions as the consumer of a graphics queue. You can use the APIs of this module to obtain and use a buffer, and output the buffer content to an OpenGL external texture.
6
7The following scenario is common for native image development:
8
9Use the native image APIs to create an **OH_NativeImage** instance as the consumer and obtain the corresponding **OHNativeWindow** instance (functioning as the producer). Use the native window APIs to fill in and flush the buffer, and then use the native image APIs to update the buffer content to an OpenGL ES texture.
10
11The native image module must be used together with the native window, native buffer, EGL, and GLES3 modules.
12
13## Available APIs
14
15| API                                                      | Description                                                        |
16| ------------------------------------------------------------ | ------------------------------------------------------------ |
17| OH_NativeImage_Create (uint32_t textureId, uint32_t textureTarget) | Creates an **OH_NativeImage** instance to be associated with the specified OpenGL ES texture ID and target.|
18| OH_NativeImage_AcquireNativeWindow (OH_NativeImage \*image)  | Obtains an **OHNativeWindow** instance associated with an **OH_NativeImage** instance. You need to call **OH_NativeWindow_DestroyNativeWindow** to release the **OHNativeWindow** instance when it is no longer needed.|
19| OH_NativeImage_AttachContext (OH_NativeImage \*image, uint32_t textureId) | Attaches an **OH_NativeImage** instance to the current OpenGL ES context. The OpenGL ES texture will be bound to an **GL_TEXTURE_EXTERNAL_OES** instance and updated through the **OH_NativeImage** instance.|
20| OH_NativeImage_DetachContext (OH_NativeImage \*image)        | Detaches an **OH_NativeImage** instance from the current OpenGL ES context.             |
21| OH_NativeImage_UpdateSurfaceImage (OH_NativeImage \*image)   | Updates the OpenGL ES texture associated with the latest frame through an **OH_NativeImage** instance.     |
22| OH_NativeImage_GetTimestamp (OH_NativeImage \*image)         | Obtains the timestamp of the texture image that recently called the **OH_NativeImage_UpdateSurfaceImage** function.|
23| OH_NativeImage_GetTransformMatrix (OH_NativeImage \*image, float matrix[16]) | Obtains the transformation matrix of the texture image that recently called the **OH_NativeImage_UpdateSurfaceImage** function.|
24| OH_NativeImage_Destroy (OH_NativeImage \*\*image)            | Destroys an **OH_NativeImage** instance created by calling **OH_NativeImage_Create**. After the instance is destroyed, the pointer to it is assigned **NULL**.|
25
26For details about the APIs, see [native_image](../reference/native-apis/_o_h___native_image.md).
27
28## How to Develop
29
30The following steps describe how to use the native Image APIs to create an **OH_NativeImage** instance as the consumer and update the data to a OpenGL external texture.
31
32**Adding Dynamic Link Libraries**
33
34Add the following libraries to **CMakeLists.txt**:
35
36```txt
37libEGL.so
38libGLESv3.so
39libnative_image.so
40libnative_window.so
41libnative_buffer.so
42```
43
44**Including Header Files**
45
46```c++
47#include <EGL/egl.h>
48#include <EGL/eglext.h>
49#include <GLES3/gl3.h>
50#include <native_image/native_image.h>
51#include <native_window/external_window.h>
52#include <native_buffer/native_buffer.h>
53```
54
551. Initialize the EGL environment.
56
57   Refer to the code snippet below.
58   ```c++
59   #include <iostream>
60   #include <string>
61   #include <EGL/egl.h>
62   #include <EGL/eglext.h>
63
64   using GetPlatformDisplayExt = PFNEGLGETPLATFORMDISPLAYEXTPROC;
65   constexpr const char *EGL_EXT_PLATFORM_WAYLAND = "EGL_EXT_platform_wayland";
66   constexpr const char *EGL_KHR_PLATFORM_WAYLAND = "EGL_KHR_platform_wayland";
67   constexpr int32_t EGL_CONTEXT_CLIENT_VERSION_NUM = 2;
68   constexpr char CHARACTER_WHITESPACE = ' ';
69   constexpr const char *CHARACTER_STRING_WHITESPACE = " ";
70   constexpr const char *EGL_GET_PLATFORM_DISPLAY_EXT = "eglGetPlatformDisplayEXT";
71   EGLContext eglContext_ = EGL_NO_CONTEXT;
72   EGLDisplay eglDisplay_ = EGL_NO_DISPLAY;
73   static inline EGLConfig config_;
74
75   // Check the EGL extension.
76   static bool CheckEglExtension(const char *extensions, const char *extension) {
77       size_t extlen = strlen(extension);
78       const char *end = extensions + strlen(extensions);
79
80       while (extensions < end) {
81           size_t n = 0;
82           if (*extensions == CHARACTER_WHITESPACE) {
83               extensions++;
84               continue;
85           }
86           n = strcspn(extensions, CHARACTER_STRING_WHITESPACE);
87           if (n == extlen && strncmp(extension, extensions, n) == 0) {
88               return true;
89           }
90           extensions += n;
91       }
92       return false;
93   }
94
95   // Obtain the display.
96   static EGLDisplay GetPlatformEglDisplay(EGLenum platform, void *native_display, const EGLint *attrib_list) {
97       static GetPlatformDisplayExt eglGetPlatformDisplayExt = NULL;
98
99       if (!eglGetPlatformDisplayExt) {
100           const char *extensions = eglQueryString(EGL_NO_DISPLAY, EGL_EXTENSIONS);
101           if (extensions && (CheckEglExtension(extensions, EGL_EXT_PLATFORM_WAYLAND) ||
102                              CheckEglExtension(extensions, EGL_KHR_PLATFORM_WAYLAND))) {
103               eglGetPlatformDisplayExt = (GetPlatformDisplayExt)eglGetProcAddress(EGL_GET_PLATFORM_DISPLAY_EXT);
104           }
105       }
106
107       if (eglGetPlatformDisplayExt) {
108           return eglGetPlatformDisplayExt(platform, native_display, attrib_list);
109       }
110
111       return eglGetDisplay((EGLNativeDisplayType)native_display);
112   }
113
114   static void InitEGLEnv() {
115       // Obtain the display.
116       eglDisplay_ = GetPlatformEglDisplay(EGL_PLATFORM_OHOS_KHR, EGL_DEFAULT_DISPLAY, NULL);
117       if (eglDisplay_ == EGL_NO_DISPLAY) {
118           std::cout << "Failed to create EGLDisplay gl errno : " << eglGetError() << std::endl;
119       }
120
121       EGLint major, minor;
122       // Initialize the EGL display.
123       if (eglInitialize(eglDisplay_, &major, &minor) == EGL_FALSE) {
124           std::cout << "Failed to initialize EGLDisplay" << std::endl;
125       }
126
127       // Bind the OpenGL ES API.
128       if (eglBindAPI(EGL_OPENGL_ES_API) == EGL_FALSE) {
129           std::cout << "Failed to bind OpenGL ES API" << std::endl;
130       }
131
132       unsigned int ret;
133       EGLint count;
134       EGLint config_attribs[] = {EGL_SURFACE_TYPE,
135                                  EGL_WINDOW_BIT,
136                                  EGL_RED_SIZE,
137                                  8,
138                                  EGL_GREEN_SIZE,
139                                  8,
140                                  EGL_BLUE_SIZE,
141                                  8,
142                                  EGL_ALPHA_SIZE,
143                                  8,
144                                  EGL_RENDERABLE_TYPE,
145                                  EGL_OPENGL_ES3_BIT,
146                                  EGL_NONE};
147
148       // Obtain a valid system configuration.
149       ret = eglChooseConfig(eglDisplay_, config_attribs, &config_, 1, &count);
150       if (!(ret && static_cast<unsigned int>(count) >= 1)) {
151           std::cout << "Failed to eglChooseConfig" << std::endl;
152       }
153
154       static const EGLint context_attribs[] = {EGL_CONTEXT_CLIENT_VERSION, EGL_CONTEXT_CLIENT_VERSION_NUM, EGL_NONE};
155
156       // Create a context.
157       eglContext_ = eglCreateContext(eglDisplay_, config_, EGL_NO_CONTEXT, context_attribs);
158       if (eglContext_ == EGL_NO_CONTEXT) {
159           std::cout << "Failed to create egl context %{public}x, error:" << eglGetError() << std::endl;
160       }
161
162       // Associate the context.
163       eglMakeCurrent(eglDisplay_, EGL_NO_SURFACE, EGL_NO_SURFACE, eglContext_);
164
165       // The EGL environment initialization is complete.
166       std::cout << "Create EGL context successfully, version" << major << "." << minor << std::endl;
167   }
168   ```
169
1702. Create an **OH_NativeImage** instance.
171
172   ```c++
173   // Create an OpenGL ES texture.
174   GLuint textureId;
175   glGenTextures(1, &textureId);
176   // Create an OH_NativeImage instance, which will be associated with an OpenGL ES texture.
177   OH_NativeImage* image = OH_NativeImage_Create(textureId, GL_TEXTURE_2D);
178   ```
179
1803. Obtain the **OHNativeWindow** instance that functions as the producer.
181
182   ```c++
183   // Obtain an OHNativeWindow instance.
184   OHNativeWindow* nativeWindow = OH_NativeImage_AcquireNativeWindow(image);
185   ```
186
1874. Set the width and height of the **OHNativeWindow** instance.
188
189   ```c++
190   int code = SET_BUFFER_GEOMETRY;
191   int32_t width = 800;
192   int32_t height = 600;
193   int32_t ret = OH_NativeWindow_NativeWindowHandleOpt(nativeWindow, code, width, height);
194   ```
195
1965. Write the produced content to an **OHNativeWindowBuffer** instance.
197
198   1. Obtain an **OHNativeWindowBuffer** instance from the **OHNativeWindow** instance.
199
200      ```c++
201      OHNativeWindowBuffer *buffer = nullptr;
202      int fenceFd;
203      // Obtain an OHNativeWindowBuffer instance by calling OH_NativeWindow_NativeWindowRequestBuffer.
204      OH_NativeWindow_NativeWindowRequestBuffer(nativeWindow, &buffer, &fenceFd);
205
206      BufferHandle *handle = OH_NativeWindow_GetBufferHandleFromNative(buffer);
207      ```
208
209   2. Write the produced content to the **OHNativeWindowBuffer** instance.
210
211      ```c++
212      #include <sys/mman.h>
213
214      // Use mmap() to obtain the memory virtual address of buffer handle.
215      void *mappedAddr = mmap(handle->virAddr, handle->size, PROT_READ | PROT_WRITE, MAP_SHARED, handle->fd, 0);
216      if (mappedAddr == MAP_FAILED) {
217          // mmap failed
218      }
219      static uint32_t value = 0x00;
220      value++;
221      uint32_t *pixel = static_cast<uint32_t *>(mappedAddr);
222      for (uint32_t x = 0; x < width; x++) {
223          for (uint32_t y = 0; y < height; y++) {
224              *pixel++ = value;
225          }
226      }
227      // Unmap the memory when the memory is no longer required.
228      int result = munmap(mappedAddr, handle->size);
229      if (result == -1) {
230          // munmap failed
231      }
232      ```
233
234   3. Flush the **OHNativeWindowBuffer** to the **OHNativeWindow**.
235
236      ```c++
237      // Set the refresh region. If Rect in Region is a null pointer or rectNumber is 0, all contents in the NativeWindowBuffer are changed.
238      Region region{nullptr, 0};
239      // Flush the buffer to the consumer through OH_NativeWindow_NativeWindowFlushBuffer, for example, by displaying it on the screen.
240      OH_NativeWindow_NativeWindowFlushBuffer(nativeWindow, buffer, fenceFd, region);
241      ```
242
243   4. Destroy the **OHNativeWindow** instance when it is no longer needed.
244
245      ```c++
246      OH_NativeWindow_DestroyNativeWindow(nativeWindow);
247      ```
248
2496. Update the content to the OpenGL texture.
250
251   ```c++
252   // Update the content to the OpenGL texture.
253   ret = OH_NativeImage_UpdateSurfaceImage(image);
254   if (ret != 0) {
255       std::cout << "OH_NativeImage_UpdateSurfaceImage failed" << std::endl;
256   }
257   // Obtain the timestamp and transformation matrix of the texture image that recently called the **OH_NativeImage_UpdateSurfaceImage** function.
258   int64_t timeStamp = OH_NativeImage_GetTimestamp(image);
259   float matrix[16];
260   ret = OH_NativeImage_GetTransformMatrix(image, matrix);
261   if (ret != 0) {
262       std::cout << "OH_NativeImage_GetTransformMatrix failed" << std::endl;
263   }
264   ```
265
2667. Detach the **OH_NativeImage** from the current OpenGL texture and attach it to a new external texture.
267
268   ```c++
269   // Detach an OH_NativeImage instance from the current OpenGL ES context.
270   ret = OH_NativeImage_DetachContext(image);
271   if (ret != 0) {
272       std::cout << "OH_NativeImage_DetachContext failed" << std::endl;
273   }
274   // Attach an OH_NativeImage instance to the current OpenGL ES context. The OpenGL ES texture will be bound to an GL_TEXTURE_EXTERNAL_OES instance and updated through the OH_NativeImage instance.
275   GLuint textureId2;
276   glGenTextures(1, &textureId2);
277   ret = OH_NativeImage_AttachContext(image, textureId2);
278   ```
279
2808. Destroy the **OH_NativeImage** instance when it is no longer needed.
281
282   ```c++
283   // Destroy the OH_NativeImage instance.
284   OH_NativeImage_Destroy(&image);
285   ```
286