1*d83cc019SAndroid Build Coastguard Worker #include <ion/ion.h>
2*d83cc019SAndroid Build Coastguard Worker #include <linux/ion_4.12.h>
3*d83cc019SAndroid Build Coastguard Worker
4*d83cc019SAndroid Build Coastguard Worker #include "ion.h"
5*d83cc019SAndroid Build Coastguard Worker
ion_get_heap_id(int ion_fd,uint32_t heap_type)6*d83cc019SAndroid Build Coastguard Worker int ion_get_heap_id(int ion_fd, uint32_t heap_type)
7*d83cc019SAndroid Build Coastguard Worker {
8*d83cc019SAndroid Build Coastguard Worker int heap_count = 0;
9*d83cc019SAndroid Build Coastguard Worker int ret = -1;
10*d83cc019SAndroid Build Coastguard Worker
11*d83cc019SAndroid Build Coastguard Worker if (ion_query_heap_cnt(ion_fd, &heap_count))
12*d83cc019SAndroid Build Coastguard Worker {
13*d83cc019SAndroid Build Coastguard Worker return -1;
14*d83cc019SAndroid Build Coastguard Worker }
15*d83cc019SAndroid Build Coastguard Worker
16*d83cc019SAndroid Build Coastguard Worker struct ion_heap_data *heap_data =
17*d83cc019SAndroid Build Coastguard Worker malloc(heap_count * sizeof(struct ion_heap_data));
18*d83cc019SAndroid Build Coastguard Worker
19*d83cc019SAndroid Build Coastguard Worker if (ion_query_get_heaps(ion_fd, heap_count, heap_data))
20*d83cc019SAndroid Build Coastguard Worker {
21*d83cc019SAndroid Build Coastguard Worker free(heap_data);
22*d83cc019SAndroid Build Coastguard Worker return -1;
23*d83cc019SAndroid Build Coastguard Worker }
24*d83cc019SAndroid Build Coastguard Worker
25*d83cc019SAndroid Build Coastguard Worker for (struct ion_heap_data *hi = heap_data;
26*d83cc019SAndroid Build Coastguard Worker (hi - heap_data) < heap_count;
27*d83cc019SAndroid Build Coastguard Worker ++hi)
28*d83cc019SAndroid Build Coastguard Worker {
29*d83cc019SAndroid Build Coastguard Worker if (hi->type == heap_type)
30*d83cc019SAndroid Build Coastguard Worker {
31*d83cc019SAndroid Build Coastguard Worker ret = hi->heap_id;
32*d83cc019SAndroid Build Coastguard Worker break;
33*d83cc019SAndroid Build Coastguard Worker }
34*d83cc019SAndroid Build Coastguard Worker }
35*d83cc019SAndroid Build Coastguard Worker
36*d83cc019SAndroid Build Coastguard Worker free(heap_data);
37*d83cc019SAndroid Build Coastguard Worker
38*d83cc019SAndroid Build Coastguard Worker return ret;
39*d83cc019SAndroid Build Coastguard Worker }
40*d83cc019SAndroid Build Coastguard Worker
41*d83cc019SAndroid Build Coastguard Worker static const int kBitsInAnInt = (sizeof(int) * 8);
42*d83cc019SAndroid Build Coastguard Worker
ion_alloc_one_fd(int ion_fd,size_t size,int heap_id,int * ion_buffer_fd)43*d83cc019SAndroid Build Coastguard Worker int ion_alloc_one_fd(int ion_fd, size_t size, int heap_id, int *ion_buffer_fd)
44*d83cc019SAndroid Build Coastguard Worker {
45*d83cc019SAndroid Build Coastguard Worker if (heap_id < 0 || heap_id >= kBitsInAnInt)
46*d83cc019SAndroid Build Coastguard Worker {
47*d83cc019SAndroid Build Coastguard Worker return -1;
48*d83cc019SAndroid Build Coastguard Worker }
49*d83cc019SAndroid Build Coastguard Worker
50*d83cc019SAndroid Build Coastguard Worker const int align = 0;
51*d83cc019SAndroid Build Coastguard Worker const int heap_mask = 1 << heap_id;
52*d83cc019SAndroid Build Coastguard Worker const int flags = 0;
53*d83cc019SAndroid Build Coastguard Worker
54*d83cc019SAndroid Build Coastguard Worker return ion_alloc_fd(ion_fd, size, align, heap_mask, flags, ion_buffer_fd);
55*d83cc019SAndroid Build Coastguard Worker }
56*d83cc019SAndroid Build Coastguard Worker
ion_mmap(void ** ptr,int ion_buffer_fd,size_t size)57*d83cc019SAndroid Build Coastguard Worker int ion_mmap(void **ptr, int ion_buffer_fd, size_t size)
58*d83cc019SAndroid Build Coastguard Worker {
59*d83cc019SAndroid Build Coastguard Worker void *const k_addr = 0;
60*d83cc019SAndroid Build Coastguard Worker const int k_prot = PROT_READ | PROT_WRITE;
61*d83cc019SAndroid Build Coastguard Worker const int k_flags = MAP_SHARED;
62*d83cc019SAndroid Build Coastguard Worker
63*d83cc019SAndroid Build Coastguard Worker void *ret = mmap(k_addr, size, k_prot, k_flags, ion_buffer_fd, 0);
64*d83cc019SAndroid Build Coastguard Worker
65*d83cc019SAndroid Build Coastguard Worker if (ret == MAP_FAILED)
66*d83cc019SAndroid Build Coastguard Worker {
67*d83cc019SAndroid Build Coastguard Worker return -1;
68*d83cc019SAndroid Build Coastguard Worker }
69*d83cc019SAndroid Build Coastguard Worker
70*d83cc019SAndroid Build Coastguard Worker *ptr = ret;
71*d83cc019SAndroid Build Coastguard Worker return 0;
72*d83cc019SAndroid Build Coastguard Worker }
73*d83cc019SAndroid Build Coastguard Worker
ion_munmap(void * ptr,size_t size)74*d83cc019SAndroid Build Coastguard Worker int ion_munmap(void *ptr, size_t size)
75*d83cc019SAndroid Build Coastguard Worker {
76*d83cc019SAndroid Build Coastguard Worker if (munmap(ptr, size))
77*d83cc019SAndroid Build Coastguard Worker {
78*d83cc019SAndroid Build Coastguard Worker return -1;
79*d83cc019SAndroid Build Coastguard Worker }
80*d83cc019SAndroid Build Coastguard Worker
81*d83cc019SAndroid Build Coastguard Worker return 0;
82*d83cc019SAndroid Build Coastguard Worker }
83*d83cc019SAndroid Build Coastguard Worker
drm_check_prime_caps(int drm_fd)84*d83cc019SAndroid Build Coastguard Worker int drm_check_prime_caps(int drm_fd)
85*d83cc019SAndroid Build Coastguard Worker {
86*d83cc019SAndroid Build Coastguard Worker struct drm_get_cap drm_get_cap_arg = {
87*d83cc019SAndroid Build Coastguard Worker .capability = DRM_CAP_PRIME,
88*d83cc019SAndroid Build Coastguard Worker .value = 0
89*d83cc019SAndroid Build Coastguard Worker };
90*d83cc019SAndroid Build Coastguard Worker
91*d83cc019SAndroid Build Coastguard Worker if (drmIoctl(drm_fd, DRM_IOCTL_GET_CAP, &drm_get_cap_arg))
92*d83cc019SAndroid Build Coastguard Worker {
93*d83cc019SAndroid Build Coastguard Worker return -1;
94*d83cc019SAndroid Build Coastguard Worker }
95*d83cc019SAndroid Build Coastguard Worker
96*d83cc019SAndroid Build Coastguard Worker if (!(drm_get_cap_arg.value & DRM_PRIME_CAP_IMPORT) ||
97*d83cc019SAndroid Build Coastguard Worker !(drm_get_cap_arg.value & DRM_PRIME_CAP_EXPORT))
98*d83cc019SAndroid Build Coastguard Worker {
99*d83cc019SAndroid Build Coastguard Worker return -1;
100*d83cc019SAndroid Build Coastguard Worker }
101*d83cc019SAndroid Build Coastguard Worker
102*d83cc019SAndroid Build Coastguard Worker return 0;
103*d83cc019SAndroid Build Coastguard Worker }
104*d83cc019SAndroid Build Coastguard Worker
gem_handle_for_ion_buffer(int drm_fd,uint32_t * gem_handle,int ion_buffer_fd)105*d83cc019SAndroid Build Coastguard Worker int gem_handle_for_ion_buffer(int drm_fd, uint32_t *gem_handle, int ion_buffer_fd)
106*d83cc019SAndroid Build Coastguard Worker {
107*d83cc019SAndroid Build Coastguard Worker struct drm_prime_handle drm_prime_fd_to_handle_arg = {
108*d83cc019SAndroid Build Coastguard Worker .handle = 0,
109*d83cc019SAndroid Build Coastguard Worker .flags = 0,
110*d83cc019SAndroid Build Coastguard Worker .fd = ion_buffer_fd
111*d83cc019SAndroid Build Coastguard Worker };
112*d83cc019SAndroid Build Coastguard Worker
113*d83cc019SAndroid Build Coastguard Worker if (drmIoctl(drm_fd,
114*d83cc019SAndroid Build Coastguard Worker DRM_IOCTL_PRIME_FD_TO_HANDLE,
115*d83cc019SAndroid Build Coastguard Worker &drm_prime_fd_to_handle_arg))
116*d83cc019SAndroid Build Coastguard Worker {
117*d83cc019SAndroid Build Coastguard Worker return -1;
118*d83cc019SAndroid Build Coastguard Worker }
119*d83cc019SAndroid Build Coastguard Worker
120*d83cc019SAndroid Build Coastguard Worker *gem_handle = drm_prime_fd_to_handle_arg.handle;
121*d83cc019SAndroid Build Coastguard Worker
122*d83cc019SAndroid Build Coastguard Worker return 0;
123*d83cc019SAndroid Build Coastguard Worker }
124*d83cc019SAndroid Build Coastguard Worker
ion_fd_for_gem_handle(int drm_fd,int * ion_fd,uint32_t gem_handle)125*d83cc019SAndroid Build Coastguard Worker int ion_fd_for_gem_handle(int drm_fd, int *ion_fd, uint32_t gem_handle)
126*d83cc019SAndroid Build Coastguard Worker {
127*d83cc019SAndroid Build Coastguard Worker struct drm_prime_handle drm_prime_handle_to_fd_arg = {
128*d83cc019SAndroid Build Coastguard Worker .handle = gem_handle,
129*d83cc019SAndroid Build Coastguard Worker .flags = 0,
130*d83cc019SAndroid Build Coastguard Worker .fd = 0
131*d83cc019SAndroid Build Coastguard Worker };
132*d83cc019SAndroid Build Coastguard Worker
133*d83cc019SAndroid Build Coastguard Worker if (drmIoctl(drm_fd,
134*d83cc019SAndroid Build Coastguard Worker DRM_IOCTL_PRIME_HANDLE_TO_FD,
135*d83cc019SAndroid Build Coastguard Worker &drm_prime_handle_to_fd_arg))
136*d83cc019SAndroid Build Coastguard Worker {
137*d83cc019SAndroid Build Coastguard Worker return -1;
138*d83cc019SAndroid Build Coastguard Worker }
139*d83cc019SAndroid Build Coastguard Worker
140*d83cc019SAndroid Build Coastguard Worker *ion_fd = drm_prime_handle_to_fd_arg.fd;
141*d83cc019SAndroid Build Coastguard Worker
142*d83cc019SAndroid Build Coastguard Worker return 0;
143*d83cc019SAndroid Build Coastguard Worker }
144*d83cc019SAndroid Build Coastguard Worker
drm_fb_for_ion_buffer(int drm_fd,uint32_t * fb_id,int ion_buffer_fd,const struct fb_configuration * fb_config)145*d83cc019SAndroid Build Coastguard Worker int drm_fb_for_ion_buffer(int drm_fd, uint32_t *fb_id, int ion_buffer_fd,
146*d83cc019SAndroid Build Coastguard Worker const struct fb_configuration *fb_config)
147*d83cc019SAndroid Build Coastguard Worker {
148*d83cc019SAndroid Build Coastguard Worker uint32_t gem_handle = 0;
149*d83cc019SAndroid Build Coastguard Worker
150*d83cc019SAndroid Build Coastguard Worker if (gem_handle_for_ion_buffer(drm_fd, &gem_handle, ion_buffer_fd))
151*d83cc019SAndroid Build Coastguard Worker {
152*d83cc019SAndroid Build Coastguard Worker return -1;
153*d83cc019SAndroid Build Coastguard Worker }
154*d83cc019SAndroid Build Coastguard Worker
155*d83cc019SAndroid Build Coastguard Worker int ret = drm_fb_for_gem_handle(drm_fd, fb_id, gem_handle, fb_config);
156*d83cc019SAndroid Build Coastguard Worker
157*d83cc019SAndroid Build Coastguard Worker gem_release_handle(drm_fd, gem_handle);
158*d83cc019SAndroid Build Coastguard Worker return ret;
159*d83cc019SAndroid Build Coastguard Worker }
160*d83cc019SAndroid Build Coastguard Worker
drm_release_fb(int drm_fd,uint32_t fb_id)161*d83cc019SAndroid Build Coastguard Worker void drm_release_fb(int drm_fd, uint32_t fb_id)
162*d83cc019SAndroid Build Coastguard Worker {
163*d83cc019SAndroid Build Coastguard Worker (void)drmIoctl(drm_fd, DRM_IOCTL_MODE_RMFB, &fb_id);
164*d83cc019SAndroid Build Coastguard Worker }
165*d83cc019SAndroid Build Coastguard Worker
ion_clone_fd_via_gem(int drm_fd,int * cloned_fd,int ion_buffer_fd)166*d83cc019SAndroid Build Coastguard Worker int ion_clone_fd_via_gem(int drm_fd, int *cloned_fd, int ion_buffer_fd)
167*d83cc019SAndroid Build Coastguard Worker {
168*d83cc019SAndroid Build Coastguard Worker uint32_t gem_handle = 0;
169*d83cc019SAndroid Build Coastguard Worker if (gem_handle_for_ion_buffer(drm_fd, &gem_handle, ion_buffer_fd))
170*d83cc019SAndroid Build Coastguard Worker {
171*d83cc019SAndroid Build Coastguard Worker return -1;
172*d83cc019SAndroid Build Coastguard Worker }
173*d83cc019SAndroid Build Coastguard Worker
174*d83cc019SAndroid Build Coastguard Worker int ret = ion_fd_for_gem_handle(drm_fd, cloned_fd, gem_handle);
175*d83cc019SAndroid Build Coastguard Worker gem_release_handle(drm_fd, gem_handle);
176*d83cc019SAndroid Build Coastguard Worker
177*d83cc019SAndroid Build Coastguard Worker return ret;
178*d83cc019SAndroid Build Coastguard Worker }
179*d83cc019SAndroid Build Coastguard Worker
180*d83cc019SAndroid Build Coastguard Worker
181