2019-06-04 11:11:33 +03:00
/* SPDX-License-Identifier: GPL-2.0-only */
2010-08-03 16:50:29 +04:00
/*
2012-04-25 13:55:42 +04:00
* Copyright ( C ) 2010 - 2012 Samsung Electronics Co . , Ltd .
2010-08-03 16:50:29 +04:00
*/
# ifndef FIMC_CORE_H_
# define FIMC_CORE_H_
2010-10-07 17:06:16 +04:00
/*#define DEBUG*/
2011-07-27 01:29:50 +04:00
# include <linux/platform_device.h>
2013-03-20 17:44:39 +04:00
# include <linux/regmap.h>
2010-11-22 20:49:06 +03:00
# include <linux/sched.h>
2011-02-23 14:24:33 +03:00
# include <linux/spinlock.h>
2013-04-01 03:31:02 +04:00
# include <linux/mfd/syscon.h>
2010-08-03 16:50:29 +04:00
# include <linux/types.h>
2010-11-22 20:49:06 +03:00
# include <linux/videodev2.h>
2010-12-01 16:14:59 +03:00
# include <linux/io.h>
2012-08-17 10:28:26 +04:00
# include <linux/sizes.h>
2011-07-27 01:08:21 +04:00
# include <media/media-entity.h>
2015-09-22 16:30:29 +03:00
# include <media/videobuf2-v4l2.h>
2011-08-25 02:25:10 +04:00
# include <media/v4l2-ctrls.h>
2010-08-03 16:50:29 +04:00
# include <media/v4l2-device.h>
# include <media/v4l2-mem2mem.h>
2010-10-07 17:06:16 +04:00
# include <media/v4l2-mediabus.h>
2015-11-14 00:40:07 +03:00
# include <media/drv-intf/exynos-fimc.h>
2010-11-22 20:49:06 +03:00
2010-08-03 16:50:29 +04:00
# define dbg(fmt, args...) \
2011-02-23 14:17:57 +03:00
pr_debug ( " %s:%d: " fmt " \n " , __func__ , __LINE__ , # # args )
2010-08-03 16:50:29 +04:00
2010-10-07 17:06:16 +04:00
/* Time to wait for next frame VSYNC interrupt while stopping operation. */
# define FIMC_SHUTDOWN_TIMEOUT ((100*HZ) / 1000)
2011-06-10 22:36:45 +04:00
# define MAX_FIMC_CLOCKS 2
2013-04-23 18:36:04 +04:00
# define FIMC_DRIVER_NAME "exynos4-fimc"
2010-10-11 20:19:27 +04:00
# define FIMC_MAX_DEVS 4
2010-08-03 16:50:29 +04:00
# define FIMC_MAX_OUT_BUFS 4
# define SCALER_MAX_HRATIO 64
# define SCALER_MAX_VRATIO 64
2010-10-08 12:01:14 +04:00
# define DMA_MIN_SIZE 8
2011-08-25 03:35:30 +04:00
# define FIMC_CAMIF_MAX_HEIGHT 0x2000
2012-09-24 18:08:45 +04:00
# define FIMC_MAX_JPEG_BUF_SIZE (10 * SZ_1M)
# define FIMC_MAX_PLANES 3
2013-03-26 15:22:21 +04:00
# define FIMC_PIX_LIMITS_MAX 4
# define FIMC_DEF_MIN_SIZE 16
# define FIMC_DEF_HEIGHT_ALIGN 2
# define FIMC_DEF_HOR_OFFS_ALIGN 1
2013-06-18 21:56:11 +04:00
# define FIMC_DEFAULT_WIDTH 640
# define FIMC_DEFAULT_HEIGHT 480
2010-08-03 16:50:29 +04:00
2010-12-27 21:34:43 +03:00
/* indices to the clocks array */
enum {
CLK_BUS ,
CLK_GATE ,
} ;
2010-10-07 17:06:16 +04:00
enum fimc_dev_flags {
2011-09-02 13:25:32 +04:00
ST_LPM ,
/* m2m node */
ST_M2M_RUN ,
2010-08-03 16:50:29 +04:00
ST_M2M_PEND ,
2011-09-02 13:25:32 +04:00
ST_M2M_SUSPENDING ,
ST_M2M_SUSPENDED ,
/* capture node */
2010-10-07 17:06:16 +04:00
ST_CAPT_PEND ,
ST_CAPT_RUN ,
ST_CAPT_STREAM ,
2011-08-26 21:51:00 +04:00
ST_CAPT_ISP_STREAM ,
2011-08-25 03:45:34 +04:00
ST_CAPT_SUSPENDED ,
2010-10-07 17:06:16 +04:00
ST_CAPT_SHUT ,
2011-09-02 13:25:32 +04:00
ST_CAPT_BUSY ,
2011-08-25 02:25:10 +04:00
ST_CAPT_APPLY_CFG ,
2011-08-26 21:57:06 +04:00
ST_CAPT_JPEG ,
2010-08-03 16:50:29 +04:00
} ;
2011-09-02 13:25:32 +04:00
# define fimc_m2m_active(dev) test_bit(ST_M2M_RUN, &(dev)->state)
2010-08-03 16:50:29 +04:00
# define fimc_m2m_pending(dev) test_bit(ST_M2M_PEND, &(dev)->state)
2010-10-07 17:06:16 +04:00
# define fimc_capture_running(dev) test_bit(ST_CAPT_RUN, &(dev)->state)
# define fimc_capture_pending(dev) test_bit(ST_CAPT_PEND, &(dev)->state)
2011-09-02 13:25:32 +04:00
# define fimc_capture_busy(dev) test_bit(ST_CAPT_BUSY, &(dev)->state)
2010-10-07 17:06:16 +04:00
2010-08-03 16:50:29 +04:00
enum fimc_datapath {
2012-04-26 13:26:29 +04:00
FIMC_IO_NONE ,
FIMC_IO_CAMERA ,
FIMC_IO_DMA ,
FIMC_IO_LCDFIFO ,
FIMC_IO_WRITEBACK ,
FIMC_IO_ISP ,
2010-08-03 16:50:29 +04:00
} ;
enum fimc_color_fmt {
2012-09-24 18:08:45 +04:00
FIMC_FMT_RGB444 = 0x10 ,
2012-04-26 13:26:29 +04:00
FIMC_FMT_RGB555 ,
FIMC_FMT_RGB565 ,
FIMC_FMT_RGB666 ,
FIMC_FMT_RGB888 ,
FIMC_FMT_RGB30_LOCAL ,
FIMC_FMT_YCBCR420 = 0x20 ,
FIMC_FMT_YCBYCR422 ,
FIMC_FMT_YCRYCB422 ,
FIMC_FMT_CBYCRY422 ,
FIMC_FMT_CRYCBY422 ,
FIMC_FMT_YCBCR444_LOCAL ,
2012-09-24 18:08:45 +04:00
FIMC_FMT_RAW8 = 0x40 ,
2012-04-26 13:26:29 +04:00
FIMC_FMT_RAW10 ,
FIMC_FMT_RAW12 ,
2012-09-24 18:08:45 +04:00
FIMC_FMT_JPEG = 0x80 ,
FIMC_FMT_YUYV_JPEG = 0x100 ,
2010-08-03 16:50:29 +04:00
} ;
2012-09-24 18:08:45 +04:00
# define fimc_fmt_is_user_defined(x) (!!((x) & 0x180))
2011-08-25 03:35:30 +04:00
# define fimc_fmt_is_rgb(x) (!!((x) & 0x10))
2010-10-08 12:01:14 +04:00
2011-08-26 21:51:00 +04:00
# define IS_M2M(__strt) ((__strt) == V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE || \
__strt = = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE )
2010-08-03 16:50:29 +04:00
/* The hardware context state. */
2010-10-08 12:01:14 +04:00
# define FIMC_PARAMS (1 << 0)
2013-01-30 16:54:06 +04:00
# define FIMC_COMPOSE (1 << 1)
2011-08-25 03:35:30 +04:00
# define FIMC_CTX_M2M (1 << 16)
# define FIMC_CTX_CAP (1 << 17)
# define FIMC_CTX_SHUT (1 << 18)
2010-08-03 16:50:29 +04:00
/* Image conversion flags */
# define FIMC_IN_DMA_ACCESS_TILED (1 << 0)
# define FIMC_IN_DMA_ACCESS_LINEAR (0 << 0)
# define FIMC_OUT_DMA_ACCESS_TILED (1 << 1)
# define FIMC_OUT_DMA_ACCESS_LINEAR (0 << 1)
# define FIMC_SCAN_MODE_PROGRESSIVE (0 << 2)
# define FIMC_SCAN_MODE_INTERLACED (1 << 2)
2010-10-08 12:01:14 +04:00
/*
* YCbCr data dynamic range for RGB - YUV color conversion .
* Y / Cb / Cr : ( 0 ~ 255 ) */
2010-08-03 16:50:29 +04:00
# define FIMC_COLOR_RANGE_WIDE (0 << 3)
/* Y (16 ~ 235), Cb/Cr (16 ~ 240) */
# define FIMC_COLOR_RANGE_NARROW (1 << 3)
/**
* struct fimc_dma_offset - pixel offset information for DMA
* @ y_h : y value horizontal offset
* @ y_v : y value vertical offset
* @ cb_h : cb value horizontal offset
* @ cb_v : cb value vertical offset
* @ cr_h : cr value horizontal offset
* @ cr_v : cr value vertical offset
*/
struct fimc_dma_offset {
int y_h ;
int y_v ;
int cb_h ;
int cb_v ;
int cr_h ;
int cr_v ;
} ;
/**
2011-05-20 13:14:59 +04:00
* struct fimc_effect - color effect information
2010-08-03 16:50:29 +04:00
* @ type : effect type
* @ pat_cb : cr value when type is " arbitrary "
* @ pat_cr : cr value when type is " arbitrary "
*/
struct fimc_effect {
u32 type ;
u8 pat_cb ;
u8 pat_cr ;
} ;
/**
* struct fimc_scaler - the configuration data for FIMC inetrnal scaler
2010-10-08 12:01:14 +04:00
* @ scaleup_h : flag indicating scaling up horizontally
* @ scaleup_v : flag indicating scaling up vertically
* @ copy_mode : flag indicating transparent DMA transfer ( no scaling
* and color format conversion )
* @ enabled : flag indicating if the scaler is used
2010-08-03 16:50:29 +04:00
* @ hfactor : horizontal shift factor
* @ vfactor : vertical shift factor
* @ pre_hratio : horizontal ratio of the prescaler
* @ pre_vratio : vertical ratio of the prescaler
* @ pre_dst_width : the prescaler ' s destination width
* @ pre_dst_height : the prescaler ' s destination height
* @ main_hratio : the main scaler ' s horizontal ratio
* @ main_vratio : the main scaler ' s vertical ratio
2010-10-08 12:01:14 +04:00
* @ real_width : source pixel ( width - offset )
* @ real_height : source pixel ( height - offset )
2010-08-03 16:50:29 +04:00
*/
struct fimc_scaler {
2010-10-22 11:10:57 +04:00
unsigned int scaleup_h : 1 ;
unsigned int scaleup_v : 1 ;
unsigned int copy_mode : 1 ;
unsigned int enabled : 1 ;
2010-08-03 16:50:29 +04:00
u32 hfactor ;
u32 vfactor ;
u32 pre_hratio ;
u32 pre_vratio ;
u32 pre_dst_width ;
u32 pre_dst_height ;
u32 main_hratio ;
u32 main_vratio ;
u32 real_width ;
u32 real_height ;
} ;
/**
2020-11-24 11:41:28 +03:00
* struct fimc_addr - the FIMC address set for DMA
* @ y : luminance plane address
* @ cb : Cb plane address
* @ cr : Cr plane address
2010-08-03 16:50:29 +04:00
*/
struct fimc_addr {
u32 y ;
u32 cb ;
u32 cr ;
} ;
/**
* struct fimc_vid_buffer - the driver ' s video buffer
2010-10-07 17:06:16 +04:00
* @ vb : v4l videobuf buffer
2011-05-20 13:14:59 +04:00
* @ list : linked list structure for buffer queue
2020-11-24 11:41:28 +03:00
* @ addr : precalculated DMA address set
2010-10-07 17:06:16 +04:00
* @ index : buffer index for the output DMA engine
2010-08-03 16:50:29 +04:00
*/
struct fimc_vid_buffer {
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 16:30:30 +03:00
struct vb2_v4l2_buffer vb ;
2010-12-01 16:14:59 +03:00
struct list_head list ;
2020-11-24 11:41:28 +03:00
struct fimc_addr addr ;
2010-10-07 17:06:16 +04:00
int index ;
2010-08-03 16:50:29 +04:00
} ;
/**
2010-10-08 12:01:14 +04:00
* struct fimc_frame - source / target frame properties
2010-08-03 16:50:29 +04:00
* @ f_width : image full width ( virtual screen size )
* @ f_height : image full height ( virtual screen size )
* @ o_width : original image width as set by S_FMT
* @ o_height : original image height as set by S_FMT
* @ offs_h : image horizontal pixel offset
* @ offs_v : image vertical pixel offset
* @ width : image pixel width
* @ height : image pixel weight
2010-12-08 20:05:08 +03:00
* @ payload : image size in bytes ( w x h x bpp )
2013-01-11 13:36:19 +04:00
* @ bytesperline : bytesperline value for each plane
2020-11-24 11:41:28 +03:00
* @ addr : image frame buffer DMA addresses
2010-08-03 16:50:29 +04:00
* @ dma_offset : DMA offset in bytes
2011-05-20 13:14:59 +04:00
* @ fmt : fimc color format pointer
2021-03-23 12:15:08 +03:00
* @ alpha : alpha value
2010-08-03 16:50:29 +04:00
*/
struct fimc_frame {
u32 f_width ;
u32 f_height ;
u32 o_width ;
u32 o_height ;
u32 offs_h ;
u32 offs_v ;
u32 width ;
u32 height ;
2012-09-24 18:08:45 +04:00
unsigned int payload [ VIDEO_MAX_PLANES ] ;
2013-01-11 13:36:19 +04:00
unsigned int bytesperline [ VIDEO_MAX_PLANES ] ;
2020-11-24 11:41:28 +03:00
struct fimc_addr addr ;
2010-08-03 16:50:29 +04:00
struct fimc_dma_offset dma_offset ;
struct fimc_fmt * fmt ;
2011-12-01 21:02:24 +04:00
u8 alpha ;
2010-08-03 16:50:29 +04:00
} ;
/**
* struct fimc_m2m_device - v4l2 memory - to - memory device data
* @ vfd : the video device node for v4l2 m2m mode
* @ m2m_dev : v4l2 memory - to - memory device data
* @ ctx : hardware context data
* @ refcnt : the reference counter
*/
struct fimc_m2m_device {
2012-07-26 14:59:11 +04:00
struct video_device vfd ;
2010-08-03 16:50:29 +04:00
struct v4l2_m2m_dev * m2m_dev ;
struct fimc_ctx * ctx ;
int refcnt ;
} ;
2013-03-20 17:44:39 +04:00
# define FIMC_SD_PAD_SINK_CAM 0
# define FIMC_SD_PAD_SINK_FIFO 1
# define FIMC_SD_PAD_SOURCE 2
# define FIMC_SD_PADS_NUM 3
2011-08-25 03:35:30 +04:00
2010-10-07 17:06:16 +04:00
/**
* struct fimc_vid_cap - camera capture device information
* @ ctx : hardware context data
2011-08-25 03:35:30 +04:00
* @ subdev : subdev exposing the FIMC processing block
2013-05-31 18:37:18 +04:00
* @ ve : exynos video device entity structure
2011-07-27 01:08:21 +04:00
* @ vd_pad : fimc video capture node pad
2011-08-25 03:35:30 +04:00
* @ sd_pads : fimc video processing block pads
2013-03-20 17:44:39 +04:00
* @ ci_fmt : image format at the FIMC camera input ( and the scaler output )
* @ wb_fmt : image format at the FIMC ISP Writeback input
* @ source_config : external image source related configuration structure
2010-10-07 17:06:16 +04:00
* @ pending_buf_q : the pending buffer queue head
* @ active_buf_q : the queue head of buffers scheduled in hardware
* @ vbq : the capture am video buffer queue
* @ active_buf_cnt : number of video buffers scheduled in hardware
* @ buf_index : index for managing the output DMA buffers
* @ frame_count : the frame counter for statistics
* @ reqbufs_count : the number of buffers requested in REQBUFS ioctl
2021-03-23 12:15:08 +03:00
* @ streaming : is streaming in progress ?
2011-08-25 02:28:18 +04:00
* @ input : capture input type , grp_id of the attached subdev
2011-09-01 13:01:08 +04:00
* @ user_subdev_api : true if subdevs are not configured by the host driver
2010-10-07 17:06:16 +04:00
*/
struct fimc_vid_cap {
struct fimc_ctx * ctx ;
2012-04-21 01:57:25 +04:00
struct v4l2_subdev subdev ;
2013-05-31 18:37:18 +04:00
struct exynos_video_entity ve ;
2011-07-27 01:08:21 +04:00
struct media_pad vd_pad ;
2011-08-25 03:35:30 +04:00
struct media_pad sd_pads [ FIMC_SD_PADS_NUM ] ;
2013-03-20 17:44:39 +04:00
struct v4l2_mbus_framefmt ci_fmt ;
struct v4l2_mbus_framefmt wb_fmt ;
struct fimc_source_info source_config ;
2010-10-07 17:06:16 +04:00
struct list_head pending_buf_q ;
struct list_head active_buf_q ;
2010-12-01 16:14:59 +03:00
struct vb2_queue vbq ;
2010-10-07 17:06:16 +04:00
int active_buf_cnt ;
int buf_index ;
unsigned int frame_count ;
unsigned int reqbufs_count ;
2013-03-25 23:50:50 +04:00
bool streaming ;
2011-08-25 02:28:18 +04:00
u32 input ;
2011-09-01 13:01:08 +04:00
bool user_subdev_api ;
2010-10-07 17:06:16 +04:00
} ;
2010-10-11 20:19:27 +04:00
/**
* struct fimc_pix_limit - image pixel size limits in various IP configurations
*
* @ scaler_en_w : max input pixel width when the scaler is enabled
* @ scaler_dis_w : max input pixel width when the scaler is disabled
* @ in_rot_en_h : max input width with the input rotator is on
* @ in_rot_dis_w : max input width with the input rotator is off
* @ out_rot_en_w : max output width with the output rotator on
* @ out_rot_dis_w : max output width with the output rotator off
*/
struct fimc_pix_limit {
u16 scaler_en_w ;
u16 scaler_dis_w ;
u16 in_rot_en_h ;
u16 in_rot_dis_w ;
u16 out_rot_en_w ;
u16 out_rot_dis_w ;
} ;
2010-08-03 16:50:29 +04:00
/**
2012-04-27 16:33:23 +04:00
* struct fimc_variant - FIMC device variant information
2010-08-03 16:50:29 +04:00
* @ has_inp_rot : set if has input rotator
* @ has_out_rot : set if has output rotator
2010-12-28 17:27:13 +03:00
* @ has_mainscaler_ext : 1 if extended mainscaler ratios in CIEXTEN register
* are present in this IP revision
2011-09-01 13:01:08 +04:00
* @ has_cam_if : set if this instance has a camera input interface
2012-08-02 17:27:46 +04:00
* @ has_isp_wb : set if this instance has ISP writeback input
2010-10-11 20:19:27 +04:00
* @ pix_limit : pixel size constraints for the scaler
2010-08-03 16:50:29 +04:00
* @ min_inp_pixsize : minimum input pixel size
* @ min_out_pixsize : minimum output pixel size
2017-02-28 01:28:58 +03:00
* @ hor_offs_align : horizontal pixel offset alignment
2011-05-27 20:12:23 +04:00
* @ min_vsize_align : minimum vertical pixel size alignment
2010-08-03 16:50:29 +04:00
*/
2012-04-27 16:33:23 +04:00
struct fimc_variant {
2010-08-03 16:50:29 +04:00
unsigned int has_inp_rot : 1 ;
unsigned int has_out_rot : 1 ;
2010-12-28 17:27:13 +03:00
unsigned int has_mainscaler_ext : 1 ;
2011-09-01 13:01:08 +04:00
unsigned int has_cam_if : 1 ;
2012-08-02 17:27:46 +04:00
unsigned int has_isp_wb : 1 ;
const struct fimc_pix_limit * pix_limit ;
2010-08-03 16:50:29 +04:00
u16 min_inp_pixsize ;
u16 min_out_pixsize ;
2010-10-11 20:19:27 +04:00
u16 hor_offs_align ;
2011-05-27 20:12:23 +04:00
u16 min_vsize_align ;
2010-08-03 16:50:29 +04:00
} ;
/**
2012-04-27 16:33:23 +04:00
* struct fimc_drvdata - per device type driver data
* @ variant : variant information for this device
* @ num_entities : number of fimc instances available in a SoC
* @ lclk_frequency : local bus clock frequency
2013-03-26 15:22:21 +04:00
* @ cistatus2 : 1 if the FIMC IPs have CISTATUS2 register
* @ dma_pix_hoff : the horizontal DMA offset unit : 1 - pixels , 0 - bytes
* @ alpha_color : 1 if alpha color component is supported
* @ out_buf_count : maximum number of output DMA buffers supported
2010-08-03 16:50:29 +04:00
*/
2012-04-27 16:33:23 +04:00
struct fimc_drvdata {
2012-08-02 17:27:46 +04:00
const struct fimc_variant * variant [ FIMC_MAX_DEVS ] ;
2012-04-27 16:33:23 +04:00
int num_entities ;
unsigned long lclk_frequency ;
2013-03-26 15:22:21 +04:00
/* Fields common to all FIMC IP instances */
u8 cistatus2 ;
u8 dma_pix_hoff ;
u8 alpha_color ;
u8 out_buf_count ;
2010-08-03 16:50:29 +04:00
} ;
2012-04-27 16:33:23 +04:00
# define fimc_get_drvdata(_pdev) \
( ( struct fimc_drvdata * ) platform_get_device_id ( _pdev ) - > driver_data )
2011-09-01 13:01:08 +04:00
2010-08-03 16:50:29 +04:00
struct fimc_ctx ;
/**
2010-10-08 12:01:14 +04:00
* struct fimc_dev - abstraction for FIMC entity
2010-08-03 16:50:29 +04:00
* @ slock : the spinlock protecting this data structure
* @ lock : the mutex protecting this data structure
* @ pdev : pointer to the FIMC platform device
2010-10-07 17:06:16 +04:00
* @ pdata : pointer to the device platform data
2013-03-20 17:44:39 +04:00
* @ sysreg : pointer to the SYSREG regmap
2011-05-20 13:14:59 +04:00
* @ variant : the IP variant information
2021-03-23 12:15:08 +03:00
* @ drv_data : driver data
2010-12-27 21:34:43 +03:00
* @ id : FIMC device index ( 0. . FIMC_MAX_DEVS )
2011-05-20 13:14:59 +04:00
* @ clock : clocks required for FIMC operation
2010-08-03 16:50:29 +04:00
* @ regs : the mapped hardware registers
2011-05-20 13:14:59 +04:00
* @ irq_queue : interrupt handler waitqueue
2011-06-10 22:36:48 +04:00
* @ v4l2_dev : root v4l2_device
2010-08-03 16:50:29 +04:00
* @ m2m : memory - to - memory V4L2 device information
2010-10-07 17:06:16 +04:00
* @ vid_cap : camera capture device information
* @ state : flags used to synchronize m2m and capture mode operation
2010-08-03 16:50:29 +04:00
*/
struct fimc_dev {
spinlock_t slock ;
struct mutex lock ;
struct platform_device * pdev ;
2010-12-27 20:42:15 +03:00
struct s5p_platform_fimc * pdata ;
2013-03-20 17:44:39 +04:00
struct regmap * sysreg ;
2012-08-02 17:27:46 +04:00
const struct fimc_variant * variant ;
2013-03-26 15:22:21 +04:00
const struct fimc_drvdata * drv_data ;
2013-04-16 10:02:20 +04:00
int id ;
2010-12-27 21:34:43 +03:00
struct clk * clock [ MAX_FIMC_CLOCKS ] ;
2010-08-03 16:50:29 +04:00
void __iomem * regs ;
2010-10-07 17:06:16 +04:00
wait_queue_head_t irq_queue ;
2011-06-10 22:36:48 +04:00
struct v4l2_device * v4l2_dev ;
2010-08-03 16:50:29 +04:00
struct fimc_m2m_device m2m ;
2010-10-07 17:06:16 +04:00
struct fimc_vid_cap vid_cap ;
2010-08-03 16:50:29 +04:00
unsigned long state ;
} ;
2012-04-02 13:41:22 +04:00
/**
* struct fimc_ctrls - v4l2 controls structure
* @ handler : the control handler
* @ colorfx : image effect control
* @ colorfx_cbcr : Cb / Cr coefficients control
* @ rotate : image rotation control
* @ hflip : horizontal flip control
* @ vflip : vertical flip control
* @ alpha : RGB alpha control
* @ ready : true if @ handler is initialized
*/
struct fimc_ctrls {
struct v4l2_ctrl_handler handler ;
struct {
struct v4l2_ctrl * colorfx ;
struct v4l2_ctrl * colorfx_cbcr ;
} ;
struct v4l2_ctrl * rotate ;
struct v4l2_ctrl * hflip ;
struct v4l2_ctrl * vflip ;
struct v4l2_ctrl * alpha ;
bool ready ;
} ;
2010-08-03 16:50:29 +04:00
/**
2021-03-11 13:20:10 +03:00
* struct fimc_ctx - the device context data
2010-08-03 16:50:29 +04:00
* @ s_frame : source frame properties
* @ d_frame : destination frame properties
* @ out_order_1p : output 1 - plane YCBCR order
* @ out_order_2p : output 2 - plane YCBCR order
2021-03-23 12:15:08 +03:00
* @ in_order_1p : input 1 - plane YCBCR order
2010-08-03 16:50:29 +04:00
* @ in_order_2p : input 2 - plane YCBCR order
* @ in_path : input mode ( DMA or camera )
* @ out_path : output mode ( DMA or FIFO )
* @ scaler : image scaler properties
* @ effect : image effect
* @ rotation : image clockwise rotation in degrees
2011-08-25 02:25:10 +04:00
* @ hflip : indicates image horizontal flip if set
* @ vflip : indicates image vertical flip if set
2010-10-08 12:01:14 +04:00
* @ flags : additional flags for image conversion
2010-08-03 16:50:29 +04:00
* @ state : flags to keep track of user configuration
* @ fimc_dev : the FIMC device this context applies to
2011-06-10 22:36:50 +04:00
* @ fh : v4l2 file handle
2012-04-02 13:41:22 +04:00
* @ ctrls : v4l2 controls structure
2010-08-03 16:50:29 +04:00
*/
struct fimc_ctx {
struct fimc_frame s_frame ;
struct fimc_frame d_frame ;
u32 out_order_1p ;
u32 out_order_2p ;
u32 in_order_1p ;
u32 in_order_2p ;
enum fimc_datapath in_path ;
enum fimc_datapath out_path ;
struct fimc_scaler scaler ;
struct fimc_effect effect ;
int rotation ;
2011-08-25 02:25:10 +04:00
unsigned int hflip : 1 ;
unsigned int vflip : 1 ;
2010-08-03 16:50:29 +04:00
u32 flags ;
u32 state ;
struct fimc_dev * fimc_dev ;
2011-06-10 22:36:50 +04:00
struct v4l2_fh fh ;
2012-04-02 13:41:22 +04:00
struct fimc_ctrls ctrls ;
2010-08-03 16:50:29 +04:00
} ;
2011-06-10 22:36:50 +04:00
# define fh_to_ctx(__fh) container_of(__fh, struct fimc_ctx, fh)
2011-08-25 03:35:30 +04:00
static inline void set_frame_bounds ( struct fimc_frame * f , u32 width , u32 height )
{
f - > o_width = width ;
f - > o_height = height ;
f - > f_width = width ;
f - > f_height = height ;
}
static inline void set_frame_crop ( struct fimc_frame * f ,
u32 left , u32 top , u32 width , u32 height )
{
f - > offs_h = left ;
f - > offs_v = top ;
f - > width = width ;
f - > height = height ;
}
static inline u32 fimc_get_format_depth ( struct fimc_fmt * ff )
{
u32 i , depth = 0 ;
if ( ff ! = NULL )
for ( i = 0 ; i < ff - > colplanes ; i + + )
depth + = ff - > depth [ i ] ;
return depth ;
}
2011-02-23 14:24:33 +03:00
static inline bool fimc_capture_active ( struct fimc_dev * fimc )
{
unsigned long flags ;
bool ret ;
spin_lock_irqsave ( & fimc - > slock , flags ) ;
ret = ! ! ( fimc - > state & ( 1 < < ST_CAPT_RUN ) | |
fimc - > state & ( 1 < < ST_CAPT_PEND ) ) ;
spin_unlock_irqrestore ( & fimc - > slock , flags ) ;
return ret ;
}
2012-03-19 20:11:40 +04:00
static inline void fimc_ctx_state_set ( u32 state , struct fimc_ctx * ctx )
2011-02-23 14:24:33 +03:00
{
unsigned long flags ;
2012-03-19 20:11:40 +04:00
spin_lock_irqsave ( & ctx - > fimc_dev - > slock , flags ) ;
2011-02-23 14:24:33 +03:00
ctx - > state | = state ;
2012-03-19 20:11:40 +04:00
spin_unlock_irqrestore ( & ctx - > fimc_dev - > slock , flags ) ;
2011-02-23 14:24:33 +03:00
}
static inline bool fimc_ctx_state_is_set ( u32 mask , struct fimc_ctx * ctx )
{
unsigned long flags ;
bool ret ;
2012-03-19 20:11:40 +04:00
spin_lock_irqsave ( & ctx - > fimc_dev - > slock , flags ) ;
2011-02-23 14:24:33 +03:00
ret = ( ctx - > state & mask ) = = mask ;
2012-03-19 20:11:40 +04:00
spin_unlock_irqrestore ( & ctx - > fimc_dev - > slock , flags ) ;
2011-02-23 14:24:33 +03:00
return ret ;
}
2010-08-03 16:50:29 +04:00
static inline int tiled_fmt ( struct fimc_fmt * fmt )
{
2010-12-08 20:05:08 +03:00
return fmt - > fourcc = = V4L2_PIX_FMT_NV12MT ;
2010-08-03 16:50:29 +04:00
}
2012-09-24 18:08:45 +04:00
static inline bool fimc_jpeg_fourcc ( u32 pixelformat )
{
return ( pixelformat = = V4L2_PIX_FMT_JPEG | |
pixelformat = = V4L2_PIX_FMT_S5C_UYVY_JPG ) ;
}
static inline bool fimc_user_defined_mbus_fmt ( u32 code )
{
2014-11-10 20:28:31 +03:00
return ( code = = MEDIA_BUS_FMT_JPEG_1X8 | |
code = = MEDIA_BUS_FMT_S5C_UYVY_JPEG_1X8 ) ;
2012-09-24 18:08:45 +04:00
}
2011-12-01 21:02:24 +04:00
/* Return the alpha component bit mask */
static inline int fimc_get_alpha_mask ( struct fimc_fmt * fmt )
{
switch ( fmt - > color ) {
2012-04-26 13:26:29 +04:00
case FIMC_FMT_RGB444 : return 0x0f ;
case FIMC_FMT_RGB555 : return 0x01 ;
case FIMC_FMT_RGB888 : return 0xff ;
2011-12-01 21:02:24 +04:00
default : return 0 ;
} ;
}
2010-10-08 12:01:14 +04:00
static inline struct fimc_frame * ctx_get_frame ( struct fimc_ctx * ctx ,
enum v4l2_buf_type type )
2010-08-06 17:50:46 +04:00
{
struct fimc_frame * frame ;
2018-10-04 22:37:10 +03:00
if ( type = = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE | |
type = = V4L2_BUF_TYPE_VIDEO_OUTPUT ) {
2011-02-23 14:24:33 +03:00
if ( fimc_ctx_state_is_set ( FIMC_CTX_M2M , ctx ) )
2010-10-07 17:06:16 +04:00
frame = & ctx - > s_frame ;
else
return ERR_PTR ( - EINVAL ) ;
2018-10-04 22:37:10 +03:00
} else if ( type = = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE | |
type = = V4L2_BUF_TYPE_VIDEO_CAPTURE ) {
2010-08-06 17:50:46 +04:00
frame = & ctx - > d_frame ;
} else {
2011-06-10 22:36:48 +04:00
v4l2_err ( ctx - > fimc_dev - > v4l2_dev ,
2010-08-06 17:50:46 +04:00
" Wrong buffer/video queue type (%d) \n " , type ) ;
return ERR_PTR ( - EINVAL ) ;
}
return frame ;
}
2010-10-07 17:06:16 +04:00
/* -----------------------------------------------------*/
/* fimc-core.c */
2010-12-08 20:05:08 +03:00
int fimc_vidioc_enum_fmt_mplane ( struct file * file , void * priv ,
struct v4l2_fmtdesc * f ) ;
2011-08-25 02:25:10 +04:00
int fimc_ctrls_create ( struct fimc_ctx * ctx ) ;
void fimc_ctrls_delete ( struct fimc_ctx * ctx ) ;
void fimc_ctrls_activate ( struct fimc_ctx * ctx , bool active ) ;
2011-12-01 21:02:24 +04:00
void fimc_alpha_ctrl_update ( struct fimc_ctx * ctx ) ;
2013-01-11 13:36:19 +04:00
void __fimc_get_format ( struct fimc_frame * frame , struct v4l2_format * f ) ;
2011-08-26 21:51:00 +04:00
void fimc_adjust_mplane_format ( struct fimc_fmt * fmt , u32 width , u32 height ,
struct v4l2_pix_format_mplane * pix ) ;
2012-04-23 00:07:09 +04:00
struct fimc_fmt * fimc_find_format ( const u32 * pixelformat , const u32 * mbus_code ,
2011-06-13 18:09:40 +04:00
unsigned int mask , int index ) ;
2012-05-08 22:51:24 +04:00
struct fimc_fmt * fimc_get_format ( unsigned int index ) ;
2010-10-07 17:06:16 +04:00
2011-08-26 21:57:06 +04:00
int fimc_check_scaler_ratio ( struct fimc_ctx * ctx , int sw , int sh ,
int dw , int dh , int rotation ) ;
2010-10-07 17:06:16 +04:00
int fimc_set_scaler_info ( struct fimc_ctx * ctx ) ;
int fimc_prepare_config ( struct fimc_ctx * ctx , u32 flags ) ;
2010-12-01 16:14:59 +03:00
int fimc_prepare_addr ( struct fimc_ctx * ctx , struct vb2_buffer * vb ,
2020-11-24 11:41:28 +03:00
struct fimc_frame * frame , struct fimc_addr * addr ) ;
2011-06-10 22:36:53 +04:00
void fimc_prepare_dma_offset ( struct fimc_ctx * ctx , struct fimc_frame * f ) ;
void fimc_set_yuv_order ( struct fimc_ctx * ctx ) ;
2012-05-08 22:51:24 +04:00
void fimc_capture_irq_handler ( struct fimc_dev * fimc , int deq_buf ) ;
2011-06-10 22:36:53 +04:00
2011-06-10 22:36:48 +04:00
int fimc_register_m2m_device ( struct fimc_dev * fimc ,
struct v4l2_device * v4l2_dev ) ;
void fimc_unregister_m2m_device ( struct fimc_dev * fimc ) ;
2011-09-01 13:01:08 +04:00
int fimc_register_driver ( void ) ;
void fimc_unregister_driver ( void ) ;
2010-10-07 17:06:16 +04:00
2013-04-01 03:31:02 +04:00
# ifdef CONFIG_MFD_SYSCON
static inline struct regmap * fimc_get_sysreg_regmap ( struct device_node * node )
{
return syscon_regmap_lookup_by_phandle ( node , " samsung,sysreg " ) ;
}
# else
# define fimc_get_sysreg_regmap(node) (NULL)
# endif
2012-05-08 22:51:24 +04:00
/* -----------------------------------------------------*/
/* fimc-m2m.c */
void fimc_m2m_job_finish ( struct fimc_ctx * ctx , int vb_state ) ;
2010-10-07 17:06:16 +04:00
/* -----------------------------------------------------*/
/* fimc-capture.c */
2012-04-21 01:57:25 +04:00
int fimc_initialize_capture_subdev ( struct fimc_dev * fimc ) ;
void fimc_unregister_capture_subdev ( struct fimc_dev * fimc ) ;
2011-08-25 02:25:10 +04:00
int fimc_capture_ctrls_create ( struct fimc_dev * fimc ) ;
2011-06-10 22:36:58 +04:00
void fimc_sensor_notify ( struct v4l2_subdev * sd , unsigned int notification ,
void * arg ) ;
2011-09-02 13:25:32 +04:00
int fimc_capture_suspend ( struct fimc_dev * fimc ) ;
int fimc_capture_resume ( struct fimc_dev * fimc ) ;
2010-10-08 12:01:14 +04:00
2010-10-07 17:06:16 +04:00
/*
2011-06-10 22:36:59 +04:00
* Buffer list manipulation functions . Must be called with fimc . slock held .
2010-10-07 17:06:16 +04:00
*/
2011-06-10 22:36:59 +04:00
/**
* fimc_active_queue_add - add buffer to the capture active buffers queue
2021-03-23 12:15:08 +03:00
* @ vid_cap : camera capture device information
2011-06-10 22:36:59 +04:00
* @ buf : buffer to add to the active buffers list
*/
static inline void fimc_active_queue_add ( struct fimc_vid_cap * vid_cap ,
struct fimc_vid_buffer * buf )
2010-10-07 17:06:16 +04:00
{
2010-12-01 16:14:59 +03:00
list_add_tail ( & buf - > list , & vid_cap - > active_buf_q ) ;
2010-10-07 17:06:16 +04:00
vid_cap - > active_buf_cnt + + ;
}
2011-06-10 22:36:59 +04:00
/**
* fimc_active_queue_pop - pop buffer from the capture active buffers queue
2021-03-23 12:15:08 +03:00
* @ vid_cap : camera capture device information
2011-06-10 22:36:59 +04:00
*
* The caller must assure the active_buf_q list is not empty .
2010-10-07 17:06:16 +04:00
*/
2011-06-10 22:36:59 +04:00
static inline struct fimc_vid_buffer * fimc_active_queue_pop (
struct fimc_vid_cap * vid_cap )
2010-10-07 17:06:16 +04:00
{
struct fimc_vid_buffer * buf ;
buf = list_entry ( vid_cap - > active_buf_q . next ,
2010-12-01 16:14:59 +03:00
struct fimc_vid_buffer , list ) ;
list_del ( & buf - > list ) ;
2010-10-07 17:06:16 +04:00
vid_cap - > active_buf_cnt - - ;
return buf ;
}
2011-06-10 22:36:59 +04:00
/**
* fimc_pending_queue_add - add buffer to the capture pending buffers queue
2021-03-23 12:15:08 +03:00
* @ vid_cap : camera capture device information
2011-06-10 22:36:59 +04:00
* @ buf : buffer to add to the pending buffers list
*/
2010-10-07 17:06:16 +04:00
static inline void fimc_pending_queue_add ( struct fimc_vid_cap * vid_cap ,
struct fimc_vid_buffer * buf )
{
2010-12-01 16:14:59 +03:00
list_add_tail ( & buf - > list , & vid_cap - > pending_buf_q ) ;
2010-10-07 17:06:16 +04:00
}
2011-06-10 22:36:59 +04:00
/**
* fimc_pending_queue_pop - pop buffer from the capture pending buffers queue
2021-03-23 12:15:08 +03:00
* @ vid_cap : camera capture device information
2011-06-10 22:36:59 +04:00
*
* The caller must assure the pending_buf_q list is not empty .
*/
static inline struct fimc_vid_buffer * fimc_pending_queue_pop (
struct fimc_vid_cap * vid_cap )
2010-10-07 17:06:16 +04:00
{
struct fimc_vid_buffer * buf ;
buf = list_entry ( vid_cap - > pending_buf_q . next ,
2010-12-01 16:14:59 +03:00
struct fimc_vid_buffer , list ) ;
list_del ( & buf - > list ) ;
2010-10-07 17:06:16 +04:00
return buf ;
}
2010-08-03 16:50:29 +04:00
# endif /* FIMC_CORE_H_ */