xref: /OK3568_Linux_fs/external/mpp/inc/mpp_buffer.h (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1 /*
2  * Copyright 2015 Rockchip Electronics Co. LTD
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #ifndef __MPP_BUFFER_H__
18 #define __MPP_BUFFER_H__
19 
20 #include "rk_type.h"
21 #include "mpp_err.h"
22 
23 /*
24  * MppBuffer module has several functions:
25  *
26  * 1. buffer get / put / reference management / external commit / get info.
27  *    this part is the basic user interface for MppBuffer.
28  *
29  *    function:
30  *
31  *    mpp_buffer_get
32  *    mpp_buffer_put
33  *    mpp_buffer_inc_ref
34  *    mpp_buffer_commit
35  *    mpp_buffer_info_get
36  *
37  * 2. user buffer working flow control abstraction.
38  *    buffer should attach to certain group, and buffer mode control the buffer usage flow.
39  *    this part is also a part of user interface.
40  *
41  *    function:
42  *
43  *    mpp_buffer_group_get
44  *    mpp_buffer_group_normal_get
45  *    mpp_buffer_group_limit_get
46  *    mpp_buffer_group_put
47  *    mpp_buffer_group_limit_config
48  *
49  * 3. buffer allocator management
50  *    this part is for allocator on different os, it does not have user interface
51  *    it will support normal buffer, Android ion buffer, Linux v4l2 vb2 buffer
52  *    user can only use MppBufferType to choose.
53  *
54  */
55 
56 /*
57  * mpp buffer group support two work flow mode:
58  *
59  * normal flow: all buffer are generated by MPP
60  *              under this mode, buffer pool is maintained internally
61  *
62  *              typical call flow:
63  *
64  *              mpp_buffer_group_get()          return A
65  *              mpp_buffer_get(A)               return a    ref +1 -> used
66  *              mpp_buffer_inc_ref(a)                       ref +1
67  *              mpp_buffer_put(a)                           ref -1
68  *              mpp_buffer_put(a)                           ref -1 -> unused
69  *              mpp_buffer_group_put(A)
70  *
71  * commit flow: all buffer are commited out of MPP
72  *              under this mode, buffers is commit by external api.
73  *              normally MPP only use it but not generate it.
74  *
75  *              typical call flow:
76  *
77  *              ==== external allocator ====
78  *              mpp_buffer_group_get()          return A
79  *              mpp_buffer_commit(A, x)
80  *              mpp_buffer_commit(A, y)
81  *
82  *              ======= internal user ======
83  *              mpp_buffer_get(A)               return a
84  *              mpp_buffer_get(A)               return b
85  *              mpp_buffer_put(a)
86  *              mpp_buffer_put(b)
87  *
88  *              ==== external allocator ====
89  *              mpp_buffer_group_put(A)
90  *
91  *              NOTE: commit interface required group handle to record group information
92  */
93 
94 /*
95  * mpp buffer group has two buffer limit mode: normal and limit
96  *
97  * normal mode: allows any buffer size and always general new buffer is no unused buffer
98  *              is available.
99  *              This mode normally use with normal flow and is used for table / stream buffer
100  *
101  * limit mode : restrict the buffer's size and count in the buffer group. if try to calloc
102  *              buffer with different size or extra count it will fail.
103  *              This mode normally use with commit flow and is used for frame buffer
104  */
105 
106 /*
107  * NOTE: normal mode is recommanded to work with normal flow, working with limit  mode is not.
108  *       limit  mode is recommanded to work with commit flow, working with normal mode is not.
109  */
110 typedef enum {
111     MPP_BUFFER_INTERNAL,
112     MPP_BUFFER_EXTERNAL,
113     MPP_BUFFER_MODE_BUTT,
114 } MppBufferMode;
115 
116 /*
117  * the mpp buffer has serval types:
118  *
119  * normal   : normal malloc buffer for unit test or hardware simulation
120  * ion      : use ion device under Android/Linux, MppBuffer will encapsulte ion file handle
121  * ext_dma  : the DMABUF(DMA buffers) come from the application
122  * drm      : use the drm device interface for memory management
123  */
124 typedef enum {
125     MPP_BUFFER_TYPE_NORMAL,
126     MPP_BUFFER_TYPE_ION,
127     MPP_BUFFER_TYPE_EXT_DMA,
128     MPP_BUFFER_TYPE_DRM,
129     MPP_BUFFER_TYPE_DMA_HEAP,
130     MPP_BUFFER_TYPE_BUTT,
131 } MppBufferType;
132 
133 #define MPP_BUFFER_TYPE_MASK            0x0000FFFF
134 
135 /*
136  * MPP_BUFFER_FLAGS cooperate with MppBufferType
137  * 16 high bits of MppBufferType are used in flags
138  *
139  * eg:
140  * DRM CMA buffer : MPP_BUFFER_TYPE_DRM | MPP_BUFFER_FLAGS_CONTIG
141  *                  = 0x00010003
142  * DRM SECURE buffer: MPP_BUFFER_TYPE_DRM | MPP_BUFFER_FLAGS_SECURE
143  *                  = 0x00080003
144  *
145  * The dma buffer source can also be set by format: flags | type.
146  * dma buffer source flags:
147  * MPP_BUFFER_FLAGS_CONTIG means cma
148  * MPP_BUFFER_FLAGS_CACHABLE means cachable
149  * MPP_BUFFER_FLAGS_DMA32 means dma32
150  *
151  * flags originate from drm_rockchip_gem_mem_type
152  */
153 #define MPP_BUFFER_FLAGS_MASK           0x003f0000      //ROCKCHIP_BO_MASK << 16
154 #define MPP_BUFFER_FLAGS_CONTIG         0x00010000      //ROCKCHIP_BO_CONTIG << 16
155 #define MPP_BUFFER_FLAGS_CACHABLE       0x00020000      //ROCKCHIP_BO_CACHABLE << 16
156 #define MPP_BUFFER_FLAGS_WC             0x00040000      //ROCKCHIP_BO_WC << 16
157 #define MPP_BUFFER_FLAGS_SECURE         0x00080000      //ROCKCHIP_BO_SECURE << 16
158 #define MPP_BUFFER_FLAGS_ALLOC_KMAP     0x00100000      //ROCKCHIP_BO_ALLOC_KMAP << 16
159 #define MPP_BUFFER_FLAGS_DMA32          0x00200000      //ROCKCHIP_BO_DMA32 << 16
160 
161 /*
162  * MppBufferInfo variable's meaning is different in different MppBufferType
163  *
164  * Common
165  * index - the buffer index used to track buffer in buffer pool
166  * size  - the buffer size
167  *
168  * MPP_BUFFER_TYPE_NORMAL
169  *
170  * ptr  - virtual address of normal malloced buffer
171  * fd   - unused and set to -1, the allocator would return its
172  *         internal buffer counter number
173  *
174  * MPP_BUFFER_TYPE_ION
175  *
176  * ptr  - virtual address of ion buffer in user space
177  * hnd  - ion handle in user space
178  * fd   - ion buffer file handle for map / unmap
179  *
180  */
181 typedef struct MppBufferInfo_t {
182     MppBufferType   type;
183     size_t          size;
184     void            *ptr;
185     void            *hnd;
186     int             fd;
187     int             index;
188 } MppBufferInfo;
189 
190 #define BUFFER_GROUP_SIZE_DEFAULT           (SZ_1M*80)
191 
192 /*
193  * mpp_buffer_import_with_tag(MppBufferGroup group, MppBufferInfo *info, MppBuffer *buffer)
194  *
195  * 1. group - specified the MppBuffer to be attached to.
196  *    group can be NULL then this buffer will attached to default legecy group
197  *    Default to NULL on mpp_buffer_import case
198  *
199  * 2. info  - input information for the output MppBuffer
200  *    info can NOT be NULL. It must contain at least one of ptr/fd.
201  *
202  * 3. buffer - generated MppBuffer from MppBufferInfo.
203  *    buffer can be NULL then the buffer is commit to group with unused status.
204  *    Otherwise generated buffer will be directly got and ref_count increased.
205  *    Default to NULL on mpp_buffer_commit case
206  *
207  * mpp_buffer_commit usage:
208  *
209  * Add a external buffer info to group. This buffer will be on unused status.
210  * Typical usage is on Android. MediaPlayer gralloc Graphic buffer then commit these buffer
211  * to decoder's buffer group. Then decoder will recycle these buffer and return buffer reference
212  * to MediaPlayer for display.
213  *
214  * mpp_buffer_import usage:
215  *
216  * Transfer a external buffer info to MppBuffer but it is not expected to attached to certain
217  * buffer group. So the group is set to NULL. Then this buffer can be used for MppFrame/MppPacket.
218  * Typical usage is for image processing. Image processing normally will be a oneshot operation
219  * It does not need complicated group management. But in other hand mpp still need to know the
220  * imported buffer is leak or not and trace its usage inside mpp process. So we attach this kind
221  * of buffer to default misc buffer group for management.
222  */
223 #define mpp_buffer_commit(group, info) \
224         mpp_buffer_import_with_tag(group, info, NULL, MODULE_TAG, __FUNCTION__)
225 
226 #define mpp_buffer_import(buffer, info) \
227         mpp_buffer_import_with_tag(NULL, info, buffer, MODULE_TAG, __FUNCTION__)
228 
229 #define mpp_buffer_get(group, buffer, size) \
230         mpp_buffer_get_with_tag(group, buffer, size, MODULE_TAG, __FUNCTION__)
231 
232 #define mpp_buffer_put(buffer) \
233         mpp_buffer_put_with_caller(buffer, __FUNCTION__)
234 
235 #define mpp_buffer_inc_ref(buffer) \
236         mpp_buffer_inc_ref_with_caller(buffer, __FUNCTION__)
237 
238 #define mpp_buffer_info_get(buffer, info) \
239         mpp_buffer_info_get_with_caller(buffer, info, __FUNCTION__)
240 
241 #define mpp_buffer_read(buffer, offset, data, size) \
242         mpp_buffer_read_with_caller(buffer, offset, data, size, __FUNCTION__)
243 
244 #define mpp_buffer_write(buffer, offset, data, size) \
245         mpp_buffer_write_with_caller(buffer, offset, data, size, __FUNCTION__)
246 
247 #define mpp_buffer_get_ptr(buffer) \
248         mpp_buffer_get_ptr_with_caller(buffer, __FUNCTION__)
249 
250 #define mpp_buffer_get_fd(buffer) \
251         mpp_buffer_get_fd_with_caller(buffer, __FUNCTION__)
252 
253 #define mpp_buffer_get_size(buffer) \
254         mpp_buffer_get_size_with_caller(buffer, __FUNCTION__)
255 
256 #define mpp_buffer_get_index(buffer) \
257         mpp_buffer_get_index_with_caller(buffer, __FUNCTION__)
258 
259 #define mpp_buffer_set_index(buffer, index) \
260         mpp_buffer_set_index_with_caller(buffer, index, __FUNCTION__)
261 
262 #define mpp_buffer_get_offset(buffer) \
263         mpp_buffer_get_offset_with_caller(buffer, __FUNCTION__)
264 
265 #define mpp_buffer_set_offset(buffer, offset) \
266         mpp_buffer_set_offset_with_caller(buffer, offset, __FUNCTION__)
267 
268 #define mpp_buffer_group_get_internal(group, type, ...) \
269         mpp_buffer_group_get(group, type, MPP_BUFFER_INTERNAL, MODULE_TAG, __FUNCTION__)
270 
271 #define mpp_buffer_group_get_external(group, type, ...) \
272         mpp_buffer_group_get(group, type, MPP_BUFFER_EXTERNAL, MODULE_TAG, __FUNCTION__)
273 
274 #ifdef __cplusplus
275 extern "C" {
276 #endif
277 
278 /*
279  * MppBuffer interface
280  * these interface will change value of group and buffer so before calling functions
281  * parameter need to be checked.
282  *
283  * IMPORTANT:
284  * mpp_buffer_import_with_tag - compounded interface for commit and import
285  *
286  */
287 MPP_RET mpp_buffer_import_with_tag(MppBufferGroup group, MppBufferInfo *info, MppBuffer *buffer,
288                                    const char *tag, const char *caller);
289 MPP_RET mpp_buffer_get_with_tag(MppBufferGroup group, MppBuffer *buffer, size_t size,
290                                 const char *tag, const char *caller);
291 MPP_RET mpp_buffer_put_with_caller(MppBuffer buffer, const char *caller);
292 MPP_RET mpp_buffer_inc_ref_with_caller(MppBuffer buffer, const char *caller);
293 
294 MPP_RET mpp_buffer_info_get_with_caller(MppBuffer buffer, MppBufferInfo *info, const char *caller);
295 MPP_RET mpp_buffer_read_with_caller(MppBuffer buffer, size_t offset, void *data, size_t size, const char *caller);
296 MPP_RET mpp_buffer_write_with_caller(MppBuffer buffer, size_t offset, void *data, size_t size, const char *caller);
297 void   *mpp_buffer_get_ptr_with_caller(MppBuffer buffer, const char *caller);
298 int     mpp_buffer_get_fd_with_caller(MppBuffer buffer, const char *caller);
299 size_t  mpp_buffer_get_size_with_caller(MppBuffer buffer, const char *caller);
300 int     mpp_buffer_get_index_with_caller(MppBuffer buffer, const char *caller);
301 MPP_RET mpp_buffer_set_index_with_caller(MppBuffer buffer, int index, const char *caller);
302 size_t  mpp_buffer_get_offset_with_caller(MppBuffer buffer, const char *caller);
303 MPP_RET mpp_buffer_set_offset_with_caller(MppBuffer buffer, size_t offset, const char *caller);
304 
305 MPP_RET mpp_buffer_group_get(MppBufferGroup *group, MppBufferType type, MppBufferMode mode,
306                              const char *tag, const char *caller);
307 MPP_RET mpp_buffer_group_put(MppBufferGroup group);
308 MPP_RET mpp_buffer_group_clear(MppBufferGroup group);
309 RK_S32  mpp_buffer_group_unused(MppBufferGroup group);
310 size_t  mpp_buffer_group_usage(MppBufferGroup group);
311 MppBufferMode mpp_buffer_group_mode(MppBufferGroup group);
312 MppBufferType mpp_buffer_group_type(MppBufferGroup group);
313 
314 /*
315  * size  : 0 - no limit, other - max buffer size
316  * count : 0 - no limit, other - max buffer count
317  */
318 MPP_RET mpp_buffer_group_limit_config(MppBufferGroup group, size_t size, RK_S32 count);
319 
320 RK_U32 mpp_buffer_total_now();
321 RK_U32 mpp_buffer_total_max();
322 
323 #ifdef __cplusplus
324 }
325 #endif
326 
327 #endif /*__MPP_BUFFER_H__*/
328