16 #include <linux/i2c.h>
19 #include <linux/slab.h>
20 #include <linux/videodev2.h>
22 #include <linux/module.h>
29 #define MT9V032_PIXEL_ARRAY_HEIGHT 492
30 #define MT9V032_PIXEL_ARRAY_WIDTH 782
32 #define MT9V032_SYSCLK_FREQ_DEF 26600000
34 #define MT9V032_CHIP_VERSION 0x00
35 #define MT9V032_CHIP_ID_REV1 0x1311
36 #define MT9V032_CHIP_ID_REV3 0x1313
37 #define MT9V032_COLUMN_START 0x01
38 #define MT9V032_COLUMN_START_MIN 1
39 #define MT9V032_COLUMN_START_DEF 1
40 #define MT9V032_COLUMN_START_MAX 752
41 #define MT9V032_ROW_START 0x02
42 #define MT9V032_ROW_START_MIN 4
43 #define MT9V032_ROW_START_DEF 5
44 #define MT9V032_ROW_START_MAX 482
45 #define MT9V032_WINDOW_HEIGHT 0x03
46 #define MT9V032_WINDOW_HEIGHT_MIN 1
47 #define MT9V032_WINDOW_HEIGHT_DEF 480
48 #define MT9V032_WINDOW_HEIGHT_MAX 480
49 #define MT9V032_WINDOW_WIDTH 0x04
50 #define MT9V032_WINDOW_WIDTH_MIN 1
51 #define MT9V032_WINDOW_WIDTH_DEF 752
52 #define MT9V032_WINDOW_WIDTH_MAX 752
53 #define MT9V032_HORIZONTAL_BLANKING 0x05
54 #define MT9V032_HORIZONTAL_BLANKING_MIN 43
55 #define MT9V032_HORIZONTAL_BLANKING_DEF 94
56 #define MT9V032_HORIZONTAL_BLANKING_MAX 1023
57 #define MT9V032_VERTICAL_BLANKING 0x06
58 #define MT9V032_VERTICAL_BLANKING_MIN 4
59 #define MT9V032_VERTICAL_BLANKING_DEF 45
60 #define MT9V032_VERTICAL_BLANKING_MAX 3000
61 #define MT9V032_CHIP_CONTROL 0x07
62 #define MT9V032_CHIP_CONTROL_MASTER_MODE (1 << 3)
63 #define MT9V032_CHIP_CONTROL_DOUT_ENABLE (1 << 7)
64 #define MT9V032_CHIP_CONTROL_SEQUENTIAL (1 << 8)
65 #define MT9V032_SHUTTER_WIDTH1 0x08
66 #define MT9V032_SHUTTER_WIDTH2 0x09
67 #define MT9V032_SHUTTER_WIDTH_CONTROL 0x0a
68 #define MT9V032_TOTAL_SHUTTER_WIDTH 0x0b
69 #define MT9V032_TOTAL_SHUTTER_WIDTH_MIN 1
70 #define MT9V032_TOTAL_SHUTTER_WIDTH_DEF 480
71 #define MT9V032_TOTAL_SHUTTER_WIDTH_MAX 32767
72 #define MT9V032_RESET 0x0c
73 #define MT9V032_READ_MODE 0x0d
74 #define MT9V032_READ_MODE_ROW_BIN_MASK (3 << 0)
75 #define MT9V032_READ_MODE_ROW_BIN_SHIFT 0
76 #define MT9V032_READ_MODE_COLUMN_BIN_MASK (3 << 2)
77 #define MT9V032_READ_MODE_COLUMN_BIN_SHIFT 2
78 #define MT9V032_READ_MODE_ROW_FLIP (1 << 4)
79 #define MT9V032_READ_MODE_COLUMN_FLIP (1 << 5)
80 #define MT9V032_READ_MODE_DARK_COLUMNS (1 << 6)
81 #define MT9V032_READ_MODE_DARK_ROWS (1 << 7)
82 #define MT9V032_PIXEL_OPERATION_MODE 0x0f
83 #define MT9V032_PIXEL_OPERATION_MODE_COLOR (1 << 2)
84 #define MT9V032_PIXEL_OPERATION_MODE_HDR (1 << 6)
85 #define MT9V032_ANALOG_GAIN 0x35
86 #define MT9V032_ANALOG_GAIN_MIN 16
87 #define MT9V032_ANALOG_GAIN_DEF 16
88 #define MT9V032_ANALOG_GAIN_MAX 64
89 #define MT9V032_MAX_ANALOG_GAIN 0x36
90 #define MT9V032_MAX_ANALOG_GAIN_MAX 127
91 #define MT9V032_FRAME_DARK_AVERAGE 0x42
92 #define MT9V032_DARK_AVG_THRESH 0x46
93 #define MT9V032_DARK_AVG_LOW_THRESH_MASK (255 << 0)
94 #define MT9V032_DARK_AVG_LOW_THRESH_SHIFT 0
95 #define MT9V032_DARK_AVG_HIGH_THRESH_MASK (255 << 8)
96 #define MT9V032_DARK_AVG_HIGH_THRESH_SHIFT 8
97 #define MT9V032_ROW_NOISE_CORR_CONTROL 0x70
98 #define MT9V032_ROW_NOISE_CORR_ENABLE (1 << 5)
99 #define MT9V032_ROW_NOISE_CORR_USE_BLK_AVG (1 << 7)
100 #define MT9V032_PIXEL_CLOCK 0x74
101 #define MT9V032_PIXEL_CLOCK_INV_LINE (1 << 0)
102 #define MT9V032_PIXEL_CLOCK_INV_FRAME (1 << 1)
103 #define MT9V032_PIXEL_CLOCK_XOR_LINE (1 << 2)
104 #define MT9V032_PIXEL_CLOCK_CONT_LINE (1 << 3)
105 #define MT9V032_PIXEL_CLOCK_INV_PXL_CLK (1 << 4)
106 #define MT9V032_TEST_PATTERN 0x7f
107 #define MT9V032_TEST_PATTERN_DATA_MASK (1023 << 0)
108 #define MT9V032_TEST_PATTERN_DATA_SHIFT 0
109 #define MT9V032_TEST_PATTERN_USE_DATA (1 << 10)
110 #define MT9V032_TEST_PATTERN_GRAY_MASK (3 << 11)
111 #define MT9V032_TEST_PATTERN_GRAY_NONE (0 << 11)
112 #define MT9V032_TEST_PATTERN_GRAY_VERTICAL (1 << 11)
113 #define MT9V032_TEST_PATTERN_GRAY_HORIZONTAL (2 << 11)
114 #define MT9V032_TEST_PATTERN_GRAY_DIAGONAL (3 << 11)
115 #define MT9V032_TEST_PATTERN_ENABLE (1 << 13)
116 #define MT9V032_TEST_PATTERN_FLIP (1 << 14)
117 #define MT9V032_AEC_AGC_ENABLE 0xaf
118 #define MT9V032_AEC_ENABLE (1 << 0)
119 #define MT9V032_AGC_ENABLE (1 << 1)
120 #define MT9V032_THERMAL_INFO 0xc1
157 s32 data = i2c_smbus_read_word_swapped(client, reg);
158 dev_dbg(&client->
dev,
"%s: read 0x%04x from 0x%02x\n", __func__,
166 dev_dbg(&client->
dev,
"%s: writing 0x%04x to 0x%02x\n", __func__,
168 return i2c_smbus_write_word_swapped(client, reg, data);
186 mt9v032_update_aec_agc(
struct mt9v032 *mt9v032,
u16 which,
int enable)
206 mt9v032_update_hblank(
struct mt9v032 *mt9v032)
215 #define EXT_CLK 25000000
217 static int mt9v032_power_on(
struct mt9v032 *mt9v032)
222 if (mt9v032->
pdata->set_clock) {
239 static void mt9v032_power_off(
struct mt9v032 *mt9v032)
241 if (mt9v032->
pdata->set_clock)
245 static int __mt9v032_set_power(
struct mt9v032 *mt9v032,
bool on)
251 mt9v032_power_off(mt9v032);
255 ret = mt9v032_power_on(mt9v032);
260 if (mt9v032->
pdata && mt9v032->
pdata->clk_pol) {
279 static struct v4l2_mbus_framefmt *
280 __mt9v032_get_pad_format(
struct mt9v032 *mt9v032,
struct v4l2_subdev_fh *fh,
285 return v4l2_subdev_get_try_format(fh, pad);
294 __mt9v032_get_pad_crop(
struct mt9v032 *mt9v032,
struct v4l2_subdev_fh *fh,
299 return v4l2_subdev_get_try_crop(fh, pad);
301 return &mt9v032->
crop;
312 struct i2c_client *client = v4l2_get_subdevdata(subdev);
313 struct mt9v032 *mt9v032 = to_mt9v032(subdev);
321 return mt9v032_set_chip_control(mt9v032, mode, 0);
349 ret = mt9v032_update_hblank(mt9v032);
354 return mt9v032_set_chip_control(mt9v032, 0, mode);
357 static int mt9v032_enum_mbus_code(
struct v4l2_subdev *subdev,
359 struct v4l2_subdev_mbus_code_enum *
code)
368 static int mt9v032_enum_frame_size(
struct v4l2_subdev *subdev,
370 struct v4l2_subdev_frame_size_enum *fse)
376 fse->max_width = fse->min_width;
378 fse->max_height = fse->min_height;
383 static int mt9v032_get_format(
struct v4l2_subdev *subdev,
387 struct mt9v032 *mt9v032 = to_mt9v032(subdev);
389 format->
format = *__mt9v032_get_pad_format(mt9v032, fh, format->
pad,
394 static void mt9v032_configure_pixel_rate(
struct mt9v032 *mt9v032,
401 mt9v032->
sysclk / hratio);
403 dev_warn(&client->
dev,
"failed to set pixel rate (%d)\n", ret);
406 static int mt9v032_set_format(
struct v4l2_subdev *subdev,
410 struct mt9v032 *mt9v032 = to_mt9v032(subdev);
411 struct v4l2_mbus_framefmt *__format;
418 __crop = __mt9v032_get_pad_crop(mt9v032, fh, format->
pad,
432 __format = __mt9v032_get_pad_format(mt9v032, fh, format->
pad,
434 __format->width = __crop->
width / hratio;
435 __format->height = __crop->
height / vratio;
437 mt9v032_configure_pixel_rate(mt9v032, hratio);
439 format->
format = *__format;
444 static int mt9v032_get_crop(
struct v4l2_subdev *subdev,
448 struct mt9v032 *mt9v032 = to_mt9v032(subdev);
450 crop->
rect = *__mt9v032_get_pad_crop(mt9v032, fh, crop->
pad,
455 static int mt9v032_set_crop(
struct v4l2_subdev *subdev,
459 struct mt9v032 *mt9v032 = to_mt9v032(subdev);
460 struct v4l2_mbus_framefmt *__format;
483 __crop = __mt9v032_get_pad_crop(mt9v032, fh, crop->
pad, crop->
which);
485 if (rect.width != __crop->
width || rect.height != __crop->
height) {
489 __format = __mt9v032_get_pad_format(mt9v032, fh, crop->
pad,
491 __format->width = rect.width;
492 __format->height = rect.height;
494 mt9v032_configure_pixel_rate(mt9v032, 1);
507 #define V4L2_CID_TEST_PATTERN_COLOR (V4L2_CID_USER_BASE | 0x1001)
511 struct mt9v032 *mt9v032 =
535 return mt9v032_update_hblank(mt9v032);
583 .s_ctrl = mt9v032_s_ctrl,
586 static const char *
const mt9v032_test_pattern_menu[] = {
588 "Gray Vertical Shade",
589 "Gray Horizontal Shade",
590 "Gray Diagonal Shade",
595 .ops = &mt9v032_ctrl_ops,
598 .name =
"Test Pattern Color",
610 static int mt9v032_set_power(
struct v4l2_subdev *subdev,
int on)
612 struct mt9v032 *mt9v032 = to_mt9v032(subdev);
621 ret = __mt9v032_set_power(mt9v032, !!on);
639 static int mt9v032_registered(
struct v4l2_subdev *subdev)
641 struct i2c_client *client = v4l2_get_subdevdata(subdev);
642 struct mt9v032 *mt9v032 = to_mt9v032(subdev);
646 dev_info(&client->
dev,
"Probing MT9V032 at address 0x%02x\n",
649 ret = mt9v032_power_on(mt9v032);
651 dev_err(&client->
dev,
"MT9V032 power up failed\n");
658 dev_err(&client->
dev,
"MT9V032 not detected, wrong version "
663 mt9v032_power_off(mt9v032);
665 dev_info(&client->
dev,
"MT9V032 detected at address 0x%02x\n",
668 mt9v032_configure_pixel_rate(mt9v032, 1);
675 struct v4l2_mbus_framefmt *
format;
678 crop = v4l2_subdev_get_try_crop(fh, 0);
684 format = v4l2_subdev_get_try_format(fh, 0);
691 return mt9v032_set_power(subdev, 1);
696 return mt9v032_set_power(subdev, 0);
700 .s_power = mt9v032_set_power,
704 .s_stream = mt9v032_s_stream,
708 .enum_mbus_code = mt9v032_enum_mbus_code,
709 .enum_frame_size = mt9v032_enum_frame_size,
710 .get_fmt = mt9v032_get_format,
711 .set_fmt = mt9v032_set_format,
712 .get_crop = mt9v032_get_crop,
713 .set_crop = mt9v032_set_crop,
717 .core = &mt9v032_subdev_core_ops,
718 .video = &mt9v032_subdev_video_ops,
719 .pad = &mt9v032_subdev_pad_ops,
723 .registered = mt9v032_registered,
724 .open = mt9v032_open,
725 .close = mt9v032_close,
732 static int mt9v032_probe(
struct i2c_client *client,
736 struct mt9v032 *mt9v032;
740 if (!i2c_check_functionality(client->
adapter,
743 "I2C-Adapter doesn't support I2C_FUNC_SMBUS_WORD\n");
747 mt9v032 = kzalloc(
sizeof(*mt9v032),
GFP_KERNEL);
778 ARRAY_SIZE(mt9v032_test_pattern_menu) - 1, 0, 0,
779 mt9v032_test_pattern_menu);
781 &mt9v032_test_pattern_color,
NULL);
790 unsigned int def = 0;
808 if (mt9v032->
ctrls.error)
810 __func__, mt9v032->
ctrls.error);
828 mt9v032->
subdev.internal_ops = &mt9v032_subdev_internal_ops;
839 static int mt9v032_remove(
struct i2c_client *client)
841 struct v4l2_subdev *subdev = i2c_get_clientdata(client);
842 struct mt9v032 *mt9v032 = to_mt9v032(subdev);
860 .probe = mt9v032_probe,
861 .remove = mt9v032_remove,
862 .id_table = mt9v032_id,