m5602_s5k4aa.c 9.9 KB
Newer Older
1 2 3
/*
 * Driver for the s5k4aa sensor
 *
4
 * Copyright (C) 2008 Erik Andrén
5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
 * Copyright (C) 2007 Ilyes Gouta. Based on the m5603x Linux Driver Project.
 * Copyright (C) 2005 m5603x Linux Driver Project <m5602@x3ng.com.br>
 *
 * Portions of code to USB interface and ALi driver software,
 * Copyright (c) 2006 Willem Duinker
 * v4l2 interface modeled after the V4L2 driver
 * for SN9C10x PC Camera Controllers
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License as
 * published by the Free Software Foundation, version 2.
 *
 */

#include "m5602_s5k4aa.h"

21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52
static
    const
	struct dmi_system_id s5k4aa_vflip_dmi_table[] = {
	{
		.ident = "Fujitsu-Siemens Amilo Xa 2528",
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "FUJITSU SIEMENS"),
			DMI_MATCH(DMI_PRODUCT_NAME, "AMILO Xa 2528")
		}
	}, {
		.ident = "Fujitsu-Siemens Amilo Xi 2550",
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "FUJITSU SIEMENS"),
			DMI_MATCH(DMI_PRODUCT_NAME, "AMILO Xi 2550")
		}
	}, {
		.ident = "MSI GX700",
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Micro-Star International"),
			DMI_MATCH(DMI_PRODUCT_NAME, "GX700"),
			DMI_MATCH(DMI_BIOS_DATE, "07/26/2007")
		}
	}, {
		.ident = "MSI GX700/GX705/EX700",
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Micro-Star International"),
			DMI_MATCH(DMI_PRODUCT_NAME, "GX700/GX705/EX700")
		}
	},
	{ }
};

53
static void s5k4aa_dump_registers(struct sd *sd);
54

55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85
int s5k4aa_probe(struct sd *sd)
{
	u8 prod_id[6] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
	const u8 expected_prod_id[6] = {0x00, 0x10, 0x00, 0x4b, 0x33, 0x75};
	int i, err = 0;

	if (force_sensor) {
		if (force_sensor == S5K4AA_SENSOR) {
			info("Forcing a %s sensor", s5k4aa.name);
			goto sensor_found;
		}
		/* If we want to force another sensor, don't try to probe this
		 * one */
		return -ENODEV;
	}

	info("Probing for a s5k4aa sensor");

	/* Preinit the sensor */
	for (i = 0; i < ARRAY_SIZE(preinit_s5k4aa) && !err; i++) {
		u8 data[2] = {0x00, 0x00};

		switch (preinit_s5k4aa[i][0]) {
		case BRIDGE:
			err = m5602_write_bridge(sd,
						 preinit_s5k4aa[i][1],
						 preinit_s5k4aa[i][2]);
			break;

		case SENSOR:
			data[0] = preinit_s5k4aa[i][2];
86
			err = m5602_write_sensor(sd,
87 88 89 90 91 92 93
						  preinit_s5k4aa[i][1],
						  data, 1);
			break;

		case SENSOR_LONG:
			data[0] = preinit_s5k4aa[i][2];
			data[1] = preinit_s5k4aa[i][3];
94
			err = m5602_write_sensor(sd,
95 96 97 98 99 100 101 102 103 104
						  preinit_s5k4aa[i][1],
						  data, 2);
			break;
		default:
			info("Invalid stream command, exiting init");
			return -EINVAL;
		}
	}

	/* Test some registers, but we don't know their exact meaning yet */
105
	if (m5602_read_sensor(sd, 0x00, prod_id, sizeof(prod_id)))
106 107 108 109 110 111
		return -ENODEV;

	if (memcmp(prod_id, expected_prod_id, sizeof(prod_id)))
		return -ENODEV;
	else
		info("Detected a s5k4aa sensor");
112

113 114 115
sensor_found:
	sd->gspca_dev.cam.cam_mode = s5k4aa.modes;
	sd->gspca_dev.cam.nmodes = s5k4aa.nmodes;
116 117 118
	sd->desc->ctrls = s5k4aa.ctrls;
	sd->desc->nctrls = s5k4aa.nctrls;

119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137
	return 0;
}

