>From d8f37b4340ea4cfd28d6e620f1b3224d946b9fab Mon Sep 17 00:00:00 2001 From: Xiaolin Zhang <xiaolin.zhang@xxxxxxxxx> Date: Thu, 30 Apr 2009 12:31:21 +0800 Subject: [PATCH] sensor pseduo driver in camera imaging on Intel moorestown platform. The moorestown platform with dual cameras will have one the same side as the display and the second on the oppsoite side of the display. The pseduo driver provides the uniform interface for isp kernel driver. Signed-off-by: Xiaolin Zhang <xiaolin.zhang@xxxxxxxxx> --- drivers/media/video/Makefile | 1 + drivers/media/video/mrstci/Kconfig | 1 + drivers/media/video/mrstci/include/ci_sensor_ioc.h | 57 + drivers/media/video/mrstci/include/sensordev.h | 119 +++ drivers/media/video/mrstci/mrstsensor/Kconfig | 9 + drivers/media/video/mrstci/mrstsensor/Makefile | 3 + drivers/media/video/mrstci/mrstsensor/mrstsensor.c | 1094 ++++++++++++++++++++ .../media/video/mrstci/mrstsensor/sensordev_priv.h | 37 + 8 files changed, 1321 insertions(+), 0 deletions(-) create mode 100644 drivers/media/video/mrstci/include/ci_sensor_ioc.h create mode 100644 drivers/media/video/mrstci/include/sensordev.h create mode 100644 drivers/media/video/mrstci/mrstsensor/Kconfig create mode 100644 drivers/media/video/mrstci/mrstsensor/Makefile create mode 100644 drivers/media/video/mrstci/mrstsensor/mrstsensor.c create mode 100644 drivers/media/video/mrstci/mrstsensor/sensordev_priv.h diff --git a/drivers/media/video/Makefile b/drivers/media/video/Makefile index f06f1cb..34a3461 100644 --- a/drivers/media/video/Makefile +++ b/drivers/media/video/Makefile @@ -152,6 +152,7 @@ obj-$(CONFIG_VIDEO_AU0828) += au0828/ obj-$(CONFIG_USB_VIDEO_CLASS) += uvc/ obj-$(CONFIG_VIDEO_MRST_ISP) += mrstci/mrstisp/ +obj-$(CONFIG_VIDEO_MRST_SENSOR) += mrstci/mrstsensor/ EXTRA_CFLAGS += -Idrivers/media/dvb/dvb-core EXTRA_CFLAGS += -Idrivers/media/dvb/frontends diff --git a/drivers/media/video/mrstci/Kconfig b/drivers/media/video/mrstci/Kconfig index 8f0a620..bf01447 100644 --- a/drivers/media/video/mrstci/Kconfig +++ b/drivers/media/video/mrstci/Kconfig @@ -10,6 +10,7 @@ if VIDEO_MRSTCI && VIDEO_V4L2 source "drivers/media/video/mrstci/mrstisp/Kconfig" +source "drivers/media/video/mrstci/mrstsensor/Kconfig" endif # VIDEO_MRSTCI diff --git a/drivers/media/video/mrstci/include/ci_sensor_ioc.h b/drivers/media/video/mrstci/include/ci_sensor_ioc.h new file mode 100644 index 0000000..80d3b0f --- /dev/null +++ b/drivers/media/video/mrstci/include/ci_sensor_ioc.h @@ -0,0 +1,57 @@ +/* + * Support for Moorestown Langwell Camera Imaging ISP subsystem. + * + * Copyright (c) 2009 Intel Corporation. All Rights Reserved. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License version + * 2 as published by the Free Software Foundation. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + * + * + * Xiaolin Zhang <xiaolin.zhang@xxxxxxxxx> + */ + +/* Sensor IOCTL */ +#ifndef _SENSOR_IOC_H +#define _SENSOR_IOC_H + +#ifdef __cplusplus +extern "C" { +#endif + +#define SENSOR_MAGIC 0x83 + +#define SENIOC_QUERYCAP _IOR(SENSOR_MAGIC, 0, struct ci_sensor_caps) +#define SENIOC_G_CONFIG _IOR(SENSOR_MAGIC, 1, struct ci_sensor_config) +#define SENIOC_S_CONFIG _IOW(SENSOR_MAGIC, 2, struct ci_sensor_config) +#define SENIOC_STREAM_ON _IO(SENSOR_MAGIC, 3) +#define SENIOC_STREAM_OFF _IO(SENSOR_MAGIC, 4) +#define SENIOC_G_REG _IOWR(SENSOR_MAGIC, 5, struct ci_sensor_reg) +#define SENIOC_S_REG _IOW(SENSOR_MAGIC, 6, struct ci_sensor_reg) +/* Get current focus position */ +#define SENIOC_MDI_G_FOCUS _IOR(SENSOR_MAGIC, 9, unsigned int) +/* Set focus to the given position */ +#define SENIOC_MDI_S_FOCUS _IOW(SENSOR_MAGIC, 10, unsigned int) +/* Trigger a forced calibration of focus hardware */ +#define SENIOC_MDI_CALIBRATE _IO(SENSOR_MAGIC, 11) +#define SENIOC_MDI_MAX_STEP _IOR(SENSOR_MAGIC, 12, unsigned int) +/* Get the max step hardware can support */ +#define SENIOC_ENUMPARM _IOWR(SENSOR_MAGIC, 13, struct ci_sensor_parm) +#define SENIOC_G_PARM _IOWR(SENSOR_MAGIC, 14, struct ci_sensor_parm) +#define SENIOC_S_PARM _IOW(SENSOR_MAGIC, 15, struct ci_sensor_parm) + +#ifdef __cplusplus +} +#endif + +#endif diff --git a/drivers/media/video/mrstci/include/sensordev.h b/drivers/media/video/mrstci/include/sensordev.h new file mode 100644 index 0000000..f4cf603 --- /dev/null +++ b/drivers/media/video/mrstci/include/sensordev.h @@ -0,0 +1,119 @@ +/* + * Support for Moorestown Langwell Camera Imaging ISP subsystem. + * + * Copyright (c) 2009 Intel Corporation. All Rights Reserved. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License version + * 2 as published by the Free Software Foundation. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + * + * + * Xiaolin Zhang <xiaolin.zhang@xxxxxxxxx> + */ + +#ifndef _SENSOR_DEV_H +#define _SENSOR_DEV_H + +#include "ci_sensor_common.h" + +/* I2C setting for both sensor and motor */ +struct i2c_setting{ + struct i2c_client *sensor_client; + struct i2c_client *motor_client; +}; + +/* Sensor description structure */ +struct sensor_device{ + /* Note: Pointers to not supported parts remain NULL! */ + struct device class_dev; + struct device *dev; + + /* Name of the camera */ + const char name[32]; + /* Type of the camera, SoC or Raw */ + int type; + /* Minor number for sensor device */ + int minor; + /* Desired camera input clock for the camera */ + unsigned int clock_freq; + /* Pointer to default configuration */ + const struct ci_sensor_config *def_config; + /* I2C for sensor and motor */ + struct i2c_setting *i2c; + + /* pointers to the sensor specific functions */ + /* Device open */ + int (*open) (struct i2c_setting *, void *); + /* Close device and power-off */ + int (*release) (struct i2c_setting *, void *); + int (*on) (struct i2c_setting *); + int (*off) (struct i2c_setting *); + /* Return sensor capablity */ + int (*querycap) (struct i2c_client *, struct ci_sensor_caps *); + /* Return sensor current configuration */ + int (*get_config) (struct i2c_client *, struct ci_sensor_config *); + /* Set sensor configration into sensor */ + int (*set_config) (struct i2c_client *, + const struct ci_sensor_config *); + /* Raw sensor specific function to set black level control */ + int (*set_blc) (struct i2c_client *, + const struct ci_sensor_blc_mean *); + /* Raw sensor specific function to set auto white balance */ + int (*set_awb) (struct i2c_client *, + const struct ci_sensor_awb_mean *); + /* Raw sensor specific function to set auto exposure control */ + int (*set_aec) (struct i2c_client *, + const struct ci_sensor_aec_mean *); + /* Return sensor control info */ + int (*enum_parm) (struct i2c_client *, struct ci_sensor_parm *); + /* return sensor control value */ + int (*get_parm) (struct i2c_client *, struct ci_sensor_parm *); + /* Set sensor control value into sensor */ + int (*set_parm) (struct i2c_client *, struct ci_sensor_parm *); + /* + * Raw sensor specific function to get lens correction configuration + * value + */ + int (*get_ls_corr_config) (struct i2c_client *, + struct ci_sensor_ls_corr_config *); + /* Return sensor supoorted resolution */ + int (*try_res) (struct i2c_client *, unsigned short *, + unsigned short *); + /* Set resolution into sensor */ + int (*set_res) (struct i2c_client *, const int, const int); + + int (*mdi_get_focus) (struct i2c_client *, unsigned int *focus); + int (*mdi_set_focus) (struct i2c_client *, unsigned int *focus); + + int (*mdi_calibrate) (struct i2c_client *); + int (*mdi_max_step) (struct i2c_client *, unsigned int *step); + + int (*read) (struct i2c_client *, uint32_t addr, uint32_t *val); + int (*write) (struct i2c_client *, uint32_t addr, uint32_t val); + int (*suspend) (void); + int (*resume) (void); + /* Sensor private data */ + void *data; + struct mutex lock; + int state; +}; + +/* + * Sensor register function for to register device into camera sensor + * framework + */ +int sensor_register_device(struct sensor_device *, int); +void sensor_unregister_device(struct sensor_device *); +int ci_sensor_res2size(u32, u16 *, u16 *); +/* _SENSOR_DEV_H */ +#endif diff --git a/drivers/media/video/mrstci/mrstsensor/Kconfig b/drivers/media/video/mrstci/mrstsensor/Kconfig new file mode 100644 index 0000000..97f8817 --- /dev/null +++ b/drivers/media/video/mrstci/mrstsensor/Kconfig @@ -0,0 +1,9 @@ +config VIDEO_MRST_SENSOR + tristate "Moorestown Sensor Pseduo Driver" + depends on VIDEO_MRST_ISP + default y + ---help--- + Say Y here if you want to support for cameras based on the Intel Moorestown platform. + + To compile this driver as a module, choose M here: the + module will be called mrstsensor.ko. diff --git a/drivers/media/video/mrstci/mrstsensor/Makefile b/drivers/media/video/mrstci/mrstsensor/Makefile new file mode 100644 index 0000000..d0295d4 --- /dev/null +++ b/drivers/media/video/mrstci/mrstsensor/Makefile @@ -0,0 +1,3 @@ +obj-$(CONFIG_VIDEO_MRST_SENSOR) += mrstsensor.o + +EXTRA_CFLAGS += -I$(src)/../include \ No newline at end of file diff --git a/drivers/media/video/mrstci/mrstsensor/mrstsensor.c b/drivers/media/video/mrstci/mrstsensor/mrstsensor.c new file mode 100644 index 0000000..40aee6d --- /dev/null +++ b/drivers/media/video/mrstci/mrstsensor/mrstsensor.c @@ -0,0 +1,1094 @@ +/* + * Support for Moorestown Langwell Camera Imaging ISP subsystem. + * + * Copyright (c) 2009 Intel Corporation. All Rights Reserved. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License version + * 2 as published by the Free Software Foundation. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + * + * + * Xiaolin Zhang <xiaolin.zhang@xxxxxxxxx> + */ + +#include <linux/module.h> +#include <linux/types.h> +#include <linux/kernel.h> +#include <linux/mm.h> +#include <linux/string.h> +#include <linux/errno.h> +#include <linux/init.h> +#include <linux/kmod.h> +#include <linux/device.h> +#include <linux/i2c.h> +#include <linux/fs.h> +#include <linux/videodev2.h> +#include <linux/uaccess.h> +#include <asm/system.h> +#include <linux/gpio.h> + +#include "sensordev_priv.h" + +#define GPIO_SCLK_25 44 +#define GPIO_STDBY1_PIN 48 +#define GPIO_STDBY2_PIN 49 +#define GPIO_RESET_PIN 50 + +static int def_sensor; +static struct sensor_device *sensor_device[AMOUNT_OF_CAM_DRIVERS]; +static struct sensor_device *sensor_cur; +static DEFINE_MUTEX(sensordev_lock); + +/* + * sysfs stuff + */ +static ssize_t show_name(struct device *cd, + struct device_attribute *attr, char *buf) +{ + struct sensor_device *sdp = container_of(cd, struct sensor_device, + class_dev); + /* Need to get back here */ + return sprintf(buf, "%.*s\n", (int)sizeof(sdp->name), sdp->name); +} + +static struct device_attribute sensor_device_attrs[] = { + __ATTR(name, S_IRUGO, show_name, NULL), + __ATTR_NULL +}; + +static void sensor_class_release(struct device *cd) +{ + ; +} + +static struct class sensor_class = { + .name = "mrstsensor", + .dev_attrs = sensor_device_attrs, + .dev_release = sensor_class_release, +}; + +/* + * Active devices + */ +static struct sensor_device *sensor_devdata(struct file *file) +{ + return sensor_device[iminor(file->f_path.dentry->d_inode)]; +} + +int ci_sensor_res2size(unsigned int res, unsigned short *h_size, + unsigned short *v_size) +{ + unsigned short hsize; + unsigned short vsize; + int err = 0; + + switch (res) { + case SENSOR_RES_QQCIF: + hsize = QQCIF_SIZE_H; + vsize = QQCIF_SIZE_V; + break; + case SENSOR_RES_QQVGA: + hsize = QQVGA_SIZE_H; + vsize = QQVGA_SIZE_V; + break; + case SENSOR_RES_QCIF: + hsize = QCIF_SIZE_H; + vsize = QCIF_SIZE_V; + break; + case SENSOR_RES_QVGA: + hsize = QVGA_SIZE_H; + vsize = QVGA_SIZE_V; + break; + case SENSOR_RES_CIF: + hsize = CIF_SIZE_H; + vsize = CIF_SIZE_V; + break; + case SENSOR_RES_VGA: + hsize = VGA_SIZE_H; + vsize = VGA_SIZE_V; + break; + case SENSOR_RES_SVGA: + hsize = SVGA_SIZE_H; + vsize = SVGA_SIZE_V; + break; + case SENSOR_RES_XGA: + hsize = XGA_SIZE_H; + vsize = XGA_SIZE_V; + break; + case SENSOR_RES_XGA_PLUS: + hsize = XGA_PLUS_SIZE_H; + vsize = XGA_PLUS_SIZE_V; + break; + case SENSOR_RES_SXGA: + hsize = SXGA_SIZE_H; + vsize = SXGA_SIZE_V; + break; + case SENSOR_RES_UXGA: + hsize = UXGA_SIZE_H; + vsize = UXGA_SIZE_V; + break; + case SENSOR_RES_QXGA: + hsize = QXGA_SIZE_H; + vsize = QXGA_SIZE_V; + break; + case SENSOR_RES_QSXGA: + hsize = QSXGA_SIZE_H; + vsize = QSXGA_SIZE_V; + break; + case SENSOR_RES_QSXGA_PLUS: + hsize = QSXGA_PLUS_SIZE_H; + vsize = QSXGA_PLUS_SIZE_V; + break; + case SENSOR_RES_QSXGA_PLUS2: + hsize = QSXGA_PLUS2_SIZE_H; + vsize = QSXGA_PLUS2_SIZE_V; + break; + case SENSOR_RES_QSXGA_PLUS3: + hsize = QSXGA_PLUS3_SIZE_H; + vsize = QSXGA_PLUS3_SIZE_V; + break; + case SENSOR_RES_WQSXGA: + hsize = WQSXGA_SIZE_H; + vsize = WQSXGA_SIZE_V; + break; + case SENSOR_RES_QUXGA: + hsize = QUXGA_SIZE_H; + vsize = QUXGA_SIZE_V; + break; + case SENSOR_RES_WQUXGA: + hsize = WQUXGA_SIZE_H; + vsize = WQUXGA_SIZE_V; + break; + case SENSOR_RES_HXGA: + hsize = HXGA_SIZE_H; + vsize = HXGA_SIZE_V; + break; + case SENSOR_RES_RAWMAX: + hsize = RAWMAX_SIZE_H; + vsize = RAWMAX_SIZE_V; + break; + case SENSOR_RES_YUV_HMAX: + hsize = YUV_HMAX_SIZE_H; + vsize = YUV_HMAX_SIZE_V; + break; + case SENSOR_RES_YUV_VMAX: + hsize = YUV_VMAX_SIZE_H; + vsize = YUV_VMAX_SIZE_V; + break; + case SENSOR_RES_BP1: + hsize = BP1_SIZE_H; + vsize = BP1_SIZE_V; + break; + case SENSOR_RES_L_AFM: + hsize = L_AFM_SIZE_H; + vsize = L_AFM_SIZE_V; + break; + case SENSOR_RES_M_AFM: + hsize = M_AFM_SIZE_H; + vsize = M_AFM_SIZE_V; + break; + case SENSOR_RES_S_AFM: + hsize = S_AFM_SIZE_H; + vsize = S_AFM_SIZE_V; + break; + + case SENSOR_RES_QXGA_PLUS: + hsize = QXGA_PLUS_SIZE_H; + vsize = QXGA_PLUS_SIZE_V; + break; + + case SENSOR_RES_1080P: + hsize = RES_1080P_SIZE_H; + vsize = 1080; + break; + + case SENSOR_RES_720P: + hsize = RES_720P_SIZE_H; + vsize = RES_720P_SIZE_V; + break; + + default: + hsize = 0; + vsize = 0; + err = -1; + printk(KERN_ERR "ci_sensor_res2size: Resolution 0x%08x" + "unknown\n", res); + break; + } + + if (h_size != NULL) + *h_size = hsize; + if (v_size != NULL) + *v_size = vsize; + + return err; +} +EXPORT_SYMBOL(ci_sensor_res2size); + +static int __sensor_do_ioctl(struct inode *inode, struct file *file, + unsigned int cmd, void *arg) +{ + struct sensor_device *sdp = sensor_devdata(file); + struct i2c_client *sc = sdp->i2c->sensor_client; + struct i2c_client *mc = sdp->i2c->motor_client; + + int err = -EINVAL; + + if (sdp == NULL) { + printk(KERN_WARNING "Device should be open first\n"); + return -1; + } + + if (sdp->state == SENSOR_SUSPEND) { + printk(KERN_INFO "Device in suspend mode, wakeup first\n"); + /* ci_sensor_resume(); */ + return -1; + } + + switch (cmd) { + case SENIOC_G_REG: + { + struct ci_sensor_reg *reg = (struct ci_sensor_reg *)arg; + + if (!sdp->read) + break; + err = sdp->read(sc, reg->addr, ®->value); + + break; + } + case SENIOC_S_REG: + { + struct ci_sensor_reg *reg = (struct ci_sensor_reg *)arg; + + if (!sdp->write) + break; + err = sdp->write(sc, reg->addr, reg->value); + + break; + } + case SENIOC_STREAM_ON: + { + if (sensor_cur != NULL) { + printk(KERN_ERR "sensor_stream_on: Another camera is" + "in use, close it first\n"); + break; + } + + if (sensor_cur == sdp) { + /* Sensor is in use, stop here */ + printk(KERN_WARNING "sensor_stream_on: The camera is" + "already in use\n"); + err = 0; + break; + } + + if (sdp->on) + err = sdp->on(sdp->i2c); + if (err) { + printk(KERN_ERR "sensor_stream_on: Error while open" + "the device\n"); + break; + } + + /* Need to wake up this sensor */ + sensor_cur = sdp; + err = 0; + break; + } + + case SENIOC_STREAM_OFF: + { + if (sensor_cur != sdp) { + printk(KERN_WARNING "sensor_stream_off: The device is" + "not in use\n"); + break; + } + + if (sdp->off) + err = sdp->off(sdp->i2c); + if (err) { + printk(KERN_ERR "sensor_stream_off: Error while off the" + " device\n"); + break; + } + + sensor_cur = NULL; + break; + } + /* --- capabilities ----------------------------------------------- */ + case SENIOC_QUERYCAP: + { + struct ci_sensor_caps *cap = (struct ci_sensor_caps *)arg; + memset(cap, 0, sizeof(*cap)); + + if (!sdp->querycap) + break; + + err = sdp->querycap(sc, cap); + break; + } + + /* --- configuration ioctls --------------------------------------- */ + case SENIOC_G_CONFIG: + { + struct ci_sensor_config *config; + config = (struct ci_sensor_config *)arg; + memset(config, 0, sizeof(*config)); + + if (!sdp->get_config) + break; + + err = sdp->get_config(sc, config); + break; + } + case SENIOC_S_CONFIG: + { + struct ci_sensor_config *config; + config = (struct ci_sensor_config *)arg; + + if (!sdp->set_config) + break; + + err = sdp->set_config(sc, config); + break; + + } + + /* --- Parameters ioctls ------------------------------------------- */ + case SENIOC_ENUMPARM: + { + struct ci_sensor_parm *parm; + parm = (struct ci_sensor_parm *)arg; + + if (!sdp->enum_parm) + break; + + err = sdp->enum_parm(sc, parm); + break; + } + case SENIOC_G_PARM: + { + struct ci_sensor_parm *parm; + parm = (struct ci_sensor_parm *)arg; + + if (!sdp->get_parm) + break; + + err = sdp->get_parm(sc, parm); + break; + } + case SENIOC_S_PARM: + { + struct ci_sensor_parm *parm; + parm = (struct ci_sensor_parm *)arg; + + if (!sdp->set_parm) + break; + + err = sdp->set_parm(sc, parm); + break; + } + case SENIOC_MDI_G_FOCUS: + { + unsigned int *focus = (unsigned int *)arg; + + if (!sdp->mdi_get_focus) + break; + + err = sdp->mdi_get_focus(mc, focus); + break; + } + case SENIOC_MDI_S_FOCUS: + { + unsigned int *focus = (unsigned int *)arg; + + if (!sdp->mdi_set_focus) + break; + + err = sdp->mdi_set_focus(mc, focus); + break; + } + case SENIOC_MDI_CALIBRATE: + { + if (!sdp->mdi_calibrate) + break; + + err = sdp->mdi_calibrate(mc); + break; + } + case SENIOC_MDI_MAX_STEP: + { + unsigned int *step = (unsigned int *)arg; + + if (!sdp->mdi_max_step) + break; + + err = sdp->mdi_max_step(mc, step); + break; + } + default: + printk(KERN_WARNING "Unmatched IOCTL: %d in CI\n", cmd); + } /* switch */ + + return err; +} + +int ci_sensor_ioctl(struct inode *inode, struct file *file, + unsigned int cmd, unsigned long arg) +{ + char sbuf[128]; + void *mbuf = NULL; + void *parg = NULL; + int err = -EINVAL; + + /* Copy arguments into temp kernel buffer */ + switch (_IOC_DIR(cmd)) { + case _IOC_NONE: + parg = NULL; + break; + case _IOC_READ: + case _IOC_WRITE: + case (_IOC_WRITE | _IOC_READ): + if (_IOC_SIZE(cmd) <= sizeof(sbuf)) + parg = sbuf; + else { + /* too big to allocate from stack */ + mbuf = kmalloc(_IOC_SIZE(cmd), GFP_KERNEL); + if (NULL == mbuf) + return -ENOMEM; + parg = mbuf; + } + + err = -EFAULT; + if (_IOC_DIR(cmd) & _IOC_WRITE) + if (copy_from_user(parg, (void __user *)arg, + _IOC_SIZE(cmd))) + goto out; + break; + } + + /* Handles IOCTL */ + err = __sensor_do_ioctl(inode, file, cmd, parg); + if (err == -ENOIOCTLCMD) + err = -EINVAL; + + switch (_IOC_DIR(cmd)) { + case _IOC_READ: + case (_IOC_WRITE | _IOC_READ): + if (copy_to_user((void __user *)arg, parg, _IOC_SIZE(cmd))) + err = -EFAULT; + break; + } +out: + kfree(mbuf); + return err; +} +EXPORT_SYMBOL(ci_sensor_ioctl); + +int ci_sensor_get_caps(struct ci_sensor_caps *caps) +{ + int err = 0; + struct i2c_client *sc; + + if (caps == NULL) + return -EIO; + + if (sensor_cur == NULL) { + printk(KERN_WARNING "sensor_get_caps: No current camera\n"); + return -ENODEV; + } + sc = sensor_cur->i2c->sensor_client; + /* init struct with 0 */ + memset(caps, 0, sizeof(*caps)); + + if (sensor_cur->querycap != NULL) + err = sensor_cur->querycap(sc, caps); + else { + err = -EIO; + printk(KERN_WARNING "sensor_get_caps: Not supported!\n"); + } + + return err; +} +EXPORT_SYMBOL(ci_sensor_get_caps); + +int ci_sensor_get_config(struct ci_sensor_config *config) +{ + int err = 0; + struct i2c_client *sc; + + if (config == NULL) + return -EIO; + + if (sensor_cur == NULL) { + printk(KERN_WARNING "sensor_get_config: No current camera\n"); + return -ENODEV; + } + sc = sensor_cur->i2c->sensor_client; + /* init struct with 0 */ + memset(config, 0, sizeof(*config)); + + if (sensor_cur->get_config != NULL) + err = sensor_cur->get_config(sc, config); + else { + err = -EIO; + printk(KERN_WARNING "sensor_get_config: Not supported!\n"); + } + + return err; +} +EXPORT_SYMBOL(ci_sensor_get_config); + +int ci_sensor_get_ls_corr_config(struct ci_sensor_ls_corr_config *config) +{ + int err = 0; + struct i2c_client *sc; + + if (config == NULL) + return -EIO; + + if (sensor_cur == NULL) { + printk(KERN_WARNING "sensor_get_ls_config: No current" + "camera\n"); + return -ENODEV; + } + sc = sensor_cur->i2c->sensor_client; + /* init struct with 0 */ + memset(config, 0, sizeof(*config)); + + if (sensor_cur->get_ls_corr_config != NULL) + err = sensor_cur->get_ls_corr_config(sc, config); + else { + err = -EIO; + printk(KERN_WARNING "sensor_get_config: Not supported!\n"); + } + + return err; +} +EXPORT_SYMBOL(ci_sensor_get_ls_corr_config); + +int ci_sensor_suspend(void) +{ + int err = 0; + + if (sensor_cur == NULL) + return 0; + + if (sensor_cur->suspend != NULL) + err = sensor_cur->suspend(); + if (!err) + sensor_cur->state = SENSOR_SUSPEND; + + return err; +} +EXPORT_SYMBOL(ci_sensor_suspend); + +int ci_sensor_resume(void) +{ + int err = 0; + + if (sensor_cur == NULL) + return 0; + + if (sensor_cur->resume != NULL) + err = sensor_cur->resume(); + if (!err) + sensor_cur->state = SENSOR_RUNNING; + + return err; +} +EXPORT_SYMBOL(ci_sensor_resume); + +/* This will start default sensor */ +int ci_sensor_start(void) +{ + int err = 0; + int offset; + struct sensor_device *vfl; + + offset = def_sensor; + mutex_lock(&sensordev_lock); + if (!sensor_device[offset]) { + if (!sensor_device[offset + MINOR_MAX]) { + mutex_unlock(&sensordev_lock); + printk(KERN_WARNING "sensor_start: NULL device\n"); + return -EIO; + } else + offset += MINOR_MAX; + } + + vfl = sensor_device[offset]; + if (sensor_cur == vfl) { + mutex_unlock(&sensordev_lock); + /* Sensor is in use, stop here */ + printk(KERN_INFO "sensor_start: The camera is" + "already in use\n"); + return 0; + } + if (sensor_cur != NULL) { + mutex_unlock(&sensordev_lock); + printk(KERN_ERR "sensor_start: Another camera is in" + "use, close it first\n"); + return -EIO; + } + + if (vfl->open) + err = vfl->open(vfl->i2c, vfl->data); + if (err) { + mutex_unlock(&sensordev_lock); + printk(KERN_ERR "sensor_start: Error while open device\n"); + return -EIO; + } + if (vfl->on) + err = vfl->on(vfl->i2c); + if (err) { + mutex_unlock(&sensordev_lock); + printk(KERN_ERR "sensor_start: Error while open device\n"); + return -EIO; + } + + /* Need to wake up this sensor */ + sensor_cur = vfl; + sensor_cur->state = SENSOR_RUNNING; + mutex_unlock(&sensordev_lock); + + return err; +} +EXPORT_SYMBOL(ci_sensor_start); + +int ci_sensor_stop(void) +{ + int err = 0; + struct sensor_device *vfl; + int offset; + + offset = def_sensor; + mutex_lock(&sensordev_lock); + if (!sensor_device[offset]) { + if (!sensor_device[offset + MINOR_MAX]) { + mutex_unlock(&sensordev_lock); + printk(KERN_WARNING "sensor_stop: NULL device\n"); + return -EIO; + } else + offset += MINOR_MAX; + } + + vfl = sensor_device[offset]; + if (vfl == NULL) { + mutex_unlock(&sensordev_lock); + printk(KERN_WARNING "sensor_stop: No such device\n"); + return -ENODEV; + } + + if (sensor_cur != vfl) { + mutex_unlock(&sensordev_lock); + printk(KERN_WARNING "sensor_stop: The device is not in" + "use\n"); + return -EIO; + } + if (vfl->release) + err = vfl->release(vfl->i2c, vfl->data); + if (err) { + mutex_unlock(&sensordev_lock); + printk(KERN_ERR "sensor_stop: Error while release" + "device\n"); + return -EIO; + } + + vfl->state = SENSOR_INIT; + sensor_cur = NULL; + mutex_unlock(&sensordev_lock); + + return 0; +} +EXPORT_SYMBOL(ci_sensor_stop); + +int ci_sensor_try_mode(int *width, int *high) +{ + int err = 0; + struct i2c_client *sc; + + if (sensor_cur == NULL) { + printk(KERN_WARNING "sensor_try_mode: No current camera\n"); + return -ENODEV; + } + + sc = sensor_cur->i2c->sensor_client; + if (sensor_cur->try_res != NULL) + err = sensor_cur->try_res(sc, (u16 *)width, (u16 *)high); + else { + err = -EIO; + printk(KERN_WARNING "sensor_try_mode: Not supported!\n"); + } + + return err; +} +EXPORT_SYMBOL(ci_sensor_try_mode); + +int ci_sensor_set_mode(const int width, const int high) +{ + int err = 0; + struct i2c_client *sc; + + if (sensor_cur == NULL) { + printk(KERN_WARNING "sensor_set_mode: No current camera\n"); + return -ENODEV; + } + + sc = sensor_cur->i2c->sensor_client; + if (sensor_cur->set_res != NULL) + err = sensor_cur->set_res(sc, width, high); + else { + err = -EIO; + printk(KERN_WARNING "sensor_set_mode: Not supported!\n"); + } + + return err; +} +EXPORT_SYMBOL(ci_sensor_set_mode); + +int ci_sensor_queryctrl(struct v4l2_queryctrl *qc) +{ + int err = 0; + struct ci_sensor_parm *parm; + struct i2c_client *sc; + + if (qc == NULL) + return -EIO; + + if (sensor_cur == NULL) { + printk(KERN_WARNING "sensor_queryctrl: No current camera\n"); + return -ENODEV; + } + + sc = sensor_cur->i2c->sensor_client; + parm = kzalloc(sizeof(struct ci_sensor_parm), GFP_KERNEL); + if (parm == NULL) + return -ENOMEM; + parm->index = qc->id; + if (sensor_cur->enum_parm != NULL) + err = sensor_cur->enum_parm(sc, parm); + else { + err = -EIO; + printk(KERN_WARNING "sensor_queryctrl: Not supported!\n"); + } + + if (!err) { + qc->type = parm->type; + /* XXX */ + strncpy(qc->name, parm->name, 32); + qc->minimum = parm->min; + qc->maximum = parm->max; + qc->step = parm->step; + qc->default_value = parm->def_value; + qc->flags = parm->flags; + } + kfree(parm); + + return err; +} +EXPORT_SYMBOL(ci_sensor_queryctrl); + +int ci_sensor_get_ctrl(struct v4l2_control *qc) +{ + int err = 0; + struct ci_sensor_parm *parm; + struct i2c_client *sc; + + if (qc == NULL) + return -EIO; + + if (sensor_cur == NULL) { + printk(KERN_WARNING "sensor_get_ctrl: No current camera\n"); + return -ENODEV; + } + + sc = sensor_cur->i2c->sensor_client; + parm = kzalloc(sizeof(struct ci_sensor_parm), GFP_KERNEL); + if (parm == NULL) + return -ENOMEM; + parm->index = qc->id; + if (sensor_cur->get_parm != NULL) + err = sensor_cur->get_parm(sc, parm); + else { + err = -EIO; + printk(KERN_WARNING "sensor_get_ctrl: Not supported!\n"); + } + if (!err) + qc->value = parm->value; + + kfree(parm); + return err; +} +EXPORT_SYMBOL(ci_sensor_get_ctrl); + +int ci_sensor_set_ctrl(struct v4l2_control *qc) +{ + int err = 0; + struct ci_sensor_parm *parm; + struct i2c_client *sc; + + if (qc == NULL) + return -EIO; + + if (sensor_cur == NULL) { + printk(KERN_WARNING "sensor_set_ctrl: No current camera\n"); + return -ENODEV; + } + + sc = sensor_cur->i2c->sensor_client; + parm = kzalloc(sizeof(struct ci_sensor_parm), GFP_KERNEL); + if (parm == NULL) + return -ENOMEM; + parm->index = qc->id; + parm->value = qc->value; + /* XXX */ + if (sensor_cur->set_parm != NULL) + err = sensor_cur->set_parm(sc, parm); + else { + err = -EIO; + printk(KERN_WARNING "sensor_set_ctrl: Not supported!\n"); + } + + kfree(parm); + return err; +} +EXPORT_SYMBOL(ci_sensor_set_ctrl); + +/* + * Specific sensor driver register function + */ +int sensor_register_device(struct sensor_device *sdp, int type) +{ + int nr, err; + int base; + char *name; + + switch (type) { + case SENSOR_TYPE_SOC: + base = 0; + name = "SoC"; + break; + case SENSOR_TYPE_RAW: + base = MINOR_MAX; + name = "Raw"; + break; + default: + printk(KERN_ERR "%s called with unknown type: %d\n", + __func__, type); + return -1; + } + + /* Check if already have a minor number */ + if (sdp->minor != -1) + return -ENFILE; + + /* Get the minor number */ + mutex_lock(&sensordev_lock); + for (nr = base; nr < (base + MINOR_MAX); nr++) + if (NULL == sensor_device[nr]) + break; + if (nr == (base + MINOR_MAX)) { + mutex_unlock(&sensordev_lock); + return -ENFILE; + } + + sensor_device[nr] = sdp; + sdp->minor = nr; + + mutex_unlock(&sensordev_lock); + mutex_init(&sdp->lock); + + /* sysfs class */ + memset(&sdp->class_dev, 0x00, sizeof(sdp->class_dev)); + if (sdp->dev) + sdp->class_dev.parent = sdp->dev; + sdp->class_dev.class = &sensor_class; + sdp->class_dev.devt = MKDEV(SENSOR_MAJOR, sdp->minor); + dev_set_name(&sdp->class_dev, "%s%d", name, nr); + err = device_register(&sdp->class_dev); + if (err < 0) { + printk(KERN_ERR "%s: device_register failed\n", + __func__); + goto fail_minor; + } + + return 0; + +fail_minor: + mutex_lock(&sensordev_lock); + sensor_device[sdp->minor] = NULL; + sdp->minor = -1; + mutex_unlock(&sensordev_lock); + return err; +} +EXPORT_SYMBOL(sensor_register_device); + +void sensor_unregister_device(struct sensor_device *sdp) +{ + mutex_lock(&sensordev_lock); + if (sensor_device[sdp->minor] != sdp) + panic("sensordev: bad unregister"); + + if (sensor_device[sdp->minor] == sensor_cur) + sensor_cur = NULL; + sensor_device[sdp->minor] = NULL; + device_unregister(&sdp->class_dev); + mutex_unlock(&sensordev_lock); +} +EXPORT_SYMBOL(sensor_unregister_device); + +static int ci_sensor_release(struct inode *inode, struct file *file) +{ + unsigned int minor = iminor(inode); + int err = 0; + struct sensor_device *vfl; + + if (minor < 0 || minor >= AMOUNT_OF_CAM_DRIVERS) + return -ENODEV; + + vfl = sensor_device[minor]; + if (vfl == NULL) { + printk(KERN_WARNING "sensor_release: No such device\n"); + return -ENODEV; + } + + if (vfl == sensor_cur) { + if (vfl->release) + err = vfl->release(vfl->i2c, vfl->data); + if (err) { + printk(KERN_ERR "sensor_release: Error while close" + "device\n"); + return -EIO; + } + + mutex_lock(&sensordev_lock); + sensor_cur = NULL; + vfl->state = SENSOR_INIT; + mutex_unlock(&sensordev_lock); + } + return 0; +} + +/* + * Open a sensor device + */ +static int ci_sensor_open(struct inode *inode, struct file *file) +{ + unsigned int minor = iminor(inode); + int err = 0; + struct sensor_device *vfl; + + if (minor >= AMOUNT_OF_CAM_DRIVERS || minor < 0) + return -ENODEV; + + mutex_lock(&sensordev_lock); + vfl = sensor_device[minor]; + if (vfl == NULL) { + mutex_unlock(&sensordev_lock); + printk(KERN_WARNING "sensor_open: There is no such device\n"); + return -EIO; + } else { + if (sensor_cur == vfl) { + mutex_unlock(&sensordev_lock); + /* Sensor is in use, stop here */ + printk(KERN_WARNING "sensor_open: The camera is" + "already in use\n"); + return 0; + } + + if (vfl->open) + err = vfl->open(vfl->i2c, vfl->data); + if (err) { + mutex_unlock(&sensordev_lock); + printk(KERN_ERR "sensor_open: Fail to open device\n"); + return -EIO; + } + err = 0; + } + vfl->state = SENSOR_INIT; + mutex_unlock(&sensordev_lock); + + return err; +} + +/* + * Sensor fs opration + */ +static const struct file_operations sensor_fops = { + .owner = THIS_MODULE, + .llseek = no_llseek, + .open = ci_sensor_open, + .release = ci_sensor_release, + .ioctl = ci_sensor_ioctl, +}; + +/* + * Initialize sensor module + */ +static int __init sensordev_init(void) +{ + int ret; + + printk(KERN_INFO "Sensor device interface\n"); + if (register_chrdev(SENSOR_MAJOR, SENSOR_NAME, &sensor_fops)) { + printk(KERN_WARNING "sensor_dev: unable to get major %d\n", + SENSOR_MAJOR); + return -EIO; + } + + ret = class_register(&sensor_class); + if (ret < 0) { + unregister_chrdev(SENSOR_MAJOR, SENSOR_NAME); + printk(KERN_WARNING "sensor_dev: class register failed\n"); + return -EIO; + } + + /* Enable sensor related GPIO in system */ + gpio_direction_output(GPIO_STDBY1_PIN, 0); + gpio_direction_output(GPIO_STDBY2_PIN, 0); + gpio_direction_output(GPIO_RESET_PIN, 1); + gpio_direction_output(GPIO_SCLK_25, 0); + /* gpio_direction_output(GPIO_AF_PD, 1); */ + + #if 0 /* disabled the mrst gpio driver function call, need to turn on in futher */ + gpio_alt_func(GPIO_STDBY1_PIN, 0); + gpio_alt_func(GPIO_STDBY2_PIN, 0); + gpio_alt_func(GPIO_RESET_PIN, 0); + gpio_alt_func(GPIO_SCLK_25, 1); + #endif + def_sensor = 0; + sensor_cur = NULL; + return 0; +} + +static void __exit sensordev_exit(void) +{ + class_unregister(&sensor_class); + unregister_chrdev(SENSOR_MAJOR, SENSOR_NAME); + + gpio_direction_output(GPIO_STDBY1_PIN, 1); + gpio_direction_output(GPIO_STDBY2_PIN, 1); + /* gpio_direction_output(GPIO_AF_PD, 0); */ +} + +module_init(sensordev_init); +module_exit(sensordev_exit); + +MODULE_AUTHOR("Xiaolin Zhang <xiaolin.zhang@xxxxxxxxx>"); +MODULE_DESCRIPTION("Device registrar for Intel Moorestown camera sensor" + "drivers"); +MODULE_LICENSE("GPL"); + +module_param(def_sensor, int, S_IRUGO); +MODULE_PARM_DESC(def_sensor, "Default sensor number to open"); diff --git a/drivers/media/video/mrstci/mrstsensor/sensordev_priv.h b/drivers/media/video/mrstci/mrstsensor/sensordev_priv.h new file mode 100644 index 0000000..433e69b --- /dev/null +++ b/drivers/media/video/mrstci/mrstsensor/sensordev_priv.h @@ -0,0 +1,37 @@ +/* + * Support for Moorestown Langwell Camera Imaging ISP subsystem. + * + * Copyright (c) 2009 Intel Corporation. All Rights Reserved. + * + * Copyright (c) Silicon Image 2008 www.siliconimage.com + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License version + * 2 as published by the Free Software Foundation. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + * + * + * Xiaolin Zhang <xiaolin.zhang@xxxxxxxxx> + */ + +#include "sensordev.h" +#include "ci_sensor_ioc.h" + +#define SENSOR_MAJOR 260 +#define SENSOR_NAME "IntelSensor" +#define MINOR_MAX 2 +#define AMOUNT_OF_CAM_DRIVERS 4 +#define V4L2_DEF_SEN SENSOR_TYPE_SOC + +#define SENSOR_INIT 0 +#define SENSOR_RUNNING 1 +#define SENSOR_SUSPEND 2 -- 1.5.5 -- To unsubscribe from this list: send the line "unsubscribe linux-media" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html