forked from projectceladon/minigbm
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathi915_private.c
233 lines (194 loc) · 6.65 KB
/
i915_private.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
/*
* Copyright 2017 The Chromium OS Authors. All rights reserved.
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#ifdef DRV_I915
#include <assert.h>
#include <errno.h>
#include <i915_drm.h>
#include <stdio.h>
#include <string.h>
#include <sys/mman.h>
#include <drm_fourcc.h>
#include <xf86drm.h>
#include "drv_priv.h"
#include "helpers.h"
#include "util.h"
#include "i915_private.h"
static const uint32_t private_linear_source_formats[] = { DRM_FORMAT_R16, DRM_FORMAT_NV16,
DRM_FORMAT_YUV420, DRM_FORMAT_YUV422,
DRM_FORMAT_YUV444, DRM_FORMAT_NV21,
DRM_FORMAT_P010 };
static const uint32_t private_rgb24_formats[] = { DRM_FORMAT_RGB888, DRM_FORMAT_BGR888 };
static const uint32_t private_source_formats[] = { DRM_FORMAT_P010, DRM_FORMAT_NV12_Y_TILED_INTEL };
#if !defined(DRM_CAP_CURSOR_WIDTH)
#define DRM_CAP_CURSOR_WIDTH 0x8
#endif
#if !defined(DRM_CAP_CURSOR_HEIGHT)
#define DRM_CAP_CURSOR_HEIGHT 0x9
#endif
static const uint32_t kDefaultCursorWidth = 64;
static const uint32_t kDefaultCursorHeight = 64;
#define BO_USE_CAMERA_MASK BO_USE_CAMERA_READ | BO_USE_SCANOUT | BO_USE_CAMERA_WRITE
static void get_preferred_cursor_attributes(uint32_t drm_fd, uint64_t *cursor_width,
uint64_t *cursor_height)
{
uint64_t width = 0, height = 0;
if (drmGetCap(drm_fd, DRM_CAP_CURSOR_WIDTH, &width)) {
fprintf(stderr, "cannot get cursor width. \n");
} else if (drmGetCap(drm_fd, DRM_CAP_CURSOR_HEIGHT, &height)) {
fprintf(stderr, "cannot get cursor height. \n");
}
if (!width)
width = kDefaultCursorWidth;
*cursor_width = width;
if (!height)
height = kDefaultCursorHeight;
*cursor_height = height;
}
int i915_private_init(struct driver *drv, uint64_t *cursor_width, uint64_t *cursor_height)
{
get_preferred_cursor_attributes(drv->fd, cursor_width, cursor_height);
return 0;
}
int i915_private_add_combinations(struct driver *drv)
{
struct format_metadata metadata;
uint64_t render_flags, texture_flags;
render_flags = BO_USE_RENDER_MASK;
texture_flags = BO_USE_TEXTURE_MASK;
metadata.tiling = I915_TILING_NONE;
metadata.priority = 1;
metadata.modifier = DRM_FORMAT_MOD_NONE;
drv_modify_combination(drv, DRM_FORMAT_ABGR8888, &metadata, BO_USE_CURSOR | BO_USE_SCANOUT);
drv_modify_combination(drv, DRM_FORMAT_NV12, &metadata,
BO_USE_TEXTURE | BO_USE_CAMERA_MASK);
drv_modify_combination(drv, DRM_FORMAT_YUYV, &metadata,
BO_USE_TEXTURE | BO_USE_CAMERA_MASK | BO_USE_RENDERING);
drv_modify_combination(drv, DRM_FORMAT_VYUY, &metadata,
BO_USE_TEXTURE | BO_USE_CAMERA_MASK | BO_USE_RENDERING);
drv_modify_combination(drv, DRM_FORMAT_UYVY, &metadata,
BO_USE_TEXTURE | BO_USE_CAMERA_MASK | BO_USE_RENDERING);
drv_modify_combination(drv, DRM_FORMAT_YVYU, &metadata,
BO_USE_TEXTURE | BO_USE_CAMERA_MASK | BO_USE_RENDERING);
drv_modify_combination(drv, DRM_FORMAT_YVU420_ANDROID, &metadata,
BO_USE_TEXTURE | BO_USE_CAMERA_MASK);
drv_modify_combination(drv, DRM_FORMAT_RGB565, &metadata, BO_USE_CAMERA_MASK);
/* Media/Camera expect these formats support. */
metadata.tiling = I915_TILING_NONE;
metadata.priority = 1;
metadata.modifier = DRM_FORMAT_MOD_NONE;
drv_add_combinations(drv, private_linear_source_formats,
ARRAY_SIZE(private_linear_source_formats), &metadata,
texture_flags | BO_USE_CAMERA_MASK);
metadata.tiling = I915_TILING_NONE;
metadata.priority = 1;
metadata.modifier = DRM_FORMAT_MOD_NONE;
drv_add_combination(drv, DRM_FORMAT_ABGR2101010, &metadata,
BO_USE_SCANOUT | BO_USE_TEXTURE |
BO_USE_RENDERING | BO_USE_SW_MASK);
drv_add_combination(drv, DRM_FORMAT_ABGR16161616F, &metadata,
BO_USE_SCANOUT | BO_USE_TEXTURE | BO_USE_RENDERING);
metadata.tiling = I915_TILING_Y;
metadata.priority = 3;
metadata.modifier = I915_FORMAT_MOD_Y_TILED;
drv_add_combinations(drv, private_source_formats, ARRAY_SIZE(private_source_formats),
&metadata, texture_flags | BO_USE_CAMERA_MASK);
/* Android CTS tests require this. */
drv_add_combinations(drv, private_rgb24_formats, ARRAY_SIZE(private_rgb24_formats),
&metadata, BO_USE_SW_MASK);
texture_flags &= ~BO_USE_RENDERSCRIPT;
texture_flags &= ~BO_USE_SW_WRITE_OFTEN;
texture_flags &= ~BO_USE_SW_READ_OFTEN;
texture_flags &= ~BO_USE_LINEAR;
metadata.tiling = I915_TILING_X;
metadata.priority = 2;
metadata.modifier = I915_FORMAT_MOD_X_TILED;
int ret = drv_add_combinations(drv, private_linear_source_formats,
ARRAY_SIZE(private_linear_source_formats), &metadata,
texture_flags | BO_USE_CAMERA_MASK);
if (ret)
return ret;
return 0;
}
void i915_private_align_dimensions(uint32_t format, uint32_t *vertical_alignment)
{
switch (format) {
case DRM_FORMAT_NV12_Y_TILED_INTEL:
*vertical_alignment = 64;
break;
}
}
uint32_t i915_private_bpp_from_format(uint32_t format, size_t plane)
{
assert(plane < drv_num_planes_from_format(format));
switch (format) {
case DRM_FORMAT_NV12_Y_TILED_INTEL:
return (plane == 0) ? 8 : 4;
case DRM_FORMAT_P010:
return (plane == 0) ? 16 : 8;
case DRM_FORMAT_YUV420:
case DRM_FORMAT_YUV422:
case DRM_FORMAT_YUV444:
case DRM_FORMAT_NV16:
return 8;
case DRM_FORMAT_R16:
return 16;
case DRM_FORMAT_ABGR2101010:
return 32;
case DRM_FORMAT_ABGR16161616F:
return 64;
}
fprintf(stderr, "drv: UNKNOWN FORMAT %d\n", format);
return 0;
}
void i915_private_vertical_subsampling_from_format(uint32_t *vertical_subsampling, uint32_t format,
size_t plane)
{
switch (format) {
case DRM_FORMAT_NV12_Y_TILED_INTEL:
case DRM_FORMAT_YUV420:
case DRM_FORMAT_P010:
*vertical_subsampling = (plane == 0) ? 1 : 2;
break;
default:
*vertical_subsampling = 1;
}
}
size_t i915_private_num_planes_from_format(uint32_t format)
{
switch (format) {
case DRM_FORMAT_R16:
case DRM_FORMAT_ABGR2101010:
case DRM_FORMAT_ABGR16161616F:
return 1;
case DRM_FORMAT_NV12_Y_TILED_INTEL:
case DRM_FORMAT_NV16:
case DRM_FORMAT_P010:
return 2;
case DRM_FORMAT_YUV420:
case DRM_FORMAT_YUV422:
case DRM_FORMAT_YUV444:
return 3;
}
fprintf(stderr, "drv: UNKNOWN FORMAT %d\n", format);
return 0;
}
uint32_t i915_private_resolve_format(uint32_t format, uint64_t usage, uint32_t *resolved_format)
{
switch (format) {
case DRM_FORMAT_FLEX_IMPLEMENTATION_DEFINED:
/* KBL camera subsystem requires NV12. */
if (usage & (BO_USE_CAMERA_READ | BO_USE_CAMERA_WRITE)) {
*resolved_format = DRM_FORMAT_NV12;
return 1;
}
if (usage & BO_USE_TEXTURE) {
*resolved_format = DRM_FORMAT_ABGR8888;
return 1;
}
}
return 0;
}
#endif