int s5k4aa_init(struct sd *sd)
{
	int i, err = 0;

	for (i = 0; i < ARRAY_SIZE(init_s5k4aa) && !err; i++) {
		u8 data[2] = {0x00, 0x00};

		switch (init_s5k4aa[i][0]) {
		case BRIDGE:
			err = m5602_write_bridge(sd,
				init_s5k4aa[i][1],
				init_s5k4aa[i][2]);
			break;

		case SENSOR:
			data[0] = init_s5k4aa[i][2];
138
			err = m5602_write_sensor(sd,
139 140 141 142 143 144
				init_s5k4aa[i][1], data, 1);
			break;

		case SENSOR_LONG:
			data[0] = init_s5k4aa[i][2];
			data[1] = init_s5k4aa[i][3];
145
			err = m5602_write_sensor(sd,
146 147 148 149 150 151 152 153 154 155 156 157 158 159
				init_s5k4aa[i][1], data, 2);
			break;
		default:
			info("Invalid stream command, exiting init");
			return -EINVAL;
		}
	}

	if (dump_sensor)
		s5k4aa_dump_registers(sd);

	if (!err && dmi_check_system(s5k4aa_vflip_dmi_table)) {
		u8 data = 0x02;
		info("vertical flip quirk active");
160
		m5602_write_sensor(sd, S5K4AA_PAGE_MAP, &data, 1);
161
		m5602_read_sensor(sd, S5K4AA_READ_MODE, &data, 1);
162 163
		data |= S5K4AA_RM_V_FLIP;
		data &= ~S5K4AA_RM_H_FLIP;
164
		m5602_write_sensor(sd, S5K4AA_READ_MODE, &data, 1);
165 166

		/* Decrement COLSTART to preserve color order (BGGR) */
167
		m5602_read_sensor(sd, S5K4AA_COLSTART_LO, &data, 1);
168
		data--;
169
		m5602_write_sensor(sd, S5K4AA_COLSTART_LO, &data, 1);
170 171

		/* Increment ROWSTART to preserve color order (BGGR) */
172
		m5602_read_sensor(sd, S5K4AA_ROWSTART_LO, &data, 1);
173
		data++;
174
		m5602_write_sensor(sd, S5K4AA_ROWSTART_LO, &data, 1);
175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190
	}

	return (err < 0) ? err : 0;
}

int s5k4aa_power_down(struct sd *sd)
{
	return 0;
}

int s5k4aa_get_exposure(struct gspca_dev *gspca_dev, __s32 *val)
{
	struct sd *sd = (struct sd *) gspca_dev;
	u8 data = S5K4AA_PAGE_MAP_2;
	int err;

191
	err = m5602_write_sensor(sd, S5K4AA_PAGE_MAP, &data, 1);
192 193 194
	if (err < 0)
		goto out;

195
	err = m5602_read_sensor(sd, S5K4AA_EXPOSURE_HI, &data, 1);
196 197 198 199
	if (err < 0)
		goto out;

	*val = data << 8;
200
	err = m5602_read_sensor(sd, S5K4AA_EXPOSURE_LO, &data, 1);
201
	*val |= data;
202
	PDEBUG(D_V4L2, "Read exposure %d", *val);
203
out:
204
	return err;
205 206 207 208 209 210 211 212
}

int s5k4aa_set_exposure(struct gspca_dev *gspca_dev, __s32 val)
{
	struct sd *sd = (struct sd *) gspca_dev;
	u8 data = S5K4AA_PAGE_MAP_2;
	int err;

213
	PDEBUG(D_V4L2, "Set exposure to %d", val);
214
	err = m5602_write_sensor(sd, S5K4AA_PAGE_MAP, &data, 1);
215 216 217
	if (err < 0)
		goto out;
	data = (val >> 8) & 0xff;
218
	err = m5602_write_sensor(sd, S5K4AA_EXPOSURE_HI, &data, 1);
219 220 221
	if (err < 0)
		goto out;
	data = val & 0xff;
222
	err = m5602_write_sensor(sd, S5K4AA_EXPOSURE_LO, &data, 1);
223
out:
224
	return err;
225 226 227 228 229 230 231 232
}

