Lines Matching +full:cam +full:- +full:0

1 // SPDX-License-Identifier: GPL-2.0-only
10 * Copyright 2006-11 One Laptop Per Child Association, Inc.
11 * Copyright 2006-11 Jonathan Corbet <corbet@lwn.net>
28 #include <media/v4l2-device.h>
36 #include "mcam-core.h"
38 #define CAFE_VERSION 0x000002
57 * Most of the camera controller registers are defined in mcam-core.h,
66 #define REG_GPR 0xb4
67 #define GPR_C1EN 0x00000020 /* Pad 1 (power down) enable */
68 #define GPR_C0EN 0x00000010 /* Pad 0 (reset) enable */
69 #define GPR_C1 0x00000002 /* Control 1 value */
71 * Control 0 is wired to reset on OLPC machines. For ov7x sensors,
74 #define GPR_C0 0x00000001 /* Control 0 value */
80 #define REG_TWSIC0 0xb8 /* TWSI (smbus) control 0 */
81 #define TWSIC0_EN 0x00000001 /* TWSI enable */
82 #define TWSIC0_MODE 0x00000002 /* 1 = 16-bit, 0 = 8-bit */
83 #define TWSIC0_SID 0x000003fc /* Slave ID */
92 #define TWSIC0_CLKDIV 0x0007fc00 /* Clock divider */
93 #define TWSIC0_MASKACK 0x00400000 /* Mask ack from sensor */
94 #define TWSIC0_OVMAGIC 0x00800000 /* Make it work on OV sensors */
96 #define REG_TWSIC1 0xbc /* TWSI control 1 */
97 #define TWSIC1_DATA 0x0000ffff /* Data to/from camchip */
98 #define TWSIC1_ADDR 0x00ff0000 /* Address (register) */
100 #define TWSIC1_READ 0x01000000 /* Set for read op */
101 #define TWSIC1_WSTAT 0x02000000 /* Write status */
102 #define TWSIC1_RVALID 0x04000000 /* Read data valid */
103 #define TWSIC1_ERROR 0x08000000 /* Something screwed up */
108 #define REG_GL_CSR 0x3004 /* Control/status register */
109 #define GCSR_SRS 0x00000001 /* SW Reset set */
110 #define GCSR_SRC 0x00000002 /* SW Reset clear */
111 #define GCSR_MRS 0x00000004 /* Master reset set */
112 #define GCSR_MRC 0x00000008 /* HW Reset clear */
113 #define GCSR_CCIC_EN 0x00004000 /* CCIC Clock enable */
114 #define REG_GL_IMASK 0x300c /* Interrupt mask register */
115 #define GIMSK_CCIC_EN 0x00000004 /* CCIC Interrupt enable */
117 #define REG_GL_FCR 0x3038 /* GPIO functional control register */
118 #define GFCR_GPIO_ON 0x08 /* Camera GPIO enabled */
119 #define REG_GL_GPIOR 0x315c /* GPIO register */
120 #define GGPIO_OUT 0x80000 /* GPIO output */
121 #define GGPIO_VAL 0x00008 /* Output pin value */
129 #define cam_err(cam, fmt, arg...) \ argument
130 dev_err(&(cam)->pdev->dev, fmt, ##arg);
131 #define cam_warn(cam, fmt, arg...) \ argument
132 dev_warn(&(cam)->pdev->dev, fmt, ##arg);
134 /* -------------------------------------------------------------------- */
153 spin_lock_irqsave(&mcam->dev_lock, flags); in cafe_smbus_write_done()
155 spin_unlock_irqrestore(&mcam->dev_lock, flags); in cafe_smbus_write_done()
159 static int cafe_smbus_write_data(struct cafe_camera *cam, in cafe_smbus_write_data() argument
164 struct mcam_camera *mcam = &cam->mcam; in cafe_smbus_write_data()
166 spin_lock_irqsave(&mcam->dev_lock, flags); in cafe_smbus_write_data()
177 spin_unlock_irqrestore(&mcam->dev_lock, flags); in cafe_smbus_write_data()
181 * Use a busy-wait because we often send a large quantity of small in cafe_smbus_write_data()
182 * commands at-once; using msleep() would cause a lot of context in cafe_smbus_write_data()
184 * boot-time and capture-start delays. in cafe_smbus_write_data()
197 wait_event_timeout(cam->smbus_wait, cafe_smbus_write_done(mcam), in cafe_smbus_write_data()
200 spin_lock_irqsave(&mcam->dev_lock, flags); in cafe_smbus_write_data()
202 spin_unlock_irqrestore(&mcam->dev_lock, flags); in cafe_smbus_write_data()
205 cam_err(cam, "SMBUS write (%02x/%02x/%02x) timed out\n", addr, in cafe_smbus_write_data()
207 return -EIO; in cafe_smbus_write_data()
210 cam_err(cam, "SMBUS write (%02x/%02x/%02x) error\n", addr, in cafe_smbus_write_data()
212 return -EIO; in cafe_smbus_write_data()
214 return 0; in cafe_smbus_write_data()
230 spin_lock_irqsave(&mcam->dev_lock, flags); in cafe_smbus_read_done()
232 spin_unlock_irqrestore(&mcam->dev_lock, flags); in cafe_smbus_read_done()
238 static int cafe_smbus_read_data(struct cafe_camera *cam, in cafe_smbus_read_data() argument
243 struct mcam_camera *mcam = &cam->mcam; in cafe_smbus_read_data()
245 spin_lock_irqsave(&mcam->dev_lock, flags); in cafe_smbus_read_data()
256 spin_unlock_irqrestore(&mcam->dev_lock, flags); in cafe_smbus_read_data()
258 wait_event_timeout(cam->smbus_wait, in cafe_smbus_read_data()
260 spin_lock_irqsave(&mcam->dev_lock, flags); in cafe_smbus_read_data()
262 spin_unlock_irqrestore(&mcam->dev_lock, flags); in cafe_smbus_read_data()
265 cam_err(cam, "SMBUS read (%02x/%02x) error\n", addr, command); in cafe_smbus_read_data()
266 return -EIO; in cafe_smbus_read_data()
269 cam_err(cam, "SMBUS read (%02x/%02x) timed out\n", addr, in cafe_smbus_read_data()
271 return -EIO; in cafe_smbus_read_data()
273 *value = rval & 0xff; in cafe_smbus_read_data()
274 return 0; in cafe_smbus_read_data()
285 struct cafe_camera *cam = i2c_get_adapdata(adapter); in cafe_smbus_xfer() local
286 int ret = -EINVAL; in cafe_smbus_xfer()
290 * it can do word too, but the cam chip has no use for that. in cafe_smbus_xfer()
293 cam_err(cam, "funky xfer size %d\n", size); in cafe_smbus_xfer()
294 return -EINVAL; in cafe_smbus_xfer()
298 ret = cafe_smbus_write_data(cam, addr, command, data->byte); in cafe_smbus_xfer()
300 ret = cafe_smbus_read_data(cam, addr, command, &data->byte); in cafe_smbus_xfer()
305 static void cafe_smbus_enable_irq(struct cafe_camera *cam) in cafe_smbus_enable_irq() argument
309 spin_lock_irqsave(&cam->mcam.dev_lock, flags); in cafe_smbus_enable_irq()
310 mcam_reg_set_bit(&cam->mcam, REG_IRQMASK, TWSIIRQS); in cafe_smbus_enable_irq()
311 spin_unlock_irqrestore(&cam->mcam.dev_lock, flags); in cafe_smbus_enable_irq()
325 static int cafe_smbus_setup(struct cafe_camera *cam) in cafe_smbus_setup() argument
332 return -ENOMEM; in cafe_smbus_setup()
333 adap->owner = THIS_MODULE; in cafe_smbus_setup()
334 adap->algo = &cafe_smbus_algo; in cafe_smbus_setup()
335 strscpy(adap->name, "cafe_ccic", sizeof(adap->name)); in cafe_smbus_setup()
336 adap->dev.parent = &cam->pdev->dev; in cafe_smbus_setup()
337 i2c_set_adapdata(adap, cam); in cafe_smbus_setup()
345 cam->i2c_adapter = adap; in cafe_smbus_setup()
346 cafe_smbus_enable_irq(cam); in cafe_smbus_setup()
347 return 0; in cafe_smbus_setup()
350 static void cafe_smbus_shutdown(struct cafe_camera *cam) in cafe_smbus_shutdown() argument
352 i2c_del_adapter(cam->i2c_adapter); in cafe_smbus_shutdown()
353 kfree(cam->i2c_adapter); in cafe_smbus_shutdown()
358 * Controller-level stuff
365 spin_lock_irqsave(&mcam->dev_lock, flags); in cafe_ctlr_init()
367 * Added magic to bring up the hardware on the B-Test board in cafe_ctlr_init()
369 mcam_reg_write(mcam, 0x3038, 0x8); in cafe_ctlr_init()
370 mcam_reg_write(mcam, 0x315c, 0x80008); in cafe_ctlr_init()
383 spin_unlock_irqrestore(&mcam->dev_lock, flags); in cafe_ctlr_init()
385 spin_lock_irqsave(&mcam->dev_lock, flags); in cafe_ctlr_init()
392 mcam_reg_write(mcam, REG_IRQMASK, 0); in cafe_ctlr_init()
393 spin_unlock_irqrestore(&mcam->dev_lock, flags); in cafe_ctlr_init()
406 * Put the sensor into operational mode (assumes OLPC-style in cafe_ctlr_power_up()
407 * wiring). Control 0 is reset - set to 1 to operate. in cafe_ctlr_power_up()
408 * Control 1 is power down, set to 0 to operate. in cafe_ctlr_power_up()
413 return 0; in cafe_ctlr_power_up()
430 struct cafe_camera *cam = data; in cafe_irq() local
431 struct mcam_camera *mcam = &cam->mcam; in cafe_irq()
434 spin_lock(&mcam->dev_lock); in cafe_irq()
436 handled = cam->registered && mccic_irq(mcam, irqs); in cafe_irq()
439 wake_up(&cam->smbus_wait); in cafe_irq()
442 spin_unlock(&mcam->dev_lock); in cafe_irq()
446 /* -------------------------------------------------------------------------- */
466 .addr = 0x42 >> 1,
470 /* -------------------------------------------------------------------------- */
479 struct cafe_camera *cam; in cafe_pci_probe() local
487 ret = -ENOMEM; in cafe_pci_probe()
488 cam = kzalloc(sizeof(struct cafe_camera), GFP_KERNEL); in cafe_pci_probe()
489 if (cam == NULL) in cafe_pci_probe()
491 pci_set_drvdata(pdev, cam); in cafe_pci_probe()
492 cam->pdev = pdev; in cafe_pci_probe()
493 mcam = &cam->mcam; in cafe_pci_probe()
494 mcam->chip_id = MCAM_CAFE; in cafe_pci_probe()
495 spin_lock_init(&mcam->dev_lock); in cafe_pci_probe()
496 init_waitqueue_head(&cam->smbus_wait); in cafe_pci_probe()
497 mcam->plat_power_up = cafe_ctlr_power_up; in cafe_pci_probe()
498 mcam->plat_power_down = cafe_ctlr_power_down; in cafe_pci_probe()
499 mcam->dev = &pdev->dev; in cafe_pci_probe()
505 mcam->buffer_mode = B_vmalloc; in cafe_pci_probe()
514 ret = -EIO; in cafe_pci_probe()
515 mcam->regs = pci_iomap(pdev, 0, 0); in cafe_pci_probe()
516 if (!mcam->regs) { in cafe_pci_probe()
517 printk(KERN_ERR "Unable to ioremap cafe-ccic regs\n"); in cafe_pci_probe()
520 mcam->regs_size = pci_resource_len(pdev, 0); in cafe_pci_probe()
521 ret = request_irq(pdev->irq, cafe_irq, IRQF_SHARED, "cafe-ccic", cam); in cafe_pci_probe()
535 ret = cafe_smbus_setup(cam); in cafe_pci_probe()
539 ret = v4l2_device_register(mcam->dev, &mcam->v4l2_dev); in cafe_pci_probe()
543 v4l2_async_nf_init(&mcam->notifier, &mcam->v4l2_dev); in cafe_pci_probe()
545 asd = v4l2_async_nf_add_i2c(&mcam->notifier, in cafe_pci_probe()
546 i2c_adapter_id(cam->i2c_adapter), in cafe_pci_probe()
558 clkdev_create(mcam->mclk, "xclk", "%d-%04x", in cafe_pci_probe()
559 i2c_adapter_id(cam->i2c_adapter), ov7670_info.addr); in cafe_pci_probe()
561 i2c_dev = i2c_new_client_device(cam->i2c_adapter, &ov7670_info); in cafe_pci_probe()
567 cam->registered = 1; in cafe_pci_probe()
568 return 0; in cafe_pci_probe()
573 v4l2_device_unregister(&mcam->v4l2_dev); in cafe_pci_probe()
575 cafe_smbus_shutdown(cam); in cafe_pci_probe()
578 free_irq(pdev->irq, cam); in cafe_pci_probe()
580 pci_iounmap(pdev, mcam->regs); in cafe_pci_probe()
584 kfree(cam); in cafe_pci_probe()
593 static void cafe_shutdown(struct cafe_camera *cam) in cafe_shutdown() argument
595 mccic_shutdown(&cam->mcam); in cafe_shutdown()
596 v4l2_device_unregister(&cam->mcam.v4l2_dev); in cafe_shutdown()
597 cafe_smbus_shutdown(cam); in cafe_shutdown()
598 free_irq(cam->pdev->irq, cam); in cafe_shutdown()
599 pci_iounmap(cam->pdev, cam->mcam.regs); in cafe_shutdown()
605 struct cafe_camera *cam = pci_get_drvdata(pdev); in cafe_pci_remove() local
607 if (cam == NULL) { in cafe_pci_remove()
611 cafe_shutdown(cam); in cafe_pci_remove()
612 kfree(cam); in cafe_pci_remove()
621 struct cafe_camera *cam = dev_get_drvdata(dev); in cafe_pci_suspend() local
623 mccic_suspend(&cam->mcam); in cafe_pci_suspend()
624 return 0; in cafe_pci_suspend()
630 struct cafe_camera *cam = dev_get_drvdata(dev); in cafe_pci_resume() local
632 cafe_ctlr_init(&cam->mcam); in cafe_pci_resume()
633 return mccic_resume(&cam->mcam); in cafe_pci_resume()
639 { 0, }
647 .name = "cafe1000-ccic",
668 ret = 0; in cafe_init()