Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2012-2014 Mentor Graphics Inc. |
| 3 | * Copyright (C) 2005-2009 Freescale Semiconductor, Inc. |
| 4 | * |
| 5 | * This program is free software; you can redistribute it and/or modify it |
| 6 | * under the terms of the GNU General Public License as published by the |
| 7 | * Free Software Foundation; either version 2 of the License, or (at your |
| 8 | * option) any later version. |
| 9 | * |
| 10 | * This program is distributed in the hope that it will be useful, but |
| 11 | * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY |
| 12 | * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License |
| 13 | * for more details. |
| 14 | */ |
| 15 | #include <linux/export.h> |
| 16 | #include <linux/module.h> |
| 17 | #include <linux/types.h> |
| 18 | #include <linux/errno.h> |
| 19 | #include <linux/delay.h> |
| 20 | #include <linux/io.h> |
| 21 | #include <linux/err.h> |
| 22 | #include <linux/platform_device.h> |
| 23 | #include <linux/videodev2.h> |
| 24 | #include <uapi/linux/v4l2-mediabus.h> |
| 25 | #include <linux/clk.h> |
| 26 | #include <linux/clk-provider.h> |
| 27 | #include <linux/clkdev.h> |
| 28 | |
| 29 | #include "ipu-prv.h" |
| 30 | |
| 31 | struct ipu_csi { |
| 32 | void __iomem *base; |
| 33 | int id; |
| 34 | u32 module; |
| 35 | struct clk *clk_ipu; /* IPU bus clock */ |
| 36 | spinlock_t lock; |
| 37 | bool inuse; |
| 38 | struct ipu_soc *ipu; |
| 39 | }; |
| 40 | |
| 41 | /* CSI Register Offsets */ |
| 42 | #define CSI_SENS_CONF 0x0000 |
| 43 | #define CSI_SENS_FRM_SIZE 0x0004 |
| 44 | #define CSI_ACT_FRM_SIZE 0x0008 |
| 45 | #define CSI_OUT_FRM_CTRL 0x000c |
| 46 | #define CSI_TST_CTRL 0x0010 |
| 47 | #define CSI_CCIR_CODE_1 0x0014 |
| 48 | #define CSI_CCIR_CODE_2 0x0018 |
| 49 | #define CSI_CCIR_CODE_3 0x001c |
| 50 | #define CSI_MIPI_DI 0x0020 |
| 51 | #define CSI_SKIP 0x0024 |
| 52 | #define CSI_CPD_CTRL 0x0028 |
| 53 | #define CSI_CPD_RC(n) (0x002c + ((n)*4)) |
| 54 | #define CSI_CPD_RS(n) (0x004c + ((n)*4)) |
| 55 | #define CSI_CPD_GRC(n) (0x005c + ((n)*4)) |
| 56 | #define CSI_CPD_GRS(n) (0x007c + ((n)*4)) |
| 57 | #define CSI_CPD_GBC(n) (0x008c + ((n)*4)) |
| 58 | #define CSI_CPD_GBS(n) (0x00Ac + ((n)*4)) |
| 59 | #define CSI_CPD_BC(n) (0x00Bc + ((n)*4)) |
| 60 | #define CSI_CPD_BS(n) (0x00Dc + ((n)*4)) |
| 61 | #define CSI_CPD_OFFSET1 0x00ec |
| 62 | #define CSI_CPD_OFFSET2 0x00f0 |
| 63 | |
| 64 | /* CSI Register Fields */ |
| 65 | #define CSI_SENS_CONF_DATA_FMT_SHIFT 8 |
| 66 | #define CSI_SENS_CONF_DATA_FMT_MASK 0x00000700 |
| 67 | #define CSI_SENS_CONF_DATA_FMT_RGB_YUV444 0L |
| 68 | #define CSI_SENS_CONF_DATA_FMT_YUV422_YUYV 1L |
| 69 | #define CSI_SENS_CONF_DATA_FMT_YUV422_UYVY 2L |
| 70 | #define CSI_SENS_CONF_DATA_FMT_BAYER 3L |
| 71 | #define CSI_SENS_CONF_DATA_FMT_RGB565 4L |
| 72 | #define CSI_SENS_CONF_DATA_FMT_RGB555 5L |
| 73 | #define CSI_SENS_CONF_DATA_FMT_RGB444 6L |
| 74 | #define CSI_SENS_CONF_DATA_FMT_JPEG 7L |
| 75 | |
| 76 | #define CSI_SENS_CONF_VSYNC_POL_SHIFT 0 |
| 77 | #define CSI_SENS_CONF_HSYNC_POL_SHIFT 1 |
| 78 | #define CSI_SENS_CONF_DATA_POL_SHIFT 2 |
| 79 | #define CSI_SENS_CONF_PIX_CLK_POL_SHIFT 3 |
| 80 | #define CSI_SENS_CONF_SENS_PRTCL_MASK 0x00000070 |
| 81 | #define CSI_SENS_CONF_SENS_PRTCL_SHIFT 4 |
| 82 | #define CSI_SENS_CONF_PACK_TIGHT_SHIFT 7 |
| 83 | #define CSI_SENS_CONF_DATA_WIDTH_SHIFT 11 |
| 84 | #define CSI_SENS_CONF_EXT_VSYNC_SHIFT 15 |
| 85 | #define CSI_SENS_CONF_DIVRATIO_SHIFT 16 |
| 86 | |
| 87 | #define CSI_SENS_CONF_DIVRATIO_MASK 0x00ff0000 |
| 88 | #define CSI_SENS_CONF_DATA_DEST_SHIFT 24 |
| 89 | #define CSI_SENS_CONF_DATA_DEST_MASK 0x07000000 |
| 90 | #define CSI_SENS_CONF_JPEG8_EN_SHIFT 27 |
| 91 | #define CSI_SENS_CONF_JPEG_EN_SHIFT 28 |
| 92 | #define CSI_SENS_CONF_FORCE_EOF_SHIFT 29 |
| 93 | #define CSI_SENS_CONF_DATA_EN_POL_SHIFT 31 |
| 94 | |
| 95 | #define CSI_DATA_DEST_IC 2 |
| 96 | #define CSI_DATA_DEST_IDMAC 4 |
| 97 | |
| 98 | #define CSI_CCIR_ERR_DET_EN 0x01000000 |
| 99 | #define CSI_HORI_DOWNSIZE_EN 0x80000000 |
| 100 | #define CSI_VERT_DOWNSIZE_EN 0x40000000 |
| 101 | #define CSI_TEST_GEN_MODE_EN 0x01000000 |
| 102 | |
| 103 | #define CSI_HSC_MASK 0x1fff0000 |
| 104 | #define CSI_HSC_SHIFT 16 |
| 105 | #define CSI_VSC_MASK 0x00000fff |
| 106 | #define CSI_VSC_SHIFT 0 |
| 107 | |
| 108 | #define CSI_TEST_GEN_R_MASK 0x000000ff |
| 109 | #define CSI_TEST_GEN_R_SHIFT 0 |
| 110 | #define CSI_TEST_GEN_G_MASK 0x0000ff00 |
| 111 | #define CSI_TEST_GEN_G_SHIFT 8 |
| 112 | #define CSI_TEST_GEN_B_MASK 0x00ff0000 |
| 113 | #define CSI_TEST_GEN_B_SHIFT 16 |
| 114 | |
| 115 | #define CSI_MAX_RATIO_SKIP_SMFC_MASK 0x00000007 |
| 116 | #define CSI_MAX_RATIO_SKIP_SMFC_SHIFT 0 |
| 117 | #define CSI_SKIP_SMFC_MASK 0x000000f8 |
| 118 | #define CSI_SKIP_SMFC_SHIFT 3 |
| 119 | #define CSI_ID_2_SKIP_MASK 0x00000300 |
| 120 | #define CSI_ID_2_SKIP_SHIFT 8 |
| 121 | |
| 122 | #define CSI_COLOR_FIRST_ROW_MASK 0x00000002 |
| 123 | #define CSI_COLOR_FIRST_COMP_MASK 0x00000001 |
| 124 | |
| 125 | /* MIPI CSI-2 data types */ |
| 126 | #define MIPI_DT_YUV420 0x18 /* YYY.../UYVY.... */ |
| 127 | #define MIPI_DT_YUV420_LEGACY 0x1a /* UYY.../VYY... */ |
| 128 | #define MIPI_DT_YUV422 0x1e /* UYVY... */ |
| 129 | #define MIPI_DT_RGB444 0x20 |
| 130 | #define MIPI_DT_RGB555 0x21 |
| 131 | #define MIPI_DT_RGB565 0x22 |
| 132 | #define MIPI_DT_RGB666 0x23 |
| 133 | #define MIPI_DT_RGB888 0x24 |
| 134 | #define MIPI_DT_RAW6 0x28 |
| 135 | #define MIPI_DT_RAW7 0x29 |
| 136 | #define MIPI_DT_RAW8 0x2a |
| 137 | #define MIPI_DT_RAW10 0x2b |
| 138 | #define MIPI_DT_RAW12 0x2c |
| 139 | #define MIPI_DT_RAW14 0x2d |
| 140 | |
| 141 | /* |
| 142 | * Bitfield of CSI bus signal polarities and modes. |
| 143 | */ |
| 144 | struct ipu_csi_bus_config { |
| 145 | unsigned data_width:4; |
| 146 | unsigned clk_mode:3; |
| 147 | unsigned ext_vsync:1; |
| 148 | unsigned vsync_pol:1; |
| 149 | unsigned hsync_pol:1; |
| 150 | unsigned pixclk_pol:1; |
| 151 | unsigned data_pol:1; |
| 152 | unsigned sens_clksrc:1; |
| 153 | unsigned pack_tight:1; |
| 154 | unsigned force_eof:1; |
| 155 | unsigned data_en_pol:1; |
| 156 | |
| 157 | unsigned data_fmt; |
| 158 | unsigned mipi_dt; |
| 159 | }; |
| 160 | |
| 161 | /* |
| 162 | * Enumeration of CSI data bus widths. |
| 163 | */ |
| 164 | enum ipu_csi_data_width { |
| 165 | IPU_CSI_DATA_WIDTH_4 = 0, |
| 166 | IPU_CSI_DATA_WIDTH_8 = 1, |
| 167 | IPU_CSI_DATA_WIDTH_10 = 3, |
| 168 | IPU_CSI_DATA_WIDTH_12 = 5, |
| 169 | IPU_CSI_DATA_WIDTH_16 = 9, |
| 170 | }; |
| 171 | |
| 172 | /* |
| 173 | * Enumeration of CSI clock modes. |
| 174 | */ |
| 175 | enum ipu_csi_clk_mode { |
| 176 | IPU_CSI_CLK_MODE_GATED_CLK, |
| 177 | IPU_CSI_CLK_MODE_NONGATED_CLK, |
| 178 | IPU_CSI_CLK_MODE_CCIR656_PROGRESSIVE, |
| 179 | IPU_CSI_CLK_MODE_CCIR656_INTERLACED, |
| 180 | IPU_CSI_CLK_MODE_CCIR1120_PROGRESSIVE_DDR, |
| 181 | IPU_CSI_CLK_MODE_CCIR1120_PROGRESSIVE_SDR, |
| 182 | IPU_CSI_CLK_MODE_CCIR1120_INTERLACED_DDR, |
| 183 | IPU_CSI_CLK_MODE_CCIR1120_INTERLACED_SDR, |
| 184 | }; |
| 185 | |
| 186 | static inline u32 ipu_csi_read(struct ipu_csi *csi, unsigned offset) |
| 187 | { |
| 188 | return readl(csi->base + offset); |
| 189 | } |
| 190 | |
| 191 | static inline void ipu_csi_write(struct ipu_csi *csi, u32 value, |
| 192 | unsigned offset) |
| 193 | { |
| 194 | writel(value, csi->base + offset); |
| 195 | } |
| 196 | |
| 197 | /* |
| 198 | * Set mclk division ratio for generating test mode mclk. Only used |
| 199 | * for test generator. |
| 200 | */ |
| 201 | static int ipu_csi_set_testgen_mclk(struct ipu_csi *csi, u32 pixel_clk, |
| 202 | u32 ipu_clk) |
| 203 | { |
| 204 | u32 temp; |
Andrzej Hajda | 9b6d0d3 | 2015-09-21 15:33:48 +0200 | [diff] [blame] | 205 | int div_ratio; |
Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 206 | |
| 207 | div_ratio = (ipu_clk / pixel_clk) - 1; |
| 208 | |
| 209 | if (div_ratio > 0xFF || div_ratio < 0) { |
| 210 | dev_err(csi->ipu->dev, |
| 211 | "value of pixel_clk extends normal range\n"); |
| 212 | return -EINVAL; |
| 213 | } |
| 214 | |
| 215 | temp = ipu_csi_read(csi, CSI_SENS_CONF); |
| 216 | temp &= ~CSI_SENS_CONF_DIVRATIO_MASK; |
| 217 | ipu_csi_write(csi, temp | (div_ratio << CSI_SENS_CONF_DIVRATIO_SHIFT), |
| 218 | CSI_SENS_CONF); |
| 219 | |
| 220 | return 0; |
| 221 | } |
| 222 | |
| 223 | /* |
| 224 | * Find the CSI data format and data width for the given V4L2 media |
| 225 | * bus pixel format code. |
| 226 | */ |
| 227 | static int mbus_code_to_bus_cfg(struct ipu_csi_bus_config *cfg, u32 mbus_code) |
| 228 | { |
| 229 | switch (mbus_code) { |
Boris BREZILLON | 3e47608 | 2014-11-10 14:28:34 -0300 | [diff] [blame] | 230 | case MEDIA_BUS_FMT_BGR565_2X8_BE: |
| 231 | case MEDIA_BUS_FMT_BGR565_2X8_LE: |
| 232 | case MEDIA_BUS_FMT_RGB565_2X8_BE: |
| 233 | case MEDIA_BUS_FMT_RGB565_2X8_LE: |
Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 234 | cfg->data_fmt = CSI_SENS_CONF_DATA_FMT_RGB565; |
| 235 | cfg->mipi_dt = MIPI_DT_RGB565; |
| 236 | cfg->data_width = IPU_CSI_DATA_WIDTH_8; |
| 237 | break; |
Boris BREZILLON | 3e47608 | 2014-11-10 14:28:34 -0300 | [diff] [blame] | 238 | case MEDIA_BUS_FMT_RGB444_2X8_PADHI_BE: |
| 239 | case MEDIA_BUS_FMT_RGB444_2X8_PADHI_LE: |
Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 240 | cfg->data_fmt = CSI_SENS_CONF_DATA_FMT_RGB444; |
| 241 | cfg->mipi_dt = MIPI_DT_RGB444; |
| 242 | cfg->data_width = IPU_CSI_DATA_WIDTH_8; |
| 243 | break; |
Boris BREZILLON | 3e47608 | 2014-11-10 14:28:34 -0300 | [diff] [blame] | 244 | case MEDIA_BUS_FMT_RGB555_2X8_PADHI_BE: |
| 245 | case MEDIA_BUS_FMT_RGB555_2X8_PADHI_LE: |
Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 246 | cfg->data_fmt = CSI_SENS_CONF_DATA_FMT_RGB555; |
| 247 | cfg->mipi_dt = MIPI_DT_RGB555; |
| 248 | cfg->data_width = IPU_CSI_DATA_WIDTH_8; |
| 249 | break; |
Boris BREZILLON | 3e47608 | 2014-11-10 14:28:34 -0300 | [diff] [blame] | 250 | case MEDIA_BUS_FMT_UYVY8_2X8: |
Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 251 | cfg->data_fmt = CSI_SENS_CONF_DATA_FMT_YUV422_UYVY; |
| 252 | cfg->mipi_dt = MIPI_DT_YUV422; |
| 253 | cfg->data_width = IPU_CSI_DATA_WIDTH_8; |
| 254 | break; |
Boris BREZILLON | 3e47608 | 2014-11-10 14:28:34 -0300 | [diff] [blame] | 255 | case MEDIA_BUS_FMT_YUYV8_2X8: |
Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 256 | cfg->data_fmt = CSI_SENS_CONF_DATA_FMT_YUV422_YUYV; |
| 257 | cfg->mipi_dt = MIPI_DT_YUV422; |
| 258 | cfg->data_width = IPU_CSI_DATA_WIDTH_8; |
| 259 | break; |
Boris BREZILLON | 3e47608 | 2014-11-10 14:28:34 -0300 | [diff] [blame] | 260 | case MEDIA_BUS_FMT_UYVY8_1X16: |
Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 261 | cfg->data_fmt = CSI_SENS_CONF_DATA_FMT_YUV422_UYVY; |
| 262 | cfg->mipi_dt = MIPI_DT_YUV422; |
| 263 | cfg->data_width = IPU_CSI_DATA_WIDTH_16; |
| 264 | break; |
Boris BREZILLON | 3e47608 | 2014-11-10 14:28:34 -0300 | [diff] [blame] | 265 | case MEDIA_BUS_FMT_YUYV8_1X16: |
Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 266 | cfg->data_fmt = CSI_SENS_CONF_DATA_FMT_YUV422_YUYV; |
| 267 | cfg->mipi_dt = MIPI_DT_YUV422; |
| 268 | cfg->data_width = IPU_CSI_DATA_WIDTH_16; |
| 269 | break; |
Boris BREZILLON | 3e47608 | 2014-11-10 14:28:34 -0300 | [diff] [blame] | 270 | case MEDIA_BUS_FMT_SBGGR8_1X8: |
| 271 | case MEDIA_BUS_FMT_SGBRG8_1X8: |
| 272 | case MEDIA_BUS_FMT_SGRBG8_1X8: |
| 273 | case MEDIA_BUS_FMT_SRGGB8_1X8: |
Philippe De Muyter | 3e8b224 | 2015-09-18 15:41:55 +0200 | [diff] [blame] | 274 | case MEDIA_BUS_FMT_Y8_1X8: |
Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 275 | cfg->data_fmt = CSI_SENS_CONF_DATA_FMT_BAYER; |
| 276 | cfg->mipi_dt = MIPI_DT_RAW8; |
| 277 | cfg->data_width = IPU_CSI_DATA_WIDTH_8; |
| 278 | break; |
Boris BREZILLON | 3e47608 | 2014-11-10 14:28:34 -0300 | [diff] [blame] | 279 | case MEDIA_BUS_FMT_SBGGR10_DPCM8_1X8: |
| 280 | case MEDIA_BUS_FMT_SGBRG10_DPCM8_1X8: |
| 281 | case MEDIA_BUS_FMT_SGRBG10_DPCM8_1X8: |
| 282 | case MEDIA_BUS_FMT_SRGGB10_DPCM8_1X8: |
| 283 | case MEDIA_BUS_FMT_SBGGR10_2X8_PADHI_BE: |
| 284 | case MEDIA_BUS_FMT_SBGGR10_2X8_PADHI_LE: |
| 285 | case MEDIA_BUS_FMT_SBGGR10_2X8_PADLO_BE: |
| 286 | case MEDIA_BUS_FMT_SBGGR10_2X8_PADLO_LE: |
Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 287 | cfg->data_fmt = CSI_SENS_CONF_DATA_FMT_BAYER; |
| 288 | cfg->mipi_dt = MIPI_DT_RAW10; |
| 289 | cfg->data_width = IPU_CSI_DATA_WIDTH_8; |
| 290 | break; |
Boris BREZILLON | 3e47608 | 2014-11-10 14:28:34 -0300 | [diff] [blame] | 291 | case MEDIA_BUS_FMT_SBGGR10_1X10: |
| 292 | case MEDIA_BUS_FMT_SGBRG10_1X10: |
| 293 | case MEDIA_BUS_FMT_SGRBG10_1X10: |
| 294 | case MEDIA_BUS_FMT_SRGGB10_1X10: |
Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 295 | cfg->data_fmt = CSI_SENS_CONF_DATA_FMT_BAYER; |
| 296 | cfg->mipi_dt = MIPI_DT_RAW10; |
| 297 | cfg->data_width = IPU_CSI_DATA_WIDTH_10; |
| 298 | break; |
Boris BREZILLON | 3e47608 | 2014-11-10 14:28:34 -0300 | [diff] [blame] | 299 | case MEDIA_BUS_FMT_SBGGR12_1X12: |
| 300 | case MEDIA_BUS_FMT_SGBRG12_1X12: |
| 301 | case MEDIA_BUS_FMT_SGRBG12_1X12: |
| 302 | case MEDIA_BUS_FMT_SRGGB12_1X12: |
Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 303 | cfg->data_fmt = CSI_SENS_CONF_DATA_FMT_BAYER; |
| 304 | cfg->mipi_dt = MIPI_DT_RAW12; |
| 305 | cfg->data_width = IPU_CSI_DATA_WIDTH_12; |
| 306 | break; |
Boris BREZILLON | 3e47608 | 2014-11-10 14:28:34 -0300 | [diff] [blame] | 307 | case MEDIA_BUS_FMT_JPEG_1X8: |
Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 308 | /* TODO */ |
| 309 | cfg->data_fmt = CSI_SENS_CONF_DATA_FMT_JPEG; |
| 310 | cfg->mipi_dt = MIPI_DT_RAW8; |
| 311 | cfg->data_width = IPU_CSI_DATA_WIDTH_8; |
| 312 | break; |
| 313 | default: |
| 314 | return -EINVAL; |
| 315 | } |
| 316 | |
| 317 | return 0; |
| 318 | } |
| 319 | |
| 320 | /* |
| 321 | * Fill a CSI bus config struct from mbus_config and mbus_framefmt. |
| 322 | */ |
| 323 | static void fill_csi_bus_cfg(struct ipu_csi_bus_config *csicfg, |
| 324 | struct v4l2_mbus_config *mbus_cfg, |
| 325 | struct v4l2_mbus_framefmt *mbus_fmt) |
| 326 | { |
| 327 | memset(csicfg, 0, sizeof(*csicfg)); |
| 328 | |
| 329 | mbus_code_to_bus_cfg(csicfg, mbus_fmt->code); |
| 330 | |
| 331 | switch (mbus_cfg->type) { |
| 332 | case V4L2_MBUS_PARALLEL: |
| 333 | csicfg->ext_vsync = 1; |
| 334 | csicfg->vsync_pol = (mbus_cfg->flags & |
| 335 | V4L2_MBUS_VSYNC_ACTIVE_LOW) ? 1 : 0; |
| 336 | csicfg->hsync_pol = (mbus_cfg->flags & |
| 337 | V4L2_MBUS_HSYNC_ACTIVE_LOW) ? 1 : 0; |
| 338 | csicfg->pixclk_pol = (mbus_cfg->flags & |
| 339 | V4L2_MBUS_PCLK_SAMPLE_FALLING) ? 1 : 0; |
| 340 | csicfg->clk_mode = IPU_CSI_CLK_MODE_GATED_CLK; |
| 341 | break; |
| 342 | case V4L2_MBUS_BT656: |
| 343 | csicfg->ext_vsync = 0; |
| 344 | if (V4L2_FIELD_HAS_BOTH(mbus_fmt->field)) |
| 345 | csicfg->clk_mode = IPU_CSI_CLK_MODE_CCIR656_INTERLACED; |
| 346 | else |
| 347 | csicfg->clk_mode = IPU_CSI_CLK_MODE_CCIR656_PROGRESSIVE; |
| 348 | break; |
| 349 | case V4L2_MBUS_CSI2: |
| 350 | /* |
| 351 | * MIPI CSI-2 requires non gated clock mode, all other |
| 352 | * parameters are not applicable for MIPI CSI-2 bus. |
| 353 | */ |
| 354 | csicfg->clk_mode = IPU_CSI_CLK_MODE_NONGATED_CLK; |
| 355 | break; |
| 356 | default: |
| 357 | /* will never get here, keep compiler quiet */ |
| 358 | break; |
| 359 | } |
| 360 | } |
| 361 | |
| 362 | int ipu_csi_init_interface(struct ipu_csi *csi, |
| 363 | struct v4l2_mbus_config *mbus_cfg, |
| 364 | struct v4l2_mbus_framefmt *mbus_fmt) |
| 365 | { |
| 366 | struct ipu_csi_bus_config cfg; |
| 367 | unsigned long flags; |
Steve Longerbeam | aede45b | 2016-07-19 18:11:06 -0700 | [diff] [blame^] | 368 | u32 width, height, data = 0; |
Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 369 | |
| 370 | fill_csi_bus_cfg(&cfg, mbus_cfg, mbus_fmt); |
| 371 | |
Steve Longerbeam | aede45b | 2016-07-19 18:11:06 -0700 | [diff] [blame^] | 372 | /* set default sensor frame width and height */ |
| 373 | width = mbus_fmt->width; |
| 374 | height = mbus_fmt->height; |
| 375 | |
Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 376 | /* Set the CSI_SENS_CONF register remaining fields */ |
| 377 | data |= cfg.data_width << CSI_SENS_CONF_DATA_WIDTH_SHIFT | |
| 378 | cfg.data_fmt << CSI_SENS_CONF_DATA_FMT_SHIFT | |
| 379 | cfg.data_pol << CSI_SENS_CONF_DATA_POL_SHIFT | |
| 380 | cfg.vsync_pol << CSI_SENS_CONF_VSYNC_POL_SHIFT | |
| 381 | cfg.hsync_pol << CSI_SENS_CONF_HSYNC_POL_SHIFT | |
| 382 | cfg.pixclk_pol << CSI_SENS_CONF_PIX_CLK_POL_SHIFT | |
| 383 | cfg.ext_vsync << CSI_SENS_CONF_EXT_VSYNC_SHIFT | |
| 384 | cfg.clk_mode << CSI_SENS_CONF_SENS_PRTCL_SHIFT | |
| 385 | cfg.pack_tight << CSI_SENS_CONF_PACK_TIGHT_SHIFT | |
| 386 | cfg.force_eof << CSI_SENS_CONF_FORCE_EOF_SHIFT | |
| 387 | cfg.data_en_pol << CSI_SENS_CONF_DATA_EN_POL_SHIFT; |
| 388 | |
| 389 | spin_lock_irqsave(&csi->lock, flags); |
| 390 | |
| 391 | ipu_csi_write(csi, data, CSI_SENS_CONF); |
| 392 | |
Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 393 | /* Set CCIR registers */ |
| 394 | |
| 395 | switch (cfg.clk_mode) { |
| 396 | case IPU_CSI_CLK_MODE_CCIR656_PROGRESSIVE: |
| 397 | ipu_csi_write(csi, 0x40030, CSI_CCIR_CODE_1); |
| 398 | ipu_csi_write(csi, 0xFF0000, CSI_CCIR_CODE_3); |
| 399 | break; |
| 400 | case IPU_CSI_CLK_MODE_CCIR656_INTERLACED: |
| 401 | if (mbus_fmt->width == 720 && mbus_fmt->height == 576) { |
| 402 | /* |
| 403 | * PAL case |
| 404 | * |
| 405 | * Field0BlankEnd = 0x6, Field0BlankStart = 0x2, |
| 406 | * Field0ActiveEnd = 0x4, Field0ActiveStart = 0 |
| 407 | * Field1BlankEnd = 0x7, Field1BlankStart = 0x3, |
| 408 | * Field1ActiveEnd = 0x5, Field1ActiveStart = 0x1 |
| 409 | */ |
Steve Longerbeam | aede45b | 2016-07-19 18:11:06 -0700 | [diff] [blame^] | 410 | height = 625; /* framelines for PAL */ |
| 411 | |
Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 412 | ipu_csi_write(csi, 0x40596 | CSI_CCIR_ERR_DET_EN, |
| 413 | CSI_CCIR_CODE_1); |
| 414 | ipu_csi_write(csi, 0xD07DF, CSI_CCIR_CODE_2); |
| 415 | ipu_csi_write(csi, 0xFF0000, CSI_CCIR_CODE_3); |
Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 416 | } else if (mbus_fmt->width == 720 && mbus_fmt->height == 480) { |
| 417 | /* |
| 418 | * NTSC case |
| 419 | * |
| 420 | * Field0BlankEnd = 0x7, Field0BlankStart = 0x3, |
| 421 | * Field0ActiveEnd = 0x5, Field0ActiveStart = 0x1 |
| 422 | * Field1BlankEnd = 0x6, Field1BlankStart = 0x2, |
| 423 | * Field1ActiveEnd = 0x4, Field1ActiveStart = 0 |
| 424 | */ |
Steve Longerbeam | aede45b | 2016-07-19 18:11:06 -0700 | [diff] [blame^] | 425 | height = 525; /* framelines for NTSC */ |
| 426 | |
Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 427 | ipu_csi_write(csi, 0xD07DF | CSI_CCIR_ERR_DET_EN, |
| 428 | CSI_CCIR_CODE_1); |
| 429 | ipu_csi_write(csi, 0x40596, CSI_CCIR_CODE_2); |
| 430 | ipu_csi_write(csi, 0xFF0000, CSI_CCIR_CODE_3); |
| 431 | } else { |
| 432 | dev_err(csi->ipu->dev, |
| 433 | "Unsupported CCIR656 interlaced video mode\n"); |
| 434 | spin_unlock_irqrestore(&csi->lock, flags); |
| 435 | return -EINVAL; |
| 436 | } |
| 437 | break; |
| 438 | case IPU_CSI_CLK_MODE_CCIR1120_PROGRESSIVE_DDR: |
| 439 | case IPU_CSI_CLK_MODE_CCIR1120_PROGRESSIVE_SDR: |
| 440 | case IPU_CSI_CLK_MODE_CCIR1120_INTERLACED_DDR: |
| 441 | case IPU_CSI_CLK_MODE_CCIR1120_INTERLACED_SDR: |
| 442 | ipu_csi_write(csi, 0x40030 | CSI_CCIR_ERR_DET_EN, |
| 443 | CSI_CCIR_CODE_1); |
| 444 | ipu_csi_write(csi, 0xFF0000, CSI_CCIR_CODE_3); |
| 445 | break; |
| 446 | case IPU_CSI_CLK_MODE_GATED_CLK: |
| 447 | case IPU_CSI_CLK_MODE_NONGATED_CLK: |
| 448 | ipu_csi_write(csi, 0, CSI_CCIR_CODE_1); |
| 449 | break; |
| 450 | } |
| 451 | |
Steve Longerbeam | aede45b | 2016-07-19 18:11:06 -0700 | [diff] [blame^] | 452 | /* Setup sensor frame size */ |
| 453 | ipu_csi_write(csi, (width - 1) | ((height - 1) << 16), |
| 454 | CSI_SENS_FRM_SIZE); |
| 455 | |
Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 456 | dev_dbg(csi->ipu->dev, "CSI_SENS_CONF = 0x%08X\n", |
| 457 | ipu_csi_read(csi, CSI_SENS_CONF)); |
| 458 | dev_dbg(csi->ipu->dev, "CSI_ACT_FRM_SIZE = 0x%08X\n", |
| 459 | ipu_csi_read(csi, CSI_ACT_FRM_SIZE)); |
| 460 | |
| 461 | spin_unlock_irqrestore(&csi->lock, flags); |
| 462 | |
| 463 | return 0; |
| 464 | } |
| 465 | EXPORT_SYMBOL_GPL(ipu_csi_init_interface); |
| 466 | |
| 467 | bool ipu_csi_is_interlaced(struct ipu_csi *csi) |
| 468 | { |
| 469 | unsigned long flags; |
| 470 | u32 sensor_protocol; |
| 471 | |
| 472 | spin_lock_irqsave(&csi->lock, flags); |
| 473 | sensor_protocol = |
| 474 | (ipu_csi_read(csi, CSI_SENS_CONF) & |
| 475 | CSI_SENS_CONF_SENS_PRTCL_MASK) >> |
| 476 | CSI_SENS_CONF_SENS_PRTCL_SHIFT; |
| 477 | spin_unlock_irqrestore(&csi->lock, flags); |
| 478 | |
| 479 | switch (sensor_protocol) { |
| 480 | case IPU_CSI_CLK_MODE_GATED_CLK: |
| 481 | case IPU_CSI_CLK_MODE_NONGATED_CLK: |
| 482 | case IPU_CSI_CLK_MODE_CCIR656_PROGRESSIVE: |
| 483 | case IPU_CSI_CLK_MODE_CCIR1120_PROGRESSIVE_DDR: |
| 484 | case IPU_CSI_CLK_MODE_CCIR1120_PROGRESSIVE_SDR: |
| 485 | return false; |
| 486 | case IPU_CSI_CLK_MODE_CCIR656_INTERLACED: |
| 487 | case IPU_CSI_CLK_MODE_CCIR1120_INTERLACED_DDR: |
| 488 | case IPU_CSI_CLK_MODE_CCIR1120_INTERLACED_SDR: |
| 489 | return true; |
| 490 | default: |
| 491 | dev_err(csi->ipu->dev, |
| 492 | "CSI %d sensor protocol unsupported\n", csi->id); |
| 493 | return false; |
| 494 | } |
| 495 | } |
| 496 | EXPORT_SYMBOL_GPL(ipu_csi_is_interlaced); |
| 497 | |
| 498 | void ipu_csi_get_window(struct ipu_csi *csi, struct v4l2_rect *w) |
| 499 | { |
| 500 | unsigned long flags; |
| 501 | u32 reg; |
| 502 | |
| 503 | spin_lock_irqsave(&csi->lock, flags); |
| 504 | |
| 505 | reg = ipu_csi_read(csi, CSI_ACT_FRM_SIZE); |
| 506 | w->width = (reg & 0xFFFF) + 1; |
| 507 | w->height = (reg >> 16 & 0xFFFF) + 1; |
| 508 | |
| 509 | reg = ipu_csi_read(csi, CSI_OUT_FRM_CTRL); |
| 510 | w->left = (reg & CSI_HSC_MASK) >> CSI_HSC_SHIFT; |
| 511 | w->top = (reg & CSI_VSC_MASK) >> CSI_VSC_SHIFT; |
| 512 | |
| 513 | spin_unlock_irqrestore(&csi->lock, flags); |
| 514 | } |
| 515 | EXPORT_SYMBOL_GPL(ipu_csi_get_window); |
| 516 | |
| 517 | void ipu_csi_set_window(struct ipu_csi *csi, struct v4l2_rect *w) |
| 518 | { |
| 519 | unsigned long flags; |
| 520 | u32 reg; |
| 521 | |
| 522 | spin_lock_irqsave(&csi->lock, flags); |
| 523 | |
| 524 | ipu_csi_write(csi, (w->width - 1) | ((w->height - 1) << 16), |
| 525 | CSI_ACT_FRM_SIZE); |
| 526 | |
| 527 | reg = ipu_csi_read(csi, CSI_OUT_FRM_CTRL); |
| 528 | reg &= ~(CSI_HSC_MASK | CSI_VSC_MASK); |
| 529 | reg |= ((w->top << CSI_VSC_SHIFT) | (w->left << CSI_HSC_SHIFT)); |
| 530 | ipu_csi_write(csi, reg, CSI_OUT_FRM_CTRL); |
| 531 | |
| 532 | spin_unlock_irqrestore(&csi->lock, flags); |
| 533 | } |
| 534 | EXPORT_SYMBOL_GPL(ipu_csi_set_window); |
| 535 | |
| 536 | void ipu_csi_set_test_generator(struct ipu_csi *csi, bool active, |
| 537 | u32 r_value, u32 g_value, u32 b_value, |
| 538 | u32 pix_clk) |
| 539 | { |
| 540 | unsigned long flags; |
| 541 | u32 ipu_clk = clk_get_rate(csi->clk_ipu); |
| 542 | u32 temp; |
| 543 | |
| 544 | spin_lock_irqsave(&csi->lock, flags); |
| 545 | |
| 546 | temp = ipu_csi_read(csi, CSI_TST_CTRL); |
| 547 | |
Fabio Estevam | 6dd0d0e | 2015-05-09 13:08:25 -0300 | [diff] [blame] | 548 | if (!active) { |
Steve Longerbeam | 2ffd48f | 2014-08-19 10:52:40 -0700 | [diff] [blame] | 549 | temp &= ~CSI_TEST_GEN_MODE_EN; |
| 550 | ipu_csi_write(csi, temp, CSI_TST_CTRL); |
| 551 | } else { |
| 552 | /* Set sensb_mclk div_ratio */ |
| 553 | ipu_csi_set_testgen_mclk(csi, pix_clk, ipu_clk); |
| 554 | |
| 555 | temp &= ~(CSI_TEST_GEN_R_MASK | CSI_TEST_GEN_G_MASK | |
| 556 | CSI_TEST_GEN_B_MASK); |
| 557 | temp |= CSI_TEST_GEN_MODE_EN; |
| 558 | temp |= (r_value << CSI_TEST_GEN_R_SHIFT) | |
| 559 | (g_value << CSI_TEST_GEN_G_SHIFT) | |
| 560 | (b_value << CSI_TEST_GEN_B_SHIFT); |
| 561 | ipu_csi_write(csi, temp, CSI_TST_CTRL); |
| 562 | } |
| 563 | |
| 564 | spin_unlock_irqrestore(&csi->lock, flags); |
| 565 | } |
| 566 | EXPORT_SYMBOL_GPL(ipu_csi_set_test_generator); |
| 567 | |
| 568 | int ipu_csi_set_mipi_datatype(struct ipu_csi *csi, u32 vc, |
| 569 | struct v4l2_mbus_framefmt *mbus_fmt) |
| 570 | { |
| 571 | struct ipu_csi_bus_config cfg; |
| 572 | unsigned long flags; |
| 573 | u32 temp; |
| 574 | |
| 575 | if (vc > 3) |
| 576 | return -EINVAL; |
| 577 | |
| 578 | mbus_code_to_bus_cfg(&cfg, mbus_fmt->code); |
| 579 | |
| 580 | spin_lock_irqsave(&csi->lock, flags); |
| 581 | |
| 582 | temp = ipu_csi_read(csi, CSI_MIPI_DI); |
| 583 | temp &= ~(0xff << (vc * 8)); |
| 584 | temp |= (cfg.mipi_dt << (vc * 8)); |
| 585 | ipu_csi_write(csi, temp, CSI_MIPI_DI); |
| 586 | |
| 587 | spin_unlock_irqrestore(&csi->lock, flags); |
| 588 | |
| 589 | return 0; |
| 590 | } |
| 591 | EXPORT_SYMBOL_GPL(ipu_csi_set_mipi_datatype); |
| 592 | |
| 593 | int ipu_csi_set_skip_smfc(struct ipu_csi *csi, u32 skip, |
| 594 | u32 max_ratio, u32 id) |
| 595 | { |
| 596 | unsigned long flags; |
| 597 | u32 temp; |
| 598 | |
| 599 | if (max_ratio > 5 || id > 3) |
| 600 | return -EINVAL; |
| 601 | |
| 602 | spin_lock_irqsave(&csi->lock, flags); |
| 603 | |
| 604 | temp = ipu_csi_read(csi, CSI_SKIP); |
| 605 | temp &= ~(CSI_MAX_RATIO_SKIP_SMFC_MASK | CSI_ID_2_SKIP_MASK | |
| 606 | CSI_SKIP_SMFC_MASK); |
| 607 | temp |= (max_ratio << CSI_MAX_RATIO_SKIP_SMFC_SHIFT) | |
| 608 | (id << CSI_ID_2_SKIP_SHIFT) | |
| 609 | (skip << CSI_SKIP_SMFC_SHIFT); |
| 610 | ipu_csi_write(csi, temp, CSI_SKIP); |
| 611 | |
| 612 | spin_unlock_irqrestore(&csi->lock, flags); |
| 613 | |
| 614 | return 0; |
| 615 | } |
| 616 | EXPORT_SYMBOL_GPL(ipu_csi_set_skip_smfc); |
| 617 | |
| 618 | int ipu_csi_set_dest(struct ipu_csi *csi, enum ipu_csi_dest csi_dest) |
| 619 | { |
| 620 | unsigned long flags; |
| 621 | u32 csi_sens_conf, dest; |
| 622 | |
| 623 | if (csi_dest == IPU_CSI_DEST_IDMAC) |
| 624 | dest = CSI_DATA_DEST_IDMAC; |
| 625 | else |
| 626 | dest = CSI_DATA_DEST_IC; /* IC or VDIC */ |
| 627 | |
| 628 | spin_lock_irqsave(&csi->lock, flags); |
| 629 | |
| 630 | csi_sens_conf = ipu_csi_read(csi, CSI_SENS_CONF); |
| 631 | csi_sens_conf &= ~CSI_SENS_CONF_DATA_DEST_MASK; |
| 632 | csi_sens_conf |= (dest << CSI_SENS_CONF_DATA_DEST_SHIFT); |
| 633 | ipu_csi_write(csi, csi_sens_conf, CSI_SENS_CONF); |
| 634 | |
| 635 | spin_unlock_irqrestore(&csi->lock, flags); |
| 636 | |
| 637 | return 0; |
| 638 | } |
| 639 | EXPORT_SYMBOL_GPL(ipu_csi_set_dest); |
| 640 | |
| 641 | int ipu_csi_enable(struct ipu_csi *csi) |
| 642 | { |
| 643 | ipu_module_enable(csi->ipu, csi->module); |
| 644 | |
| 645 | return 0; |
| 646 | } |
| 647 | EXPORT_SYMBOL_GPL(ipu_csi_enable); |
| 648 | |
| 649 | int ipu_csi_disable(struct ipu_csi *csi) |
| 650 | { |
| 651 | ipu_module_disable(csi->ipu, csi->module); |
| 652 | |
| 653 | return 0; |
| 654 | } |
| 655 | EXPORT_SYMBOL_GPL(ipu_csi_disable); |
| 656 | |
| 657 | struct ipu_csi *ipu_csi_get(struct ipu_soc *ipu, int id) |
| 658 | { |
| 659 | unsigned long flags; |
| 660 | struct ipu_csi *csi, *ret; |
| 661 | |
| 662 | if (id > 1) |
| 663 | return ERR_PTR(-EINVAL); |
| 664 | |
| 665 | csi = ipu->csi_priv[id]; |
| 666 | ret = csi; |
| 667 | |
| 668 | spin_lock_irqsave(&csi->lock, flags); |
| 669 | |
| 670 | if (csi->inuse) { |
| 671 | ret = ERR_PTR(-EBUSY); |
| 672 | goto unlock; |
| 673 | } |
| 674 | |
| 675 | csi->inuse = true; |
| 676 | unlock: |
| 677 | spin_unlock_irqrestore(&csi->lock, flags); |
| 678 | return ret; |
| 679 | } |
| 680 | EXPORT_SYMBOL_GPL(ipu_csi_get); |
| 681 | |
| 682 | void ipu_csi_put(struct ipu_csi *csi) |
| 683 | { |
| 684 | unsigned long flags; |
| 685 | |
| 686 | spin_lock_irqsave(&csi->lock, flags); |
| 687 | csi->inuse = false; |
| 688 | spin_unlock_irqrestore(&csi->lock, flags); |
| 689 | } |
| 690 | EXPORT_SYMBOL_GPL(ipu_csi_put); |
| 691 | |
| 692 | int ipu_csi_init(struct ipu_soc *ipu, struct device *dev, int id, |
| 693 | unsigned long base, u32 module, struct clk *clk_ipu) |
| 694 | { |
| 695 | struct ipu_csi *csi; |
| 696 | |
| 697 | if (id > 1) |
| 698 | return -ENODEV; |
| 699 | |
| 700 | csi = devm_kzalloc(dev, sizeof(*csi), GFP_KERNEL); |
| 701 | if (!csi) |
| 702 | return -ENOMEM; |
| 703 | |
| 704 | ipu->csi_priv[id] = csi; |
| 705 | |
| 706 | spin_lock_init(&csi->lock); |
| 707 | csi->module = module; |
| 708 | csi->id = id; |
| 709 | csi->clk_ipu = clk_ipu; |
| 710 | csi->base = devm_ioremap(dev, base, PAGE_SIZE); |
| 711 | if (!csi->base) |
| 712 | return -ENOMEM; |
| 713 | |
| 714 | dev_dbg(dev, "CSI%d base: 0x%08lx remapped to %p\n", |
| 715 | id, base, csi->base); |
| 716 | csi->ipu = ipu; |
| 717 | |
| 718 | return 0; |
| 719 | } |
| 720 | |
| 721 | void ipu_csi_exit(struct ipu_soc *ipu, int id) |
| 722 | { |
| 723 | } |
| 724 | |
| 725 | void ipu_csi_dump(struct ipu_csi *csi) |
| 726 | { |
| 727 | dev_dbg(csi->ipu->dev, "CSI_SENS_CONF: %08x\n", |
| 728 | ipu_csi_read(csi, CSI_SENS_CONF)); |
| 729 | dev_dbg(csi->ipu->dev, "CSI_SENS_FRM_SIZE: %08x\n", |
| 730 | ipu_csi_read(csi, CSI_SENS_FRM_SIZE)); |
| 731 | dev_dbg(csi->ipu->dev, "CSI_ACT_FRM_SIZE: %08x\n", |
| 732 | ipu_csi_read(csi, CSI_ACT_FRM_SIZE)); |
| 733 | dev_dbg(csi->ipu->dev, "CSI_OUT_FRM_CTRL: %08x\n", |
| 734 | ipu_csi_read(csi, CSI_OUT_FRM_CTRL)); |
| 735 | dev_dbg(csi->ipu->dev, "CSI_TST_CTRL: %08x\n", |
| 736 | ipu_csi_read(csi, CSI_TST_CTRL)); |
| 737 | dev_dbg(csi->ipu->dev, "CSI_CCIR_CODE_1: %08x\n", |
| 738 | ipu_csi_read(csi, CSI_CCIR_CODE_1)); |
| 739 | dev_dbg(csi->ipu->dev, "CSI_CCIR_CODE_2: %08x\n", |
| 740 | ipu_csi_read(csi, CSI_CCIR_CODE_2)); |
| 741 | dev_dbg(csi->ipu->dev, "CSI_CCIR_CODE_3: %08x\n", |
| 742 | ipu_csi_read(csi, CSI_CCIR_CODE_3)); |
| 743 | dev_dbg(csi->ipu->dev, "CSI_MIPI_DI: %08x\n", |
| 744 | ipu_csi_read(csi, CSI_MIPI_DI)); |
| 745 | dev_dbg(csi->ipu->dev, "CSI_SKIP: %08x\n", |
| 746 | ipu_csi_read(csi, CSI_SKIP)); |
| 747 | } |
| 748 | EXPORT_SYMBOL_GPL(ipu_csi_dump); |