int s5k4aa_get_vflip(struct gspca_dev *gspca_dev, __s32 *val)
{
	struct sd *sd = (struct sd *) gspca_dev;
	u8 data = S5K4AA_PAGE_MAP_2;
	int err;

233
	err = m5602_write_sensor(sd, S5K4AA_PAGE_MAP, &data, 1);
234 235 236
	if (err < 0)
		goto out;

237
	err = m5602_read_sensor(sd, S5K4AA_PAGE_MAP, &data, 1);
238
	*val = (data & S5K4AA_RM_V_FLIP) >> 7;
239
	PDEBUG(D_V4L2, "Read vertical flip %d", *val);
240 241

out:
242
	return err;
243 244 245 246 247 248 249 250
}

int s5k4aa_set_vflip(struct gspca_dev *gspca_dev, __s32 val)
{
	struct sd *sd = (struct sd *) gspca_dev;
	u8 data = S5K4AA_PAGE_MAP_2;
	int err;

251
	PDEBUG(D_V4L2, "Set vertical flip to %d", val);
252
	err = m5602_write_sensor(sd, S5K4AA_PAGE_MAP, &data, 1);
253 254
	if (err < 0)
		goto out;
255
	err = m5602_write_sensor(sd, S5K4AA_READ_MODE, &data, 1);
256 257 258 259
	if (err < 0)
		goto out;
	data = ((data & ~S5K4AA_RM_V_FLIP)
			| ((val & 0x01) << 7));
260
	err = m5602_write_sensor(sd, S5K4AA_READ_MODE, &data, 1);
261 262 263 264
	if (err < 0)
		goto out;

	if (val) {
265
		err = m5602_read_sensor(sd, S5K4AA_ROWSTART_LO, &data, 1);
266 267 268 269
		if (err < 0)
			goto out;

		data++;
270
		err = m5602_write_sensor(sd, S5K4AA_ROWSTART_LO, &data, 1);
271
	} else {
272
		err = m5602_read_sensor(sd, S5K4AA_ROWSTART_LO, &data, 1);
273 274 275 276
		if (err < 0)
			goto out;

		data--;
277
		err = m5602_write_sensor(sd, S5K4AA_ROWSTART_LO, &data, 1);
278 279
	}
out:
280
	return err;
281 282 283 284 285 286 287 288
}

int s5k4aa_get_hflip(struct gspca_dev *gspca_dev, __s32 *val)
{
	struct sd *sd = (struct sd *) gspca_dev;
	u8 data = S5K4AA_PAGE_MAP_2;
	int err;

289
	err = m5602_write_sensor(sd, S5K4AA_PAGE_MAP, &data, 1);
290 291 292
	if (err < 0)
		goto out;

293
	err = m5602_read_sensor(sd, S5K4AA_PAGE_MAP, &data, 1);
294
	*val = (data & S5K4AA_RM_H_FLIP) >> 6;
295
	PDEBUG(D_V4L2, "Read horizontal flip %d", *val);
296
out:
297
	return err;
298 299 300 301 302 303 304 305
}

