aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/media/video/ov7670.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/media/video/ov7670.c')
-rw-r--r--drivers/media/video/ov7670.c72
1 files changed, 67 insertions, 5 deletions
diff --git a/drivers/media/video/ov7670.c b/drivers/media/video/ov7670.c
index 7d380d76338a..89dd18c3c5cc 100644
--- a/drivers/media/video/ov7670.c
+++ b/drivers/media/video/ov7670.c
@@ -36,6 +36,11 @@ MODULE_LICENSE("GPL");
36#define QCIF_HEIGHT 144 36#define QCIF_HEIGHT 144
37 37
38/* 38/*
39 * Our nominal (default) frame rate.
40 */
41#define OV7670_FRAME_RATE 30
42
43/*
39 * The 7670 sits on i2c with ID 0x42 44 * The 7670 sits on i2c with ID 0x42
40 */ 45 */
41#define OV7670_I2C_ADDR 0x42 46#define OV7670_I2C_ADDR 0x42
@@ -291,7 +296,7 @@ static struct regval_list ov7670_default_regs[] = {
291 { 0xc9, 0x60 }, { REG_COM16, 0x38 }, 296 { 0xc9, 0x60 }, { REG_COM16, 0x38 },
292 { 0x56, 0x40 }, 297 { 0x56, 0x40 },
293 298
294 { 0x34, 0x11 }, { REG_COM11, COM11_EXP }, 299 { 0x34, 0x11 }, { REG_COM11, COM11_EXP|COM11_HZAUTO },
295 { 0xa4, 0x88 }, { 0x96, 0 }, 300 { 0xa4, 0x88 }, { 0x96, 0 },
296 { 0x97, 0x30 }, { 0x98, 0x20 }, 301 { 0x97, 0x30 }, { 0x98, 0x20 },
297 { 0x99, 0x30 }, { 0x9a, 0x84 }, 302 { 0x99, 0x30 }, { 0x9a, 0x84 },
@@ -722,6 +727,63 @@ static int ov7670_s_fmt(struct i2c_client *c, struct v4l2_format *fmt)
722} 727}
723 728
724/* 729/*
730 * Implement G/S_PARM. There is a "high quality" mode we could try
731 * to do someday; for now, we just do the frame rate tweak.
732 */
733static int ov7670_g_parm(struct i2c_client *c, struct v4l2_streamparm *parms)
734{
735 struct v4l2_captureparm *cp = &parms->parm.capture;
736 unsigned char clkrc;
737 int ret;
738
739 if (parms->type != V4L2_BUF_TYPE_VIDEO_CAPTURE)
740 return -EINVAL;
741 ret = ov7670_read(c, REG_CLKRC, &clkrc);
742 if (ret < 0)
743 return ret;
744 memset(cp, 0, sizeof(struct v4l2_captureparm));
745 cp->capability = V4L2_CAP_TIMEPERFRAME;
746 cp->timeperframe.numerator = 1;
747 cp->timeperframe.denominator = OV7670_FRAME_RATE;
748 if ((clkrc & CLK_EXT) == 0 && (clkrc & CLK_SCALE) > 1)
749 cp->timeperframe.denominator /= (clkrc & CLK_SCALE);
750 return 0;
751}
752
753static int ov7670_s_parm(struct i2c_client *c, struct v4l2_streamparm *parms)
754{
755 struct v4l2_captureparm *cp = &parms->parm.capture;
756 struct v4l2_fract *tpf = &cp->timeperframe;
757 unsigned char clkrc;
758 int ret, div;
759
760 if (parms->type != V4L2_BUF_TYPE_VIDEO_CAPTURE)
761 return -EINVAL;
762 if (cp->extendedmode != 0)
763 return -EINVAL;
764 /*
765 * CLKRC has a reserved bit, so let's preserve it.
766 */
767 ret = ov7670_read(c, REG_CLKRC, &clkrc);
768 if (ret < 0)
769 return ret;
770 if (tpf->numerator == 0 || tpf->denominator == 0)
771 div = 1; /* Reset to full rate */
772 else
773 div = (tpf->numerator*OV7670_FRAME_RATE)/tpf->denominator;
774 if (div == 0)
775 div = 1;
776 else if (div > CLK_SCALE)
777 div = CLK_SCALE;
778 clkrc = (clkrc & 0x80) | div;
779 tpf->numerator = 1;
780 tpf->denominator = OV7670_FRAME_RATE/div;
781 return ov7670_write(c, REG_CLKRC, clkrc);
782}
783
784
785
786/*
725 * Code for dealing with controls. 787 * Code for dealing with controls.
726 */ 788 */
727 789
@@ -1231,10 +1293,10 @@ static int ov7670_command(struct i2c_client *client, unsigned int cmd,
1231 return ov7670_s_ctrl(client, (struct v4l2_control *) arg); 1293 return ov7670_s_ctrl(client, (struct v4l2_control *) arg);
1232 case VIDIOC_G_CTRL: 1294 case VIDIOC_G_CTRL:
1233 return ov7670_g_ctrl(client, (struct v4l2_control *) arg); 1295 return ov7670_g_ctrl(client, (struct v4l2_control *) arg);
1234 /* Todo: 1296 case VIDIOC_S_PARM:
1235 g/s_parm 1297 return ov7670_s_parm(client, (struct v4l2_streamparm *) arg);
1236 initialization 1298 case VIDIOC_G_PARM:
1237 */ 1299 return ov7670_g_parm(client, (struct v4l2_streamparm *) arg);
1238 } 1300 }
1239 return -EINVAL; 1301 return -EINVAL;
1240} 1302}