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#include <linux/module.h>
38#include <linux/types.h>
39#include <linux/kernel.h>
40#include <linux/mm.h>
41#include <linux/string.h>
42#include <linux/errno.h>
43#include <linux/uaccess.h>
44#include <asm/pgtable.h>
45#include <asm/io.h>
46#include <asm/div64.h>
47#include <media/v4l2-common.h>
48#include <media/v4l2-device.h>
49#include <media/v4l2-ctrls.h>
50
51#include <linux/videodev2.h>
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66int v4l2_ctrl_query_fill(struct v4l2_queryctrl *qctrl, s32 _min, s32 _max, s32 _step, s32 _def)
67{
68 const char *name;
69 s64 min = _min;
70 s64 max = _max;
71 u64 step = _step;
72 s64 def = _def;
73
74 v4l2_ctrl_fill(qctrl->id, &name, &qctrl->type,
75 &min, &max, &step, &def, &qctrl->flags);
76
77 if (name == NULL)
78 return -EINVAL;
79
80 qctrl->minimum = min;
81 qctrl->maximum = max;
82 qctrl->step = step;
83 qctrl->default_value = def;
84 qctrl->reserved[0] = qctrl->reserved[1] = 0;
85 strscpy(qctrl->name, name, sizeof(qctrl->name));
86 return 0;
87}
88EXPORT_SYMBOL(v4l2_ctrl_query_fill);
89
90
91
92
93
94static unsigned int clamp_align(unsigned int x, unsigned int min,
95 unsigned int max, unsigned int align)
96{
97
98 unsigned int mask = ~((1 << align) - 1);
99
100
101 x = clamp(x, (min + ~mask) & mask, max & mask);
102
103
104 if (align)
105 x = (x + (1 << (align - 1))) & mask;
106
107 return x;
108}
109
110static unsigned int clamp_roundup(unsigned int x, unsigned int min,
111 unsigned int max, unsigned int alignment)
112{
113 x = clamp(x, min, max);
114 if (alignment)
115 x = round_up(x, alignment);
116
117 return x;
118}
119
120void v4l_bound_align_image(u32 *w, unsigned int wmin, unsigned int wmax,
121 unsigned int walign,
122 u32 *h, unsigned int hmin, unsigned int hmax,
123 unsigned int halign, unsigned int salign)
124{
125 *w = clamp_align(*w, wmin, wmax, walign);
126 *h = clamp_align(*h, hmin, hmax, halign);
127
128
129 if (!salign)
130 return;
131
132
133 walign = __ffs(*w);
134 halign = __ffs(*h);
135
136 if (walign + halign < salign) {
137
138 unsigned int wmaxa = __fls(wmax ^ (wmin - 1));
139
140 unsigned int hmaxa = __fls(hmax ^ (hmin - 1));
141
142
143 do {
144 if (halign >= hmaxa ||
145 (walign <= halign && walign < wmaxa)) {
146 *w = clamp_align(*w, wmin, wmax, walign + 1);
147 walign = __ffs(*w);
148 } else {
149 *h = clamp_align(*h, hmin, hmax, halign + 1);
150 halign = __ffs(*h);
151 }
152 } while (halign + walign < salign);
153 }
154}
155EXPORT_SYMBOL_GPL(v4l_bound_align_image);
156
157const void *
158__v4l2_find_nearest_size(const void *array, size_t array_size,
159 size_t entry_size, size_t width_offset,
160 size_t height_offset, s32 width, s32 height)
161{
162 u32 error, min_error = U32_MAX;
163 const void *best = NULL;
164 unsigned int i;
165
166 if (!array)
167 return NULL;
168
169 for (i = 0; i < array_size; i++, array += entry_size) {
170 const u32 *entry_width = array + width_offset;
171 const u32 *entry_height = array + height_offset;
172
173 error = abs(*entry_width - width) + abs(*entry_height - height);
174 if (error > min_error)
175 continue;
176
177 min_error = error;
178 best = array;
179 if (!error)
180 break;
181 }
182
183 return best;
184}
185EXPORT_SYMBOL_GPL(__v4l2_find_nearest_size);
186
187int v4l2_g_parm_cap(struct video_device *vdev,
188 struct v4l2_subdev *sd, struct v4l2_streamparm *a)
189{
190 struct v4l2_subdev_frame_interval ival = { 0 };
191 int ret;
192
193 if (a->type != V4L2_BUF_TYPE_VIDEO_CAPTURE &&
194 a->type != V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE)
195 return -EINVAL;
196
197 if (vdev->device_caps & V4L2_CAP_READWRITE)
198 a->parm.capture.readbuffers = 2;
199 if (v4l2_subdev_has_op(sd, video, g_frame_interval))
200 a->parm.capture.capability = V4L2_CAP_TIMEPERFRAME;
201 ret = v4l2_subdev_call(sd, video, g_frame_interval, &ival);
202 if (!ret)
203 a->parm.capture.timeperframe = ival.interval;
204 return ret;
205}
206EXPORT_SYMBOL_GPL(v4l2_g_parm_cap);
207
208int v4l2_s_parm_cap(struct video_device *vdev,
209 struct v4l2_subdev *sd, struct v4l2_streamparm *a)
210{
211 struct v4l2_subdev_frame_interval ival = {
212 .interval = a->parm.capture.timeperframe
213 };
214 int ret;
215
216 if (a->type != V4L2_BUF_TYPE_VIDEO_CAPTURE &&
217 a->type != V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE)
218 return -EINVAL;
219
220 memset(&a->parm, 0, sizeof(a->parm));
221 if (vdev->device_caps & V4L2_CAP_READWRITE)
222 a->parm.capture.readbuffers = 2;
223 else
224 a->parm.capture.readbuffers = 0;
225
226 if (v4l2_subdev_has_op(sd, video, g_frame_interval))
227 a->parm.capture.capability = V4L2_CAP_TIMEPERFRAME;
228 ret = v4l2_subdev_call(sd, video, s_frame_interval, &ival);
229 if (!ret)
230 a->parm.capture.timeperframe = ival.interval;
231 return ret;
232}
233EXPORT_SYMBOL_GPL(v4l2_s_parm_cap);
234
235const struct v4l2_format_info *v4l2_format_info(u32 format)
236{
237 static const struct v4l2_format_info formats[] = {
238
239 { .format = V4L2_PIX_FMT_BGR24, .mem_planes = 1, .comp_planes = 1, .bpp = { 3, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
240 { .format = V4L2_PIX_FMT_RGB24, .mem_planes = 1, .comp_planes = 1, .bpp = { 3, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
241 { .format = V4L2_PIX_FMT_HSV24, .mem_planes = 1, .comp_planes = 1, .bpp = { 3, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
242 { .format = V4L2_PIX_FMT_BGR32, .mem_planes = 1, .comp_planes = 1, .bpp = { 4, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
243 { .format = V4L2_PIX_FMT_XBGR32, .mem_planes = 1, .comp_planes = 1, .bpp = { 4, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
244 { .format = V4L2_PIX_FMT_BGRX32, .mem_planes = 1, .comp_planes = 1, .bpp = { 4, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
245 { .format = V4L2_PIX_FMT_RGB32, .mem_planes = 1, .comp_planes = 1, .bpp = { 4, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
246 { .format = V4L2_PIX_FMT_XRGB32, .mem_planes = 1, .comp_planes = 1, .bpp = { 4, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
247 { .format = V4L2_PIX_FMT_RGBX32, .mem_planes = 1, .comp_planes = 1, .bpp = { 4, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
248 { .format = V4L2_PIX_FMT_HSV32, .mem_planes = 1, .comp_planes = 1, .bpp = { 4, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
249 { .format = V4L2_PIX_FMT_ARGB32, .mem_planes = 1, .comp_planes = 1, .bpp = { 4, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
250 { .format = V4L2_PIX_FMT_RGBA32, .mem_planes = 1, .comp_planes = 1, .bpp = { 4, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
251 { .format = V4L2_PIX_FMT_ABGR32, .mem_planes = 1, .comp_planes = 1, .bpp = { 4, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
252 { .format = V4L2_PIX_FMT_BGRA32, .mem_planes = 1, .comp_planes = 1, .bpp = { 4, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
253 { .format = V4L2_PIX_FMT_GREY, .mem_planes = 1, .comp_planes = 1, .bpp = { 1, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
254
255
256 { .format = V4L2_PIX_FMT_YUYV, .mem_planes = 1, .comp_planes = 1, .bpp = { 2, 0, 0, 0 }, .hdiv = 2, .vdiv = 1 },
257 { .format = V4L2_PIX_FMT_YVYU, .mem_planes = 1, .comp_planes = 1, .bpp = { 2, 0, 0, 0 }, .hdiv = 2, .vdiv = 1 },
258 { .format = V4L2_PIX_FMT_UYVY, .mem_planes = 1, .comp_planes = 1, .bpp = { 2, 0, 0, 0 }, .hdiv = 2, .vdiv = 1 },
259 { .format = V4L2_PIX_FMT_VYUY, .mem_planes = 1, .comp_planes = 1, .bpp = { 2, 0, 0, 0 }, .hdiv = 2, .vdiv = 1 },
260
261
262 { .format = V4L2_PIX_FMT_NV12, .mem_planes = 1, .comp_planes = 2, .bpp = { 1, 2, 0, 0 }, .hdiv = 2, .vdiv = 2 },
263 { .format = V4L2_PIX_FMT_NV21, .mem_planes = 1, .comp_planes = 2, .bpp = { 1, 2, 0, 0 }, .hdiv = 2, .vdiv = 2 },
264 { .format = V4L2_PIX_FMT_NV16, .mem_planes = 1, .comp_planes = 2, .bpp = { 1, 2, 0, 0 }, .hdiv = 2, .vdiv = 1 },
265 { .format = V4L2_PIX_FMT_NV61, .mem_planes = 1, .comp_planes = 2, .bpp = { 1, 2, 0, 0 }, .hdiv = 2, .vdiv = 1 },
266 { .format = V4L2_PIX_FMT_NV24, .mem_planes = 1, .comp_planes = 2, .bpp = { 1, 2, 0, 0 }, .hdiv = 1, .vdiv = 1 },
267 { .format = V4L2_PIX_FMT_NV42, .mem_planes = 1, .comp_planes = 2, .bpp = { 1, 2, 0, 0 }, .hdiv = 1, .vdiv = 1 },
268
269 { .format = V4L2_PIX_FMT_YUV410, .mem_planes = 1, .comp_planes = 3, .bpp = { 1, 1, 1, 0 }, .hdiv = 4, .vdiv = 4 },
270 { .format = V4L2_PIX_FMT_YVU410, .mem_planes = 1, .comp_planes = 3, .bpp = { 1, 1, 1, 0 }, .hdiv = 4, .vdiv = 4 },
271 { .format = V4L2_PIX_FMT_YUV411P, .mem_planes = 1, .comp_planes = 3, .bpp = { 1, 1, 1, 0 }, .hdiv = 4, .vdiv = 1 },
272 { .format = V4L2_PIX_FMT_YUV420, .mem_planes = 1, .comp_planes = 3, .bpp = { 1, 1, 1, 0 }, .hdiv = 2, .vdiv = 2 },
273 { .format = V4L2_PIX_FMT_YVU420, .mem_planes = 1, .comp_planes = 3, .bpp = { 1, 1, 1, 0 }, .hdiv = 2, .vdiv = 2 },
274 { .format = V4L2_PIX_FMT_YUV422P, .mem_planes = 1, .comp_planes = 3, .bpp = { 1, 1, 1, 0 }, .hdiv = 2, .vdiv = 1 },
275
276
277 { .format = V4L2_PIX_FMT_YUV420M, .mem_planes = 3, .comp_planes = 3, .bpp = { 1, 1, 1, 0 }, .hdiv = 2, .vdiv = 2 },
278 { .format = V4L2_PIX_FMT_YVU420M, .mem_planes = 3, .comp_planes = 3, .bpp = { 1, 1, 1, 0 }, .hdiv = 2, .vdiv = 2 },
279 { .format = V4L2_PIX_FMT_YUV422M, .mem_planes = 3, .comp_planes = 3, .bpp = { 1, 1, 1, 0 }, .hdiv = 2, .vdiv = 1 },
280 { .format = V4L2_PIX_FMT_YVU422M, .mem_planes = 3, .comp_planes = 3, .bpp = { 1, 1, 1, 0 }, .hdiv = 2, .vdiv = 1 },
281 { .format = V4L2_PIX_FMT_YUV444M, .mem_planes = 3, .comp_planes = 3, .bpp = { 1, 1, 1, 0 }, .hdiv = 1, .vdiv = 1 },
282 { .format = V4L2_PIX_FMT_YVU444M, .mem_planes = 3, .comp_planes = 3, .bpp = { 1, 1, 1, 0 }, .hdiv = 1, .vdiv = 1 },
283
284 { .format = V4L2_PIX_FMT_NV12M, .mem_planes = 2, .comp_planes = 2, .bpp = { 1, 2, 0, 0 }, .hdiv = 2, .vdiv = 2 },
285 { .format = V4L2_PIX_FMT_NV21M, .mem_planes = 2, .comp_planes = 2, .bpp = { 1, 2, 0, 0 }, .hdiv = 2, .vdiv = 2 },
286 { .format = V4L2_PIX_FMT_NV16M, .mem_planes = 2, .comp_planes = 2, .bpp = { 1, 2, 0, 0 }, .hdiv = 2, .vdiv = 1 },
287 { .format = V4L2_PIX_FMT_NV61M, .mem_planes = 2, .comp_planes = 2, .bpp = { 1, 2, 0, 0 }, .hdiv = 2, .vdiv = 1 },
288
289
290 { .format = V4L2_PIX_FMT_SBGGR8, .mem_planes = 1, .comp_planes = 1, .bpp = { 1, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
291 { .format = V4L2_PIX_FMT_SGBRG8, .mem_planes = 1, .comp_planes = 1, .bpp = { 1, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
292 { .format = V4L2_PIX_FMT_SGRBG8, .mem_planes = 1, .comp_planes = 1, .bpp = { 1, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
293 { .format = V4L2_PIX_FMT_SRGGB8, .mem_planes = 1, .comp_planes = 1, .bpp = { 1, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
294 { .format = V4L2_PIX_FMT_SBGGR10, .mem_planes = 1, .comp_planes = 1, .bpp = { 2, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
295 { .format = V4L2_PIX_FMT_SGBRG10, .mem_planes = 1, .comp_planes = 1, .bpp = { 2, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
296 { .format = V4L2_PIX_FMT_SGRBG10, .mem_planes = 1, .comp_planes = 1, .bpp = { 2, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
297 { .format = V4L2_PIX_FMT_SRGGB10, .mem_planes = 1, .comp_planes = 1, .bpp = { 2, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
298 { .format = V4L2_PIX_FMT_SBGGR10ALAW8, .mem_planes = 1, .comp_planes = 1, .bpp = { 1, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
299 { .format = V4L2_PIX_FMT_SGBRG10ALAW8, .mem_planes = 1, .comp_planes = 1, .bpp = { 1, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
300 { .format = V4L2_PIX_FMT_SGRBG10ALAW8, .mem_planes = 1, .comp_planes = 1, .bpp = { 1, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
301 { .format = V4L2_PIX_FMT_SRGGB10ALAW8, .mem_planes = 1, .comp_planes = 1, .bpp = { 1, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
302 { .format = V4L2_PIX_FMT_SBGGR10DPCM8, .mem_planes = 1, .comp_planes = 1, .bpp = { 1, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
303 { .format = V4L2_PIX_FMT_SGBRG10DPCM8, .mem_planes = 1, .comp_planes = 1, .bpp = { 1, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
304 { .format = V4L2_PIX_FMT_SGRBG10DPCM8, .mem_planes = 1, .comp_planes = 1, .bpp = { 1, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
305 { .format = V4L2_PIX_FMT_SRGGB10DPCM8, .mem_planes = 1, .comp_planes = 1, .bpp = { 1, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
306 { .format = V4L2_PIX_FMT_SBGGR12, .mem_planes = 1, .comp_planes = 1, .bpp = { 2, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
307 { .format = V4L2_PIX_FMT_SGBRG12, .mem_planes = 1, .comp_planes = 1, .bpp = { 2, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
308 { .format = V4L2_PIX_FMT_SGRBG12, .mem_planes = 1, .comp_planes = 1, .bpp = { 2, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
309 { .format = V4L2_PIX_FMT_SRGGB12, .mem_planes = 1, .comp_planes = 1, .bpp = { 2, 0, 0, 0 }, .hdiv = 1, .vdiv = 1 },
310 };
311 unsigned int i;
312
313 for (i = 0; i < ARRAY_SIZE(formats); ++i)
314 if (formats[i].format == format)
315 return &formats[i];
316 return NULL;
317}
318EXPORT_SYMBOL(v4l2_format_info);
319
320static inline unsigned int v4l2_format_block_width(const struct v4l2_format_info *info, int plane)
321{
322 if (!info->block_w[plane])
323 return 1;
324 return info->block_w[plane];
325}
326
327static inline unsigned int v4l2_format_block_height(const struct v4l2_format_info *info, int plane)
328{
329 if (!info->block_h[plane])
330 return 1;
331 return info->block_h[plane];
332}
333
334void v4l2_apply_frmsize_constraints(u32 *width, u32 *height,
335 const struct v4l2_frmsize_stepwise *frmsize)
336{
337 if (!frmsize)
338 return;
339
340
341
342
343
344 *width = clamp_roundup(*width, frmsize->min_width, frmsize->max_width,
345 frmsize->step_width);
346 *height = clamp_roundup(*height, frmsize->min_height, frmsize->max_height,
347 frmsize->step_height);
348}
349EXPORT_SYMBOL_GPL(v4l2_apply_frmsize_constraints);
350
351int v4l2_fill_pixfmt_mp(struct v4l2_pix_format_mplane *pixfmt,
352 u32 pixelformat, u32 width, u32 height)
353{
354 const struct v4l2_format_info *info;
355 struct v4l2_plane_pix_format *plane;
356 int i;
357
358 info = v4l2_format_info(pixelformat);
359 if (!info)
360 return -EINVAL;
361
362 pixfmt->width = width;
363 pixfmt->height = height;
364 pixfmt->pixelformat = pixelformat;
365 pixfmt->num_planes = info->mem_planes;
366
367 if (info->mem_planes == 1) {
368 plane = &pixfmt->plane_fmt[0];
369 plane->bytesperline = ALIGN(width, v4l2_format_block_width(info, 0)) * info->bpp[0];
370 plane->sizeimage = 0;
371
372 for (i = 0; i < info->comp_planes; i++) {
373 unsigned int hdiv = (i == 0) ? 1 : info->hdiv;
374 unsigned int vdiv = (i == 0) ? 1 : info->vdiv;
375 unsigned int aligned_width;
376 unsigned int aligned_height;
377
378 aligned_width = ALIGN(width, v4l2_format_block_width(info, i));
379 aligned_height = ALIGN(height, v4l2_format_block_height(info, i));
380
381 plane->sizeimage += info->bpp[i] *
382 DIV_ROUND_UP(aligned_width, hdiv) *
383 DIV_ROUND_UP(aligned_height, vdiv);
384 }
385 } else {
386 for (i = 0; i < info->comp_planes; i++) {
387 unsigned int hdiv = (i == 0) ? 1 : info->hdiv;
388 unsigned int vdiv = (i == 0) ? 1 : info->vdiv;
389 unsigned int aligned_width;
390 unsigned int aligned_height;
391
392 aligned_width = ALIGN(width, v4l2_format_block_width(info, i));
393 aligned_height = ALIGN(height, v4l2_format_block_height(info, i));
394
395 plane = &pixfmt->plane_fmt[i];
396 plane->bytesperline =
397 info->bpp[i] * DIV_ROUND_UP(aligned_width, hdiv);
398 plane->sizeimage =
399 plane->bytesperline * DIV_ROUND_UP(aligned_height, vdiv);
400 }
401 }
402 return 0;
403}
404EXPORT_SYMBOL_GPL(v4l2_fill_pixfmt_mp);
405
406int v4l2_fill_pixfmt(struct v4l2_pix_format *pixfmt, u32 pixelformat,
407 u32 width, u32 height)
408{
409 const struct v4l2_format_info *info;
410 int i;
411
412 info = v4l2_format_info(pixelformat);
413 if (!info)
414 return -EINVAL;
415
416
417 if (info->mem_planes > 1)
418 return -EINVAL;
419
420 pixfmt->width = width;
421 pixfmt->height = height;
422 pixfmt->pixelformat = pixelformat;
423 pixfmt->bytesperline = ALIGN(width, v4l2_format_block_width(info, 0)) * info->bpp[0];
424 pixfmt->sizeimage = 0;
425
426 for (i = 0; i < info->comp_planes; i++) {
427 unsigned int hdiv = (i == 0) ? 1 : info->hdiv;
428 unsigned int vdiv = (i == 0) ? 1 : info->vdiv;
429 unsigned int aligned_width;
430 unsigned int aligned_height;
431
432 aligned_width = ALIGN(width, v4l2_format_block_width(info, i));
433 aligned_height = ALIGN(height, v4l2_format_block_height(info, i));
434
435 pixfmt->sizeimage += info->bpp[i] *
436 DIV_ROUND_UP(aligned_width, hdiv) *
437 DIV_ROUND_UP(aligned_height, vdiv);
438 }
439 return 0;
440}
441EXPORT_SYMBOL_GPL(v4l2_fill_pixfmt);
442