1 /*
2  * Mesa 3-D graphics library
3  * Version:  7.12
4  *
5  * Copyright (C) 2010-2011 LunarG Inc.
6  *
7  * Permission is hereby granted, free of charge, to any person obtaining a
8  * copy of this software and associated documentation files (the "Software"),
9  * to deal in the Software without restriction, including without limitation
10  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
11  * and/or sell copies of the Software, and to permit persons to whom the
12  * Software is furnished to do so, subject to the following conditions:
13  *
14  * The above copyright notice and this permission notice shall be included
15  * in all copies or substantial portions of the Software.
16  *
17  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
19  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
20  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
21  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
22  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
23  * DEALINGS IN THE SOFTWARE.
24  *
25  * Authors:
26  *    Chia-I Wu <olv@lunarg.com>
27  */
28 
29 #include "pipe/p_compiler.h"
30 #include "pipe/p_state.h"
31 #include "util/u_memory.h"
32 #include "util/u_format.h"
33 #include "state_tracker/sw_winsys.h"
34 
35 #include <hardware/gralloc.h>
36 #include <utils/Errors.h>
37 
38 #if ANDROID_VERSION < 0x0300
39 #include <private/ui/sw_gralloc_handle.h>
40 #endif
41 
42 #include "android_sw_winsys.h"
43 
44 struct android_sw_winsys
45 {
46    struct sw_winsys base;
47 
48    const gralloc_module_t *grmod;
49 };
50 
51 struct android_sw_displaytarget
52 {
53    buffer_handle_t handle;
54    int stride;
55    int width, height;
56    int usage; /* gralloc usage */
57 
58    void *mapped;
59 };
60 
61 static INLINE struct android_sw_winsys *
android_sw_winsys(struct sw_winsys * ws)62 android_sw_winsys(struct sw_winsys *ws)
63 {
64    return (struct android_sw_winsys *) ws;
65 }
66 
67 static INLINE struct android_sw_displaytarget *
android_sw_displaytarget(struct sw_displaytarget * dt)68 android_sw_displaytarget(struct sw_displaytarget *dt)
69 {
70    return (struct android_sw_displaytarget *) dt;
71 }
72 
73 namespace android {
74 
75 static void
android_displaytarget_display(struct sw_winsys * ws,struct sw_displaytarget * dt,void * context_private)76 android_displaytarget_display(struct sw_winsys *ws,
77                               struct sw_displaytarget *dt,
78                               void *context_private)
79 {
80 }
81 
82 static struct sw_displaytarget *
android_displaytarget_create(struct sw_winsys * ws,unsigned tex_usage,enum pipe_format format,unsigned width,unsigned height,unsigned alignment,unsigned * stride)83 android_displaytarget_create(struct sw_winsys *ws,
84                              unsigned tex_usage,
85                              enum pipe_format format,
86                              unsigned width, unsigned height,
87                              unsigned alignment,
88                              unsigned *stride)
89 {
90    return NULL;
91 }
92 
93 static void
android_displaytarget_destroy(struct sw_winsys * ws,struct sw_displaytarget * dt)94 android_displaytarget_destroy(struct sw_winsys *ws,
95                               struct sw_displaytarget *dt)
96 {
97    struct android_sw_displaytarget *adt = android_sw_displaytarget(dt);
98 
99    assert(!adt->mapped);
100    FREE(adt);
101 }
102 
103 static void
android_displaytarget_unmap(struct sw_winsys * ws,struct sw_displaytarget * dt)104 android_displaytarget_unmap(struct sw_winsys *ws,
105                             struct sw_displaytarget *dt)
106 {
107    struct android_sw_winsys *droid = android_sw_winsys(ws);
108    struct android_sw_displaytarget *adt = android_sw_displaytarget(dt);
109 
110 #if ANDROID_VERSION < 0x0300
111    /* try sw_gralloc first */
112    if (adt->mapped && sw_gralloc_handle_t::validate(adt->handle) >= 0) {
113       adt->mapped = NULL;
114       return;
115    }
116 #endif
117 
118    if (adt->mapped) {
119       droid->grmod->unlock(droid->grmod, adt->handle);
120       adt->mapped = NULL;
121    }
122 }
123 
124 static void *
android_displaytarget_map(struct sw_winsys * ws,struct sw_displaytarget * dt,unsigned flags)125 android_displaytarget_map(struct sw_winsys *ws,
126                           struct sw_displaytarget *dt,
127                           unsigned flags)
128 {
129    struct android_sw_winsys *droid = android_sw_winsys(ws);
130    struct android_sw_displaytarget *adt = android_sw_displaytarget(dt);
131 
132 #if ANDROID_VERSION < 0x0300
133    /* try sw_gralloc first */
134    if (sw_gralloc_handle_t::validate(adt->handle) >= 0) {
135       const sw_gralloc_handle_t *swhandle =
136          reinterpret_cast<const sw_gralloc_handle_t *>(adt->handle);
137       adt->mapped = reinterpret_cast<void *>(swhandle->base);
138 
139       return adt->mapped;
140    }
141 #endif
142 
143    if (!adt->mapped) {
144       /* lock the buffer for CPU access */
145       droid->grmod->lock(droid->grmod, adt->handle,
146             adt->usage, 0, 0, adt->width, adt->height, &adt->mapped);
147    }
148 
149    return adt->mapped;
150 }
151 
152 static struct sw_displaytarget *
android_displaytarget_from_handle(struct sw_winsys * ws,const struct pipe_resource * templ,struct winsys_handle * whandle,unsigned * stride)153 android_displaytarget_from_handle(struct sw_winsys *ws,
154                                   const struct pipe_resource *templ,
155                                   struct winsys_handle *whandle,
156                                   unsigned *stride)
157 {
158    struct android_winsys_handle *ahandle =
159       (struct android_winsys_handle *) whandle;
160    struct android_sw_displaytarget *adt;
161 
162    adt = CALLOC_STRUCT(android_sw_displaytarget);
163    if (!adt)
164       return NULL;
165 
166    adt->handle = ahandle->handle;
167    adt->stride = ahandle->stride;
168    adt->width = templ->width0;
169    adt->height = templ->height0;
170 
171    if (templ->bind & (PIPE_BIND_RENDER_TARGET | PIPE_BIND_TRANSFER_WRITE))
172       adt->usage |= GRALLOC_USAGE_SW_WRITE_OFTEN;
173    if (templ->bind & (PIPE_BIND_SAMPLER_VIEW | PIPE_BIND_TRANSFER_READ))
174       adt->usage |= GRALLOC_USAGE_SW_READ_OFTEN;
175 
176    if (stride)
177       *stride = adt->stride;
178 
179    return reinterpret_cast<struct sw_displaytarget *>(adt);
180 }
181 
182 static boolean
android_displaytarget_get_handle(struct sw_winsys * ws,struct sw_displaytarget * dt,struct winsys_handle * whandle)183 android_displaytarget_get_handle(struct sw_winsys *ws,
184                                  struct sw_displaytarget *dt,
185                                  struct winsys_handle *whandle)
186 {
187    return FALSE;
188 }
189 
190 static boolean
android_is_displaytarget_format_supported(struct sw_winsys * ws,unsigned tex_usage,enum pipe_format format)191 android_is_displaytarget_format_supported(struct sw_winsys *ws,
192                                           unsigned tex_usage,
193                                           enum pipe_format format)
194 {
195    struct android_sw_winsys *droid = android_sw_winsys(ws);
196    int fmt = -1;
197 
198    switch (format) {
199    case PIPE_FORMAT_R8G8B8A8_UNORM:
200       fmt = HAL_PIXEL_FORMAT_RGBA_8888;
201       break;
202    case PIPE_FORMAT_R8G8B8X8_UNORM:
203       fmt = HAL_PIXEL_FORMAT_RGBX_8888;
204       break;
205    case PIPE_FORMAT_R8G8B8_UNORM:
206       fmt = HAL_PIXEL_FORMAT_RGB_888;
207       break;
208    case PIPE_FORMAT_B5G6R5_UNORM:
209       fmt = HAL_PIXEL_FORMAT_RGB_565;
210       break;
211    case PIPE_FORMAT_B8G8R8A8_UNORM:
212       fmt = HAL_PIXEL_FORMAT_BGRA_8888;
213       break;
214    default:
215       break;
216    }
217 
218    return (fmt != -1);
219 }
220 
221 static void
android_destroy(struct sw_winsys * ws)222 android_destroy(struct sw_winsys *ws)
223 {
224    struct android_sw_winsys *droid = android_sw_winsys(ws);
225 
226    FREE(droid);
227 }
228 
229 }; /* namespace android */
230 
231 using namespace android;
232 
233 struct sw_winsys *
android_create_sw_winsys(void)234 android_create_sw_winsys(void)
235 {
236    struct android_sw_winsys *droid;
237    const hw_module_t *mod;
238 
239    droid = CALLOC_STRUCT(android_sw_winsys);
240    if (!droid)
241       return NULL;
242 
243    if (hw_get_module(GRALLOC_HARDWARE_MODULE_ID, &mod)) {
244       FREE(droid);
245       return NULL;
246    }
247 
248    droid->grmod = (const gralloc_module_t *) mod;
249 
250    droid->base.destroy = android_destroy;
251    droid->base.is_displaytarget_format_supported =
252       android_is_displaytarget_format_supported;
253 
254    droid->base.displaytarget_create = android_displaytarget_create;
255    droid->base.displaytarget_destroy = android_displaytarget_destroy;
256    droid->base.displaytarget_from_handle = android_displaytarget_from_handle;
257    droid->base.displaytarget_get_handle = android_displaytarget_get_handle;
258 
259    droid->base.displaytarget_map = android_displaytarget_map;
260    droid->base.displaytarget_unmap = android_displaytarget_unmap;
261    droid->base.displaytarget_display = android_displaytarget_display;
262 
263    return &droid->base;
264 }
265