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
*
* 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 .
*/
# 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>
2010-11-22 20:49:06 +03:00
# include <linux/sched.h>
2011-02-23 14:24:33 +03:00
# include <linux/spinlock.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>
2010-12-01 16:14:59 +03:00
# include <media/videobuf2-core.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>
2010-12-27 20:42:15 +03:00
# include <media/s5p_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
2011-09-01 13:01:08 +04:00
# define FIMC_MODULE_NAME "s5p-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
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_fmt - the driver ' s internal color format data
2010-10-07 17:06:16 +04:00
* @ mbus_code : Media Bus pixel code , - 1 if not applicable
2010-08-03 16:50:29 +04:00
* @ name : format description
2010-10-07 17:06:16 +04:00
* @ fourcc : the fourcc code for this format , 0 if not applicable
2010-08-03 16:50:29 +04:00
* @ color : the corresponding fimc_color_fmt
2010-12-08 20:05:08 +03:00
* @ memplanes : number of physically non - contiguous data planes
* @ colplanes : number of physically contiguous data planes
2011-05-20 13:14:59 +04:00
* @ depth : per plane driver ' s private ' number of bits per pixel '
2012-09-24 18:08:45 +04:00
* @ mdataplanes : bitmask indicating meta data plane ( s ) , ( 1 < < plane_no )
2011-05-20 13:14:59 +04:00
* @ flags : flags indicating which operation mode format applies to
2010-08-03 16:50:29 +04:00
*/
struct fimc_fmt {
2010-10-07 17:06:16 +04:00
enum v4l2_mbus_pixelcode mbus_code ;
2010-08-03 16:50:29 +04:00
char * name ;
u32 fourcc ;
u32 color ;
2010-12-08 20:05:08 +03:00
u16 memplanes ;
u16 colplanes ;
u8 depth [ VIDEO_MAX_PLANES ] ;
2012-09-24 18:08:45 +04:00
u16 mdataplanes ;
2010-10-07 17:06:16 +04:00
u16 flags ;
2011-12-01 21:02:24 +04:00
# define FMT_FLAGS_CAM (1 << 0)
# define FMT_FLAGS_M2M_IN (1 << 1)
# define FMT_FLAGS_M2M_OUT (1 << 2)
# define FMT_FLAGS_M2M (1 << 1 | 1 << 2)
# define FMT_HAS_ALPHA (1 << 3)
2012-09-24 18:08:45 +04:00
# define FMT_FLAGS_COMPRESSED (1 << 4)
2010-08-03 16:50:29 +04:00
} ;
/**
* 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 ;
} ;
/**
* struct fimc_addr - the FIMC physical address set for DMA
* @ y : luminance plane physical address
* @ cb : Cb plane physical address
* @ cr : Cr plane physical address
*/
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
2010-10-07 17:06:16 +04:00
* @ paddr : precalculated physical address set
* @ index : buffer index for the output DMA engine
2010-08-03 16:50:29 +04:00
*/
struct fimc_vid_buffer {
2010-12-01 16:14:59 +03:00
struct vb2_buffer vb ;
struct list_head list ;
2010-10-07 17:06:16 +04:00
struct fimc_addr paddr ;
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
2011-05-20 13:14:59 +04:00
* @ paddr : image frame buffer physical 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
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 ] ;
2010-08-03 16:50:29 +04:00
struct fimc_addr paddr ;
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 ;
} ;
2011-08-25 03:35:30 +04:00
# define FIMC_SD_PAD_SINK 0
# define FIMC_SD_PAD_SOURCE 1
# define FIMC_SD_PADS_NUM 2
2010-10-07 17:06:16 +04:00
/**
* struct fimc_vid_cap - camera capture device information
* @ ctx : hardware context data
* @ vfd : video device node for camera capture mode
2011-08-25 03:35:30 +04:00
* @ subdev : subdev exposing the FIMC processing block
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
* @ mf : media bus format at the FIMC camera input ( and the scaler output ) pad
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
* @ input_index : input ( camera sensor ) index
* @ refcnt : driver ' s private reference counter
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 ;
2010-12-01 16:14:59 +03:00
struct vb2_alloc_ctx * alloc_ctx ;
2012-07-26 14:15:42 +04:00
struct video_device vfd ;
2012-04-21 01:57:25 +04:00
struct v4l2_subdev subdev ;
2011-07-27 01:08:21 +04:00
struct media_pad vd_pad ;
2011-08-25 03:35:30 +04:00
struct v4l2_mbus_framefmt mf ;
struct media_pad sd_pads [ FIMC_SD_PADS_NUM ] ;
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 ;
int input_index ;
int refcnt ;
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
* @ pix_hoff : indicate whether horizontal offset is in pixels or in bytes
* @ has_inp_rot : set if has input rotator
* @ has_out_rot : set if has output rotator
2010-11-25 16:49:21 +03:00
* @ has_cistatus2 : 1 if CISTATUS2 register is present in this IP revision
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
2010-10-11 20:19:27 +04:00
* @ hor_offs_align : horizontal pixel offset aligment
2011-05-27 20:12:23 +04:00
* @ min_vsize_align : minimum vertical pixel size alignment
2010-10-11 20:19:27 +04:00
* @ out_buf_count : the number of buffers in output DMA sequence
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 pix_hoff : 1 ;
unsigned int has_inp_rot : 1 ;
unsigned int has_out_rot : 1 ;
2010-11-25 16:49:21 +03:00
unsigned int has_cistatus2 : 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 ;
2011-12-01 21:02:24 +04:00
unsigned int has_alpha : 1 ;
2012-08-02 17:27:46 +04:00
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-10-11 20:19:27 +04:00
u16 out_buf_count ;
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
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 ;
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
2011-05-20 13:14:59 +04:00
* @ variant : the IP variant information
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
2011-05-20 13:14:59 +04:00
* @ alloc_ctx : videobuf2 memory allocator context
2011-09-01 13:01:08 +04:00
* @ pipeline : fimc video capture pipeline data structure
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 ;
2012-08-02 17:27:46 +04:00
const struct fimc_variant * variant ;
2010-12-27 21:34:43 +03:00
u16 id ;
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 ;
2010-12-01 16:14:59 +03:00
struct vb2_alloc_ctx * alloc_ctx ;
2011-09-01 13:01:08 +04:00
struct fimc_pipeline pipeline ;
2012-08-14 17:46:58 +04:00
const struct fimc_pipeline_ops * pipeline_ops ;
2010-08-03 16:50:29 +04:00
} ;
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
/**
* fimc_ctx - the device context data
* @ 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
* @ in_order_1p input 1 - plane YCBCR order
* @ 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
* @ m2m_ctx : memory - to - memory device context
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 ;
struct v4l2_m2m_ctx * m2m_ctx ;
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 )
{
return ( code = = V4L2_MBUS_FMT_JPEG_1X8 | |
code = = V4L2_MBUS_FMT_S5C_UYVY_JPEG_1X8 ) ;
}
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 ;
2010-12-08 20:05:08 +03:00
if ( V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE = = type ) {
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 ) ;
2010-12-08 20:05:08 +03:00
} else if ( V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE = = type ) {
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 ,
2010-10-07 17:06:16 +04:00
struct fimc_frame * frame , struct fimc_addr * paddr ) ;
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
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
* @ 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
*
* 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
* @ 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
*
* 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_ */