2016-04-26 21:22:19 +08:00
|
|
|
/*
|
|
|
|
* Driver for Renesas R-Car VIN
|
|
|
|
*
|
|
|
|
* Copyright (C) 2016 Renesas Electronics Corp.
|
|
|
|
* Copyright (C) 2011-2013 Renesas Solutions Corp.
|
|
|
|
* Copyright (C) 2013 Cogent Embedded, Inc., <source@cogentembedded.com>
|
|
|
|
* Copyright (C) 2008 Magnus Damm
|
|
|
|
*
|
|
|
|
* Based on the soc-camera rcar_vin driver
|
|
|
|
*
|
|
|
|
* 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; either version 2 of the License, or (at your
|
|
|
|
* option) any later version.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef __RCAR_VIN__
|
|
|
|
#define __RCAR_VIN__
|
|
|
|
|
2018-04-14 19:57:18 +08:00
|
|
|
#include <linux/kref.h>
|
|
|
|
|
2016-04-26 21:22:19 +08:00
|
|
|
#include <media/v4l2-async.h>
|
|
|
|
#include <media/v4l2-ctrls.h>
|
|
|
|
#include <media/v4l2-dev.h>
|
|
|
|
#include <media/v4l2-device.h>
|
|
|
|
#include <media/videobuf2-v4l2.h>
|
|
|
|
|
|
|
|
/* Number of HW buffers */
|
|
|
|
#define HW_BUFFER_NUM 3
|
|
|
|
|
|
|
|
/* Address alignment mask for HW buffers */
|
|
|
|
#define HW_BUFFER_MASK 0x7f
|
|
|
|
|
2018-04-14 19:57:18 +08:00
|
|
|
/* Max number on VIN instances that can be in a system */
|
|
|
|
#define RCAR_VIN_NUM 8
|
|
|
|
|
|
|
|
struct rvin_group;
|
|
|
|
|
2018-04-14 19:57:00 +08:00
|
|
|
enum model_id {
|
2016-04-26 21:22:19 +08:00
|
|
|
RCAR_H1,
|
|
|
|
RCAR_M1,
|
2016-08-15 23:06:28 +08:00
|
|
|
RCAR_GEN2,
|
2018-04-14 19:57:12 +08:00
|
|
|
RCAR_GEN3,
|
2016-04-26 21:22:19 +08:00
|
|
|
};
|
|
|
|
|
2018-04-14 19:57:20 +08:00
|
|
|
enum rvin_csi_id {
|
|
|
|
RVIN_CSI20,
|
|
|
|
RVIN_CSI21,
|
|
|
|
RVIN_CSI40,
|
|
|
|
RVIN_CSI41,
|
|
|
|
RVIN_CSI_MAX,
|
|
|
|
};
|
|
|
|
|
2016-04-26 21:22:19 +08:00
|
|
|
/**
|
|
|
|
* STOPPED - No operation in progress
|
|
|
|
* RUNNING - Operation in progress have buffers
|
|
|
|
* STOPPING - Stopping operation
|
|
|
|
*/
|
|
|
|
enum rvin_dma_state {
|
|
|
|
STOPPED = 0,
|
|
|
|
RUNNING,
|
|
|
|
STOPPING,
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* struct rvin_video_format - Data format stored in memory
|
|
|
|
* @fourcc: Pixelformat
|
|
|
|
* @bpp: Bytes per pixel
|
|
|
|
*/
|
|
|
|
struct rvin_video_format {
|
|
|
|
u32 fourcc;
|
|
|
|
u8 bpp;
|
|
|
|
};
|
|
|
|
|
2016-08-15 23:06:34 +08:00
|
|
|
/**
|
|
|
|
* struct rvin_graph_entity - Video endpoint from async framework
|
|
|
|
* @asd: sub-device descriptor for async framework
|
|
|
|
* @subdev: subdevice matched using async framework
|
2017-05-24 08:15:28 +08:00
|
|
|
* @source_pad: source pad of remote subdevice
|
|
|
|
* @sink_pad: sink pad of remote subdevice
|
2016-08-15 23:06:34 +08:00
|
|
|
*/
|
2016-04-26 21:22:19 +08:00
|
|
|
struct rvin_graph_entity {
|
|
|
|
struct v4l2_async_subdev asd;
|
|
|
|
struct v4l2_subdev *subdev;
|
2016-08-15 23:06:35 +08:00
|
|
|
|
2017-05-24 08:15:28 +08:00
|
|
|
unsigned int source_pad;
|
|
|
|
unsigned int sink_pad;
|
2016-04-26 21:22:19 +08:00
|
|
|
};
|
|
|
|
|
2018-04-14 19:57:20 +08:00
|
|
|
/**
|
|
|
|
* struct rvin_group_route - describes a route from a channel of a
|
|
|
|
* CSI-2 receiver to a VIN
|
|
|
|
*
|
|
|
|
* @csi: CSI-2 receiver ID.
|
|
|
|
* @channel: Output channel of the CSI-2 receiver.
|
|
|
|
* @vin: VIN ID.
|
|
|
|
* @mask: Bitmask of the different CHSEL register values that
|
|
|
|
* allow for a route from @csi + @chan to @vin.
|
|
|
|
*
|
|
|
|
* .. note::
|
|
|
|
* Each R-Car CSI-2 receiver has four output channels facing the VIN
|
|
|
|
* devices, each channel can carry one CSI-2 Virtual Channel (VC).
|
|
|
|
* There is no correlation between channel number and CSI-2 VC. It's
|
|
|
|
* up to the CSI-2 receiver driver to configure which VC is output
|
|
|
|
* on which channel, the VIN devices only care about output channels.
|
|
|
|
*
|
|
|
|
* There are in some cases multiple CHSEL register settings which would
|
|
|
|
* allow for the same route from @csi + @channel to @vin. For example
|
|
|
|
* on R-Car H3 both the CHSEL values 0 and 3 allow for a route from
|
|
|
|
* CSI40/VC0 to VIN0. All possible CHSEL values for a route need to be
|
|
|
|
* recorded as a bitmask in @mask, in this example bit 0 and 3 should
|
|
|
|
* be set.
|
|
|
|
*/
|
|
|
|
struct rvin_group_route {
|
|
|
|
enum rvin_csi_id csi;
|
|
|
|
unsigned int channel;
|
|
|
|
unsigned int vin;
|
|
|
|
unsigned int mask;
|
|
|
|
};
|
|
|
|
|
2018-04-14 19:57:00 +08:00
|
|
|
/**
|
|
|
|
* struct rvin_info - Information about the particular VIN implementation
|
|
|
|
* @model: VIN model
|
2018-04-14 19:57:14 +08:00
|
|
|
* @use_mc: use media controller instead of controlling subdevice
|
2018-04-14 19:57:01 +08:00
|
|
|
* @max_width: max input width the VIN supports
|
|
|
|
* @max_height: max input height the VIN supports
|
2018-04-14 19:57:20 +08:00
|
|
|
* @routes: list of possible routes from the CSI-2 recivers to
|
|
|
|
* all VINs. The list mush be NULL terminated.
|
2018-04-14 19:57:00 +08:00
|
|
|
*/
|
|
|
|
struct rvin_info {
|
|
|
|
enum model_id model;
|
2018-04-14 19:57:14 +08:00
|
|
|
bool use_mc;
|
2018-04-14 19:57:01 +08:00
|
|
|
|
|
|
|
unsigned int max_width;
|
|
|
|
unsigned int max_height;
|
2018-04-14 19:57:20 +08:00
|
|
|
const struct rvin_group_route *routes;
|
2018-04-14 19:57:00 +08:00
|
|
|
};
|
|
|
|
|
2016-04-26 21:22:19 +08:00
|
|
|
/**
|
|
|
|
* struct rvin_dev - Renesas VIN device structure
|
|
|
|
* @dev: (OF) device
|
|
|
|
* @base: device I/O register space remapped to virtual memory
|
2018-04-14 19:57:00 +08:00
|
|
|
* @info: info about VIN instance
|
2016-04-26 21:22:19 +08:00
|
|
|
*
|
|
|
|
* @vdev: V4L2 video device associated with VIN
|
|
|
|
* @v4l2_dev: V4L2 device
|
|
|
|
* @ctrl_handler: V4L2 control handler
|
|
|
|
* @notifier: V4L2 asynchronous subdevs notifier
|
2016-08-15 23:06:29 +08:00
|
|
|
* @digital: entity in the DT for local digital subdevice
|
2016-04-26 21:22:19 +08:00
|
|
|
*
|
2018-04-14 19:57:18 +08:00
|
|
|
* @group: Gen3 CSI group
|
|
|
|
* @id: Gen3 group id for this VIN
|
2018-04-14 19:57:17 +08:00
|
|
|
* @pad: media pad for the video device entity
|
|
|
|
*
|
2016-04-26 21:22:19 +08:00
|
|
|
* @lock: protects @queue
|
|
|
|
* @queue: vb2 buffers queue
|
2018-03-14 10:49:09 +08:00
|
|
|
* @scratch: cpu address for scratch buffer
|
|
|
|
* @scratch_phys: physical address of the scratch buffer
|
2016-04-26 21:22:19 +08:00
|
|
|
*
|
2018-03-14 10:49:10 +08:00
|
|
|
* @qlock: protects @queue_buf, @buf_list, @sequence
|
2016-04-26 21:22:19 +08:00
|
|
|
* @state
|
|
|
|
* @queue_buf: Keeps track of buffers given to HW slot
|
|
|
|
* @buf_list: list of queued buffers
|
|
|
|
* @sequence: V4L2 buffers sequence number
|
|
|
|
* @state: keeps track of operation state
|
|
|
|
*
|
2018-04-14 19:57:11 +08:00
|
|
|
* @mbus_cfg: media bus configuration from DT
|
|
|
|
* @mbus_code: media bus format code
|
2016-04-26 21:22:19 +08:00
|
|
|
* @format: active V4L2 pixel format
|
|
|
|
*
|
|
|
|
* @crop: active cropping
|
|
|
|
* @compose: active composing
|
2018-04-14 19:57:09 +08:00
|
|
|
* @source: active size of the video source
|
2018-04-14 19:57:10 +08:00
|
|
|
* @std: active video standard of the video source
|
2016-04-26 21:22:19 +08:00
|
|
|
*/
|
|
|
|
struct rvin_dev {
|
|
|
|
struct device *dev;
|
|
|
|
void __iomem *base;
|
2018-04-14 19:57:00 +08:00
|
|
|
const struct rvin_info *info;
|
2016-04-26 21:22:19 +08:00
|
|
|
|
|
|
|
struct video_device vdev;
|
|
|
|
struct v4l2_device v4l2_dev;
|
|
|
|
struct v4l2_ctrl_handler ctrl_handler;
|
|
|
|
struct v4l2_async_notifier notifier;
|
2017-09-02 06:41:19 +08:00
|
|
|
struct rvin_graph_entity *digital;
|
2016-04-26 21:22:19 +08:00
|
|
|
|
2018-04-14 19:57:18 +08:00
|
|
|
struct rvin_group *group;
|
|
|
|
unsigned int id;
|
2018-04-14 19:57:17 +08:00
|
|
|
struct media_pad pad;
|
|
|
|
|
2016-04-26 21:22:19 +08:00
|
|
|
struct mutex lock;
|
|
|
|
struct vb2_queue queue;
|
2018-03-14 10:49:09 +08:00
|
|
|
void *scratch;
|
|
|
|
dma_addr_t scratch_phys;
|
2016-04-26 21:22:19 +08:00
|
|
|
|
|
|
|
spinlock_t qlock;
|
|
|
|
struct vb2_v4l2_buffer *queue_buf[HW_BUFFER_NUM];
|
|
|
|
struct list_head buf_list;
|
|
|
|
unsigned int sequence;
|
|
|
|
enum rvin_dma_state state;
|
|
|
|
|
2018-04-14 19:57:11 +08:00
|
|
|
struct v4l2_mbus_config mbus_cfg;
|
|
|
|
u32 mbus_code;
|
2016-04-26 21:22:19 +08:00
|
|
|
struct v4l2_pix_format format;
|
|
|
|
|
|
|
|
struct v4l2_rect crop;
|
|
|
|
struct v4l2_rect compose;
|
2018-04-14 19:57:09 +08:00
|
|
|
struct v4l2_rect source;
|
2018-04-14 19:57:10 +08:00
|
|
|
v4l2_std_id std;
|
2016-04-26 21:22:19 +08:00
|
|
|
};
|
|
|
|
|
2017-09-02 06:41:19 +08:00
|
|
|
#define vin_to_source(vin) ((vin)->digital->subdev)
|
2016-04-26 21:22:19 +08:00
|
|
|
|
|
|
|
/* Debug */
|
|
|
|
#define vin_dbg(d, fmt, arg...) dev_dbg(d->dev, fmt, ##arg)
|
|
|
|
#define vin_info(d, fmt, arg...) dev_info(d->dev, fmt, ##arg)
|
|
|
|
#define vin_warn(d, fmt, arg...) dev_warn(d->dev, fmt, ##arg)
|
|
|
|
#define vin_err(d, fmt, arg...) dev_err(d->dev, fmt, ##arg)
|
|
|
|
|
2018-04-14 19:57:18 +08:00
|
|
|
/**
|
|
|
|
* struct rvin_group - VIN CSI2 group information
|
|
|
|
* @refcount: number of VIN instances using the group
|
|
|
|
*
|
|
|
|
* @mdev: media device which represents the group
|
|
|
|
*
|
2018-04-14 19:57:21 +08:00
|
|
|
* @lock: protects the count, notifier, vin and csi members
|
2018-04-14 19:57:18 +08:00
|
|
|
* @count: number of enabled VIN instances found in DT
|
2018-04-14 19:57:21 +08:00
|
|
|
* @notifier: pointer to the notifier of a VIN which handles the
|
|
|
|
* groups async sub-devices.
|
2018-04-14 19:57:18 +08:00
|
|
|
* @vin: VIN instances which are part of the group
|
2018-04-14 19:57:21 +08:00
|
|
|
* @csi: array of pairs of fwnode and subdev pointers
|
|
|
|
* to all CSI-2 subdevices.
|
2018-04-14 19:57:18 +08:00
|
|
|
*/
|
|
|
|
struct rvin_group {
|
|
|
|
struct kref refcount;
|
|
|
|
|
|
|
|
struct media_device mdev;
|
|
|
|
|
|
|
|
struct mutex lock;
|
|
|
|
unsigned int count;
|
2018-04-14 19:57:21 +08:00
|
|
|
struct v4l2_async_notifier *notifier;
|
2018-04-14 19:57:18 +08:00
|
|
|
struct rvin_dev *vin[RCAR_VIN_NUM];
|
2018-04-14 19:57:21 +08:00
|
|
|
|
|
|
|
struct {
|
|
|
|
struct fwnode_handle *fwnode;
|
|
|
|
struct v4l2_subdev *subdev;
|
|
|
|
} csi[RVIN_CSI_MAX];
|
2018-04-14 19:57:18 +08:00
|
|
|
};
|
|
|
|
|
2018-04-14 19:56:57 +08:00
|
|
|
int rvin_dma_register(struct rvin_dev *vin, int irq);
|
|
|
|
void rvin_dma_unregister(struct rvin_dev *vin);
|
2016-04-26 21:22:19 +08:00
|
|
|
|
2018-04-14 19:56:57 +08:00
|
|
|
int rvin_v4l2_register(struct rvin_dev *vin);
|
|
|
|
void rvin_v4l2_unregister(struct rvin_dev *vin);
|
2016-04-26 21:22:19 +08:00
|
|
|
|
|
|
|
const struct rvin_video_format *rvin_format_from_pixel(u32 pixelformat);
|
|
|
|
|
|
|
|
/* Cropping, composing and scaling */
|
|
|
|
void rvin_crop_scale_comp(struct rvin_dev *vin);
|
|
|
|
|
2018-04-14 19:57:13 +08:00
|
|
|
int rvin_set_channel_routing(struct rvin_dev *vin, u8 chsel);
|
|
|
|
|
2016-04-26 21:22:19 +08:00
|
|
|
#endif
|