1 /*
2  * Copyright (C) 2010 ARM Limited. All rights reserved.
3  *
4  * Copyright (C) 2008 The Android Open Source Project
5  *
6  * Licensed under the Apache License, Version 2.0 (the "License");
7  * you may not use this file except in compliance with the License.
8  * You may obtain a copy of the License at
9  *
10  *      http://www.apache.org/licenses/LICENSE-2.0
11  *
12  * Unless required by applicable law or agreed to in writing, software
13  * distributed under the License is distributed on an "AS IS" BASIS,
14  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15  * See the License for the specific language governing permissions and
16  * limitations under the License.
17  */
18 
19 #include <cstdlib>
20 #include <string.h>
21 #include <errno.h>
22 #include <pthread.h>
23 #include <sys/types.h>
24 #include <sys/stat.h>
25 #include <fcntl.h>
26 
27 #include <cutils/log.h>
28 #include <cutils/atomic.h>
29 #include <hardware/hardware.h>
30 #include <hardware/gralloc.h>
31 
32 #include <sys/ioctl.h>
33 
34 #include "alloc_device.h"
35 #include "gralloc_priv.h"
36 #include "gralloc_helper.h"
37 #include "framebuffer_device.h"
38 
39 #if GRALLOC_ARM_UMP_MODULE
40 #include <ump/ump.h>
41 #include <ump/ump_ref_drv.h>
42 #endif
43 
44 #if GRALLOC_ARM_DMA_BUF_MODULE
45 #include <ion/ion.h>
46 #include "ion_4.12.h"
47 #include "dma-heap.h"
48 
49 #define ION_SYSTEM	(char*)"ion_system_heap"
50 #define ION_CMA		(char*)"linux,cma"
51 
52 #define DMABUF_SYSTEM	(char*)"system"
53 #define DMABUF_CMA	(char*)"linux,cma"
54 static enum {
55 	INTERFACE_UNKNOWN,
56 	INTERFACE_ION_LEGACY,
57 	INTERFACE_ION_MODERN,
58 	INTERFACE_DMABUF_HEAPS
59 } interface_ver;
60 
61 static int system_heap_id;
62 static int cma_heap_id;
63 #endif
64 
65 #if GRALLOC_SIMULATE_FAILURES
66 #include <cutils/properties.h>
67 
68 /* system property keys for controlling simulated UMP allocation failures */
69 #define PROP_MALI_TEST_GRALLOC_FAIL_FIRST     "mali.test.gralloc.fail_first"
70 #define PROP_MALI_TEST_GRALLOC_FAIL_INTERVAL  "mali.test.gralloc.fail_interval"
71 
__ump_alloc_should_fail()72 static int __ump_alloc_should_fail()
73 {
74 
75 	static unsigned int call_count  = 0;
76 	unsigned int        first_fail  = 0;
77 	int                 fail_period = 0;
78 	int                 fail        = 0;
79 
80 	++call_count;
81 
82 	/* read the system properties that control failure simulation */
83 	{
84 		char prop_value[PROPERTY_VALUE_MAX];
85 
86 		if (property_get(PROP_MALI_TEST_GRALLOC_FAIL_FIRST, prop_value, "0") > 0)
87 		{
88 			sscanf(prop_value, "%11u", &first_fail);
89 		}
90 
91 		if (property_get(PROP_MALI_TEST_GRALLOC_FAIL_INTERVAL, prop_value, "0") > 0)
92 		{
93 			sscanf(prop_value, "%11u", &fail_period);
94 		}
95 	}
96 
97 	/* failure simulation is enabled by setting the first_fail property to non-zero */
98 	if (first_fail > 0)
99 	{
100 		LOGI("iteration %u (fail=%u, period=%u)\n", call_count, first_fail, fail_period);
101 
102 		fail = (call_count == first_fail) ||
103 		       (call_count > first_fail && fail_period > 0 && 0 == (call_count - first_fail) % fail_period);
104 
105 		if (fail)
106 		{
107 			AERR("failed ump_ref_drv_allocate on iteration #%d\n", call_count);
108 		}
109 	}
110 
111 	return fail;
112 }
113 #endif
114 
115 #ifdef FBIOGET_DMABUF
fb_get_framebuffer_dmabuf(private_module_t * m,private_handle_t * hnd)116 static int fb_get_framebuffer_dmabuf(private_module_t *m, private_handle_t *hnd)
117 {
118 	struct fb_dmabuf_export fb_dma_buf;
119 	int res;
120 	res = ioctl(m->framebuffer->fd, FBIOGET_DMABUF, &fb_dma_buf);
121 
122 	if (res == 0)
123 	{
124 		hnd->share_fd = fb_dma_buf.fd;
125 		return 0;
126 	}
127 	else
128 	{
129 		AINF("FBIOGET_DMABUF ioctl failed(%d). See gralloc_priv.h and the integration manual for vendor framebuffer "
130 		     "integration",
131 		     res);
132 		return -1;
133 	}
134 }
135 #endif
136 
137 #if GRALLOC_ARM_DMA_BUF_MODULE
138 #define DEVPATH "/dev/dma_heap"
dma_heap_open(const char * name)139 int dma_heap_open(const char* name)
140 {
141 	int ret, fd;
142 	char buf[256];
143 
144 	ret = sprintf(buf, "%s/%s", DEVPATH, name);
145 	if (ret < 0) {
146 		AERR("sprintf failed!\n");
147 		return ret;
148 	}
149 
150 	fd = open(buf, O_RDONLY);
151 	if (fd < 0)
152 		AERR("open %s failed!\n", buf);
153 	return fd;
154 }
155 
dma_heap_alloc(int fd,size_t len,unsigned int flags,int * dmabuf_fd)156 int dma_heap_alloc(int fd, size_t len, unsigned int flags, int *dmabuf_fd)
157 {
158 	struct dma_heap_allocation_data data = {
159 		.len = len,
160 		.fd_flags = O_RDWR | O_CLOEXEC,
161 		.heap_flags = flags,
162 	};
163 	int ret;
164 
165 	if (dmabuf_fd == NULL)
166 		return -EINVAL;
167 
168 	ret = ioctl(fd, DMA_HEAP_IOCTL_ALLOC, &data);
169 	if (ret < 0)
170 		return ret;
171 	*dmabuf_fd = (int)data.fd;
172 	return ret;
173 }
174 
alloc_ion_fd(int ion_fd,size_t size,unsigned int heap_mask,unsigned int flags,int * shared_fd)175 static int alloc_ion_fd(int ion_fd, size_t size, unsigned int heap_mask, unsigned int flags, int *shared_fd)
176 {
177 	int heap;
178 
179 	if (interface_ver == INTERFACE_DMABUF_HEAPS) {
180 		int fd = system_heap_id;
181 		unsigned long flg = 0;
182 		if (heap_mask == ION_HEAP_TYPE_DMA_MASK)
183 			fd = cma_heap_id;
184 
185 		return dma_heap_alloc(fd, size, flg, shared_fd);
186 	}
187 
188 	if (interface_ver == INTERFACE_ION_MODERN) {
189 		heap = 1 << system_heap_id;
190 		if (heap_mask == ION_HEAP_TYPE_DMA_MASK)
191 			heap = 1 << cma_heap_id;
192 	} else {
193 		heap = heap_mask;
194 	}
195 	return ion_alloc_fd(ion_fd, size, 0, heap, flags, shared_fd);
196 }
197 #endif
198 
gralloc_alloc_buffer(alloc_device_t * dev,size_t size,int usage,buffer_handle_t * pHandle)199 static int gralloc_alloc_buffer(alloc_device_t *dev, size_t size, int usage, buffer_handle_t *pHandle)
200 {
201 #if GRALLOC_ARM_DMA_BUF_MODULE
202 	{
203 		private_module_t *m = reinterpret_cast<private_module_t *>(dev->common.module);
204 		void *cpu_ptr = MAP_FAILED;
205 		int shared_fd;
206 		int ret;
207 		unsigned int heap_mask;
208 		int lock_state = 0;
209 		int map_mask = 0;
210 
211 		if (usage & GRALLOC_USAGE_PROTECTED) {
212 #if defined(ION_HEAP_SECURE_MASK)
213 			heap_mask = ION_HEAP_SECURE_MASK;
214 #else
215 			AERR("The platform does NOT support protected ION memory.");
216 			return -1;
217 #endif
218 		}
219 		else if (usage & GRALLOC_USAGE_HW_FB) {
220 			heap_mask = ION_HEAP_TYPE_DMA_MASK;
221 		}
222 		else {
223 			heap_mask = ION_HEAP_SYSTEM_MASK;
224 		}
225 
226 		ret = alloc_ion_fd(m->ion_client, size, heap_mask, 0, &shared_fd);
227 		if (ret != 0) {
228 			AERR("Failed to ion_alloc_fd from ion_client:%d", m->ion_client);
229 			return -1;
230 		}
231 
232 		if (!(usage & GRALLOC_USAGE_PROTECTED))
233 		{
234 			map_mask = PROT_READ | PROT_WRITE;
235 		}
236 		else
237 		{
238 			map_mask = PROT_WRITE;
239 		}
240 
241 		cpu_ptr = mmap(NULL, size, map_mask, MAP_SHARED, shared_fd, 0);
242 
243 		if (MAP_FAILED == cpu_ptr)
244 		{
245 			AERR("ion_map( %d ) failed", m->ion_client);
246 
247 			close(shared_fd);
248 			return -1;
249 		}
250 
251 		lock_state = private_handle_t::LOCK_STATE_MAPPED;
252 
253 		private_handle_t *hnd = new private_handle_t(private_handle_t::PRIV_FLAGS_USES_ION, usage, size, cpu_ptr, lock_state);
254 
255 		if (NULL != hnd)
256 		{
257 			hnd->share_fd = shared_fd;
258 			*pHandle = hnd;
259 			return 0;
260 		}
261 		else
262 		{
263 			AERR("Gralloc out of mem for ion_client:%d", m->ion_client);
264 		}
265 
266 		close(shared_fd);
267 
268 		ret = munmap(cpu_ptr, size);
269 
270 		if (0 != ret)
271 		{
272 			AERR("munmap failed for base:%p size: %lu", cpu_ptr, (unsigned long)size);
273 		}
274 
275 		return -1;
276 	}
277 #endif
278 
279 #if GRALLOC_ARM_UMP_MODULE
280 	MALI_IGNORE(dev);
281 	{
282 		ump_handle ump_mem_handle;
283 		void *cpu_ptr;
284 		ump_secure_id ump_id;
285 		ump_alloc_constraints constraints;
286 
287 		size = round_up_to_page_size(size);
288 
289 		if ((usage & GRALLOC_USAGE_SW_READ_MASK) == GRALLOC_USAGE_SW_READ_OFTEN)
290 		{
291 			constraints =  UMP_REF_DRV_CONSTRAINT_USE_CACHE;
292 		}
293 		else
294 		{
295 			constraints = UMP_REF_DRV_CONSTRAINT_NONE;
296 		}
297 
298 #ifdef GRALLOC_SIMULATE_FAILURES
299 
300 		/* if the failure condition matches, fail this iteration */
301 		if (__ump_alloc_should_fail())
302 		{
303 			ump_mem_handle = UMP_INVALID_MEMORY_HANDLE;
304 		}
305 		else
306 #endif
307 		{
308 			if (usage & GRALLOC_USAGE_PROTECTED)
309 			{
310 				AERR("gralloc_alloc_buffer() does not support to allocate protected UMP memory.");
311 			}
312 			else
313 			{
314 				ump_mem_handle = ump_ref_drv_allocate(size, constraints);
315 
316 				if (UMP_INVALID_MEMORY_HANDLE != ump_mem_handle)
317 				{
318 					cpu_ptr = ump_mapped_pointer_get(ump_mem_handle);
319 
320 					if (NULL != cpu_ptr)
321 					{
322 						ump_id = ump_secure_id_get(ump_mem_handle);
323 
324 						if (UMP_INVALID_SECURE_ID != ump_id)
325 						{
326 							private_handle_t *hnd = new private_handle_t(private_handle_t::PRIV_FLAGS_USES_UMP, usage, size, cpu_ptr,
327 							        private_handle_t::LOCK_STATE_MAPPED, ump_id, ump_mem_handle);
328 
329 							if (NULL != hnd)
330 							{
331 								*pHandle = hnd;
332 								return 0;
333 							}
334 							else
335 							{
336 								AERR("gralloc_alloc_buffer() failed to allocate handle. ump_handle = %p, ump_id = %d", ump_mem_handle, ump_id);
337 							}
338 						}
339 						else
340 						{
341 							AERR("gralloc_alloc_buffer() failed to retrieve valid secure id. ump_handle = %p", ump_mem_handle);
342 						}
343 
344 						ump_mapped_pointer_release(ump_mem_handle);
345 					}
346 					else
347 					{
348 						AERR("gralloc_alloc_buffer() failed to map UMP memory. ump_handle = %p", ump_mem_handle);
349 					}
350 
351 					ump_reference_release(ump_mem_handle);
352 				}
353 				else
354 				{
355 					AERR("gralloc_alloc_buffer() failed to allocate UMP memory. size:%d constraints: %d", size, constraints);
356 				}
357 			}
358 		}
359 
360 		return -1;
361 	}
362 #endif
363 
364 }
365 
366 #ifndef DISABLE_FRAMEBUFFER_HAL
gralloc_alloc_framebuffer_locked(alloc_device_t * dev,size_t size,int usage,buffer_handle_t * pHandle)367 static int gralloc_alloc_framebuffer_locked(alloc_device_t *dev, size_t size, int usage, buffer_handle_t *pHandle)
368 {
369 	private_module_t *m = reinterpret_cast<private_module_t *>(dev->common.module);
370 
371 	// allocate the framebuffer
372 	if (m->framebuffer == NULL)
373 	{
374 		// initialize the framebuffer, the framebuffer is mapped once and forever.
375 		int err = init_frame_buffer_locked(m);
376 
377 		if (err < 0)
378 		{
379 			return err;
380 		}
381 	}
382 
383 	uint32_t bufferMask = m->bufferMask;
384 	const uint32_t numBuffers = m->numBuffers;
385 	const size_t bufferSize = m->finfo.line_length * m->info.yres;
386 
387 	if (numBuffers == 1)
388 	{
389 		// If we have only one buffer, we never use page-flipping. Instead,
390 		// we return a regular buffer which will be memcpy'ed to the main
391 		// screen when post is called.
392 		int newUsage = (usage & ~GRALLOC_USAGE_HW_FB) | GRALLOC_USAGE_HW_2D;
393 		AERR("fallback to single buffering. Virtual Y-res too small %d", m->info.yres);
394 		return gralloc_alloc_buffer(dev, bufferSize, newUsage, pHandle);
395 	}
396 
397 	if (bufferMask >= ((1LU << numBuffers) - 1))
398 	{
399 		// We ran out of buffers, reset bufferMask.
400 		bufferMask = 0;
401 		m->bufferMask = 0;
402 	}
403 
404 	void *vaddr = m->framebuffer->base;
405 
406 	// find a free slot
407 	for (uint32_t i = 0 ; i < numBuffers ; i++)
408 	{
409 		if ((bufferMask & (1LU << i)) == 0)
410 		{
411 			m->bufferMask |= (1LU << i);
412 			break;
413 		}
414 
415 		vaddr = (void *)((uintptr_t)vaddr + bufferSize);
416 	}
417 
418 	// The entire framebuffer memory is already mapped, now create a buffer object for parts of this memory
419 	private_handle_t *hnd = new private_handle_t(private_handle_t::PRIV_FLAGS_FRAMEBUFFER, usage, size, vaddr,
420 	        0, m->framebuffer->fd, (uintptr_t)vaddr - (uintptr_t) m->framebuffer->base, m->framebuffer->fb_paddr);
421 
422 #if GRALLOC_ARM_UMP_MODULE
423 	hnd->ump_id = m->framebuffer->ump_id;
424 
425 	/* create a backing ump memory handle if the framebuffer is exposed as a secure ID */
426 	if ((int)UMP_INVALID_SECURE_ID != hnd->ump_id)
427 	{
428 		hnd->ump_mem_handle = (int)ump_handle_create_from_secure_id(hnd->ump_id);
429 
430 		if ((int)UMP_INVALID_MEMORY_HANDLE == hnd->ump_mem_handle)
431 		{
432 			AINF("warning: unable to create UMP handle from secure ID %i\n", hnd->ump_id);
433 		}
434 	}
435 
436 #endif
437 
438 #if GRALLOC_ARM_DMA_BUF_MODULE
439 	{
440 #ifdef FBIOGET_DMABUF
441 		/*
442 		 * Perform allocator specific actions. If these fail we fall back to a regular buffer
443 		 * which will be memcpy'ed to the main screen when fb_post is called.
444 		 */
445 		if (fb_get_framebuffer_dmabuf(m, hnd) == -1)
446 		{
447 			int newUsage = (usage & ~GRALLOC_USAGE_HW_FB) | GRALLOC_USAGE_HW_2D;
448 
449 			AINF("Fallback to single buffering. Unable to map framebuffer memory to handle:%p", hnd);
450 			return gralloc_alloc_buffer(dev, bufferSize, newUsage, pHandle);
451 		}
452 #endif
453 	}
454 
455 	// correct numFds/numInts when there is no dmabuf fd
456 	if (hnd->share_fd < 0)
457 	{
458 		hnd->numFds--;
459 		hnd->numInts++;
460 	}
461 #endif
462 
463 	*pHandle = hnd;
464 
465 	return 0;
466 }
467 
gralloc_alloc_framebuffer(alloc_device_t * dev,size_t size,int usage,buffer_handle_t * pHandle)468 static int gralloc_alloc_framebuffer(alloc_device_t *dev, size_t size, int usage, buffer_handle_t *pHandle)
469 {
470 	private_module_t *m = reinterpret_cast<private_module_t *>(dev->common.module);
471 	pthread_mutex_lock(&m->lock);
472 	int err = gralloc_alloc_framebuffer_locked(dev, size, usage, pHandle);
473 	pthread_mutex_unlock(&m->lock);
474 	return err;
475 }
476 #endif /* DISABLE_FRAMEBUFFER_HAL */
477 
alloc_device_alloc(alloc_device_t * dev,int w,int h,int format,int usage,buffer_handle_t * pHandle,int * pStride)478 static int alloc_device_alloc(alloc_device_t *dev, int w, int h, int format, int usage, buffer_handle_t *pHandle, int *pStride)
479 {
480 	if (!pHandle || !pStride)
481 	{
482 		return -EINVAL;
483 	}
484 
485 	size_t size;
486 	size_t stride;
487 	int bpp = 1;
488 
489 	if (format == HAL_PIXEL_FORMAT_YCrCb_420_SP || format == HAL_PIXEL_FORMAT_YV12
490 	        /* HAL_PIXEL_FORMAT_YCbCr_420_SP, HAL_PIXEL_FORMAT_YCbCr_420_P, HAL_PIXEL_FORMAT_YCbCr_422_I are not defined in Android.
491 	         * To enable Mali DDK EGLImage support for those formats, firstly, you have to add them in Android system/core/include/system/graphics.h.
492 	         * Then, define SUPPORT_LEGACY_FORMAT in the same header file(Mali DDK will also check this definition).
493 	         */
494 #ifdef SUPPORT_LEGACY_FORMAT
495 	        || format == HAL_PIXEL_FORMAT_YCbCr_420_SP || format == HAL_PIXEL_FORMAT_YCbCr_420_P || format == HAL_PIXEL_FORMAT_YCbCr_422_I
496 #endif
497 	   )
498 	{
499 		switch (format)
500 		{
501 			case HAL_PIXEL_FORMAT_YCrCb_420_SP:
502 				stride = GRALLOC_ALIGN(w, 16);
503 				size = GRALLOC_ALIGN(h, 16) * (stride + GRALLOC_ALIGN(stride / 2, 16));
504 				break;
505 
506 			case HAL_PIXEL_FORMAT_YV12:
507 #ifdef SUPPORT_LEGACY_FORMAT
508 			case HAL_PIXEL_FORMAT_YCbCr_420_P:
509 #endif
510 				/*
511 				 * Since Utgard has limitation that "64-byte alignment is enforced on texture and mipmap addresses", here to make sure
512 				 * the v, u plane start addresses are 64-byte aligned.
513 				 */
514 				stride = GRALLOC_ALIGN(w, (h % 8 == 0) ? GRALLOC_ALIGN_BASE_16 :
515 										 ((h % 4 == 0) ? GRALLOC_ALIGN_BASE_64 : GRALLOC_ALIGN_BASE_128));
516 				size = GRALLOC_ALIGN(h, 2) * (stride + GRALLOC_ALIGN(stride / 2, 16));
517 
518 				break;
519 #ifdef SUPPORT_LEGACY_FORMAT
520 
521 			case HAL_PIXEL_FORMAT_YCbCr_420_SP:
522 				stride = GRALLOC_ALIGN(w, 16);
523 				size = GRALLOC_ALIGN(h, 16) * (stride + GRALLOC_ALIGN(stride / 2, 16));
524 				break;
525 
526 			case HAL_PIXEL_FORMAT_YCbCr_422_I:
527 				stride = GRALLOC_ALIGN(w, 16);
528 				size = h * stride * 2;
529 
530 				break;
531 #endif
532 
533 			default:
534 				return -EINVAL;
535 		}
536 	}
537 	else
538 	{
539 
540 		switch (format)
541 		{
542 			case HAL_PIXEL_FORMAT_RGBA_8888:
543 			case HAL_PIXEL_FORMAT_RGBX_8888:
544 			case HAL_PIXEL_FORMAT_BGRA_8888:
545 				bpp = 4;
546 				break;
547 
548 			case HAL_PIXEL_FORMAT_RGB_888:
549 				bpp = 3;
550 				break;
551 
552 			case HAL_PIXEL_FORMAT_RGB_565:
553 #if PLATFORM_SDK_VERSION < 19
554 			case HAL_PIXEL_FORMAT_RGBA_5551:
555 			case HAL_PIXEL_FORMAT_RGBA_4444:
556 #endif
557 				bpp = 2;
558 				break;
559 
560 			case HAL_PIXEL_FORMAT_BLOB:
561 				if (h != 1) {
562 					AERR("Height for HAL_PIXEL_FORMAT_BLOB must be 1. h=%d", h);
563 					return -EINVAL;
564 				}
565 				break;
566 
567 			default:
568 				AERR("The format is not supported yet: format=%d\n",  format);
569 				return -EINVAL;
570 		}
571 
572 		if (format == HAL_PIXEL_FORMAT_BLOB) {
573 			stride = 0; /* No 'rows', it's effectively a long one dimensional array */
574 			size = w;
575 		}else{
576 			size_t bpr = GRALLOC_ALIGN(w * bpp, 64);
577 			size = bpr * h;
578 			stride = bpr / bpp;
579 		}
580 	}
581 
582 	int err;
583 
584 #ifndef DISABLE_FRAMEBUFFER_HAL
585 
586 	if (usage & GRALLOC_USAGE_HW_FB)
587 	{
588 		err = gralloc_alloc_framebuffer(dev, size, usage, pHandle);
589 	}
590 	else
591 #endif
592 
593 	{
594 		err = gralloc_alloc_buffer(dev, size, usage, pHandle);
595 	}
596 
597 	if (err < 0)
598 	{
599 		return err;
600 	}
601 
602 	/* match the framebuffer format */
603 	if (usage & GRALLOC_USAGE_HW_FB)
604 	{
605 #ifdef GRALLOC_16_BITS
606 		format = HAL_PIXEL_FORMAT_RGB_565;
607 #else
608 		format = HAL_PIXEL_FORMAT_BGRA_8888;
609 #endif
610 	}
611 
612 	private_handle_t *hnd = (private_handle_t *)*pHandle;
613 	int               private_usage = usage & (GRALLOC_USAGE_PRIVATE_0 |
614 	                                  GRALLOC_USAGE_PRIVATE_1);
615 
616 	switch (private_usage)
617 	{
618 		case 0:
619 			hnd->yuv_info = MALI_YUV_BT601_NARROW;
620 			break;
621 
622 		case GRALLOC_USAGE_PRIVATE_1:
623 			hnd->yuv_info = MALI_YUV_BT601_WIDE;
624 			break;
625 
626 		case GRALLOC_USAGE_PRIVATE_0:
627 			hnd->yuv_info = MALI_YUV_BT709_NARROW;
628 			break;
629 
630 		case (GRALLOC_USAGE_PRIVATE_0 | GRALLOC_USAGE_PRIVATE_1):
631 			hnd->yuv_info = MALI_YUV_BT709_WIDE;
632 			break;
633 	}
634 
635 	hnd->width = w;
636 	hnd->height = h;
637 	hnd->format = format;
638 	hnd->stride = stride;
639 	hnd->byte_stride = GRALLOC_ALIGN(w*bpp,64);
640 	*pStride = stride;
641 	return 0;
642 }
643 
alloc_device_free(alloc_device_t __unused * dev,buffer_handle_t handle)644 static int alloc_device_free(alloc_device_t __unused *dev, buffer_handle_t handle)
645 {
646 	if (private_handle_t::validate(handle) < 0)
647 	{
648 		return -EINVAL;
649 	}
650 
651 	private_handle_t const *hnd = reinterpret_cast<private_handle_t const *>(handle);
652 
653 	if (hnd->flags & private_handle_t::PRIV_FLAGS_FRAMEBUFFER)
654 	{
655 #if GRALLOC_ARM_UMP_MODULE
656 
657 		if ((int)UMP_INVALID_MEMORY_HANDLE != hnd->ump_mem_handle)
658 		{
659 			ump_reference_release((ump_handle)hnd->ump_mem_handle);
660 		}
661 
662 #endif
663 	}
664 	else if (hnd->flags & private_handle_t::PRIV_FLAGS_USES_UMP)
665 	{
666 #if GRALLOC_ARM_UMP_MODULE
667 
668 		/* Buffer might be unregistered so we need to check for invalid ump handle*/
669 		if ((int)UMP_INVALID_MEMORY_HANDLE != hnd->ump_mem_handle)
670 		{
671 			ump_mapped_pointer_release((ump_handle)hnd->ump_mem_handle);
672 			ump_reference_release((ump_handle)hnd->ump_mem_handle);
673 		}
674 
675 #else
676 		AERR("Can't free ump memory for handle:%p. Not supported.", hnd);
677 #endif
678 	}
679 	else if (hnd->flags & private_handle_t::PRIV_FLAGS_USES_ION)
680 	{
681 #if GRALLOC_ARM_DMA_BUF_MODULE
682 		/* Buffer might be unregistered so we need to check for invalid ump handle*/
683 		if (0 != hnd->base)
684 		{
685 			if (0 != munmap((void *)hnd->base, hnd->size))
686 			{
687 				AERR("Failed to munmap handle %p", hnd);
688 			}
689 		}
690 
691 		close(hnd->share_fd);
692 
693 		memset((void *)hnd, 0, sizeof(*hnd));
694 #else
695 		AERR("Can't free dma_buf memory for handle:0x%x. Not supported.", (unsigned int)hnd);
696 #endif
697 
698 	}
699 
700 	delete hnd;
701 
702 	return 0;
703 }
704 
alloc_device_close(struct hw_device_t * device)705 static int alloc_device_close(struct hw_device_t *device)
706 {
707 	alloc_device_t *dev = reinterpret_cast<alloc_device_t *>(device);
708 
709 	if (dev)
710 	{
711 #if GRALLOC_ARM_DMA_BUF_MODULE
712 		private_module_t *m = reinterpret_cast<private_module_t *>(device);
713 
714 		if (0 != ion_close(m->ion_client))
715 		{
716 			AERR("Failed to close ion_client: %d", m->ion_client);
717 		}
718 
719 		close(m->ion_client);
720 #endif
721 		delete dev;
722 #if GRALLOC_ARM_UMP_MODULE
723 		ump_close(); // Our UMP memory refs will be released automatically here...
724 #endif
725 	}
726 
727 	return 0;
728 }
729 
730 #if GRALLOC_ARM_DMA_BUF_MODULE
find_heap_id(int ion_client,char * name)731 static int find_heap_id(int ion_client, char* name)
732 {
733 	int i, ret, cnt, heap_id = -1;
734 	struct ion_heap_data *data;
735 
736 	ret = ion_query_heap_cnt(ion_client, &cnt);
737 
738 	if (ret)
739 	{
740 		AERR("ion count query failed with %s", strerror(errno));
741 		return -1;
742 	}
743 
744 	data = (struct ion_heap_data *)malloc(cnt * sizeof(*data));
745 	if (!data)
746 	{
747 		AERR("Error allocating data %s\n", strerror(errno));
748 		return -1;
749 	}
750 
751 	ret = ion_query_get_heaps(ion_client, cnt, data);
752 	if (ret)
753 	{
754 		AERR("Error querying heaps from ion %s", strerror(errno));
755 	}
756 	else
757 	{
758 		for (i = 0; i < cnt; i++) {
759 			struct ion_heap_data *dat = (struct ion_heap_data *)data;
760 			if (strcmp(dat[i].name, name) == 0) {
761 				heap_id = dat[i].heap_id;
762 				break;
763 			}
764 		}
765 
766 		if (i > cnt)
767 		{
768 			AERR("No System Heap Found amongst %d heaps\n", cnt);
769 			heap_id = -1;
770 		}
771 	}
772 
773 	free(data);
774 	return heap_id;
775 }
776 #endif
777 
initialize_interface(private_module_t * m)778 static int initialize_interface(private_module_t *m)
779 {
780 	int fd;
781 
782 	if (interface_ver != INTERFACE_UNKNOWN)
783 		return 0;
784 
785 	/* test for dma-heaps*/
786 	fd = dma_heap_open(DMABUF_SYSTEM);
787 	if (fd >= 0) {
788 		AINF("Using DMA-BUF Heaps.\n");
789 		interface_ver = INTERFACE_DMABUF_HEAPS;
790 		system_heap_id = fd;
791 		cma_heap_id = dma_heap_open(DMABUF_CMA);
792 		/* Open other dma heaps here */
793 		return 0;
794 	}
795 
796 	/* test for modern vs legacy ION */
797 	m->ion_client = ion_open();
798 	if (m->ion_client < 0) {
799 		AERR("ion_open failed with %s", strerror(errno));
800 		return -1;
801 	}
802 	if (!ion_is_legacy(m->ion_client)) {
803 		system_heap_id = find_heap_id(m->ion_client, ION_SYSTEM);
804 		cma_heap_id = find_heap_id(m->ion_client, ION_CMA);
805 		if (system_heap_id < 0) {
806 			ion_close(m->ion_client);
807 			m->ion_client = -1;
808 			AERR( "ion_open failed: no system heap found" );
809 			return -1;
810 		}
811 		if (cma_heap_id < 0) {
812 			AERR("No cma heap found, falling back to system");
813 			cma_heap_id = system_heap_id;
814 		}
815 		AINF("Using ION Modern interface.\n");
816 		interface_ver = INTERFACE_ION_MODERN;
817 	} else {
818 		AINF("Using ION Legacy interface.\n");
819 		interface_ver = INTERFACE_ION_LEGACY;
820 	}
821 	return 0;
822 }
823 
alloc_device_open(hw_module_t const * module,const char * name,hw_device_t ** device)824 int alloc_device_open(hw_module_t const *module, const char *name, hw_device_t **device)
825 {
826 	MALI_IGNORE(name);
827 	alloc_device_t *dev;
828 
829 	dev = new alloc_device_t;
830 
831 	if (NULL == dev)
832 	{
833 		return -1;
834 	}
835 
836 #if GRALLOC_ARM_UMP_MODULE
837 	ump_result ump_res = ump_open();
838 
839 	if (UMP_OK != ump_res)
840 	{
841 		AERR("UMP open failed with %d", ump_res);
842 		delete dev;
843 		return -1;
844 	}
845 
846 #endif
847 
848 	/* initialize our state here */
849 	memset(dev, 0, sizeof(*dev));
850 
851 	/* initialize the procs */
852 	dev->common.tag = HARDWARE_DEVICE_TAG;
853 	dev->common.version = 0;
854 	dev->common.module = const_cast<hw_module_t *>(module);
855 	dev->common.close = alloc_device_close;
856 	dev->alloc = alloc_device_alloc;
857 	dev->free = alloc_device_free;
858 
859 #if GRALLOC_ARM_DMA_BUF_MODULE
860 	private_module_t *m = reinterpret_cast<private_module_t *>(dev->common.module);
861 
862 	if (initialize_interface(m) < 0) {
863 		delete dev;
864 		return -1;
865 	}
866 #endif
867 
868 	*device = &dev->common;
869 
870 	return 0;
871 }
872