int s5k4aa_set_hflip(struct gspca_dev *gspca_dev, __s32 val)
{
	struct sd *sd = (struct sd *) gspca_dev;
	u8 data = S5K4AA_PAGE_MAP_2;
	int err;

306
	PDEBUG(D_V4L2, "Set horizontal flip to %d",
307
	       val);
308
	err = m5602_write_sensor(sd, S5K4AA_PAGE_MAP, &data, 1);
309 310
	if (err < 0)
		goto out;
311
	err = m5602_write_sensor(sd, S5K4AA_READ_MODE, &data, 1);
312 313 314 315
	if (err < 0)
		goto out;

	data = ((data & ~S5K4AA_RM_H_FLIP) | ((val & 0x01) << 6));
316
	err = m5602_write_sensor(sd, S5K4AA_READ_MODE, &data, 1);
317 318 319 320
	if (err < 0)
		goto out;

	if (val) {
321
		err = m5602_read_sensor(sd, S5K4AA_COLSTART_LO, &data, 1);
322 323 324
		if (err < 0)
			goto out;
		data++;
325
		err = m5602_write_sensor(sd, S5K4AA_COLSTART_LO, &data, 1);
326 327 328
		if (err < 0)
			goto out;
	} else {
329
		err = m5602_read_sensor(sd, S5K4AA_COLSTART_LO, &data, 1);
330 331 332
		if (err < 0)
			goto out;
		data--;
333
		err = m5602_write_sensor(sd, S5K4AA_COLSTART_LO, &data, 1);
334 335
	}
out:
336
	return err;
337 338 339 340 341 342 343 344
}

int s5k4aa_get_gain(struct gspca_dev *gspca_dev, __s32 *val)
{
	struct sd *sd = (struct sd *) gspca_dev;
	u8 data = S5K4AA_PAGE_MAP_2;
	int err;

345
	err = m5602_write_sensor(sd, S5K4AA_PAGE_MAP, &data, 1);
346 347 348
	if (err < 0)
		goto out;

349
	err = m5602_read_sensor(sd, S5K4AA_GAIN_2, &data, 1);
350
	*val = data;
351
	PDEBUG(D_V4L2, "Read gain %d", *val);
352 353

out:
354
	return err;
355 356 357 358 359 360 361 362
}

int s5k4aa_set_gain(struct gspca_dev *gspca_dev, __s32 val)
{
	struct sd *sd = (struct sd *) gspca_dev;
	u8 data = S5K4AA_PAGE_MAP_2;
	int err;

363
	PDEBUG(D_V4L2, "Set gain to %d", val);
364
	err = m5602_write_sensor(sd, S5K4AA_PAGE_MAP, &data, 1);
365 366 367 368
	if (err < 0)
		goto out;

	data = val & 0xff;
369
	err = m5602_write_sensor(sd, S5K4AA_GAIN_2, &data, 1);
370 371

out:
372
	return err;
373 374
}

375
static void s5k4aa_dump_registers(struct sd *sd)
376 377 378
{
	int address;
	u8 page, old_page;
379
	m5602_read_sensor(sd, S5K4AA_PAGE_MAP, &old_page, 1);
380
	for (page = 0; page < 16; page++) {
381
		m5602_write_sensor(sd, S5K4AA_PAGE_MAP, &page, 1);
382 383 384
		info("Dumping the s5k4aa register state for page 0x%x", page);
		for (address = 0; address <= 0xff; address++) {
			u8 value = 0;
385
			m5602_read_sensor(sd, address, &value, 1);
386 387 388 389 390 391 392
			info("register 0x%x contains 0x%x",
			     address, value);
		}
	}
	info("s5k4aa register state dump complete");

	for (page = 0; page < 16; page++) {
393
		m5602_write_sensor(sd, S5K4AA_PAGE_MAP, &page, 1);
394 395 396 397 398
		info("Probing for which registers that are "
		     "read/write for page 0x%x", page);
		for (address = 0; address <= 0xff; address++) {
			u8 old_value, ctrl_value, test_value = 0xff;

399
			m5602_read_sensor(sd, address, &old_value, 1);
400
			m5602_write_sensor(sd, address, &test_value, 1);
401
			m5602_read_sensor(sd, address, &ctrl_value, 1);
402 403 404 405 406 407 408

			if (ctrl_value == test_value)
				info("register 0x%x is writeable", address);
			else
				info("register 0x%x is read only", address);

			/* Restore original value */
409
			m5602_write_sensor(sd, address, &old_value, 1);
410 411 412
		}
	}
	info("Read/write register probing complete");
413
	m5602_write_sensor(sd, S5K4AA_PAGE_MAP, &old_page, 1);
414
}