2010-08-03 16:50:29 +04:00
/*
2011-04-28 16:06:19 +04:00
* Copyright ( C ) 2010 - 2011 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>
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
# include "regs-fimc.h"
# define err(fmt, args...) \
printk ( KERN_ERR " %s:%d: " fmt " \n " , __func__ , __LINE__ , # # args )
# 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
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 ,
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 ,
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 {
2010-10-07 17:06:16 +04:00
FIMC_CAMERA ,
2010-08-03 16:50:29 +04:00
FIMC_DMA ,
FIMC_LCDFIFO ,
FIMC_WRITEBACK
} ;
enum fimc_color_fmt {
2010-10-08 12:01:14 +04:00
S5P_FIMC_RGB565 = 0x10 ,
2010-08-03 16:50:29 +04:00
S5P_FIMC_RGB666 ,
S5P_FIMC_RGB888 ,
2010-10-08 12:01:14 +04:00
S5P_FIMC_RGB30_LOCAL ,
S5P_FIMC_YCBCR420 = 0x20 ,
2010-08-03 16:50:29 +04:00
S5P_FIMC_YCBYCR422 ,
S5P_FIMC_YCRYCB422 ,
S5P_FIMC_CBYCRY422 ,
S5P_FIMC_CRYCBY422 ,
S5P_FIMC_YCBCR444_LOCAL ,
} ;
2010-10-08 12:01:14 +04:00
# define fimc_fmt_is_rgb(x) ((x) & 0x10)
2010-08-03 16:50:29 +04:00
/* Cb/Cr chrominance components order for 2 plane Y/CbCr 4:2:2 formats. */
# define S5P_FIMC_LSB_CRCB S5P_CIOCTRL_ORDER422_2P_LSB_CRCB
/* The embedded image effect selection */
# define S5P_FIMC_EFFECT_ORIGINAL S5P_CIIMGEFF_FIN_BYPASS
# define S5P_FIMC_EFFECT_ARBITRARY S5P_CIIMGEFF_FIN_ARBITRARY
# define S5P_FIMC_EFFECT_NEGATIVE S5P_CIIMGEFF_FIN_NEGATIVE
# define S5P_FIMC_EFFECT_ARTFREEZE S5P_CIIMGEFF_FIN_ARTFREEZE
# define S5P_FIMC_EFFECT_EMBOSSING S5P_CIIMGEFF_FIN_EMBOSSING
# define S5P_FIMC_EFFECT_SIKHOUETTE S5P_CIIMGEFF_FIN_SILHOUETTE
/* The hardware context state. */
2010-10-08 12:01:14 +04:00
# define FIMC_PARAMS (1 << 0)
# define FIMC_SRC_ADDR (1 << 1)
# define FIMC_DST_ADDR (1 << 2)
# define FIMC_SRC_FMT (1 << 3)
# define FIMC_DST_FMT (1 << 4)
# define FIMC_CTX_M2M (1 << 5)
# define FIMC_CTX_CAP (1 << 6)
2011-02-23 14:24:33 +03:00
# define FIMC_CTX_SHUT (1 << 7)
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 '
* @ 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 ] ;
2010-10-07 17:06:16 +04:00
u16 flags ;
# define FMT_FLAGS_CAM (1 << 0)
# define FMT_FLAGS_M2M (1 << 1)
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 )
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 ;
2010-12-08 20:05:08 +03:00
unsigned long payload [ 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 ;
} ;
/**
* 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 {
struct video_device * vfd ;
struct v4l2_m2m_dev * m2m_dev ;
struct fimc_ctx * ctx ;
int refcnt ;
} ;
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
* @ sd : pointer to camera sensor subdevice currently in use
2011-07-27 01:08:21 +04:00
* @ vd_pad : fimc video capture node pad
2010-10-07 17:06:16 +04:00
* @ fmt : Media Bus format configured at selected image sensor
* @ 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 ;
2010-10-07 17:06:16 +04:00
struct video_device * vfd ;
2010-12-01 16:14:59 +03:00
struct v4l2_subdev * sd ; ;
2011-07-27 01:08:21 +04:00
struct media_pad vd_pad ;
2010-10-07 17:06:16 +04:00
struct v4l2_mbus_framefmt fmt ;
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
/**
* struct samsung_fimc_variant - camera interface variant information
*
* @ 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
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
* @ out_buf_count : the number of buffers in output DMA sequence
2010-08-03 16:50:29 +04:00
*/
struct samsung_fimc_variant {
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 ;
2010-10-11 20:19:27 +04:00
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 ;
u16 out_buf_count ;
2010-08-03 16:50:29 +04:00
} ;
/**
2010-10-08 12:01:14 +04:00
* struct samsung_fimc_driverdata - per device type driver data for init time .
2010-08-03 16:50:29 +04:00
*
* @ variant : the variant information for this driver .
* @ dev_cnt : number of fimc sub - devices available in SoC
2010-10-07 17:06:16 +04:00
* @ lclk_frequency : fimc bus clock frequency
2010-08-03 16:50:29 +04:00
*/
struct samsung_fimc_driverdata {
struct samsung_fimc_variant * variant [ FIMC_MAX_DEVS ] ;
2010-10-07 17:06:16 +04:00
unsigned long lclk_frequency ;
2010-10-11 20:19:27 +04:00
int num_entities ;
2010-08-03 16:50:29 +04:00
} ;
2011-09-01 13:01:08 +04:00
struct fimc_pipeline {
struct media_pipeline * pipe ;
struct v4l2_subdev * sensor ;
struct v4l2_subdev * csis ;
} ;
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 )
* @ num_clocks : the number of clocks managed by this device instance
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
* @ regs_res : the resource claimed for IO registers
2011-05-20 13:14:59 +04:00
* @ irq : FIMC interrupt number
* @ 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 ;
2010-08-03 16:50:29 +04:00
struct samsung_fimc_variant * variant ;
2010-12-27 21:34:43 +03:00
u16 id ;
u16 num_clocks ;
struct clk * clock [ MAX_FIMC_CLOCKS ] ;
2010-08-03 16:50:29 +04:00
void __iomem * regs ;
struct resource * regs_res ;
int irq ;
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 ;
2010-08-03 16:50:29 +04:00
} ;
/**
* fimc_ctx - the device context data
2011-05-20 13:14:59 +04:00
* @ slock : spinlock protecting this data structure
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
* @ 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
2011-08-25 02:25:10 +04:00
* @ ctrl_handler : v4l2 controls handler
* @ ctrl_rotate image rotation control
* @ ctrl_hflip horizontal flip control
* @ ctrl_vflip vartical flip control
* @ ctrls_rdy : true if the control handler is initialized
2010-08-03 16:50:29 +04:00
*/
struct fimc_ctx {
spinlock_t slock ;
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 ;
2011-08-25 02:25:10 +04:00
struct v4l2_ctrl_handler ctrl_handler ;
struct v4l2_ctrl * ctrl_rotate ;
struct v4l2_ctrl * ctrl_hflip ;
struct v4l2_ctrl * ctrl_vflip ;
bool ctrls_rdy ;
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-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 ;
}
static inline void fimc_ctx_state_lock_set ( u32 state , struct fimc_ctx * ctx )
{
unsigned long flags ;
spin_lock_irqsave ( & ctx - > slock , flags ) ;
ctx - > state | = state ;
spin_unlock_irqrestore ( & ctx - > slock , flags ) ;
}
static inline bool fimc_ctx_state_is_set ( u32 mask , struct fimc_ctx * ctx )
{
unsigned long flags ;
bool ret ;
spin_lock_irqsave ( & ctx - > slock , flags ) ;
ret = ( ctx - > state & mask ) = = mask ;
spin_unlock_irqrestore ( & ctx - > slock , flags ) ;
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
}
static inline void fimc_hw_clear_irq ( struct fimc_dev * dev )
{
u32 cfg = readl ( dev - > regs + S5P_CIGCTRL ) ;
cfg | = S5P_CIGCTRL_IRQ_CLR ;
writel ( cfg , dev - > regs + S5P_CIGCTRL ) ;
}
2010-10-08 12:01:14 +04:00
static inline void fimc_hw_enable_scaler ( struct fimc_dev * dev , bool on )
2010-08-03 16:50:29 +04:00
{
u32 cfg = readl ( dev - > regs + S5P_CISCCTRL ) ;
2010-10-08 12:01:14 +04:00
if ( on )
cfg | = S5P_CISCCTRL_SCALERSTART ;
else
cfg & = ~ S5P_CISCCTRL_SCALERSTART ;
2010-08-03 16:50:29 +04:00
writel ( cfg , dev - > regs + S5P_CISCCTRL ) ;
}
2010-10-08 12:01:14 +04:00
static inline void fimc_hw_activate_input_dma ( struct fimc_dev * dev , bool on )
2010-08-03 16:50:29 +04:00
{
2010-10-08 12:01:14 +04:00
u32 cfg = readl ( dev - > regs + S5P_MSCTRL ) ;
if ( on )
cfg | = S5P_MSCTRL_ENVID ;
else
cfg & = ~ S5P_MSCTRL_ENVID ;
writel ( cfg , dev - > regs + S5P_MSCTRL ) ;
2010-08-03 16:50:29 +04:00
}
static inline void fimc_hw_dis_capture ( struct fimc_dev * dev )
{
u32 cfg = readl ( dev - > regs + S5P_CIIMGCPT ) ;
cfg & = ~ ( S5P_CIIMGCPT_IMGCPTEN | S5P_CIIMGCPT_IMGCPTEN_SC ) ;
writel ( cfg , dev - > regs + S5P_CIIMGCPT ) ;
}
2010-10-11 20:19:27 +04:00
/**
* fimc_hw_set_dma_seq - configure output DMA buffer sequence
* @ mask : each bit corresponds to one of 32 output buffer registers set
* 1 to include buffer in the sequence , 0 to disable
*
* This function mask output DMA ring buffers , i . e . it allows to configure
* which of the output buffer address registers will be used by the DMA
* engine .
*/
static inline void fimc_hw_set_dma_seq ( struct fimc_dev * dev , u32 mask )
{
writel ( mask , dev - > regs + S5P_CIFCNTSEQ ) ;
}
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-11-25 16:49:21 +03:00
/* Return an index to the buffer actually being written. */
2010-10-07 17:06:16 +04:00
static inline u32 fimc_hw_get_frame_index ( struct fimc_dev * dev )
{
2010-11-25 16:49:21 +03:00
u32 reg ;
if ( dev - > variant - > has_cistatus2 ) {
reg = readl ( dev - > regs + S5P_CISTATUS2 ) & 0x3F ;
return reg > 0 ? - - reg : reg ;
} else {
reg = readl ( dev - > regs + S5P_CISTATUS ) ;
return ( reg & S5P_CISTATUS_FRAMECNT_MASK ) > >
S5P_CISTATUS_FRAMECNT_SHIFT ;
}
2010-10-07 17:06:16 +04:00
}
2010-08-03 16:50:29 +04:00
/* -----------------------------------------------------*/
/* fimc-reg.c */
2010-10-08 12:01:14 +04:00
void fimc_hw_reset ( struct fimc_dev * fimc ) ;
2010-08-03 16:50:29 +04:00
void fimc_hw_set_rotation ( struct fimc_ctx * ctx ) ;
void fimc_hw_set_target_format ( struct fimc_ctx * ctx ) ;
void fimc_hw_set_out_dma ( struct fimc_ctx * ctx ) ;
2010-10-08 12:01:14 +04:00
void fimc_hw_en_lastirq ( struct fimc_dev * fimc , int enable ) ;
void fimc_hw_en_irq ( struct fimc_dev * fimc , int enable ) ;
2010-12-28 17:27:13 +03:00
void fimc_hw_set_prescaler ( struct fimc_ctx * ctx ) ;
void fimc_hw_set_mainscaler ( struct fimc_ctx * ctx ) ;
2010-08-03 16:50:29 +04:00
void fimc_hw_en_capture ( struct fimc_ctx * ctx ) ;
void fimc_hw_set_effect ( struct fimc_ctx * ctx ) ;
void fimc_hw_set_in_dma ( struct fimc_ctx * ctx ) ;
void fimc_hw_set_input_path ( struct fimc_ctx * ctx ) ;
void fimc_hw_set_output_path ( struct fimc_ctx * ctx ) ;
2010-10-08 12:01:14 +04:00
void fimc_hw_set_input_addr ( struct fimc_dev * fimc , struct fimc_addr * paddr ) ;
void fimc_hw_set_output_addr ( struct fimc_dev * fimc , struct fimc_addr * paddr ,
2010-12-08 20:05:08 +03:00
int index ) ;
2010-10-07 17:06:16 +04:00
int fimc_hw_set_camera_source ( struct fimc_dev * fimc ,
2010-12-27 20:42:15 +03:00
struct s5p_fimc_isp_info * cam ) ;
2010-10-07 17:06:16 +04:00
int fimc_hw_set_camera_offset ( struct fimc_dev * fimc , struct fimc_frame * f ) ;
int fimc_hw_set_camera_polarity ( struct fimc_dev * fimc ,
2010-12-27 20:42:15 +03:00
struct s5p_fimc_isp_info * cam ) ;
2010-10-07 17:06:16 +04:00
int fimc_hw_set_camera_type ( struct fimc_dev * fimc ,
2010-12-27 20:42:15 +03:00
struct s5p_fimc_isp_info * cam ) ;
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-06-10 22:36:50 +04:00
int fimc_try_fmt_mplane ( struct fimc_ctx * ctx , struct v4l2_format * f ) ;
2010-10-07 17:06:16 +04:00
int fimc_try_crop ( struct fimc_ctx * ctx , struct v4l2_crop * cr ) ;
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-06-10 22:36:50 +04:00
int fimc_fill_format ( struct fimc_frame * frame , struct v4l2_format * f ) ;
2010-10-07 17:06:16 +04:00
2011-06-13 18:09:40 +04:00
struct fimc_fmt * fimc_find_format ( u32 * pixelformat , u32 * mbus_code ,
unsigned int mask , int index ) ;
2010-10-07 17:06:16 +04:00
2010-12-29 04:12:43 +03:00
int fimc_check_scaler_ratio ( int sw , int sh , int dw , int dh , int rot ) ;
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 ) ;
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
/* -----------------------------------------------------*/
/* fimc-capture.c */
2011-06-10 22:36:48 +04:00
int fimc_register_capture_device ( struct fimc_dev * fimc ,
struct v4l2_device * v4l2_dev ) ;
2010-10-07 17:06:16 +04:00
void fimc_unregister_capture_device ( struct fimc_dev * fimc ) ;
2011-08-25 02:25:10 +04:00
int fimc_capture_ctrls_create ( struct fimc_dev * fimc ) ;
2010-10-07 17:06:16 +04:00
int fimc_vid_cap_buf_queue ( struct fimc_dev * fimc ,
struct fimc_vid_buffer * fimc_vb ) ;
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
/* Locking: the caller holds fimc->slock */
static inline void fimc_activate_capture ( struct fimc_ctx * ctx )
{
fimc_hw_enable_scaler ( ctx - > fimc_dev , ctx - > scaler . enabled ) ;
fimc_hw_en_capture ( ctx ) ;
}
static inline void fimc_deactivate_capture ( struct fimc_dev * fimc )
{
fimc_hw_en_lastirq ( fimc , true ) ;
fimc_hw_dis_capture ( fimc ) ;
fimc_hw_enable_scaler ( fimc , false ) ;
fimc_hw_en_lastirq ( fimc , false ) ;
}
2010-08-03 16:50:29 +04:00
2010-10-07 17:06:16 +04:00
/*
2010-12-01 16:14:59 +03:00
* Add buf to the capture active buffers queue .
* Locking : Need to be called with fimc_dev : : slock held .
2010-10-07 17:06:16 +04:00
*/
static inline void active_queue_add ( struct fimc_vid_cap * vid_cap ,
2010-12-01 16:14:59 +03:00
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 + + ;
}
/*
* Pop a video buffer from the capture active buffers queue
2010-12-01 16:14:59 +03:00
* Locking : Need to be called with fimc_dev : : slock held .
2010-10-07 17:06:16 +04:00
*/
static inline struct fimc_vid_buffer *
active_queue_pop ( struct fimc_vid_cap * vid_cap )
{
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 ;
}
/* Add video buffer to the capture pending buffers queue */
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
}
/* Add video buffer to the capture pending buffers queue */
static inline struct fimc_vid_buffer *
pending_queue_pop ( struct fimc_vid_cap * vid_cap )
{
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_ */