xref: /OK3568_Linux_fs/external/linux-rga/samples/utils/allocator/dma_alloc.cpp (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun  * Copyright (C) 2022 Rockchip Electronics Co., Ltd.
3*4882a593Smuzhiyun  * Authors:
4*4882a593Smuzhiyun  *  Cerf Yu <cerf.yu@rock-chips.com>
5*4882a593Smuzhiyun  *
6*4882a593Smuzhiyun  * Licensed under the Apache License, Version 2.0 (the "License");
7*4882a593Smuzhiyun  * you may not use this file except in compliance with the License.
8*4882a593Smuzhiyun  * You may obtain a copy of the License at
9*4882a593Smuzhiyun  *
10*4882a593Smuzhiyun  *      http://www.apache.org/licenses/LICENSE-2.0
11*4882a593Smuzhiyun  *
12*4882a593Smuzhiyun  * Unless required by applicable law or agreed to in writing, software
13*4882a593Smuzhiyun  * distributed under the License is distributed on an "AS IS" BASIS,
14*4882a593Smuzhiyun  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15*4882a593Smuzhiyun  * See the License for the specific language governing permissions and
16*4882a593Smuzhiyun  * limitations under the License.
17*4882a593Smuzhiyun  */
18*4882a593Smuzhiyun 
19*4882a593Smuzhiyun #include <getopt.h>
20*4882a593Smuzhiyun #include <sys/mman.h>
21*4882a593Smuzhiyun #include <sys/socket.h>
22*4882a593Smuzhiyun #include <sys/stat.h>
23*4882a593Smuzhiyun #include <time.h>
24*4882a593Smuzhiyun #include <stdbool.h>
25*4882a593Smuzhiyun #include <assert.h>
26*4882a593Smuzhiyun #include <endian.h>
27*4882a593Smuzhiyun #include <errno.h>
28*4882a593Smuzhiyun #include <fcntl.h>
29*4882a593Smuzhiyun #include <stdarg.h>
30*4882a593Smuzhiyun #include <stdio.h>
31*4882a593Smuzhiyun #include <stdlib.h>
32*4882a593Smuzhiyun #include <string.h>
33*4882a593Smuzhiyun #include <sys/ioctl.h>
34*4882a593Smuzhiyun #include <sys/stat.h>
35*4882a593Smuzhiyun #include <sys/types.h>
36*4882a593Smuzhiyun #include <sys/poll.h>
37*4882a593Smuzhiyun #include <unistd.h>
38*4882a593Smuzhiyun #include <stdbool.h>
39*4882a593Smuzhiyun #include <sys/eventfd.h>
40*4882a593Smuzhiyun 
41*4882a593Smuzhiyun #include <sched.h>
42*4882a593Smuzhiyun #include <pthread.h>
43*4882a593Smuzhiyun 
44*4882a593Smuzhiyun #include <stdint.h>
45*4882a593Smuzhiyun #include <math.h>
46*4882a593Smuzhiyun #include <memory.h>
47*4882a593Smuzhiyun #include <sys/time.h>
48*4882a593Smuzhiyun 
49*4882a593Smuzhiyun #include "dma_alloc.h"
50*4882a593Smuzhiyun #include "RgaUtils.h"
51*4882a593Smuzhiyun 
52*4882a593Smuzhiyun typedef unsigned long long __u64;
53*4882a593Smuzhiyun typedef  unsigned int __u32;
54*4882a593Smuzhiyun 
55*4882a593Smuzhiyun struct dma_heap_allocation_data {
56*4882a593Smuzhiyun 	__u64 len;
57*4882a593Smuzhiyun 	__u32 fd;
58*4882a593Smuzhiyun 	__u32 fd_flags;
59*4882a593Smuzhiyun 	__u64 heap_flags;
60*4882a593Smuzhiyun };
61*4882a593Smuzhiyun 
62*4882a593Smuzhiyun #define DMA_HEAP_IOC_MAGIC		'H'
63*4882a593Smuzhiyun #define DMA_HEAP_IOCTL_ALLOC	_IOWR(DMA_HEAP_IOC_MAGIC, 0x0,\
64*4882a593Smuzhiyun 				      struct dma_heap_allocation_data)
65*4882a593Smuzhiyun 
66*4882a593Smuzhiyun #define DMA_BUF_SYNC_READ      (1 << 0)
67*4882a593Smuzhiyun #define DMA_BUF_SYNC_WRITE     (2 << 0)
68*4882a593Smuzhiyun #define DMA_BUF_SYNC_RW        (DMA_BUF_SYNC_READ | DMA_BUF_SYNC_WRITE)
69*4882a593Smuzhiyun #define DMA_BUF_SYNC_START     (0 << 2)
70*4882a593Smuzhiyun #define DMA_BUF_SYNC_END       (1 << 2)
71*4882a593Smuzhiyun 
72*4882a593Smuzhiyun struct dma_buf_sync {
73*4882a593Smuzhiyun 	__u64 flags;
74*4882a593Smuzhiyun };
75*4882a593Smuzhiyun 
76*4882a593Smuzhiyun #define DMA_BUF_BASE		'b'
77*4882a593Smuzhiyun #define DMA_BUF_IOCTL_SYNC	_IOW(DMA_BUF_BASE, 0, struct dma_buf_sync)
78*4882a593Smuzhiyun 
79*4882a593Smuzhiyun #define CMA_HEAP_SIZE	1024 * 1024
80*4882a593Smuzhiyun 
dma_sync_device_to_cpu(int fd)81*4882a593Smuzhiyun int dma_sync_device_to_cpu(int fd) {
82*4882a593Smuzhiyun     struct dma_buf_sync sync = {0};
83*4882a593Smuzhiyun 
84*4882a593Smuzhiyun     sync.flags = DMA_BUF_SYNC_START | DMA_BUF_SYNC_RW;
85*4882a593Smuzhiyun     return ioctl(fd, DMA_BUF_IOCTL_SYNC, &sync);
86*4882a593Smuzhiyun }
87*4882a593Smuzhiyun 
dma_sync_cpu_to_device(int fd)88*4882a593Smuzhiyun int dma_sync_cpu_to_device(int fd) {
89*4882a593Smuzhiyun     struct dma_buf_sync sync = {0};
90*4882a593Smuzhiyun 
91*4882a593Smuzhiyun     sync.flags = DMA_BUF_SYNC_END | DMA_BUF_SYNC_RW;
92*4882a593Smuzhiyun     return ioctl(fd, DMA_BUF_IOCTL_SYNC, &sync);
93*4882a593Smuzhiyun }
94*4882a593Smuzhiyun 
dma_buf_alloc(const char * path,size_t size,int * fd,void ** va)95*4882a593Smuzhiyun int dma_buf_alloc(const char *path, size_t size, int *fd, void **va) {
96*4882a593Smuzhiyun     int ret;
97*4882a593Smuzhiyun     int prot;
98*4882a593Smuzhiyun     void *mmap_va;
99*4882a593Smuzhiyun     int dma_heap_fd = -1;
100*4882a593Smuzhiyun     struct dma_heap_allocation_data buf_data;
101*4882a593Smuzhiyun 
102*4882a593Smuzhiyun     /* open dma_heap fd */
103*4882a593Smuzhiyun     dma_heap_fd = open(path, O_RDWR);
104*4882a593Smuzhiyun     if (dma_heap_fd < 0) {
105*4882a593Smuzhiyun         printf("open %s fail!\n", path);
106*4882a593Smuzhiyun         return dma_heap_fd;
107*4882a593Smuzhiyun     }
108*4882a593Smuzhiyun 
109*4882a593Smuzhiyun     /* alloc buffer */
110*4882a593Smuzhiyun     memset(&buf_data, 0x0, sizeof(struct dma_heap_allocation_data));
111*4882a593Smuzhiyun 
112*4882a593Smuzhiyun     buf_data.len = size;
113*4882a593Smuzhiyun     buf_data.fd_flags = O_CLOEXEC | O_RDWR;
114*4882a593Smuzhiyun     ret = ioctl(dma_heap_fd, DMA_HEAP_IOCTL_ALLOC, &buf_data);
115*4882a593Smuzhiyun     if (ret < 0) {
116*4882a593Smuzhiyun         printf("RK_DMA_HEAP_ALLOC_BUFFER failed\n");
117*4882a593Smuzhiyun         return ret;
118*4882a593Smuzhiyun     }
119*4882a593Smuzhiyun 
120*4882a593Smuzhiyun     /* mmap va */
121*4882a593Smuzhiyun     if (fcntl(buf_data.fd, F_GETFL) & O_RDWR)
122*4882a593Smuzhiyun         prot = PROT_READ | PROT_WRITE;
123*4882a593Smuzhiyun     else
124*4882a593Smuzhiyun         prot = PROT_READ;
125*4882a593Smuzhiyun 
126*4882a593Smuzhiyun     /* mmap contiguors buffer to user */
127*4882a593Smuzhiyun     mmap_va = (void *)mmap(NULL, buf_data.len, prot, MAP_SHARED, buf_data.fd, 0);
128*4882a593Smuzhiyun     if (mmap_va == MAP_FAILED) {
129*4882a593Smuzhiyun         printf("mmap failed: %s\n", strerror(errno));
130*4882a593Smuzhiyun         return -errno;
131*4882a593Smuzhiyun     }
132*4882a593Smuzhiyun 
133*4882a593Smuzhiyun     *va = mmap_va;
134*4882a593Smuzhiyun     *fd = buf_data.fd;
135*4882a593Smuzhiyun 
136*4882a593Smuzhiyun     close(dma_heap_fd);
137*4882a593Smuzhiyun 
138*4882a593Smuzhiyun     return 0;
139*4882a593Smuzhiyun }
140*4882a593Smuzhiyun 
dma_buf_free(size_t size,int * fd,void * va)141*4882a593Smuzhiyun void dma_buf_free(size_t size, int *fd, void *va) {
142*4882a593Smuzhiyun     int len;
143*4882a593Smuzhiyun 
144*4882a593Smuzhiyun     len =  size;
145*4882a593Smuzhiyun     munmap(va, len);
146*4882a593Smuzhiyun 
147*4882a593Smuzhiyun     close(*fd);
148*4882a593Smuzhiyun     *fd = -1;
149*4882a593Smuzhiyun }
150*4882a593Smuzhiyun 
151*4882a593Smuzhiyun 
152*4882a593Smuzhiyun 
153