xref: /aosp_15_r20/external/igt-gpu-tools/lib/ion.c (revision d83cc019efdc2edc6c4b16e9034a3ceb8d35d77c)
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