|
@@ -13,10 +13,11 @@
|
|
|
#include <linux/slab.h>
|
|
|
#include <linux/videodev2.h>
|
|
|
|
|
|
-#include <media/v4l2-subdev.h>
|
|
|
-#include <media/v4l2-chip-ident.h>
|
|
|
+#include <media/rj54n1cb0c.h>
|
|
|
#include <media/soc_camera.h>
|
|
|
#include <media/soc_mediabus.h>
|
|
|
+#include <media/v4l2-subdev.h>
|
|
|
+#include <media/v4l2-chip-ident.h>
|
|
|
|
|
|
#define RJ54N1_DEV_CODE 0x0400
|
|
|
#define RJ54N1_DEV_CODE2 0x0401
|
|
@@ -39,6 +40,7 @@
|
|
|
#define RJ54N1_H_OBEN_OFS 0x0413
|
|
|
#define RJ54N1_V_OBEN_OFS 0x0414
|
|
|
#define RJ54N1_RESIZE_CONTROL 0x0415
|
|
|
+#define RJ54N1_STILL_CONTROL 0x0417
|
|
|
#define RJ54N1_INC_USE_SEL_H 0x0425
|
|
|
#define RJ54N1_INC_USE_SEL_L 0x0426
|
|
|
#define RJ54N1_MIRROR_STILL_MODE 0x0427
|
|
@@ -50,10 +52,21 @@
|
|
|
#define RJ54N1_RA_SEL_UL 0x0530
|
|
|
#define RJ54N1_BYTE_SWAP 0x0531
|
|
|
#define RJ54N1_OUT_SIGPO 0x053b
|
|
|
+#define RJ54N1_WB_SEL_WEIGHT_I 0x054e
|
|
|
+#define RJ54N1_BIT8_WB 0x0569
|
|
|
+#define RJ54N1_HCAPS_WB 0x056a
|
|
|
+#define RJ54N1_VCAPS_WB 0x056b
|
|
|
+#define RJ54N1_HCAPE_WB 0x056c
|
|
|
+#define RJ54N1_VCAPE_WB 0x056d
|
|
|
+#define RJ54N1_EXPOSURE_CONTROL 0x058c
|
|
|
#define RJ54N1_FRAME_LENGTH_S_H 0x0595
|
|
|
#define RJ54N1_FRAME_LENGTH_S_L 0x0596
|
|
|
#define RJ54N1_FRAME_LENGTH_P_H 0x0597
|
|
|
#define RJ54N1_FRAME_LENGTH_P_L 0x0598
|
|
|
+#define RJ54N1_PEAK_H 0x05b7
|
|
|
+#define RJ54N1_PEAK_50 0x05b8
|
|
|
+#define RJ54N1_PEAK_60 0x05b9
|
|
|
+#define RJ54N1_PEAK_DIFF 0x05ba
|
|
|
#define RJ54N1_IOC 0x05ef
|
|
|
#define RJ54N1_TG_BYPASS 0x0700
|
|
|
#define RJ54N1_PLL_L 0x0701
|
|
@@ -69,6 +82,7 @@
|
|
|
#define RJ54N1_OCLK_SEL_EN 0x0713
|
|
|
#define RJ54N1_CLK_RST 0x0717
|
|
|
#define RJ54N1_RESET_STANDBY 0x0718
|
|
|
+#define RJ54N1_FWFLG 0x07fe
|
|
|
|
|
|
#define E_EXCLK (1 << 7)
|
|
|
#define SOFT_STDBY (1 << 4)
|
|
@@ -79,11 +93,18 @@
|
|
|
#define RESIZE_HOLD_SEL (1 << 2)
|
|
|
#define RESIZE_GO (1 << 1)
|
|
|
|
|
|
+/*
|
|
|
+ * When cropping, the camera automatically centers the cropped region, there
|
|
|
+ * doesn't seem to be a way to specify an explicit location of the rectangle.
|
|
|
+ */
|
|
|
#define RJ54N1_COLUMN_SKIP 0
|
|
|
#define RJ54N1_ROW_SKIP 0
|
|
|
#define RJ54N1_MAX_WIDTH 1600
|
|
|
#define RJ54N1_MAX_HEIGHT 1200
|
|
|
|
|
|
+#define PLL_L 2
|
|
|
+#define PLL_N 0x31
|
|
|
+
|
|
|
/* I2C addresses: 0x50, 0x51, 0x60, 0x61 */
|
|
|
|
|
|
/* RJ54N1CB0C has only one fixed colorspace per pixelcode */
|
|
@@ -118,7 +139,7 @@ static const struct rj54n1_datafmt rj54n1_colour_fmts[] = {
|
|
|
};
|
|
|
|
|
|
struct rj54n1_clock_div {
|
|
|
- u8 ratio_tg;
|
|
|
+ u8 ratio_tg; /* can be 0 or an odd number */
|
|
|
u8 ratio_t;
|
|
|
u8 ratio_r;
|
|
|
u8 ratio_op;
|
|
@@ -127,12 +148,14 @@ struct rj54n1_clock_div {
|
|
|
|
|
|
struct rj54n1 {
|
|
|
struct v4l2_subdev subdev;
|
|
|
+ struct rj54n1_clock_div clk_div;
|
|
|
const struct rj54n1_datafmt *fmt;
|
|
|
struct v4l2_rect rect; /* Sensor window */
|
|
|
+ unsigned int tgclk_mhz;
|
|
|
+ bool auto_wb;
|
|
|
unsigned short width; /* Output window */
|
|
|
unsigned short height;
|
|
|
unsigned short resize; /* Sensor * 1024 / resize = Output */
|
|
|
- struct rj54n1_clock_div clk_div;
|
|
|
unsigned short scale;
|
|
|
u8 bank;
|
|
|
};
|
|
@@ -189,7 +212,7 @@ const static struct rj54n1_reg_val bank_7[] = {
|
|
|
{0x714, 0xff},
|
|
|
{0x715, 0xff},
|
|
|
{0x716, 0x1f},
|
|
|
- {0x7FE, 0x02},
|
|
|
+ {0x7FE, 2},
|
|
|
};
|
|
|
|
|
|
const static struct rj54n1_reg_val bank_8[] = {
|
|
@@ -377,7 +400,7 @@ const static struct rj54n1_reg_val bank_8[] = {
|
|
|
{0x8BB, 0x00},
|
|
|
{0x8BC, 0xFF},
|
|
|
{0x8BD, 0x00},
|
|
|
- {0x8FE, 0x02},
|
|
|
+ {0x8FE, 2},
|
|
|
};
|
|
|
|
|
|
const static struct rj54n1_reg_val bank_10[] = {
|
|
@@ -470,8 +493,10 @@ static int rj54n1_enum_fmt(struct v4l2_subdev *sd, int index,
|
|
|
|
|
|
static int rj54n1_s_stream(struct v4l2_subdev *sd, int enable)
|
|
|
{
|
|
|
- /* TODO: start / stop streaming */
|
|
|
- return 0;
|
|
|
+ struct i2c_client *client = sd->priv;
|
|
|
+
|
|
|
+ /* Switch between preview and still shot modes */
|
|
|
+ return reg_set(client, RJ54N1_STILL_CONTROL, (!enable) << 7, 0x80);
|
|
|
}
|
|
|
|
|
|
static int rj54n1_set_bus_param(struct soc_camera_device *icd,
|
|
@@ -530,6 +555,44 @@ static int rj54n1_commit(struct i2c_client *client)
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
+static int rj54n1_sensor_scale(struct v4l2_subdev *sd, u32 *in_w, u32 *in_h,
|
|
|
+ u32 *out_w, u32 *out_h);
|
|
|
+
|
|
|
+static int rj54n1_s_crop(struct v4l2_subdev *sd, struct v4l2_crop *a)
|
|
|
+{
|
|
|
+ struct i2c_client *client = sd->priv;
|
|
|
+ struct rj54n1 *rj54n1 = to_rj54n1(client);
|
|
|
+ struct v4l2_rect *rect = &a->c;
|
|
|
+ unsigned int dummy, output_w, output_h,
|
|
|
+ input_w = rect->width, input_h = rect->height;
|
|
|
+ int ret;
|
|
|
+
|
|
|
+ /* arbitrary minimum width and height, edges unimportant */
|
|
|
+ soc_camera_limit_side(&dummy, &input_w,
|
|
|
+ RJ54N1_COLUMN_SKIP, 8, RJ54N1_MAX_WIDTH);
|
|
|
+
|
|
|
+ soc_camera_limit_side(&dummy, &input_h,
|
|
|
+ RJ54N1_ROW_SKIP, 8, RJ54N1_MAX_HEIGHT);
|
|
|
+
|
|
|
+ output_w = (input_w * 1024 + rj54n1->resize / 2) / rj54n1->resize;
|
|
|
+ output_h = (input_h * 1024 + rj54n1->resize / 2) / rj54n1->resize;
|
|
|
+
|
|
|
+ dev_dbg(&client->dev, "Scaling for %ux%u : %u = %ux%u\n",
|
|
|
+ input_w, input_h, rj54n1->resize, output_w, output_h);
|
|
|
+
|
|
|
+ ret = rj54n1_sensor_scale(sd, &input_w, &input_h, &output_w, &output_h);
|
|
|
+ if (ret < 0)
|
|
|
+ return ret;
|
|
|
+
|
|
|
+ rj54n1->width = output_w;
|
|
|
+ rj54n1->height = output_h;
|
|
|
+ rj54n1->resize = ret;
|
|
|
+ rj54n1->rect.width = input_w;
|
|
|
+ rj54n1->rect.height = input_h;
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
static int rj54n1_g_crop(struct v4l2_subdev *sd, struct v4l2_crop *a)
|
|
|
{
|
|
|
struct i2c_client *client = sd->priv;
|
|
@@ -579,11 +642,44 @@ static int rj54n1_sensor_scale(struct v4l2_subdev *sd, u32 *in_w, u32 *in_h,
|
|
|
u32 *out_w, u32 *out_h)
|
|
|
{
|
|
|
struct i2c_client *client = sd->priv;
|
|
|
+ struct rj54n1 *rj54n1 = to_rj54n1(client);
|
|
|
unsigned int skip, resize, input_w = *in_w, input_h = *in_h,
|
|
|
output_w = *out_w, output_h = *out_h;
|
|
|
- u16 inc_sel;
|
|
|
+ u16 inc_sel, wb_bit8, wb_left, wb_right, wb_top, wb_bottom;
|
|
|
+ unsigned int peak, peak_50, peak_60;
|
|
|
int ret;
|
|
|
|
|
|
+ /*
|
|
|
+ * We have a problem with crops, where the window is larger than 512x384
|
|
|
+ * and output window is larger than a half of the input one. In this
|
|
|
+ * case we have to either reduce the input window to equal or below
|
|
|
+ * 512x384 or the output window to equal or below 1/2 of the input.
|
|
|
+ */
|
|
|
+ if (output_w > max(512U, input_w / 2)) {
|
|
|
+ if (2 * output_w > RJ54N1_MAX_WIDTH) {
|
|
|
+ input_w = RJ54N1_MAX_WIDTH;
|
|
|
+ output_w = RJ54N1_MAX_WIDTH / 2;
|
|
|
+ } else {
|
|
|
+ input_w = output_w * 2;
|
|
|
+ }
|
|
|
+
|
|
|
+ dev_dbg(&client->dev, "Adjusted output width: in %u, out %u\n",
|
|
|
+ input_w, output_w);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (output_h > max(384U, input_h / 2)) {
|
|
|
+ if (2 * output_h > RJ54N1_MAX_HEIGHT) {
|
|
|
+ input_h = RJ54N1_MAX_HEIGHT;
|
|
|
+ output_h = RJ54N1_MAX_HEIGHT / 2;
|
|
|
+ } else {
|
|
|
+ input_h = output_h * 2;
|
|
|
+ }
|
|
|
+
|
|
|
+ dev_dbg(&client->dev, "Adjusted output height: in %u, out %u\n",
|
|
|
+ input_h, output_h);
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Idea: use the read mode for snapshots, handle separate geometries */
|
|
|
ret = rj54n1_set_rect(client, RJ54N1_X_OUTPUT_SIZE_S_L,
|
|
|
RJ54N1_Y_OUTPUT_SIZE_S_L,
|
|
|
RJ54N1_XY_OUTPUT_SIZE_S_H, output_w, output_h);
|
|
@@ -595,17 +691,27 @@ static int rj54n1_sensor_scale(struct v4l2_subdev *sd, u32 *in_w, u32 *in_h,
|
|
|
if (ret < 0)
|
|
|
return ret;
|
|
|
|
|
|
- if (output_w > input_w || output_h > input_h) {
|
|
|
+ if (output_w > input_w && output_h > input_h) {
|
|
|
input_w = output_w;
|
|
|
input_h = output_h;
|
|
|
|
|
|
resize = 1024;
|
|
|
} else {
|
|
|
unsigned int resize_x, resize_y;
|
|
|
- resize_x = input_w * 1024 / output_w;
|
|
|
- resize_y = input_h * 1024 / output_h;
|
|
|
-
|
|
|
- resize = min(resize_x, resize_y);
|
|
|
+ resize_x = (input_w * 1024 + output_w / 2) / output_w;
|
|
|
+ resize_y = (input_h * 1024 + output_h / 2) / output_h;
|
|
|
+
|
|
|
+ /* We want max(resize_x, resize_y), check if it still fits */
|
|
|
+ if (resize_x > resize_y &&
|
|
|
+ (output_h * resize_x + 512) / 1024 > RJ54N1_MAX_HEIGHT)
|
|
|
+ resize = (RJ54N1_MAX_HEIGHT * 1024 + output_h / 2) /
|
|
|
+ output_h;
|
|
|
+ else if (resize_y > resize_x &&
|
|
|
+ (output_w * resize_y + 512) / 1024 > RJ54N1_MAX_WIDTH)
|
|
|
+ resize = (RJ54N1_MAX_WIDTH * 1024 + output_w / 2) /
|
|
|
+ output_w;
|
|
|
+ else
|
|
|
+ resize = max(resize_x, resize_y);
|
|
|
|
|
|
/* Prohibited value ranges */
|
|
|
switch (resize) {
|
|
@@ -618,12 +724,9 @@ static int rj54n1_sensor_scale(struct v4l2_subdev *sd, u32 *in_w, u32 *in_h,
|
|
|
case 8160 ... 8191:
|
|
|
resize = 8159;
|
|
|
break;
|
|
|
- case 16320 ... 16383:
|
|
|
+ case 16320 ... 16384:
|
|
|
resize = 16319;
|
|
|
}
|
|
|
-
|
|
|
- input_w = output_w * resize / 1024;
|
|
|
- input_h = output_h * resize / 1024;
|
|
|
}
|
|
|
|
|
|
/* Set scaling */
|
|
@@ -636,9 +739,18 @@ static int rj54n1_sensor_scale(struct v4l2_subdev *sd, u32 *in_w, u32 *in_h,
|
|
|
|
|
|
/*
|
|
|
* Configure a skipping bitmask. The sensor will select a skipping value
|
|
|
- * among set bits automatically.
|
|
|
+ * among set bits automatically. This is very unclear in the datasheet
|
|
|
+ * too. I was told, in this register one enables all skipping values,
|
|
|
+ * that are required for a specific resize, and the camera selects
|
|
|
+ * automatically, which ones to use. But it is unclear how to identify,
|
|
|
+ * which cropping values are needed. Secondly, why don't we just set all
|
|
|
+ * bits and let the camera choose? Would it increase processing time and
|
|
|
+ * reduce the framerate? Using 0xfffc for INC_USE_SEL doesn't seem to
|
|
|
+ * improve the image quality or stability for larger frames (see comment
|
|
|
+ * above), but I didn't check the framerate.
|
|
|
*/
|
|
|
skip = min(resize / 1024, (unsigned)15);
|
|
|
+
|
|
|
inc_sel = 1 << skip;
|
|
|
|
|
|
if (inc_sel <= 2)
|
|
@@ -650,6 +762,43 @@ static int rj54n1_sensor_scale(struct v4l2_subdev *sd, u32 *in_w, u32 *in_h,
|
|
|
if (!ret)
|
|
|
ret = reg_write(client, RJ54N1_INC_USE_SEL_H, inc_sel >> 8);
|
|
|
|
|
|
+ if (!rj54n1->auto_wb) {
|
|
|
+ /* Auto white balance window */
|
|
|
+ wb_left = output_w / 16;
|
|
|
+ wb_right = (3 * output_w / 4 - 3) / 4;
|
|
|
+ wb_top = output_h / 16;
|
|
|
+ wb_bottom = (3 * output_h / 4 - 3) / 4;
|
|
|
+ wb_bit8 = ((wb_left >> 2) & 0x40) | ((wb_top >> 4) & 0x10) |
|
|
|
+ ((wb_right >> 6) & 4) | ((wb_bottom >> 8) & 1);
|
|
|
+
|
|
|
+ if (!ret)
|
|
|
+ ret = reg_write(client, RJ54N1_BIT8_WB, wb_bit8);
|
|
|
+ if (!ret)
|
|
|
+ ret = reg_write(client, RJ54N1_HCAPS_WB, wb_left);
|
|
|
+ if (!ret)
|
|
|
+ ret = reg_write(client, RJ54N1_VCAPS_WB, wb_top);
|
|
|
+ if (!ret)
|
|
|
+ ret = reg_write(client, RJ54N1_HCAPE_WB, wb_right);
|
|
|
+ if (!ret)
|
|
|
+ ret = reg_write(client, RJ54N1_VCAPE_WB, wb_bottom);
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Antiflicker */
|
|
|
+ peak = 12 * RJ54N1_MAX_WIDTH * (1 << 14) * resize / rj54n1->tgclk_mhz /
|
|
|
+ 10000;
|
|
|
+ peak_50 = peak / 6;
|
|
|
+ peak_60 = peak / 5;
|
|
|
+
|
|
|
+ if (!ret)
|
|
|
+ ret = reg_write(client, RJ54N1_PEAK_H,
|
|
|
+ ((peak_50 >> 4) & 0xf0) | (peak_60 >> 8));
|
|
|
+ if (!ret)
|
|
|
+ ret = reg_write(client, RJ54N1_PEAK_50, peak_50);
|
|
|
+ if (!ret)
|
|
|
+ ret = reg_write(client, RJ54N1_PEAK_60, peak_60);
|
|
|
+ if (!ret)
|
|
|
+ ret = reg_write(client, RJ54N1_PEAK_DIFF, peak / 150);
|
|
|
+
|
|
|
/* Start resizing */
|
|
|
if (!ret)
|
|
|
ret = reg_write(client, RJ54N1_RESIZE_CONTROL,
|
|
@@ -658,8 +807,6 @@ static int rj54n1_sensor_scale(struct v4l2_subdev *sd, u32 *in_w, u32 *in_h,
|
|
|
if (ret < 0)
|
|
|
return ret;
|
|
|
|
|
|
- dev_dbg(&client->dev, "resize %u, skip %u\n", resize, skip);
|
|
|
-
|
|
|
/* Constant taken from manufacturer's example */
|
|
|
msleep(230);
|
|
|
|
|
@@ -667,11 +814,14 @@ static int rj54n1_sensor_scale(struct v4l2_subdev *sd, u32 *in_w, u32 *in_h,
|
|
|
if (ret < 0)
|
|
|
return ret;
|
|
|
|
|
|
- *in_w = input_w;
|
|
|
- *in_h = input_h;
|
|
|
+ *in_w = (output_w * resize + 512) / 1024;
|
|
|
+ *in_h = (output_h * resize + 512) / 1024;
|
|
|
*out_w = output_w;
|
|
|
*out_h = output_h;
|
|
|
|
|
|
+ dev_dbg(&client->dev, "Scaled for %ux%u : %u = %ux%u, skip %u\n",
|
|
|
+ *in_w, *in_h, resize, output_w, output_h, skip);
|
|
|
+
|
|
|
return resize;
|
|
|
}
|
|
|
|
|
@@ -682,14 +832,14 @@ static int rj54n1_set_clock(struct i2c_client *client)
|
|
|
|
|
|
/* Enable external clock */
|
|
|
ret = reg_write(client, RJ54N1_RESET_STANDBY, E_EXCLK | SOFT_STDBY);
|
|
|
- /* Leave stand-by */
|
|
|
+ /* Leave stand-by. Note: use this when implementing suspend / resume */
|
|
|
if (!ret)
|
|
|
ret = reg_write(client, RJ54N1_RESET_STANDBY, E_EXCLK);
|
|
|
|
|
|
if (!ret)
|
|
|
- ret = reg_write(client, RJ54N1_PLL_L, 2);
|
|
|
+ ret = reg_write(client, RJ54N1_PLL_L, PLL_L);
|
|
|
if (!ret)
|
|
|
- ret = reg_write(client, RJ54N1_PLL_N, 0x31);
|
|
|
+ ret = reg_write(client, RJ54N1_PLL_N, PLL_N);
|
|
|
|
|
|
/* TGCLK dividers */
|
|
|
if (!ret)
|
|
@@ -748,6 +898,7 @@ static int rj54n1_set_clock(struct i2c_client *client)
|
|
|
"Resetting RJ54N1CB0C clock failed: %d!\n", ret);
|
|
|
return -EIO;
|
|
|
}
|
|
|
+
|
|
|
/* Start the PLL */
|
|
|
ret = reg_set(client, RJ54N1_OCLK_DSP, 1, 1);
|
|
|
|
|
@@ -760,6 +911,7 @@ static int rj54n1_set_clock(struct i2c_client *client)
|
|
|
|
|
|
static int rj54n1_reg_init(struct i2c_client *client)
|
|
|
{
|
|
|
+ struct rj54n1 *rj54n1 = to_rj54n1(client);
|
|
|
int ret = rj54n1_set_clock(client);
|
|
|
|
|
|
if (!ret)
|
|
@@ -782,14 +934,26 @@ static int rj54n1_reg_init(struct i2c_client *client)
|
|
|
if (!ret)
|
|
|
ret = reg_write(client, RJ54N1_Y_GAIN, 0x84);
|
|
|
|
|
|
- /* Mirror the image back: default is upside down and left-to-right... */
|
|
|
+ /*
|
|
|
+ * Mirror the image back: default is upside down and left-to-right...
|
|
|
+ * Set manual preview / still shot switching
|
|
|
+ */
|
|
|
if (!ret)
|
|
|
- ret = reg_set(client, RJ54N1_MIRROR_STILL_MODE, 3, 3);
|
|
|
+ ret = reg_write(client, RJ54N1_MIRROR_STILL_MODE, 0x27);
|
|
|
|
|
|
if (!ret)
|
|
|
ret = reg_write_multiple(client, bank_4, ARRAY_SIZE(bank_4));
|
|
|
+
|
|
|
+ /* Auto exposure area */
|
|
|
+ if (!ret)
|
|
|
+ ret = reg_write(client, RJ54N1_EXPOSURE_CONTROL, 0x80);
|
|
|
+ /* Check current auto WB config */
|
|
|
if (!ret)
|
|
|
+ ret = reg_read(client, RJ54N1_WB_SEL_WEIGHT_I);
|
|
|
+ if (ret >= 0) {
|
|
|
+ rj54n1->auto_wb = ret & 0x80;
|
|
|
ret = reg_write_multiple(client, bank_5, ARRAY_SIZE(bank_5));
|
|
|
+ }
|
|
|
if (!ret)
|
|
|
ret = reg_write_multiple(client, bank_8, ARRAY_SIZE(bank_8));
|
|
|
|
|
@@ -806,8 +970,9 @@ static int rj54n1_reg_init(struct i2c_client *client)
|
|
|
ret = reg_write(client, RJ54N1_RESET_STANDBY,
|
|
|
E_EXCLK | DSP_RSTX | TG_RSTX | SEN_RSTX);
|
|
|
|
|
|
+ /* Start register update? Same register as 0x?FE in many bank_* sets */
|
|
|
if (!ret)
|
|
|
- ret = reg_write(client, 0x7fe, 2);
|
|
|
+ ret = reg_write(client, RJ54N1_FWFLG, 2);
|
|
|
|
|
|
/* Constant taken from manufacturer's example */
|
|
|
msleep(700);
|
|
@@ -815,7 +980,6 @@ static int rj54n1_reg_init(struct i2c_client *client)
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
-/* FIXME: streaming output only up to 800x600 is functional */
|
|
|
static int rj54n1_try_fmt(struct v4l2_subdev *sd,
|
|
|
struct v4l2_mbus_framefmt *mf)
|
|
|
{
|
|
@@ -861,14 +1025,13 @@ static int rj54n1_s_fmt(struct v4l2_subdev *sd,
|
|
|
* The host driver can call us without .try_fmt(), so, we have to take
|
|
|
* care ourseleves
|
|
|
*/
|
|
|
- ret = rj54n1_try_fmt(sd, mf);
|
|
|
+ rj54n1_try_fmt(sd, mf);
|
|
|
|
|
|
/*
|
|
|
* Verify if the sensor has just been powered on. TODO: replace this
|
|
|
* with proper PM, when a suitable API is available.
|
|
|
*/
|
|
|
- if (!ret)
|
|
|
- ret = reg_read(client, RJ54N1_RESET_STANDBY);
|
|
|
+ ret = reg_read(client, RJ54N1_RESET_STANDBY);
|
|
|
if (ret < 0)
|
|
|
return ret;
|
|
|
|
|
@@ -878,6 +1041,9 @@ static int rj54n1_s_fmt(struct v4l2_subdev *sd,
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
+ dev_dbg(&client->dev, "%s: code = %d, width = %u, height = %u\n",
|
|
|
+ __func__, mf->code, mf->width, mf->height);
|
|
|
+
|
|
|
/* RA_SEL_UL is only relevant for raw modes, ignored otherwise. */
|
|
|
switch (mf->code) {
|
|
|
case V4L2_MBUS_FMT_YUYV8_2X8_LE:
|
|
@@ -1062,6 +1228,14 @@ static const struct v4l2_queryctrl rj54n1_controls[] = {
|
|
|
.step = 1,
|
|
|
.default_value = 66,
|
|
|
.flags = V4L2_CTRL_FLAG_SLIDER,
|
|
|
+ }, {
|
|
|
+ .id = V4L2_CID_AUTO_WHITE_BALANCE,
|
|
|
+ .type = V4L2_CTRL_TYPE_BOOLEAN,
|
|
|
+ .name = "Auto white balance",
|
|
|
+ .minimum = 0,
|
|
|
+ .maximum = 1,
|
|
|
+ .step = 1,
|
|
|
+ .default_value = 1,
|
|
|
},
|
|
|
};
|
|
|
|
|
@@ -1075,6 +1249,7 @@ static struct soc_camera_ops rj54n1_ops = {
|
|
|
static int rj54n1_g_ctrl(struct v4l2_subdev *sd, struct v4l2_control *ctrl)
|
|
|
{
|
|
|
struct i2c_client *client = sd->priv;
|
|
|
+ struct rj54n1 *rj54n1 = to_rj54n1(client);
|
|
|
int data;
|
|
|
|
|
|
switch (ctrl->id) {
|
|
@@ -1097,6 +1272,9 @@ static int rj54n1_g_ctrl(struct v4l2_subdev *sd, struct v4l2_control *ctrl)
|
|
|
|
|
|
ctrl->value = data / 2;
|
|
|
break;
|
|
|
+ case V4L2_CID_AUTO_WHITE_BALANCE:
|
|
|
+ ctrl->value = rj54n1->auto_wb;
|
|
|
+ break;
|
|
|
}
|
|
|
|
|
|
return 0;
|
|
@@ -1106,6 +1284,7 @@ static int rj54n1_s_ctrl(struct v4l2_subdev *sd, struct v4l2_control *ctrl)
|
|
|
{
|
|
|
int data;
|
|
|
struct i2c_client *client = sd->priv;
|
|
|
+ struct rj54n1 *rj54n1 = to_rj54n1(client);
|
|
|
const struct v4l2_queryctrl *qctrl;
|
|
|
|
|
|
qctrl = soc_camera_find_qctrl(&rj54n1_ops, ctrl->id);
|
|
@@ -1136,6 +1315,13 @@ static int rj54n1_s_ctrl(struct v4l2_subdev *sd, struct v4l2_control *ctrl)
|
|
|
else if (reg_write(client, RJ54N1_Y_GAIN, ctrl->value * 2) < 0)
|
|
|
return -EIO;
|
|
|
break;
|
|
|
+ case V4L2_CID_AUTO_WHITE_BALANCE:
|
|
|
+ /* Auto WB area - whole image */
|
|
|
+ if (reg_set(client, RJ54N1_WB_SEL_WEIGHT_I, ctrl->value << 7,
|
|
|
+ 0x80) < 0)
|
|
|
+ return -EIO;
|
|
|
+ rj54n1->auto_wb = ctrl->value;
|
|
|
+ break;
|
|
|
}
|
|
|
|
|
|
return 0;
|
|
@@ -1158,6 +1344,7 @@ static struct v4l2_subdev_video_ops rj54n1_subdev_video_ops = {
|
|
|
.try_mbus_fmt = rj54n1_try_fmt,
|
|
|
.enum_mbus_fmt = rj54n1_enum_fmt,
|
|
|
.g_crop = rj54n1_g_crop,
|
|
|
+ .s_crop = rj54n1_s_crop,
|
|
|
.cropcap = rj54n1_cropcap,
|
|
|
};
|
|
|
|
|
@@ -1166,21 +1353,13 @@ static struct v4l2_subdev_ops rj54n1_subdev_ops = {
|
|
|
.video = &rj54n1_subdev_video_ops,
|
|
|
};
|
|
|
|
|
|
-static int rj54n1_pin_config(struct i2c_client *client)
|
|
|
-{
|
|
|
- /*
|
|
|
- * Experimentally found out IOCTRL wired to 0. TODO: add to platform
|
|
|
- * data: 0 or 1 << 7.
|
|
|
- */
|
|
|
- return reg_write(client, RJ54N1_IOC, 0);
|
|
|
-}
|
|
|
-
|
|
|
/*
|
|
|
* Interface active, can use i2c. If it fails, it can indeed mean, that
|
|
|
* this wasn't our capture interface, so, we wait for the right one
|
|
|
*/
|
|
|
static int rj54n1_video_probe(struct soc_camera_device *icd,
|
|
|
- struct i2c_client *client)
|
|
|
+ struct i2c_client *client,
|
|
|
+ struct rj54n1_pdata *priv)
|
|
|
{
|
|
|
int data1, data2;
|
|
|
int ret;
|
|
@@ -1201,7 +1380,8 @@ static int rj54n1_video_probe(struct soc_camera_device *icd,
|
|
|
goto ei2c;
|
|
|
}
|
|
|
|
|
|
- ret = rj54n1_pin_config(client);
|
|
|
+ /* Configure IOCTL polarity from the platform data: 0 or 1 << 7. */
|
|
|
+ ret = reg_write(client, RJ54N1_IOC, priv->ioctl_high << 7);
|
|
|
if (ret < 0)
|
|
|
goto ei2c;
|
|
|
|
|
@@ -1219,6 +1399,7 @@ static int rj54n1_probe(struct i2c_client *client,
|
|
|
struct soc_camera_device *icd = client->dev.platform_data;
|
|
|
struct i2c_adapter *adapter = to_i2c_adapter(client->dev.parent);
|
|
|
struct soc_camera_link *icl;
|
|
|
+ struct rj54n1_pdata *rj54n1_priv;
|
|
|
int ret;
|
|
|
|
|
|
if (!icd) {
|
|
@@ -1227,11 +1408,13 @@ static int rj54n1_probe(struct i2c_client *client,
|
|
|
}
|
|
|
|
|
|
icl = to_soc_camera_link(icd);
|
|
|
- if (!icl) {
|
|
|
+ if (!icl || !icl->priv) {
|
|
|
dev_err(&client->dev, "RJ54N1CB0C: missing platform data!\n");
|
|
|
return -EINVAL;
|
|
|
}
|
|
|
|
|
|
+ rj54n1_priv = icl->priv;
|
|
|
+
|
|
|
if (!i2c_check_functionality(adapter, I2C_FUNC_SMBUS_BYTE_DATA)) {
|
|
|
dev_warn(&adapter->dev,
|
|
|
"I2C-Adapter doesn't support I2C_FUNC_SMBUS_BYTE\n");
|
|
@@ -1255,8 +1438,10 @@ static int rj54n1_probe(struct i2c_client *client,
|
|
|
rj54n1->height = RJ54N1_MAX_HEIGHT;
|
|
|
rj54n1->fmt = &rj54n1_colour_fmts[0];
|
|
|
rj54n1->resize = 1024;
|
|
|
+ rj54n1->tgclk_mhz = (rj54n1_priv->mclk_freq / PLL_L * PLL_N) /
|
|
|
+ (clk_div.ratio_tg + 1) / (clk_div.ratio_t + 1);
|
|
|
|
|
|
- ret = rj54n1_video_probe(icd, client);
|
|
|
+ ret = rj54n1_video_probe(icd, client, rj54n1_priv);
|
|
|
if (ret < 0) {
|
|
|
icd->ops = NULL;
|
|
|
i2c_set_clientdata(client, NULL);
|