网络摄像头4 cmos 0v9650,driver

将ov9650的驱动编译进内核
 Device Drivers  ---> <*> Multimedia support  --->  <*>   OV9650 on the S3C2440 driver

查看这项的help,可知符号
Symbol: S3C2440_CAMERA [=y]    
在目录/opt/FriendlyArm/mini2440/linux-2.6.32.2/drivers/media/video的Makefile里面找到以下2行

obj-$(CONFIG_S3C2440_CAMERA) += s3c2440camera.o
s3c2440camera-objs    :=    s3c2440_ov9650.o sccb.o s3c2440camif.o

可见0v9650的驱动由3个c文件构成,如下
s3c2440camif.c
s3c2440_ov9650.c
sccb.c
还有两个h文件
s3c2440camif.h
sccb.h
(并且可如果编译成模块的话,生成s3c2440camera.ko)


其中
s3c2440camif.c用于从cmos接口获取图像数据和将数据传输到进程空间(在有app读取时)。比如创建misc设备,设备名字是camera。在misc中仅实现3个方法open,read,release。没有ioctl(和mmap),所以在网络摄像头2里所看到的来自app的ioctl会失败。
s3c2440_ov9650.c 读取和配置ov9650寄存器。通过iic接口传输数据。设备地址是60(#define OV9650_SCCB_ADDR    0x60).比如进行初始化和product id获取.
sccb.c 定义了去读ov9650的寄存器的具体方法,是时序模拟的iic。而s3c2440_ov9650.c里是调用这些具体方法去读写ov9650的寄存器的。
**************************************************************硬件连接******************************************
cmos ov9650的引脚

2440处理器cmos接口

用于传输ov9650的寄存器配置信息即工作参数,ov9650<--->2440
电源控制,ov9650<---2440
用户ov9650传输捕获的图像数据,ov9650--->2440

查看ov9650的spec可知其输出的数据格式


**************************************************************驱动分析****************************************************

/*
 * camif_init()
 */
static int __init camif_init(void)
{
	int ret;
	struct s3c2440camif_dev * pdev;
	struct clk * camif_upll_clk;

	printk(KERN_ALERT"initializing s3c2440 camera interface......\n");

	pdev = &camera;

	/* set gpio-j to camera mode. */
	s3c2410_gpio_cfgpin(S3C2440_GPJ0, S3C2440_GPJ0_CAMDATA0);
	s3c2410_gpio_cfgpin(S3C2440_GPJ1, S3C2440_GPJ1_CAMDATA1);
	s3c2410_gpio_cfgpin(S3C2440_GPJ2, S3C2440_GPJ2_CAMDATA2);
	s3c2410_gpio_cfgpin(S3C2440_GPJ3, S3C2440_GPJ3_CAMDATA3);
	s3c2410_gpio_cfgpin(S3C2440_GPJ4, S3C2440_GPJ4_CAMDATA4);
	s3c2410_gpio_cfgpin(S3C2440_GPJ5, S3C2440_GPJ5_CAMDATA5);
	s3c2410_gpio_cfgpin(S3C2440_GPJ6, S3C2440_GPJ6_CAMDATA6);
	s3c2410_gpio_cfgpin(S3C2440_GPJ7, S3C2440_GPJ7_CAMDATA7);
	s3c2410_gpio_cfgpin(S3C2440_GPJ8, S3C2440_GPJ8_CAMPCLK);
	s3c2410_gpio_cfgpin(S3C2440_GPJ9, S3C2440_GPJ9_CAMVSYNC);
	s3c2410_gpio_cfgpin(S3C2440_GPJ10, S3C2440_GPJ10_CAMHREF);
	s3c2410_gpio_cfgpin(S3C2440_GPJ11, S3C2440_GPJ11_CAMCLKOUT);
	s3c2410_gpio_cfgpin(S3C2440_GPJ12, S3C2440_GPJ12_CAMRESET);

	/* init camera's virtual memory. */
	if (!request_mem_region((unsigned long)S3C2440_PA_CAMIF, S3C2440_SZ_CAMIF, CARD_NAME))
	{
		ret = -EBUSY;
		goto error1;
	}

	/* remap the virtual memory. */
	camif_base_addr = (unsigned long)ioremap_nocache((unsigned long)S3C2440_PA_CAMIF, S3C2440_SZ_CAMIF);
	if (camif_base_addr == (unsigned long)NULL)
	{
		ret = -EBUSY;
		goto error2;
	}

	/* init camera clock. */
	pdev->clk = clk_get(NULL, "camif");
	if (IS_ERR(pdev->clk))
	{
		ret = -ENOENT;
		goto error3;
	}
	clk_enable(pdev->clk);

	camif_upll_clk = clk_get(NULL, "camif-upll");
	clk_set_rate(camif_upll_clk, 24000000);
	mdelay(100);

	/* init reference counter and its mutex. */
	mutex_init(&pdev->rcmutex);
	pdev->rc = 0;

	/* init image input source. */
	pdev->input = 0;

	/* init camif state and its lock. */
	pdev->state = CAMIF_STATE_FREE;

	/* init command code, command lock and the command wait queue. */
	pdev->cmdcode = CAMIF_CMD_NONE;
	init_waitqueue_head(&pdev->cmdqueue);

	/* register to videodev layer. */
	if (misc_register(&misc) < 0)
	{
		ret = -EBUSY;
		goto error4;
	}
	printk(KERN_ALERT"s3c2440 camif init done\n");
	
	sccb_init();
	hw_reset_camif();
	has_ov9650 = s3c2440_ov9650_init() >= 0;
	s3c2410_gpio_setpin(S3C2410_GPG(4), 1);
	return 0;

error4:
	clk_put(pdev->clk);
error3:
	iounmap((void *)camif_base_addr);
error2:
	release_mem_region((unsigned long)S3C2440_PA_CAMIF, S3C2440_SZ_CAMIF);
	
error1:
	return ret;
}






posted on 2011-11-25 18:44  _song  阅读(438)  评论(0编辑  收藏  举报