1 // SPDX-License-Identifier: GPL-2.0+
2 /*
3 * Copyright (c) 2011-2018 Magewell Electronics Co., Ltd. (Nanjing)
4 * All rights reserved.
5 * Author: Yong Deng <yong.deng@magewell.com>
6 */
7
8 #include <linux/clk.h>
9 #include <linux/delay.h>
10 #include <linux/dma-mapping.h>
11 #include <linux/err.h>
12 #include <linux/fs.h>
13 #include <linux/interrupt.h>
14 #include <linux/io.h>
15 #include <linux/ioctl.h>
16 #include <linux/module.h>
17 #include <linux/of.h>
18 #include <linux/of_device.h>
19 #include <linux/platform_device.h>
20 #include <linux/pm_runtime.h>
21 #include <linux/regmap.h>
22 #include <linux/reset.h>
23 #include <linux/sched.h>
24 #include <linux/sizes.h>
25 #include <linux/slab.h>
26 #include <media/v4l2-mc.h>
27
28 #include "sun6i_csi.h"
29 #include "sun6i_csi_reg.h"
30
31 /* Helpers */
32
33 /* TODO add 10&12 bit YUV, RGB support */
sun6i_csi_is_format_supported(struct sun6i_csi_device * csi_dev,u32 pixformat,u32 mbus_code)34 bool sun6i_csi_is_format_supported(struct sun6i_csi_device *csi_dev,
35 u32 pixformat, u32 mbus_code)
36 {
37 struct sun6i_csi_v4l2 *v4l2 = &csi_dev->v4l2;
38
39 /*
40 * Some video receivers have the ability to be compatible with
41 * 8bit and 16bit bus width.
42 * Identify the media bus format from device tree.
43 */
44 if ((v4l2->v4l2_ep.bus_type == V4L2_MBUS_PARALLEL
45 || v4l2->v4l2_ep.bus_type == V4L2_MBUS_BT656)
46 && v4l2->v4l2_ep.bus.parallel.bus_width == 16) {
47 switch (pixformat) {
48 case V4L2_PIX_FMT_NV12_16L16:
49 case V4L2_PIX_FMT_NV12:
50 case V4L2_PIX_FMT_NV21:
51 case V4L2_PIX_FMT_NV16:
52 case V4L2_PIX_FMT_NV61:
53 case V4L2_PIX_FMT_YUV420:
54 case V4L2_PIX_FMT_YVU420:
55 case V4L2_PIX_FMT_YUV422P:
56 switch (mbus_code) {
57 case MEDIA_BUS_FMT_UYVY8_1X16:
58 case MEDIA_BUS_FMT_VYUY8_1X16:
59 case MEDIA_BUS_FMT_YUYV8_1X16:
60 case MEDIA_BUS_FMT_YVYU8_1X16:
61 return true;
62 default:
63 dev_dbg(csi_dev->dev,
64 "Unsupported mbus code: 0x%x\n",
65 mbus_code);
66 break;
67 }
68 break;
69 default:
70 dev_dbg(csi_dev->dev, "Unsupported pixformat: 0x%x\n",
71 pixformat);
72 break;
73 }
74 return false;
75 }
76
77 switch (pixformat) {
78 case V4L2_PIX_FMT_SBGGR8:
79 return (mbus_code == MEDIA_BUS_FMT_SBGGR8_1X8);
80 case V4L2_PIX_FMT_SGBRG8:
81 return (mbus_code == MEDIA_BUS_FMT_SGBRG8_1X8);
82 case V4L2_PIX_FMT_SGRBG8:
83 return (mbus_code == MEDIA_BUS_FMT_SGRBG8_1X8);
84 case V4L2_PIX_FMT_SRGGB8:
85 return (mbus_code == MEDIA_BUS_FMT_SRGGB8_1X8);
86 case V4L2_PIX_FMT_SBGGR10:
87 return (mbus_code == MEDIA_BUS_FMT_SBGGR10_1X10);
88 case V4L2_PIX_FMT_SGBRG10:
89 return (mbus_code == MEDIA_BUS_FMT_SGBRG10_1X10);
90 case V4L2_PIX_FMT_SGRBG10:
91 return (mbus_code == MEDIA_BUS_FMT_SGRBG10_1X10);
92 case V4L2_PIX_FMT_SRGGB10:
93 return (mbus_code == MEDIA_BUS_FMT_SRGGB10_1X10);
94 case V4L2_PIX_FMT_SBGGR12:
95 return (mbus_code == MEDIA_BUS_FMT_SBGGR12_1X12);
96 case V4L2_PIX_FMT_SGBRG12:
97 return (mbus_code == MEDIA_BUS_FMT_SGBRG12_1X12);
98 case V4L2_PIX_FMT_SGRBG12:
99 return (mbus_code == MEDIA_BUS_FMT_SGRBG12_1X12);
100 case V4L2_PIX_FMT_SRGGB12:
101 return (mbus_code == MEDIA_BUS_FMT_SRGGB12_1X12);
102
103 case V4L2_PIX_FMT_YUYV:
104 return (mbus_code == MEDIA_BUS_FMT_YUYV8_2X8);
105 case V4L2_PIX_FMT_YVYU:
106 return (mbus_code == MEDIA_BUS_FMT_YVYU8_2X8);
107 case V4L2_PIX_FMT_UYVY:
108 return (mbus_code == MEDIA_BUS_FMT_UYVY8_2X8);
109 case V4L2_PIX_FMT_VYUY:
110 return (mbus_code == MEDIA_BUS_FMT_VYUY8_2X8);
111
112 case V4L2_PIX_FMT_NV12_16L16:
113 case V4L2_PIX_FMT_NV12:
114 case V4L2_PIX_FMT_NV21:
115 case V4L2_PIX_FMT_NV16:
116 case V4L2_PIX_FMT_NV61:
117 case V4L2_PIX_FMT_YUV420:
118 case V4L2_PIX_FMT_YVU420:
119 case V4L2_PIX_FMT_YUV422P:
120 switch (mbus_code) {
121 case MEDIA_BUS_FMT_UYVY8_2X8:
122 case MEDIA_BUS_FMT_VYUY8_2X8:
123 case MEDIA_BUS_FMT_YUYV8_2X8:
124 case MEDIA_BUS_FMT_YVYU8_2X8:
125 return true;
126 default:
127 dev_dbg(csi_dev->dev, "Unsupported mbus code: 0x%x\n",
128 mbus_code);
129 break;
130 }
131 break;
132
133 case V4L2_PIX_FMT_RGB565:
134 return (mbus_code == MEDIA_BUS_FMT_RGB565_2X8_LE);
135 case V4L2_PIX_FMT_RGB565X:
136 return (mbus_code == MEDIA_BUS_FMT_RGB565_2X8_BE);
137
138 case V4L2_PIX_FMT_JPEG:
139 return (mbus_code == MEDIA_BUS_FMT_JPEG_1X8);
140
141 default:
142 dev_dbg(csi_dev->dev, "Unsupported pixformat: 0x%x\n",
143 pixformat);
144 break;
145 }
146
147 return false;
148 }
149
sun6i_csi_set_power(struct sun6i_csi_device * csi_dev,bool enable)150 int sun6i_csi_set_power(struct sun6i_csi_device *csi_dev, bool enable)
151 {
152 struct device *dev = csi_dev->dev;
153 struct regmap *regmap = csi_dev->regmap;
154 int ret;
155
156 if (!enable) {
157 regmap_update_bits(regmap, CSI_EN_REG, CSI_EN_CSI_EN, 0);
158 pm_runtime_put(dev);
159
160 return 0;
161 }
162
163 ret = pm_runtime_resume_and_get(dev);
164 if (ret < 0)
165 return ret;
166
167 regmap_update_bits(regmap, CSI_EN_REG, CSI_EN_CSI_EN, CSI_EN_CSI_EN);
168
169 return 0;
170 }
171
get_csi_input_format(struct sun6i_csi_device * csi_dev,u32 mbus_code,u32 pixformat)172 static enum csi_input_fmt get_csi_input_format(struct sun6i_csi_device *csi_dev,
173 u32 mbus_code, u32 pixformat)
174 {
175 /* non-YUV */
176 if ((mbus_code & 0xF000) != 0x2000)
177 return CSI_INPUT_FORMAT_RAW;
178
179 switch (pixformat) {
180 case V4L2_PIX_FMT_YUYV:
181 case V4L2_PIX_FMT_YVYU:
182 case V4L2_PIX_FMT_UYVY:
183 case V4L2_PIX_FMT_VYUY:
184 return CSI_INPUT_FORMAT_RAW;
185 default:
186 break;
187 }
188
189 /* not support YUV420 input format yet */
190 dev_dbg(csi_dev->dev, "Select YUV422 as default input format of CSI.\n");
191 return CSI_INPUT_FORMAT_YUV422;
192 }
193
194 static enum csi_output_fmt
get_csi_output_format(struct sun6i_csi_device * csi_dev,u32 pixformat,u32 field)195 get_csi_output_format(struct sun6i_csi_device *csi_dev, u32 pixformat,
196 u32 field)
197 {
198 bool buf_interlaced = false;
199
200 if (field == V4L2_FIELD_INTERLACED
201 || field == V4L2_FIELD_INTERLACED_TB
202 || field == V4L2_FIELD_INTERLACED_BT)
203 buf_interlaced = true;
204
205 switch (pixformat) {
206 case V4L2_PIX_FMT_SBGGR8:
207 case V4L2_PIX_FMT_SGBRG8:
208 case V4L2_PIX_FMT_SGRBG8:
209 case V4L2_PIX_FMT_SRGGB8:
210 return buf_interlaced ? CSI_FRAME_RAW_8 : CSI_FIELD_RAW_8;
211 case V4L2_PIX_FMT_SBGGR10:
212 case V4L2_PIX_FMT_SGBRG10:
213 case V4L2_PIX_FMT_SGRBG10:
214 case V4L2_PIX_FMT_SRGGB10:
215 return buf_interlaced ? CSI_FRAME_RAW_10 : CSI_FIELD_RAW_10;
216 case V4L2_PIX_FMT_SBGGR12:
217 case V4L2_PIX_FMT_SGBRG12:
218 case V4L2_PIX_FMT_SGRBG12:
219 case V4L2_PIX_FMT_SRGGB12:
220 return buf_interlaced ? CSI_FRAME_RAW_12 : CSI_FIELD_RAW_12;
221
222 case V4L2_PIX_FMT_YUYV:
223 case V4L2_PIX_FMT_YVYU:
224 case V4L2_PIX_FMT_UYVY:
225 case V4L2_PIX_FMT_VYUY:
226 return buf_interlaced ? CSI_FRAME_RAW_8 : CSI_FIELD_RAW_8;
227
228 case V4L2_PIX_FMT_NV12_16L16:
229 return buf_interlaced ? CSI_FRAME_MB_YUV420 :
230 CSI_FIELD_MB_YUV420;
231 case V4L2_PIX_FMT_NV12:
232 case V4L2_PIX_FMT_NV21:
233 return buf_interlaced ? CSI_FRAME_UV_CB_YUV420 :
234 CSI_FIELD_UV_CB_YUV420;
235 case V4L2_PIX_FMT_YUV420:
236 case V4L2_PIX_FMT_YVU420:
237 return buf_interlaced ? CSI_FRAME_PLANAR_YUV420 :
238 CSI_FIELD_PLANAR_YUV420;
239 case V4L2_PIX_FMT_NV16:
240 case V4L2_PIX_FMT_NV61:
241 return buf_interlaced ? CSI_FRAME_UV_CB_YUV422 :
242 CSI_FIELD_UV_CB_YUV422;
243 case V4L2_PIX_FMT_YUV422P:
244 return buf_interlaced ? CSI_FRAME_PLANAR_YUV422 :
245 CSI_FIELD_PLANAR_YUV422;
246
247 case V4L2_PIX_FMT_RGB565:
248 case V4L2_PIX_FMT_RGB565X:
249 return buf_interlaced ? CSI_FRAME_RGB565 : CSI_FIELD_RGB565;
250
251 case V4L2_PIX_FMT_JPEG:
252 return buf_interlaced ? CSI_FRAME_RAW_8 : CSI_FIELD_RAW_8;
253
254 default:
255 dev_warn(csi_dev->dev, "Unsupported pixformat: 0x%x\n", pixformat);
256 break;
257 }
258
259 return CSI_FIELD_RAW_8;
260 }
261
get_csi_input_seq(struct sun6i_csi_device * csi_dev,u32 mbus_code,u32 pixformat)262 static enum csi_input_seq get_csi_input_seq(struct sun6i_csi_device *csi_dev,
263 u32 mbus_code, u32 pixformat)
264 {
265 /* Input sequence does not apply to non-YUV formats */
266 if ((mbus_code & 0xF000) != 0x2000)
267 return 0;
268
269 switch (pixformat) {
270 case V4L2_PIX_FMT_NV12_16L16:
271 case V4L2_PIX_FMT_NV12:
272 case V4L2_PIX_FMT_NV16:
273 case V4L2_PIX_FMT_YUV420:
274 case V4L2_PIX_FMT_YUV422P:
275 switch (mbus_code) {
276 case MEDIA_BUS_FMT_UYVY8_2X8:
277 case MEDIA_BUS_FMT_UYVY8_1X16:
278 return CSI_INPUT_SEQ_UYVY;
279 case MEDIA_BUS_FMT_VYUY8_2X8:
280 case MEDIA_BUS_FMT_VYUY8_1X16:
281 return CSI_INPUT_SEQ_VYUY;
282 case MEDIA_BUS_FMT_YUYV8_2X8:
283 case MEDIA_BUS_FMT_YUYV8_1X16:
284 return CSI_INPUT_SEQ_YUYV;
285 case MEDIA_BUS_FMT_YVYU8_1X16:
286 case MEDIA_BUS_FMT_YVYU8_2X8:
287 return CSI_INPUT_SEQ_YVYU;
288 default:
289 dev_warn(csi_dev->dev, "Unsupported mbus code: 0x%x\n",
290 mbus_code);
291 break;
292 }
293 break;
294 case V4L2_PIX_FMT_NV21:
295 case V4L2_PIX_FMT_NV61:
296 case V4L2_PIX_FMT_YVU420:
297 switch (mbus_code) {
298 case MEDIA_BUS_FMT_UYVY8_2X8:
299 case MEDIA_BUS_FMT_UYVY8_1X16:
300 return CSI_INPUT_SEQ_VYUY;
301 case MEDIA_BUS_FMT_VYUY8_2X8:
302 case MEDIA_BUS_FMT_VYUY8_1X16:
303 return CSI_INPUT_SEQ_UYVY;
304 case MEDIA_BUS_FMT_YUYV8_2X8:
305 case MEDIA_BUS_FMT_YUYV8_1X16:
306 return CSI_INPUT_SEQ_YVYU;
307 case MEDIA_BUS_FMT_YVYU8_1X16:
308 case MEDIA_BUS_FMT_YVYU8_2X8:
309 return CSI_INPUT_SEQ_YUYV;
310 default:
311 dev_warn(csi_dev->dev, "Unsupported mbus code: 0x%x\n",
312 mbus_code);
313 break;
314 }
315 break;
316
317 case V4L2_PIX_FMT_YUYV:
318 return CSI_INPUT_SEQ_YUYV;
319
320 default:
321 dev_warn(csi_dev->dev, "Unsupported pixformat: 0x%x, defaulting to YUYV\n",
322 pixformat);
323 break;
324 }
325
326 return CSI_INPUT_SEQ_YUYV;
327 }
328
sun6i_csi_setup_bus(struct sun6i_csi_device * csi_dev)329 static void sun6i_csi_setup_bus(struct sun6i_csi_device *csi_dev)
330 {
331 struct v4l2_fwnode_endpoint *endpoint = &csi_dev->v4l2.v4l2_ep;
332 struct sun6i_csi_config *config = &csi_dev->config;
333 unsigned char bus_width;
334 u32 flags;
335 u32 cfg;
336 bool input_interlaced = false;
337
338 if (config->field == V4L2_FIELD_INTERLACED
339 || config->field == V4L2_FIELD_INTERLACED_TB
340 || config->field == V4L2_FIELD_INTERLACED_BT)
341 input_interlaced = true;
342
343 bus_width = endpoint->bus.parallel.bus_width;
344
345 regmap_read(csi_dev->regmap, CSI_IF_CFG_REG, &cfg);
346
347 cfg &= ~(CSI_IF_CFG_CSI_IF_MASK | CSI_IF_CFG_MIPI_IF_MASK |
348 CSI_IF_CFG_IF_DATA_WIDTH_MASK |
349 CSI_IF_CFG_CLK_POL_MASK | CSI_IF_CFG_VREF_POL_MASK |
350 CSI_IF_CFG_HREF_POL_MASK | CSI_IF_CFG_FIELD_MASK |
351 CSI_IF_CFG_SRC_TYPE_MASK);
352
353 if (input_interlaced)
354 cfg |= CSI_IF_CFG_SRC_TYPE_INTERLACED;
355 else
356 cfg |= CSI_IF_CFG_SRC_TYPE_PROGRESSED;
357
358 switch (endpoint->bus_type) {
359 case V4L2_MBUS_PARALLEL:
360 cfg |= CSI_IF_CFG_MIPI_IF_CSI;
361
362 flags = endpoint->bus.parallel.flags;
363
364 cfg |= (bus_width == 16) ? CSI_IF_CFG_CSI_IF_YUV422_16BIT :
365 CSI_IF_CFG_CSI_IF_YUV422_INTLV;
366
367 if (flags & V4L2_MBUS_FIELD_EVEN_LOW)
368 cfg |= CSI_IF_CFG_FIELD_POSITIVE;
369
370 if (flags & V4L2_MBUS_VSYNC_ACTIVE_LOW)
371 cfg |= CSI_IF_CFG_VREF_POL_POSITIVE;
372 if (flags & V4L2_MBUS_HSYNC_ACTIVE_LOW)
373 cfg |= CSI_IF_CFG_HREF_POL_POSITIVE;
374
375 if (flags & V4L2_MBUS_PCLK_SAMPLE_RISING)
376 cfg |= CSI_IF_CFG_CLK_POL_FALLING_EDGE;
377 break;
378 case V4L2_MBUS_BT656:
379 cfg |= CSI_IF_CFG_MIPI_IF_CSI;
380
381 flags = endpoint->bus.parallel.flags;
382
383 cfg |= (bus_width == 16) ? CSI_IF_CFG_CSI_IF_BT1120 :
384 CSI_IF_CFG_CSI_IF_BT656;
385
386 if (flags & V4L2_MBUS_FIELD_EVEN_LOW)
387 cfg |= CSI_IF_CFG_FIELD_POSITIVE;
388
389 if (flags & V4L2_MBUS_PCLK_SAMPLE_FALLING)
390 cfg |= CSI_IF_CFG_CLK_POL_FALLING_EDGE;
391 break;
392 default:
393 dev_warn(csi_dev->dev, "Unsupported bus type: %d\n",
394 endpoint->bus_type);
395 break;
396 }
397
398 switch (bus_width) {
399 case 8:
400 cfg |= CSI_IF_CFG_IF_DATA_WIDTH_8BIT;
401 break;
402 case 10:
403 cfg |= CSI_IF_CFG_IF_DATA_WIDTH_10BIT;
404 break;
405 case 12:
406 cfg |= CSI_IF_CFG_IF_DATA_WIDTH_12BIT;
407 break;
408 case 16: /* No need to configure DATA_WIDTH for 16bit */
409 break;
410 default:
411 dev_warn(csi_dev->dev, "Unsupported bus width: %u\n", bus_width);
412 break;
413 }
414
415 regmap_write(csi_dev->regmap, CSI_IF_CFG_REG, cfg);
416 }
417
sun6i_csi_set_format(struct sun6i_csi_device * csi_dev)418 static void sun6i_csi_set_format(struct sun6i_csi_device *csi_dev)
419 {
420 struct sun6i_csi_config *config = &csi_dev->config;
421 u32 cfg;
422 u32 val;
423
424 regmap_read(csi_dev->regmap, CSI_CH_CFG_REG, &cfg);
425
426 cfg &= ~(CSI_CH_CFG_INPUT_FMT_MASK |
427 CSI_CH_CFG_OUTPUT_FMT_MASK | CSI_CH_CFG_VFLIP_EN |
428 CSI_CH_CFG_HFLIP_EN | CSI_CH_CFG_FIELD_SEL_MASK |
429 CSI_CH_CFG_INPUT_SEQ_MASK);
430
431 val = get_csi_input_format(csi_dev, config->code,
432 config->pixelformat);
433 cfg |= CSI_CH_CFG_INPUT_FMT(val);
434
435 val = get_csi_output_format(csi_dev, config->pixelformat,
436 config->field);
437 cfg |= CSI_CH_CFG_OUTPUT_FMT(val);
438
439 val = get_csi_input_seq(csi_dev, config->code,
440 config->pixelformat);
441 cfg |= CSI_CH_CFG_INPUT_SEQ(val);
442
443 if (config->field == V4L2_FIELD_TOP)
444 cfg |= CSI_CH_CFG_FIELD_SEL_FIELD0;
445 else if (config->field == V4L2_FIELD_BOTTOM)
446 cfg |= CSI_CH_CFG_FIELD_SEL_FIELD1;
447 else
448 cfg |= CSI_CH_CFG_FIELD_SEL_BOTH;
449
450 regmap_write(csi_dev->regmap, CSI_CH_CFG_REG, cfg);
451 }
452
sun6i_csi_set_window(struct sun6i_csi_device * csi_dev)453 static void sun6i_csi_set_window(struct sun6i_csi_device *csi_dev)
454 {
455 struct sun6i_csi_config *config = &csi_dev->config;
456 u32 bytesperline_y;
457 u32 bytesperline_c;
458 int *planar_offset = csi_dev->planar_offset;
459 u32 width = config->width;
460 u32 height = config->height;
461 u32 hor_len = width;
462
463 switch (config->pixelformat) {
464 case V4L2_PIX_FMT_YUYV:
465 case V4L2_PIX_FMT_YVYU:
466 case V4L2_PIX_FMT_UYVY:
467 case V4L2_PIX_FMT_VYUY:
468 dev_dbg(csi_dev->dev,
469 "Horizontal length should be 2 times of width for packed YUV formats!\n");
470 hor_len = width * 2;
471 break;
472 default:
473 break;
474 }
475
476 regmap_write(csi_dev->regmap, CSI_CH_HSIZE_REG,
477 CSI_CH_HSIZE_HOR_LEN(hor_len) |
478 CSI_CH_HSIZE_HOR_START(0));
479 regmap_write(csi_dev->regmap, CSI_CH_VSIZE_REG,
480 CSI_CH_VSIZE_VER_LEN(height) |
481 CSI_CH_VSIZE_VER_START(0));
482
483 planar_offset[0] = 0;
484 switch (config->pixelformat) {
485 case V4L2_PIX_FMT_NV12_16L16:
486 case V4L2_PIX_FMT_NV12:
487 case V4L2_PIX_FMT_NV21:
488 case V4L2_PIX_FMT_NV16:
489 case V4L2_PIX_FMT_NV61:
490 bytesperline_y = width;
491 bytesperline_c = width;
492 planar_offset[1] = bytesperline_y * height;
493 planar_offset[2] = -1;
494 break;
495 case V4L2_PIX_FMT_YUV420:
496 case V4L2_PIX_FMT_YVU420:
497 bytesperline_y = width;
498 bytesperline_c = width / 2;
499 planar_offset[1] = bytesperline_y * height;
500 planar_offset[2] = planar_offset[1] +
501 bytesperline_c * height / 2;
502 break;
503 case V4L2_PIX_FMT_YUV422P:
504 bytesperline_y = width;
505 bytesperline_c = width / 2;
506 planar_offset[1] = bytesperline_y * height;
507 planar_offset[2] = planar_offset[1] +
508 bytesperline_c * height;
509 break;
510 default: /* raw */
511 dev_dbg(csi_dev->dev,
512 "Calculating pixelformat(0x%x)'s bytesperline as a packed format\n",
513 config->pixelformat);
514 bytesperline_y = (sun6i_csi_get_bpp(config->pixelformat) *
515 config->width) / 8;
516 bytesperline_c = 0;
517 planar_offset[1] = -1;
518 planar_offset[2] = -1;
519 break;
520 }
521
522 regmap_write(csi_dev->regmap, CSI_CH_BUF_LEN_REG,
523 CSI_CH_BUF_LEN_BUF_LEN_C(bytesperline_c) |
524 CSI_CH_BUF_LEN_BUF_LEN_Y(bytesperline_y));
525 }
526
sun6i_csi_update_config(struct sun6i_csi_device * csi_dev,struct sun6i_csi_config * config)527 int sun6i_csi_update_config(struct sun6i_csi_device *csi_dev,
528 struct sun6i_csi_config *config)
529 {
530 if (!config)
531 return -EINVAL;
532
533 memcpy(&csi_dev->config, config, sizeof(csi_dev->config));
534
535 sun6i_csi_setup_bus(csi_dev);
536 sun6i_csi_set_format(csi_dev);
537 sun6i_csi_set_window(csi_dev);
538
539 return 0;
540 }
541
sun6i_csi_update_buf_addr(struct sun6i_csi_device * csi_dev,dma_addr_t addr)542 void sun6i_csi_update_buf_addr(struct sun6i_csi_device *csi_dev,
543 dma_addr_t addr)
544 {
545 regmap_write(csi_dev->regmap, CSI_CH_F0_BUFA_REG,
546 (addr + csi_dev->planar_offset[0]) >> 2);
547 if (csi_dev->planar_offset[1] != -1)
548 regmap_write(csi_dev->regmap, CSI_CH_F1_BUFA_REG,
549 (addr + csi_dev->planar_offset[1]) >> 2);
550 if (csi_dev->planar_offset[2] != -1)
551 regmap_write(csi_dev->regmap, CSI_CH_F2_BUFA_REG,
552 (addr + csi_dev->planar_offset[2]) >> 2);
553 }
554
sun6i_csi_set_stream(struct sun6i_csi_device * csi_dev,bool enable)555 void sun6i_csi_set_stream(struct sun6i_csi_device *csi_dev, bool enable)
556 {
557 struct regmap *regmap = csi_dev->regmap;
558
559 if (!enable) {
560 regmap_update_bits(regmap, CSI_CAP_REG, CSI_CAP_CH0_VCAP_ON, 0);
561 regmap_write(regmap, CSI_CH_INT_EN_REG, 0);
562 return;
563 }
564
565 regmap_write(regmap, CSI_CH_INT_STA_REG, 0xFF);
566 regmap_write(regmap, CSI_CH_INT_EN_REG,
567 CSI_CH_INT_EN_HB_OF_INT_EN |
568 CSI_CH_INT_EN_FIFO2_OF_INT_EN |
569 CSI_CH_INT_EN_FIFO1_OF_INT_EN |
570 CSI_CH_INT_EN_FIFO0_OF_INT_EN |
571 CSI_CH_INT_EN_FD_INT_EN |
572 CSI_CH_INT_EN_CD_INT_EN);
573
574 regmap_update_bits(regmap, CSI_CAP_REG, CSI_CAP_CH0_VCAP_ON,
575 CSI_CAP_CH0_VCAP_ON);
576 }
577
578 /* Media */
579
580 static const struct media_device_ops sun6i_csi_media_ops = {
581 .link_notify = v4l2_pipeline_link_notify,
582 };
583
584 /* V4L2 */
585
sun6i_csi_link_entity(struct sun6i_csi_device * csi_dev,struct media_entity * entity,struct fwnode_handle * fwnode)586 static int sun6i_csi_link_entity(struct sun6i_csi_device *csi_dev,
587 struct media_entity *entity,
588 struct fwnode_handle *fwnode)
589 {
590 struct media_entity *sink;
591 struct media_pad *sink_pad;
592 int src_pad_index;
593 int ret;
594
595 ret = media_entity_get_fwnode_pad(entity, fwnode, MEDIA_PAD_FL_SOURCE);
596 if (ret < 0) {
597 dev_err(csi_dev->dev,
598 "%s: no source pad in external entity %s\n", __func__,
599 entity->name);
600 return -EINVAL;
601 }
602
603 src_pad_index = ret;
604
605 sink = &csi_dev->video.video_dev.entity;
606 sink_pad = &csi_dev->video.pad;
607
608 dev_dbg(csi_dev->dev, "creating %s:%u -> %s:%u link\n",
609 entity->name, src_pad_index, sink->name, sink_pad->index);
610 ret = media_create_pad_link(entity, src_pad_index, sink,
611 sink_pad->index,
612 MEDIA_LNK_FL_ENABLED |
613 MEDIA_LNK_FL_IMMUTABLE);
614 if (ret < 0) {
615 dev_err(csi_dev->dev, "failed to create %s:%u -> %s:%u link\n",
616 entity->name, src_pad_index,
617 sink->name, sink_pad->index);
618 return ret;
619 }
620
621 return 0;
622 }
623
sun6i_subdev_notify_complete(struct v4l2_async_notifier * notifier)624 static int sun6i_subdev_notify_complete(struct v4l2_async_notifier *notifier)
625 {
626 struct sun6i_csi_device *csi_dev =
627 container_of(notifier, struct sun6i_csi_device,
628 v4l2.notifier);
629 struct sun6i_csi_v4l2 *v4l2 = &csi_dev->v4l2;
630 struct v4l2_device *v4l2_dev = &v4l2->v4l2_dev;
631 struct v4l2_subdev *sd;
632 int ret;
633
634 dev_dbg(csi_dev->dev, "notify complete, all subdevs registered\n");
635
636 sd = list_first_entry(&v4l2_dev->subdevs, struct v4l2_subdev, list);
637 if (!sd)
638 return -EINVAL;
639
640 ret = sun6i_csi_link_entity(csi_dev, &sd->entity, sd->fwnode);
641 if (ret < 0)
642 return ret;
643
644 ret = v4l2_device_register_subdev_nodes(v4l2_dev);
645 if (ret < 0)
646 return ret;
647
648 return 0;
649 }
650
651 static const struct v4l2_async_notifier_operations sun6i_csi_async_ops = {
652 .complete = sun6i_subdev_notify_complete,
653 };
654
sun6i_csi_fwnode_parse(struct device * dev,struct v4l2_fwnode_endpoint * vep,struct v4l2_async_subdev * asd)655 static int sun6i_csi_fwnode_parse(struct device *dev,
656 struct v4l2_fwnode_endpoint *vep,
657 struct v4l2_async_subdev *asd)
658 {
659 struct sun6i_csi_device *csi_dev = dev_get_drvdata(dev);
660
661 if (vep->base.port || vep->base.id) {
662 dev_warn(dev, "Only support a single port with one endpoint\n");
663 return -ENOTCONN;
664 }
665
666 switch (vep->bus_type) {
667 case V4L2_MBUS_PARALLEL:
668 case V4L2_MBUS_BT656:
669 csi_dev->v4l2.v4l2_ep = *vep;
670 return 0;
671 default:
672 dev_err(dev, "Unsupported media bus type\n");
673 return -ENOTCONN;
674 }
675 }
676
sun6i_csi_v4l2_setup(struct sun6i_csi_device * csi_dev)677 static int sun6i_csi_v4l2_setup(struct sun6i_csi_device *csi_dev)
678 {
679 struct sun6i_csi_v4l2 *v4l2 = &csi_dev->v4l2;
680 struct media_device *media_dev = &v4l2->media_dev;
681 struct v4l2_device *v4l2_dev = &v4l2->v4l2_dev;
682 struct v4l2_async_notifier *notifier = &v4l2->notifier;
683 struct device *dev = csi_dev->dev;
684 int ret;
685
686 /* Media Device */
687
688 strscpy(media_dev->model, SUN6I_CSI_DESCRIPTION,
689 sizeof(media_dev->model));
690 media_dev->hw_revision = 0;
691 media_dev->ops = &sun6i_csi_media_ops;
692 media_dev->dev = dev;
693
694 media_device_init(media_dev);
695
696 ret = media_device_register(media_dev);
697 if (ret) {
698 dev_err(dev, "failed to register media device: %d\n", ret);
699 goto error_media;
700 }
701
702 /* V4L2 Device */
703
704 v4l2_dev->mdev = media_dev;
705
706 ret = v4l2_device_register(dev, v4l2_dev);
707 if (ret) {
708 dev_err(dev, "failed to register v4l2 device: %d\n", ret);
709 goto error_media;
710 }
711
712 /* Video */
713
714 ret = sun6i_video_setup(csi_dev);
715 if (ret)
716 goto error_v4l2_device;
717
718 /* V4L2 Async */
719
720 v4l2_async_nf_init(notifier);
721 notifier->ops = &sun6i_csi_async_ops;
722
723 ret = v4l2_async_nf_parse_fwnode_endpoints(dev, notifier,
724 sizeof(struct
725 v4l2_async_subdev),
726 sun6i_csi_fwnode_parse);
727 if (ret)
728 goto error_video;
729
730 ret = v4l2_async_nf_register(v4l2_dev, notifier);
731 if (ret) {
732 dev_err(dev, "failed to register v4l2 async notifier: %d\n",
733 ret);
734 goto error_v4l2_async_notifier;
735 }
736
737 return 0;
738
739 error_v4l2_async_notifier:
740 v4l2_async_nf_cleanup(notifier);
741
742 error_video:
743 sun6i_video_cleanup(csi_dev);
744
745 error_v4l2_device:
746 v4l2_device_unregister(&v4l2->v4l2_dev);
747
748 error_media:
749 media_device_unregister(media_dev);
750 media_device_cleanup(media_dev);
751
752 return ret;
753 }
754
sun6i_csi_v4l2_cleanup(struct sun6i_csi_device * csi_dev)755 static void sun6i_csi_v4l2_cleanup(struct sun6i_csi_device *csi_dev)
756 {
757 struct sun6i_csi_v4l2 *v4l2 = &csi_dev->v4l2;
758
759 media_device_unregister(&v4l2->media_dev);
760 v4l2_async_nf_unregister(&v4l2->notifier);
761 v4l2_async_nf_cleanup(&v4l2->notifier);
762 sun6i_video_cleanup(csi_dev);
763 v4l2_device_unregister(&v4l2->v4l2_dev);
764 media_device_cleanup(&v4l2->media_dev);
765 }
766
767 /* Platform */
768
sun6i_csi_interrupt(int irq,void * private)769 static irqreturn_t sun6i_csi_interrupt(int irq, void *private)
770 {
771 struct sun6i_csi_device *csi_dev = private;
772 struct regmap *regmap = csi_dev->regmap;
773 u32 status;
774
775 regmap_read(regmap, CSI_CH_INT_STA_REG, &status);
776
777 if (!(status & 0xFF))
778 return IRQ_NONE;
779
780 if ((status & CSI_CH_INT_STA_FIFO0_OF_PD) ||
781 (status & CSI_CH_INT_STA_FIFO1_OF_PD) ||
782 (status & CSI_CH_INT_STA_FIFO2_OF_PD) ||
783 (status & CSI_CH_INT_STA_HB_OF_PD)) {
784 regmap_write(regmap, CSI_CH_INT_STA_REG, status);
785 regmap_update_bits(regmap, CSI_EN_REG, CSI_EN_CSI_EN, 0);
786 regmap_update_bits(regmap, CSI_EN_REG, CSI_EN_CSI_EN,
787 CSI_EN_CSI_EN);
788 return IRQ_HANDLED;
789 }
790
791 if (status & CSI_CH_INT_STA_FD_PD)
792 sun6i_video_frame_done(csi_dev);
793
794 regmap_write(regmap, CSI_CH_INT_STA_REG, status);
795
796 return IRQ_HANDLED;
797 }
798
sun6i_csi_suspend(struct device * dev)799 static int sun6i_csi_suspend(struct device *dev)
800 {
801 struct sun6i_csi_device *csi_dev = dev_get_drvdata(dev);
802
803 reset_control_assert(csi_dev->reset);
804 clk_disable_unprepare(csi_dev->clock_ram);
805 clk_disable_unprepare(csi_dev->clock_mod);
806
807 return 0;
808 }
809
sun6i_csi_resume(struct device * dev)810 static int sun6i_csi_resume(struct device *dev)
811 {
812 struct sun6i_csi_device *csi_dev = dev_get_drvdata(dev);
813 int ret;
814
815 ret = reset_control_deassert(csi_dev->reset);
816 if (ret) {
817 dev_err(dev, "failed to deassert reset\n");
818 return ret;
819 }
820
821 ret = clk_prepare_enable(csi_dev->clock_mod);
822 if (ret) {
823 dev_err(dev, "failed to enable module clock\n");
824 goto error_reset;
825 }
826
827 ret = clk_prepare_enable(csi_dev->clock_ram);
828 if (ret) {
829 dev_err(dev, "failed to enable ram clock\n");
830 goto error_clock_mod;
831 }
832
833 return 0;
834
835 error_clock_mod:
836 clk_disable_unprepare(csi_dev->clock_mod);
837
838 error_reset:
839 reset_control_assert(csi_dev->reset);
840
841 return ret;
842 }
843
844 static const struct dev_pm_ops sun6i_csi_pm_ops = {
845 .runtime_suspend = sun6i_csi_suspend,
846 .runtime_resume = sun6i_csi_resume,
847 };
848
849 static const struct regmap_config sun6i_csi_regmap_config = {
850 .reg_bits = 32,
851 .reg_stride = 4,
852 .val_bits = 32,
853 .max_register = 0x9c,
854 };
855
sun6i_csi_resources_setup(struct sun6i_csi_device * csi_dev,struct platform_device * platform_dev)856 static int sun6i_csi_resources_setup(struct sun6i_csi_device *csi_dev,
857 struct platform_device *platform_dev)
858 {
859 struct device *dev = csi_dev->dev;
860 const struct sun6i_csi_variant *variant;
861 void __iomem *io_base;
862 int ret;
863 int irq;
864
865 variant = of_device_get_match_data(dev);
866 if (!variant)
867 return -EINVAL;
868
869 /* Registers */
870
871 io_base = devm_platform_ioremap_resource(platform_dev, 0);
872 if (IS_ERR(io_base))
873 return PTR_ERR(io_base);
874
875 csi_dev->regmap = devm_regmap_init_mmio_clk(dev, "bus", io_base,
876 &sun6i_csi_regmap_config);
877 if (IS_ERR(csi_dev->regmap)) {
878 dev_err(dev, "failed to init register map\n");
879 return PTR_ERR(csi_dev->regmap);
880 }
881
882 /* Clocks */
883
884 csi_dev->clock_mod = devm_clk_get(dev, "mod");
885 if (IS_ERR(csi_dev->clock_mod)) {
886 dev_err(dev, "failed to acquire module clock\n");
887 return PTR_ERR(csi_dev->clock_mod);
888 }
889
890 csi_dev->clock_ram = devm_clk_get(dev, "ram");
891 if (IS_ERR(csi_dev->clock_ram)) {
892 dev_err(dev, "failed to acquire ram clock\n");
893 return PTR_ERR(csi_dev->clock_ram);
894 }
895
896 ret = clk_set_rate_exclusive(csi_dev->clock_mod,
897 variant->clock_mod_rate);
898 if (ret) {
899 dev_err(dev, "failed to set mod clock rate\n");
900 return ret;
901 }
902
903 /* Reset */
904
905 csi_dev->reset = devm_reset_control_get_shared(dev, NULL);
906 if (IS_ERR(csi_dev->reset)) {
907 dev_err(dev, "failed to acquire reset\n");
908 ret = PTR_ERR(csi_dev->reset);
909 goto error_clock_rate_exclusive;
910 }
911
912 /* Interrupt */
913
914 irq = platform_get_irq(platform_dev, 0);
915 if (irq < 0) {
916 dev_err(dev, "failed to get interrupt\n");
917 ret = -ENXIO;
918 goto error_clock_rate_exclusive;
919 }
920
921 ret = devm_request_irq(dev, irq, sun6i_csi_interrupt, 0, SUN6I_CSI_NAME,
922 csi_dev);
923 if (ret) {
924 dev_err(dev, "failed to request interrupt\n");
925 goto error_clock_rate_exclusive;
926 }
927
928 /* Runtime PM */
929
930 pm_runtime_enable(dev);
931
932 return 0;
933
934 error_clock_rate_exclusive:
935 clk_rate_exclusive_put(csi_dev->clock_mod);
936
937 return ret;
938 }
939
sun6i_csi_resources_cleanup(struct sun6i_csi_device * csi_dev)940 static void sun6i_csi_resources_cleanup(struct sun6i_csi_device *csi_dev)
941 {
942 pm_runtime_disable(csi_dev->dev);
943 clk_rate_exclusive_put(csi_dev->clock_mod);
944 }
945
sun6i_csi_probe(struct platform_device * platform_dev)946 static int sun6i_csi_probe(struct platform_device *platform_dev)
947 {
948 struct sun6i_csi_device *csi_dev;
949 struct device *dev = &platform_dev->dev;
950 int ret;
951
952 csi_dev = devm_kzalloc(dev, sizeof(*csi_dev), GFP_KERNEL);
953 if (!csi_dev)
954 return -ENOMEM;
955
956 csi_dev->dev = &platform_dev->dev;
957 platform_set_drvdata(platform_dev, csi_dev);
958
959 ret = sun6i_csi_resources_setup(csi_dev, platform_dev);
960 if (ret)
961 return ret;
962
963 ret = sun6i_csi_v4l2_setup(csi_dev);
964 if (ret)
965 goto error_resources;
966
967 return 0;
968
969 error_resources:
970 sun6i_csi_resources_cleanup(csi_dev);
971
972 return ret;
973 }
974
sun6i_csi_remove(struct platform_device * pdev)975 static int sun6i_csi_remove(struct platform_device *pdev)
976 {
977 struct sun6i_csi_device *csi_dev = platform_get_drvdata(pdev);
978
979 sun6i_csi_v4l2_cleanup(csi_dev);
980 sun6i_csi_resources_cleanup(csi_dev);
981
982 return 0;
983 }
984
985 static const struct sun6i_csi_variant sun6i_a31_csi_variant = {
986 .clock_mod_rate = 297000000,
987 };
988
989 static const struct sun6i_csi_variant sun50i_a64_csi_variant = {
990 .clock_mod_rate = 300000000,
991 };
992
993 static const struct of_device_id sun6i_csi_of_match[] = {
994 {
995 .compatible = "allwinner,sun6i-a31-csi",
996 .data = &sun6i_a31_csi_variant,
997 },
998 {
999 .compatible = "allwinner,sun8i-a83t-csi",
1000 .data = &sun6i_a31_csi_variant,
1001 },
1002 {
1003 .compatible = "allwinner,sun8i-h3-csi",
1004 .data = &sun6i_a31_csi_variant,
1005 },
1006 {
1007 .compatible = "allwinner,sun8i-v3s-csi",
1008 .data = &sun6i_a31_csi_variant,
1009 },
1010 {
1011 .compatible = "allwinner,sun50i-a64-csi",
1012 .data = &sun50i_a64_csi_variant,
1013 },
1014 {},
1015 };
1016
1017 MODULE_DEVICE_TABLE(of, sun6i_csi_of_match);
1018
1019 static struct platform_driver sun6i_csi_platform_driver = {
1020 .probe = sun6i_csi_probe,
1021 .remove = sun6i_csi_remove,
1022 .driver = {
1023 .name = SUN6I_CSI_NAME,
1024 .of_match_table = of_match_ptr(sun6i_csi_of_match),
1025 .pm = &sun6i_csi_pm_ops,
1026 },
1027 };
1028
1029 module_platform_driver(sun6i_csi_platform_driver);
1030
1031 MODULE_DESCRIPTION("Allwinner A31 Camera Sensor Interface driver");
1032 MODULE_AUTHOR("Yong Deng <yong.deng@magewell.com>");
1033 MODULE_LICENSE("GPL");
1034