2010-10-07 17:06:16 +04:00
/*
2011-04-28 16:06:19 +04:00
* Samsung S5P / EXYNOS4 SoC series camera interface ( camera capture ) driver
2010-10-07 17:06:16 +04:00
*
2011-04-28 16:06:19 +04:00
* Copyright ( C ) 2010 - 2011 Samsung Electronics Co . , Ltd .
2010-10-07 17:06:16 +04:00
* Author : Sylwester Nawrocki , < s . nawrocki @ samsung . com >
*
* 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 .
*/
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/types.h>
# include <linux/errno.h>
# include <linux/bug.h>
# include <linux/interrupt.h>
# include <linux/device.h>
2011-09-02 13:25:32 +04:00
# include <linux/pm_runtime.h>
2010-10-07 17:06:16 +04:00
# include <linux/list.h>
# include <linux/slab.h>
# include <linux/videodev2.h>
# include <media/v4l2-device.h>
# include <media/v4l2-ioctl.h>
# include <media/v4l2-mem2mem.h>
2010-12-01 16:14:59 +03:00
# include <media/videobuf2-core.h>
# include <media/videobuf2-dma-contig.h>
2010-10-07 17:06:16 +04:00
2011-08-25 02:25:10 +04:00
# include "fimc-mdevice.h"
2010-10-07 17:06:16 +04:00
# include "fimc-core.h"
2012-05-02 13:14:49 +04:00
# include "fimc-reg.h"
2010-10-07 17:06:16 +04:00
2011-06-10 22:36:53 +04:00
static int fimc_init_capture ( struct fimc_dev * fimc )
{
struct fimc_ctx * ctx = fimc - > vid_cap . ctx ;
struct fimc_sensor_info * sensor ;
unsigned long flags ;
int ret = 0 ;
if ( fimc - > pipeline . sensor = = NULL | | ctx = = NULL )
return - ENXIO ;
if ( ctx - > s_frame . fmt = = NULL )
return - EINVAL ;
sensor = v4l2_get_subdev_hostdata ( fimc - > pipeline . sensor ) ;
spin_lock_irqsave ( & fimc - > slock , flags ) ;
fimc_prepare_dma_offset ( ctx , & ctx - > d_frame ) ;
fimc_set_yuv_order ( ctx ) ;
fimc_hw_set_camera_polarity ( fimc , sensor - > pdata ) ;
fimc_hw_set_camera_type ( fimc , sensor - > pdata ) ;
fimc_hw_set_camera_source ( fimc , sensor - > pdata ) ;
fimc_hw_set_camera_offset ( fimc , & ctx - > s_frame ) ;
ret = fimc_set_scaler_info ( ctx ) ;
if ( ! ret ) {
fimc_hw_set_input_path ( ctx ) ;
fimc_hw_set_prescaler ( ctx ) ;
fimc_hw_set_mainscaler ( ctx ) ;
fimc_hw_set_target_format ( ctx ) ;
fimc_hw_set_rotation ( ctx ) ;
2011-08-26 21:57:06 +04:00
fimc_hw_set_effect ( ctx , false ) ;
2011-06-10 22:36:53 +04:00
fimc_hw_set_output_path ( ctx ) ;
fimc_hw_set_out_dma ( ctx ) ;
2011-12-01 21:02:24 +04:00
if ( fimc - > variant - > has_alpha )
fimc_hw_set_rgb_alpha ( ctx ) ;
2011-08-25 03:35:30 +04:00
clear_bit ( ST_CAPT_APPLY_CFG , & fimc - > state ) ;
2011-06-10 22:36:53 +04:00
}
spin_unlock_irqrestore ( & fimc - > slock , flags ) ;
return ret ;
}
2011-08-25 03:45:34 +04:00
static int fimc_capture_state_cleanup ( struct fimc_dev * fimc , bool suspend )
2010-10-07 17:06:16 +04:00
{
2011-08-29 15:51:49 +04:00
struct fimc_vid_cap * cap = & fimc - > vid_cap ;
2010-12-01 16:14:59 +03:00
struct fimc_vid_buffer * buf ;
2011-08-29 15:51:49 +04:00
unsigned long flags ;
2011-08-25 03:45:34 +04:00
bool streaming ;
2010-10-07 17:06:16 +04:00
spin_lock_irqsave ( & fimc - > slock , flags ) ;
2011-08-25 03:45:34 +04:00
streaming = fimc - > state & ( 1 < < ST_CAPT_ISP_STREAM ) ;
2010-10-07 17:06:16 +04:00
2011-08-25 03:45:34 +04:00
fimc - > state & = ~ ( 1 < < ST_CAPT_RUN | 1 < < ST_CAPT_SHUT |
1 < < ST_CAPT_STREAM | 1 < < ST_CAPT_ISP_STREAM ) ;
2012-03-16 18:47:51 +04:00
if ( suspend )
fimc - > state | = ( 1 < < ST_CAPT_SUSPENDED ) ;
else
2011-08-25 03:45:34 +04:00
fimc - > state & = ~ ( 1 < < ST_CAPT_PEND | 1 < < ST_CAPT_SUSPENDED ) ;
2010-12-01 16:14:59 +03:00
2011-08-25 03:45:34 +04:00
/* Release unused buffers */
while ( ! suspend & & ! list_empty ( & cap - > pending_buf_q ) ) {
2011-06-10 22:36:59 +04:00
buf = fimc_pending_queue_pop ( cap ) ;
2010-12-01 16:14:59 +03:00
vb2_buffer_done ( & buf - > vb , VB2_BUF_STATE_ERROR ) ;
}
2011-08-25 03:45:34 +04:00
/* If suspending put unused buffers onto pending queue */
2010-12-01 16:14:59 +03:00
while ( ! list_empty ( & cap - > active_buf_q ) ) {
2011-06-10 22:36:59 +04:00
buf = fimc_active_queue_pop ( cap ) ;
2011-08-25 03:45:34 +04:00
if ( suspend )
fimc_pending_queue_add ( cap , buf ) ;
else
vb2_buffer_done ( & buf - > vb , VB2_BUF_STATE_ERROR ) ;
2010-12-01 16:14:59 +03:00
}
2011-10-05 21:20:45 +04:00
fimc_hw_reset ( fimc ) ;
cap - > buf_index = 0 ;
2010-10-07 17:06:16 +04:00
spin_unlock_irqrestore ( & fimc - > slock , flags ) ;
2011-08-26 21:51:00 +04:00
2011-08-25 03:45:34 +04:00
if ( streaming )
2011-08-26 21:51:00 +04:00
return fimc_pipeline_s_stream ( fimc , 0 ) ;
else
return 0 ;
2011-08-29 15:51:49 +04:00
}
2011-08-25 03:45:34 +04:00
static int fimc_stop_capture ( struct fimc_dev * fimc , bool suspend )
2011-08-29 15:51:49 +04:00
{
unsigned long flags ;
if ( ! fimc_capture_active ( fimc ) )
return 0 ;
spin_lock_irqsave ( & fimc - > slock , flags ) ;
set_bit ( ST_CAPT_SHUT , & fimc - > state ) ;
fimc_deactivate_capture ( fimc ) ;
spin_unlock_irqrestore ( & fimc - > slock , flags ) ;
wait_event_timeout ( fimc - > irq_queue ,
! test_bit ( ST_CAPT_SHUT , & fimc - > state ) ,
2011-08-25 03:45:34 +04:00
( 2 * HZ / 10 ) ) ; /* 200 ms */
2010-10-07 17:06:16 +04:00
2011-08-25 03:45:34 +04:00
return fimc_capture_state_cleanup ( fimc , suspend ) ;
2010-10-07 17:06:16 +04:00
}
2011-08-25 03:35:30 +04:00
/**
* fimc_capture_config_update - apply the camera interface configuration
*
* To be called from within the interrupt handler with fimc . slock
* spinlock held . It updates the camera pixel crop , rotation and
* image flip in H / W .
*/
2012-05-08 22:51:24 +04:00
static int fimc_capture_config_update ( struct fimc_ctx * ctx )
2011-08-25 03:35:30 +04:00
{
struct fimc_dev * fimc = ctx - > fimc_dev ;
int ret ;
2011-10-05 21:20:45 +04:00
if ( ! test_bit ( ST_CAPT_APPLY_CFG , & fimc - > state ) )
2011-08-25 03:35:30 +04:00
return 0 ;
fimc_hw_set_camera_offset ( fimc , & ctx - > s_frame ) ;
2012-03-19 20:11:40 +04:00
2011-08-25 03:35:30 +04:00
ret = fimc_set_scaler_info ( ctx ) ;
2012-03-19 20:11:40 +04:00
if ( ret )
return ret ;
fimc_hw_set_prescaler ( ctx ) ;
fimc_hw_set_mainscaler ( ctx ) ;
fimc_hw_set_target_format ( ctx ) ;
fimc_hw_set_rotation ( ctx ) ;
fimc_prepare_dma_offset ( ctx , & ctx - > d_frame ) ;
fimc_hw_set_out_dma ( ctx ) ;
if ( fimc - > variant - > has_alpha )
fimc_hw_set_rgb_alpha ( ctx ) ;
clear_bit ( ST_CAPT_APPLY_CFG , & fimc - > state ) ;
2011-08-25 03:35:30 +04:00
return ret ;
}
2011-08-29 15:51:49 +04:00
2012-05-08 22:51:24 +04:00
void fimc_capture_irq_handler ( struct fimc_dev * fimc , int deq_buf )
{
struct fimc_vid_cap * cap = & fimc - > vid_cap ;
struct fimc_vid_buffer * v_buf ;
struct timeval * tv ;
struct timespec ts ;
if ( test_and_clear_bit ( ST_CAPT_SHUT , & fimc - > state ) ) {
wake_up ( & fimc - > irq_queue ) ;
goto done ;
}
if ( ! list_empty ( & cap - > active_buf_q ) & &
test_bit ( ST_CAPT_RUN , & fimc - > state ) & & deq_buf ) {
ktime_get_real_ts ( & ts ) ;
v_buf = fimc_active_queue_pop ( cap ) ;
tv = & v_buf - > vb . v4l2_buf . timestamp ;
tv - > tv_sec = ts . tv_sec ;
tv - > tv_usec = ts . tv_nsec / NSEC_PER_USEC ;
v_buf - > vb . v4l2_buf . sequence = cap - > frame_count + + ;
vb2_buffer_done ( & v_buf - > vb , VB2_BUF_STATE_DONE ) ;
}
if ( ! list_empty ( & cap - > pending_buf_q ) ) {
v_buf = fimc_pending_queue_pop ( cap ) ;
fimc_hw_set_output_addr ( fimc , & v_buf - > paddr , cap - > buf_index ) ;
v_buf - > index = cap - > buf_index ;
/* Move the buffer to the capture active queue */
fimc_active_queue_add ( cap , v_buf ) ;
dbg ( " next frame: %d, done frame: %d " ,
fimc_hw_get_frame_index ( fimc ) , v_buf - > index ) ;
if ( + + cap - > buf_index > = FIMC_MAX_OUT_BUFS )
cap - > buf_index = 0 ;
}
if ( cap - > active_buf_cnt = = 0 ) {
if ( deq_buf )
clear_bit ( ST_CAPT_RUN , & fimc - > state ) ;
if ( + + cap - > buf_index > = FIMC_MAX_OUT_BUFS )
cap - > buf_index = 0 ;
} else {
set_bit ( ST_CAPT_RUN , & fimc - > state ) ;
}
fimc_capture_config_update ( cap - > ctx ) ;
done :
if ( cap - > active_buf_cnt = = 1 ) {
fimc_deactivate_capture ( fimc ) ;
clear_bit ( ST_CAPT_STREAM , & fimc - > state ) ;
}
dbg ( " frame: %d, active_buf_cnt: %d " ,
fimc_hw_get_frame_index ( fimc ) , cap - > active_buf_cnt ) ;
}
2011-08-29 15:51:49 +04:00
static int start_streaming ( struct vb2_queue * q , unsigned int count )
2010-12-01 16:14:59 +03:00
{
struct fimc_ctx * ctx = q - > drv_priv ;
struct fimc_dev * fimc = ctx - > fimc_dev ;
2011-06-10 22:36:53 +04:00
struct fimc_vid_cap * vid_cap = & fimc - > vid_cap ;
2011-08-29 15:51:49 +04:00
int min_bufs ;
2010-12-01 16:14:59 +03:00
int ret ;
2011-06-10 22:36:53 +04:00
vid_cap - > frame_count = 0 ;
2011-02-07 21:59:46 +03:00
2011-06-10 22:36:53 +04:00
ret = fimc_init_capture ( fimc ) ;
2010-12-01 16:14:59 +03:00
if ( ret )
2011-08-29 15:51:49 +04:00
goto error ;
2010-12-01 16:14:59 +03:00
set_bit ( ST_CAPT_PEND , & fimc - > state ) ;
2011-08-29 15:51:49 +04:00
min_bufs = fimc - > vid_cap . reqbufs_count > 1 ? 2 : 1 ;
2011-08-26 21:51:00 +04:00
if ( vid_cap - > active_buf_cnt > = min_bufs & &
! test_and_set_bit ( ST_CAPT_STREAM , & fimc - > state ) ) {
2011-08-29 15:51:49 +04:00
fimc_activate_capture ( ctx ) ;
2011-08-26 21:51:00 +04:00
if ( ! test_and_set_bit ( ST_CAPT_ISP_STREAM , & fimc - > state ) )
fimc_pipeline_s_stream ( fimc , 1 ) ;
}
2010-12-01 16:14:59 +03:00
return 0 ;
2011-08-29 15:51:49 +04:00
error :
2011-08-25 03:45:34 +04:00
fimc_capture_state_cleanup ( fimc , false ) ;
2011-08-29 15:51:49 +04:00
return ret ;
2010-12-01 16:14:59 +03:00
}
static int stop_streaming ( struct vb2_queue * q )
{
struct fimc_ctx * ctx = q - > drv_priv ;
struct fimc_dev * fimc = ctx - > fimc_dev ;
2011-02-23 14:24:33 +03:00
if ( ! fimc_capture_active ( fimc ) )
2010-12-01 16:14:59 +03:00
return - EINVAL ;
2011-08-25 03:45:34 +04:00
return fimc_stop_capture ( fimc , false ) ;
2010-12-01 16:14:59 +03:00
}
2011-09-02 13:25:32 +04:00
int fimc_capture_suspend ( struct fimc_dev * fimc )
{
2011-08-25 03:45:34 +04:00
bool suspend = fimc_capture_busy ( fimc ) ;
int ret = fimc_stop_capture ( fimc , suspend ) ;
if ( ret )
return ret ;
return fimc_pipeline_shutdown ( fimc ) ;
2011-09-02 13:25:32 +04:00
}
2011-08-25 03:45:34 +04:00
static void buffer_queue ( struct vb2_buffer * vb ) ;
2011-09-02 13:25:32 +04:00
int fimc_capture_resume ( struct fimc_dev * fimc )
{
2011-08-25 03:45:34 +04:00
struct fimc_vid_cap * vid_cap = & fimc - > vid_cap ;
struct fimc_vid_buffer * buf ;
int i ;
if ( ! test_and_clear_bit ( ST_CAPT_SUSPENDED , & fimc - > state ) )
return 0 ;
INIT_LIST_HEAD ( & fimc - > vid_cap . active_buf_q ) ;
vid_cap - > buf_index = 0 ;
fimc_pipeline_initialize ( fimc , & fimc - > vid_cap . vfd - > entity ,
false ) ;
fimc_init_capture ( fimc ) ;
clear_bit ( ST_CAPT_SUSPENDED , & fimc - > state ) ;
for ( i = 0 ; i < vid_cap - > reqbufs_count ; i + + ) {
if ( list_empty ( & vid_cap - > pending_buf_q ) )
break ;
buf = fimc_pending_queue_pop ( vid_cap ) ;
buffer_queue ( & buf - > vb ) ;
}
2011-09-02 13:25:32 +04:00
return 0 ;
2011-08-25 03:45:34 +04:00
2011-09-02 13:25:32 +04:00
}
2012-04-23 00:07:09 +04:00
static int queue_setup ( struct vb2_queue * vq , const struct v4l2_format * pfmt ,
2011-08-24 17:30:21 +04:00
unsigned int * num_buffers , unsigned int * num_planes ,
unsigned int sizes [ ] , void * allocators [ ] )
2010-12-01 16:14:59 +03:00
{
2012-04-23 00:07:09 +04:00
const struct v4l2_pix_format_mplane * pixm = NULL ;
2010-12-01 16:14:59 +03:00
struct fimc_ctx * ctx = vq - > drv_priv ;
2012-04-23 00:07:09 +04:00
struct fimc_frame * frame = & ctx - > d_frame ;
struct fimc_fmt * fmt = frame - > fmt ;
unsigned long wh ;
2010-12-08 20:05:08 +03:00
int i ;
2010-12-01 16:14:59 +03:00
2012-04-23 00:07:09 +04:00
if ( pfmt ) {
pixm = & pfmt - > fmt . pix_mp ;
fmt = fimc_find_format ( & pixm - > pixelformat , NULL ,
FMT_FLAGS_CAM | FMT_FLAGS_M2M , - 1 ) ;
wh = pixm - > width * pixm - > height ;
} else {
wh = frame - > f_width * frame - > f_height ;
}
if ( fmt = = NULL )
2010-12-01 16:14:59 +03:00
return - EINVAL ;
2010-12-08 20:05:08 +03:00
* num_planes = fmt - > memplanes ;
2010-12-01 16:14:59 +03:00
2010-12-08 20:05:08 +03:00
for ( i = 0 ; i < fmt - > memplanes ; i + + ) {
2012-04-23 00:07:09 +04:00
unsigned int size = ( wh * fmt - > depth [ i ] ) / 8 ;
if ( pixm )
sizes [ i ] = max ( size , pixm - > plane_fmt [ i ] . sizeimage ) ;
else
sizes [ i ] = size ;
2010-12-08 20:05:08 +03:00
allocators [ i ] = ctx - > fimc_dev - > alloc_ctx ;
}
2010-12-01 16:14:59 +03:00
2010-12-08 20:05:08 +03:00
return 0 ;
2010-12-01 16:14:59 +03:00
}
static int buffer_prepare ( struct vb2_buffer * vb )
{
struct vb2_queue * vq = vb - > vb2_queue ;
struct fimc_ctx * ctx = vq - > drv_priv ;
int i ;
2011-08-26 21:51:00 +04:00
if ( ctx - > d_frame . fmt = = NULL )
2010-12-08 20:05:08 +03:00
return - EINVAL ;
2010-12-01 16:14:59 +03:00
2010-12-08 20:05:08 +03:00
for ( i = 0 ; i < ctx - > d_frame . fmt - > memplanes ; i + + ) {
2011-08-26 21:51:00 +04:00
unsigned long size = ctx - > d_frame . payload [ i ] ;
2010-12-01 16:14:59 +03:00
if ( vb2_plane_size ( vb , i ) < size ) {
2011-06-10 22:36:48 +04:00
v4l2_err ( ctx - > fimc_dev - > vid_cap . vfd ,
" User buffer too small (%ld < %ld) \n " ,
2010-12-01 16:14:59 +03:00
vb2_plane_size ( vb , i ) , size ) ;
return - EINVAL ;
}
vb2_set_plane_payload ( vb , i , size ) ;
}
return 0 ;
}
static void buffer_queue ( struct vb2_buffer * vb )
{
struct fimc_vid_buffer * buf
= container_of ( vb , struct fimc_vid_buffer , vb ) ;
2011-08-26 21:51:00 +04:00
struct fimc_ctx * ctx = vb2_get_drv_priv ( vb - > vb2_queue ) ;
struct fimc_dev * fimc = ctx - > fimc_dev ;
2010-12-01 16:14:59 +03:00
struct fimc_vid_cap * vid_cap = & fimc - > vid_cap ;
unsigned long flags ;
2011-02-07 21:59:46 +03:00
int min_bufs ;
2010-12-01 16:14:59 +03:00
spin_lock_irqsave ( & fimc - > slock , flags ) ;
2011-02-07 21:59:46 +03:00
fimc_prepare_addr ( ctx , & buf - > vb , & ctx - > d_frame , & buf - > paddr ) ;
2011-08-25 03:45:34 +04:00
if ( ! test_bit ( ST_CAPT_SUSPENDED , & fimc - > state ) & &
! test_bit ( ST_CAPT_STREAM , & fimc - > state ) & &
vid_cap - > active_buf_cnt < FIMC_MAX_OUT_BUFS ) {
2011-02-07 21:59:46 +03:00
/* Setup the buffer directly for processing. */
int buf_id = ( vid_cap - > reqbufs_count = = 1 ) ? - 1 :
vid_cap - > buf_index ;
2010-12-01 16:14:59 +03:00
2011-02-07 21:59:46 +03:00
fimc_hw_set_output_addr ( fimc , & buf - > paddr , buf_id ) ;
buf - > index = vid_cap - > buf_index ;
2011-06-10 22:36:59 +04:00
fimc_active_queue_add ( vid_cap , buf ) ;
2010-12-01 16:14:59 +03:00
2011-02-07 21:59:46 +03:00
if ( + + vid_cap - > buf_index > = FIMC_MAX_OUT_BUFS )
vid_cap - > buf_index = 0 ;
} else {
fimc_pending_queue_add ( vid_cap , buf ) ;
2010-12-01 16:14:59 +03:00
}
2011-02-07 21:59:46 +03:00
min_bufs = vid_cap - > reqbufs_count > 1 ? 2 : 1 ;
2011-08-26 21:51:00 +04:00
2011-08-29 15:51:49 +04:00
if ( vb2_is_streaming ( & vid_cap - > vbq ) & &
vid_cap - > active_buf_cnt > = min_bufs & &
2011-08-26 21:51:00 +04:00
! test_and_set_bit ( ST_CAPT_STREAM , & fimc - > state ) ) {
2011-02-07 21:59:46 +03:00
fimc_activate_capture ( ctx ) ;
2011-08-26 21:51:00 +04:00
spin_unlock_irqrestore ( & fimc - > slock , flags ) ;
2011-02-07 21:59:46 +03:00
2011-08-26 21:51:00 +04:00
if ( ! test_and_set_bit ( ST_CAPT_ISP_STREAM , & fimc - > state ) )
fimc_pipeline_s_stream ( fimc , 1 ) ;
return ;
}
2010-12-01 16:14:59 +03:00
spin_unlock_irqrestore ( & fimc - > slock , flags ) ;
}
static void fimc_lock ( struct vb2_queue * vq )
{
struct fimc_ctx * ctx = vb2_get_drv_priv ( vq ) ;
mutex_lock ( & ctx - > fimc_dev - > lock ) ;
}
static void fimc_unlock ( struct vb2_queue * vq )
{
struct fimc_ctx * ctx = vb2_get_drv_priv ( vq ) ;
mutex_unlock ( & ctx - > fimc_dev - > lock ) ;
}
static struct vb2_ops fimc_capture_qops = {
. queue_setup = queue_setup ,
. buf_prepare = buffer_prepare ,
. buf_queue = buffer_queue ,
. wait_prepare = fimc_unlock ,
. wait_finish = fimc_lock ,
. start_streaming = start_streaming ,
. stop_streaming = stop_streaming ,
} ;
2011-08-25 02:25:10 +04:00
/**
* fimc_capture_ctrls_create - initialize the control handler
* Initialize the capture video node control handler and fill it
* with the FIMC controls . Inherit any sensor ' s controls if the
* ' user_subdev_api ' flag is false ( default behaviour ) .
* This function need to be called with the graph mutex held .
*/
int fimc_capture_ctrls_create ( struct fimc_dev * fimc )
{
struct fimc_vid_cap * vid_cap = & fimc - > vid_cap ;
int ret ;
if ( WARN_ON ( vid_cap - > ctx = = NULL ) )
return - ENXIO ;
if ( vid_cap - > ctx - > ctrls_rdy )
return 0 ;
ret = fimc_ctrls_create ( vid_cap - > ctx ) ;
if ( ret | | vid_cap - > user_subdev_api )
return ret ;
return v4l2_ctrl_add_handler ( & vid_cap - > ctx - > ctrl_handler ,
fimc - > pipeline . sensor - > ctrl_handler ) ;
}
2011-08-25 03:35:30 +04:00
static int fimc_capture_set_default_format ( struct fimc_dev * fimc ) ;
2010-10-07 17:06:16 +04:00
static int fimc_capture_open ( struct file * file )
{
struct fimc_dev * fimc = video_drvdata ( file ) ;
2011-06-10 22:36:50 +04:00
int ret = v4l2_fh_open ( file ) ;
if ( ret )
return ret ;
2010-10-07 17:06:16 +04:00
dbg ( " pid: %d, state: 0x%lx " , task_pid_nr ( current ) , fimc - > state ) ;
/* Return if the corresponding video mem2mem node is already opened. */
if ( fimc_m2m_active ( fimc ) )
return - EBUSY ;
2011-08-25 03:45:34 +04:00
set_bit ( ST_CAPT_BUSY , & fimc - > state ) ;
2011-08-26 21:51:00 +04:00
pm_runtime_get_sync ( & fimc - > pdev - > dev ) ;
if ( + + fimc - > vid_cap . refcnt = = 1 ) {
ret = fimc_pipeline_initialize ( fimc ,
& fimc - > vid_cap . vfd - > entity , true ) ;
if ( ret < 0 ) {
dev_err ( & fimc - > pdev - > dev ,
" Video pipeline initialization failed \n " ) ;
pm_runtime_put_sync ( & fimc - > pdev - > dev ) ;
fimc - > vid_cap . refcnt - - ;
v4l2_fh_release ( file ) ;
2011-08-25 03:45:34 +04:00
clear_bit ( ST_CAPT_BUSY , & fimc - > state ) ;
2011-08-26 21:51:00 +04:00
return ret ;
}
2011-08-25 02:25:10 +04:00
ret = fimc_capture_ctrls_create ( fimc ) ;
2011-08-25 03:35:30 +04:00
if ( ! ret & & ! fimc - > vid_cap . user_subdev_api )
ret = fimc_capture_set_default_format ( fimc ) ;
2011-08-26 21:51:00 +04:00
}
2011-08-25 02:25:10 +04:00
return ret ;
2010-10-07 17:06:16 +04:00
}
static int fimc_capture_close ( struct file * file )
{
struct fimc_dev * fimc = video_drvdata ( file ) ;
dbg ( " pid: %d, state: 0x%lx " , task_pid_nr ( current ) , fimc - > state ) ;
if ( - - fimc - > vid_cap . refcnt = = 0 ) {
2011-08-25 03:45:34 +04:00
clear_bit ( ST_CAPT_BUSY , & fimc - > state ) ;
fimc_stop_capture ( fimc , false ) ;
2011-08-26 21:51:00 +04:00
fimc_pipeline_shutdown ( fimc ) ;
2011-08-25 03:45:34 +04:00
clear_bit ( ST_CAPT_SUSPENDED , & fimc - > state ) ;
2010-10-07 17:06:16 +04:00
}
2011-09-02 13:25:32 +04:00
pm_runtime_put ( & fimc - > pdev - > dev ) ;
2011-08-25 03:45:34 +04:00
if ( fimc - > vid_cap . refcnt = = 0 ) {
vb2_queue_release ( & fimc - > vid_cap . vbq ) ;
fimc_ctrls_delete ( fimc - > vid_cap . ctx ) ;
}
2011-06-10 22:36:50 +04:00
return v4l2_fh_release ( file ) ;
2010-10-07 17:06:16 +04:00
}
static unsigned int fimc_capture_poll ( struct file * file ,
struct poll_table_struct * wait )
{
2011-06-10 22:36:50 +04:00
struct fimc_dev * fimc = video_drvdata ( file ) ;
2010-10-07 17:06:16 +04:00
2010-12-01 16:25:18 +03:00
return vb2_poll ( & fimc - > vid_cap . vbq , file , wait ) ;
2010-10-07 17:06:16 +04:00
}
static int fimc_capture_mmap ( struct file * file , struct vm_area_struct * vma )
{
2011-06-10 22:36:50 +04:00
struct fimc_dev * fimc = video_drvdata ( file ) ;
2010-10-07 17:06:16 +04:00
2010-12-01 16:25:18 +03:00
return vb2_mmap ( & fimc - > vid_cap . vbq , vma ) ;
2010-10-07 17:06:16 +04:00
}
static const struct v4l2_file_operations fimc_capture_fops = {
. owner = THIS_MODULE ,
. open = fimc_capture_open ,
. release = fimc_capture_close ,
. poll = fimc_capture_poll ,
. unlocked_ioctl = video_ioctl2 ,
. mmap = fimc_capture_mmap ,
} ;
2011-08-25 03:35:30 +04:00
/*
* Format and crop negotiation helpers
*/
static struct fimc_fmt * fimc_capture_try_format ( struct fimc_ctx * ctx ,
u32 * width , u32 * height ,
u32 * code , u32 * fourcc , int pad )
{
bool rotation = ctx - > rotation = = 90 | | ctx - > rotation = = 270 ;
struct fimc_dev * fimc = ctx - > fimc_dev ;
struct samsung_fimc_variant * var = fimc - > variant ;
struct fimc_pix_limit * pl = var - > pix_limit ;
struct fimc_frame * dst = & ctx - > d_frame ;
u32 depth , min_w , max_w , min_h , align_h = 3 ;
u32 mask = FMT_FLAGS_CAM ;
struct fimc_fmt * ffmt ;
/* Color conversion from/to JPEG is not supported */
if ( code & & ctx - > s_frame . fmt & & pad = = FIMC_SD_PAD_SOURCE & &
fimc_fmt_is_jpeg ( ctx - > s_frame . fmt - > color ) )
* code = V4L2_MBUS_FMT_JPEG_1X8 ;
if ( fourcc & & * fourcc ! = V4L2_PIX_FMT_JPEG & & pad ! = FIMC_SD_PAD_SINK )
mask | = FMT_FLAGS_M2M ;
ffmt = fimc_find_format ( fourcc , code , mask , 0 ) ;
if ( WARN_ON ( ! ffmt ) )
return NULL ;
if ( code )
* code = ffmt - > mbus_code ;
if ( fourcc )
* fourcc = ffmt - > fourcc ;
if ( pad = = FIMC_SD_PAD_SINK ) {
max_w = fimc_fmt_is_jpeg ( ffmt - > color ) ?
pl - > scaler_dis_w : pl - > scaler_en_w ;
/* Apply the camera input interface pixel constraints */
v4l_bound_align_image ( width , max_t ( u32 , * width , 32 ) , max_w , 4 ,
height , max_t ( u32 , * height , 32 ) ,
FIMC_CAMIF_MAX_HEIGHT ,
fimc_fmt_is_jpeg ( ffmt - > color ) ? 3 : 1 ,
0 ) ;
return ffmt ;
}
/* Can't scale or crop in transparent (JPEG) transfer mode */
if ( fimc_fmt_is_jpeg ( ffmt - > color ) ) {
* width = ctx - > s_frame . f_width ;
* height = ctx - > s_frame . f_height ;
return ffmt ;
}
/* Apply the scaler and the output DMA constraints */
max_w = rotation ? pl - > out_rot_en_w : pl - > out_rot_dis_w ;
min_w = ctx - > state & FIMC_DST_CROP ? dst - > width : var - > min_out_pixsize ;
min_h = ctx - > state & FIMC_DST_CROP ? dst - > height : var - > min_out_pixsize ;
2011-05-27 20:12:23 +04:00
if ( var - > min_vsize_align = = 1 & & ! rotation )
2011-08-25 03:35:30 +04:00
align_h = fimc_fmt_is_rgb ( ffmt - > color ) ? 0 : 1 ;
depth = fimc_get_format_depth ( ffmt ) ;
v4l_bound_align_image ( width , min_w , max_w ,
ffs ( var - > min_out_pixsize ) - 1 ,
height , min_h , FIMC_CAMIF_MAX_HEIGHT ,
align_h ,
64 / ( ALIGN ( depth , 8 ) ) ) ;
dbg ( " pad%d: code: 0x%x, %dx%d. dst fmt: %dx%d " ,
pad , code ? * code : 0 , * width , * height ,
dst - > f_width , dst - > f_height ) ;
return ffmt ;
}
static void fimc_capture_try_crop ( struct fimc_ctx * ctx , struct v4l2_rect * r ,
int pad )
{
bool rotate = ctx - > rotation = = 90 | | ctx - > rotation = = 270 ;
struct fimc_dev * fimc = ctx - > fimc_dev ;
struct samsung_fimc_variant * var = fimc - > variant ;
struct fimc_pix_limit * pl = var - > pix_limit ;
struct fimc_frame * sink = & ctx - > s_frame ;
u32 max_w , max_h , min_w = 0 , min_h = 0 , min_sz ;
u32 align_sz = 0 , align_h = 4 ;
u32 max_sc_h , max_sc_v ;
/* In JPEG transparent transfer mode cropping is not supported */
if ( fimc_fmt_is_jpeg ( ctx - > d_frame . fmt - > color ) ) {
r - > width = sink - > f_width ;
r - > height = sink - > f_height ;
r - > left = r - > top = 0 ;
return ;
}
if ( pad = = FIMC_SD_PAD_SOURCE ) {
if ( ctx - > rotation ! = 90 & & ctx - > rotation ! = 270 )
align_h = 1 ;
max_sc_h = min ( SCALER_MAX_HRATIO , 1 < < ( ffs ( sink - > width ) - 3 ) ) ;
max_sc_v = min ( SCALER_MAX_VRATIO , 1 < < ( ffs ( sink - > height ) - 1 ) ) ;
min_sz = var - > min_out_pixsize ;
} else {
u32 depth = fimc_get_format_depth ( sink - > fmt ) ;
align_sz = 64 / ALIGN ( depth , 8 ) ;
min_sz = var - > min_inp_pixsize ;
min_w = min_h = min_sz ;
max_sc_h = max_sc_v = 1 ;
}
/*
* For the crop rectangle at source pad the following constraints
* must be met :
* - it must fit in the sink pad format rectangle ( f_width / f_height ) ;
* - maximum downscaling ratio is 64 ;
* - maximum crop size depends if the rotator is used or not ;
* - the sink pad format width / height must be 4 multiple of the
* prescaler ratios determined by sink pad size and source pad crop ,
* the prescaler ratio is returned by fimc_get_scaler_factor ( ) .
*/
max_w = min_t ( u32 ,
rotate ? pl - > out_rot_en_w : pl - > out_rot_dis_w ,
rotate ? sink - > f_height : sink - > f_width ) ;
max_h = min_t ( u32 , FIMC_CAMIF_MAX_HEIGHT , sink - > f_height ) ;
if ( pad = = FIMC_SD_PAD_SOURCE ) {
min_w = min_t ( u32 , max_w , sink - > f_width / max_sc_h ) ;
min_h = min_t ( u32 , max_h , sink - > f_height / max_sc_v ) ;
if ( rotate ) {
swap ( max_sc_h , max_sc_v ) ;
swap ( min_w , min_h ) ;
}
}
v4l_bound_align_image ( & r - > width , min_w , max_w , ffs ( min_sz ) - 1 ,
& r - > height , min_h , max_h , align_h ,
align_sz ) ;
/* Adjust left/top if cropping rectangle is out of bounds */
r - > left = clamp_t ( u32 , r - > left , 0 , sink - > f_width - r - > width ) ;
r - > top = clamp_t ( u32 , r - > top , 0 , sink - > f_height - r - > height ) ;
r - > left = round_down ( r - > left , var - > hor_offs_align ) ;
dbg ( " pad%d: (%d,%d)/%dx%d, sink fmt: %dx%d " ,
pad , r - > left , r - > top , r - > width , r - > height ,
sink - > f_width , sink - > f_height ) ;
}
/*
* The video node ioctl operations
*/
2010-10-07 17:06:16 +04:00
static int fimc_vidioc_querycap_capture ( struct file * file , void * priv ,
struct v4l2_capability * cap )
{
2011-06-10 22:36:50 +04:00
struct fimc_dev * fimc = video_drvdata ( file ) ;
2010-10-07 17:06:16 +04:00
strncpy ( cap - > driver , fimc - > pdev - > name , sizeof ( cap - > driver ) - 1 ) ;
strncpy ( cap - > card , fimc - > pdev - > name , sizeof ( cap - > card ) - 1 ) ;
cap - > bus_info [ 0 ] = 0 ;
2011-07-07 13:42:05 +04:00
cap - > capabilities = V4L2_CAP_STREAMING | V4L2_CAP_VIDEO_CAPTURE_MPLANE ;
2010-10-07 17:06:16 +04:00
return 0 ;
}
2011-06-13 18:09:40 +04:00
static int fimc_cap_enum_fmt_mplane ( struct file * file , void * priv ,
struct v4l2_fmtdesc * f )
{
struct fimc_fmt * fmt ;
fmt = fimc_find_format ( NULL , NULL , FMT_FLAGS_CAM | FMT_FLAGS_M2M ,
f - > index ) ;
if ( ! fmt )
return - EINVAL ;
strncpy ( f - > description , fmt - > name , sizeof ( f - > description ) - 1 ) ;
f - > pixelformat = fmt - > fourcc ;
if ( fmt - > fourcc = = V4L2_MBUS_FMT_JPEG_1X8 )
f - > flags | = V4L2_FMT_FLAG_COMPRESSED ;
return 0 ;
}
2011-08-25 03:35:30 +04:00
/**
* fimc_pipeline_try_format - negotiate and / or set formats at pipeline
* elements
* @ ctx : FIMC capture context
* @ tfmt : media bus format to try / set on subdevs
* @ fmt_id : fimc pixel format id corresponding to returned @ tfmt ( output )
* @ set : true to set format on subdevs , false to try only
*/
static int fimc_pipeline_try_format ( struct fimc_ctx * ctx ,
struct v4l2_mbus_framefmt * tfmt ,
struct fimc_fmt * * fmt_id ,
bool set )
{
struct fimc_dev * fimc = ctx - > fimc_dev ;
struct v4l2_subdev * sd = fimc - > pipeline . sensor ;
struct v4l2_subdev * csis = fimc - > pipeline . csis ;
struct v4l2_subdev_format sfmt ;
struct v4l2_mbus_framefmt * mf = & sfmt . format ;
struct fimc_fmt * ffmt = NULL ;
int ret , i = 0 ;
if ( WARN_ON ( ! sd | | ! tfmt ) )
return - EINVAL ;
2010-10-07 17:06:16 +04:00
2011-08-25 03:35:30 +04:00
memset ( & sfmt , 0 , sizeof ( sfmt ) ) ;
sfmt . format = * tfmt ;
sfmt . which = set ? V4L2_SUBDEV_FORMAT_ACTIVE : V4L2_SUBDEV_FORMAT_TRY ;
while ( 1 ) {
ffmt = fimc_find_format ( NULL , mf - > code ! = 0 ? & mf - > code : NULL ,
FMT_FLAGS_CAM , i + + ) ;
if ( ffmt = = NULL ) {
/*
* Notify user - space if common pixel code for
* host and sensor does not exist .
*/
return - EINVAL ;
}
mf - > code = tfmt - > code = ffmt - > mbus_code ;
2010-10-07 17:06:16 +04:00
2011-08-25 03:35:30 +04:00
ret = v4l2_subdev_call ( sd , pad , set_fmt , NULL , & sfmt ) ;
if ( ret )
return ret ;
if ( mf - > code ! = tfmt - > code ) {
mf - > code = 0 ;
continue ;
}
2012-01-13 00:49:28 +04:00
if ( mf - > width ! = tfmt - > width | | mf - > height ! = tfmt - > height ) {
2011-08-25 03:35:30 +04:00
u32 fcc = ffmt - > fourcc ;
tfmt - > width = mf - > width ;
tfmt - > height = mf - > height ;
ffmt = fimc_capture_try_format ( ctx ,
& tfmt - > width , & tfmt - > height ,
NULL , & fcc , FIMC_SD_PAD_SOURCE ) ;
if ( ffmt & & ffmt - > mbus_code )
mf - > code = ffmt - > mbus_code ;
2012-01-13 00:49:28 +04:00
if ( mf - > width ! = tfmt - > width | |
mf - > height ! = tfmt - > height )
2011-08-25 03:35:30 +04:00
continue ;
tfmt - > code = mf - > code ;
}
if ( csis )
ret = v4l2_subdev_call ( csis , pad , set_fmt , NULL , & sfmt ) ;
2010-10-07 17:06:16 +04:00
2011-08-25 03:35:30 +04:00
if ( mf - > code = = tfmt - > code & &
2012-01-13 00:49:28 +04:00
mf - > width = = tfmt - > width & & mf - > height = = tfmt - > height )
2011-08-25 03:35:30 +04:00
break ;
}
2010-10-07 17:06:16 +04:00
2011-08-25 03:35:30 +04:00
if ( fmt_id & & ffmt )
* fmt_id = ffmt ;
* tfmt = * mf ;
2010-10-07 17:06:16 +04:00
2011-08-25 03:35:30 +04:00
dbg ( " code: 0x%x, %dx%d, %p " , mf - > code , mf - > width , mf - > height , ffmt ) ;
return 0 ;
}
2010-10-07 17:06:16 +04:00
2011-06-10 22:36:50 +04:00
static int fimc_cap_g_fmt_mplane ( struct file * file , void * fh ,
struct v4l2_format * f )
{
struct fimc_dev * fimc = video_drvdata ( file ) ;
struct fimc_ctx * ctx = fimc - > vid_cap . ctx ;
if ( f - > type ! = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE )
return - EINVAL ;
return fimc_fill_format ( & ctx - > d_frame , f ) ;
}
static int fimc_cap_try_fmt_mplane ( struct file * file , void * fh ,
struct v4l2_format * f )
{
2011-08-25 03:35:30 +04:00
struct v4l2_pix_format_mplane * pix = & f - > fmt . pix_mp ;
2011-06-10 22:36:50 +04:00
struct fimc_dev * fimc = video_drvdata ( file ) ;
struct fimc_ctx * ctx = fimc - > vid_cap . ctx ;
2011-08-25 03:35:30 +04:00
struct v4l2_mbus_framefmt mf ;
struct fimc_fmt * ffmt = NULL ;
if ( f - > type ! = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE )
return - EINVAL ;
if ( pix - > pixelformat = = V4L2_PIX_FMT_JPEG ) {
fimc_capture_try_format ( ctx , & pix - > width , & pix - > height ,
NULL , & pix - > pixelformat ,
FIMC_SD_PAD_SINK ) ;
ctx - > s_frame . f_width = pix - > width ;
ctx - > s_frame . f_height = pix - > height ;
}
ffmt = fimc_capture_try_format ( ctx , & pix - > width , & pix - > height ,
NULL , & pix - > pixelformat ,
FIMC_SD_PAD_SOURCE ) ;
if ( ! ffmt )
return - EINVAL ;
if ( ! fimc - > vid_cap . user_subdev_api ) {
mf . width = pix - > width ;
mf . height = pix - > height ;
mf . code = ffmt - > mbus_code ;
fimc_md_graph_lock ( fimc ) ;
fimc_pipeline_try_format ( ctx , & mf , & ffmt , false ) ;
fimc_md_graph_unlock ( fimc ) ;
pix - > width = mf . width ;
pix - > height = mf . height ;
if ( ffmt )
pix - > pixelformat = ffmt - > fourcc ;
}
2011-06-10 22:36:50 +04:00
2011-08-25 03:35:30 +04:00
fimc_adjust_mplane_format ( ffmt , pix - > width , pix - > height , pix ) ;
2011-08-26 21:51:00 +04:00
return 0 ;
2011-06-10 22:36:50 +04:00
}
2011-08-26 21:57:06 +04:00
static void fimc_capture_mark_jpeg_xfer ( struct fimc_ctx * ctx , bool jpeg )
{
ctx - > scaler . enabled = ! jpeg ;
fimc_ctrls_activate ( ctx , ! jpeg ) ;
if ( jpeg )
set_bit ( ST_CAPT_JPEG , & ctx - > fimc_dev - > state ) ;
else
clear_bit ( ST_CAPT_JPEG , & ctx - > fimc_dev - > state ) ;
}
2011-08-25 03:35:30 +04:00
static int fimc_capture_set_format ( struct fimc_dev * fimc , struct v4l2_format * f )
2010-10-07 17:06:16 +04:00
{
2011-06-10 22:36:50 +04:00
struct fimc_ctx * ctx = fimc - > vid_cap . ctx ;
2011-08-25 03:35:30 +04:00
struct v4l2_pix_format_mplane * pix = & f - > fmt . pix_mp ;
struct v4l2_mbus_framefmt * mf = & fimc - > vid_cap . mf ;
struct fimc_frame * ff = & ctx - > d_frame ;
struct fimc_fmt * s_fmt = NULL ;
int ret , i ;
2010-10-07 17:06:16 +04:00
2010-12-08 20:05:08 +03:00
if ( f - > type ! = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE )
2010-10-07 17:06:16 +04:00
return - EINVAL ;
2011-08-25 03:35:30 +04:00
if ( vb2_is_busy ( & fimc - > vid_cap . vbq ) )
2010-12-08 20:05:08 +03:00
return - EBUSY ;
2010-10-07 17:06:16 +04:00
2011-08-25 03:35:30 +04:00
/* Pre-configure format at camera interface input, for JPEG only */
if ( pix - > pixelformat = = V4L2_PIX_FMT_JPEG ) {
fimc_capture_try_format ( ctx , & pix - > width , & pix - > height ,
NULL , & pix - > pixelformat ,
FIMC_SD_PAD_SINK ) ;
ctx - > s_frame . f_width = pix - > width ;
ctx - > s_frame . f_height = pix - > height ;
}
/* Try the format at the scaler and the DMA output */
ff - > fmt = fimc_capture_try_format ( ctx , & pix - > width , & pix - > height ,
NULL , & pix - > pixelformat ,
FIMC_SD_PAD_SOURCE ) ;
if ( ! ff - > fmt )
2010-12-01 16:25:18 +03:00
return - EINVAL ;
2011-12-01 21:02:24 +04:00
/* Update RGB Alpha control state and value range */
fimc_alpha_ctrl_update ( ctx ) ;
2011-08-25 03:35:30 +04:00
/* Try to match format at the host and the sensor */
if ( ! fimc - > vid_cap . user_subdev_api ) {
mf - > code = ff - > fmt - > mbus_code ;
mf - > width = pix - > width ;
mf - > height = pix - > height ;
fimc_md_graph_lock ( fimc ) ;
ret = fimc_pipeline_try_format ( ctx , mf , & s_fmt , true ) ;
fimc_md_graph_unlock ( fimc ) ;
if ( ret )
return ret ;
pix - > width = mf - > width ;
pix - > height = mf - > height ;
}
fimc_adjust_mplane_format ( ff - > fmt , pix - > width , pix - > height , pix ) ;
for ( i = 0 ; i < ff - > fmt - > colplanes ; i + + )
ff - > payload [ i ] =
( pix - > width * pix - > height * ff - > fmt - > depth [ i ] ) / 8 ;
set_frame_bounds ( ff , pix - > width , pix - > height ) ;
/* Reset the composition rectangle if not yet configured */
if ( ! ( ctx - > state & FIMC_DST_CROP ) )
set_frame_crop ( ff , 0 , 0 , pix - > width , pix - > height ) ;
2011-08-26 21:57:06 +04:00
fimc_capture_mark_jpeg_xfer ( ctx , fimc_fmt_is_jpeg ( ff - > fmt - > color ) ) ;
2011-08-25 03:35:30 +04:00
/* Reset cropping and set format at the camera interface input */
if ( ! fimc - > vid_cap . user_subdev_api ) {
ctx - > s_frame . fmt = s_fmt ;
set_frame_bounds ( & ctx - > s_frame , pix - > width , pix - > height ) ;
set_frame_crop ( & ctx - > s_frame , 0 , 0 , pix - > width , pix - > height ) ;
2011-04-08 16:11:43 +04:00
}
2010-12-08 20:05:08 +03:00
2011-08-25 03:35:30 +04:00
return ret ;
}
2010-10-07 17:06:16 +04:00
2011-08-25 03:35:30 +04:00
static int fimc_cap_s_fmt_mplane ( struct file * file , void * priv ,
struct v4l2_format * f )
{
struct fimc_dev * fimc = video_drvdata ( file ) ;
2010-10-07 17:06:16 +04:00
2011-08-25 03:35:30 +04:00
return fimc_capture_set_format ( fimc , f ) ;
2010-10-07 17:06:16 +04:00
}
static int fimc_cap_enum_input ( struct file * file , void * priv ,
2011-07-27 01:27:59 +04:00
struct v4l2_input * i )
2010-10-07 17:06:16 +04:00
{
2011-06-10 22:36:50 +04:00
struct fimc_dev * fimc = video_drvdata ( file ) ;
2011-08-26 21:51:00 +04:00
struct v4l2_subdev * sd = fimc - > pipeline . sensor ;
2010-10-07 17:06:16 +04:00
2011-07-27 01:27:59 +04:00
if ( i - > index ! = 0 )
2010-10-07 17:06:16 +04:00
return - EINVAL ;
i - > type = V4L2_INPUT_TYPE_CAMERA ;
2011-08-26 21:51:00 +04:00
if ( sd )
strlcpy ( i - > name , sd - > name , sizeof ( i - > name ) ) ;
2010-10-07 17:06:16 +04:00
return 0 ;
}
2011-07-27 01:27:59 +04:00
static int fimc_cap_s_input ( struct file * file , void * priv , unsigned int i )
2010-10-07 17:06:16 +04:00
{
2011-07-27 01:27:59 +04:00
return i = = 0 ? i : - EINVAL ;
2010-10-07 17:06:16 +04:00
}
2011-07-27 01:27:59 +04:00
static int fimc_cap_g_input ( struct file * file , void * priv , unsigned int * i )
2010-10-07 17:06:16 +04:00
{
2011-07-27 01:27:59 +04:00
* i = 0 ;
2010-10-07 17:06:16 +04:00
return 0 ;
}
2011-08-25 03:35:30 +04:00
/**
* fimc_pipeline_validate - check for formats inconsistencies
* between source and sink pad of each link
*
* Return 0 if all formats match or - EPIPE otherwise .
*/
static int fimc_pipeline_validate ( struct fimc_dev * fimc )
{
struct v4l2_subdev_format sink_fmt , src_fmt ;
struct fimc_vid_cap * vid_cap = & fimc - > vid_cap ;
struct v4l2_subdev * sd ;
struct media_pad * pad ;
int ret ;
/* Start with the video capture node pad */
pad = media_entity_remote_source ( & vid_cap - > vd_pad ) ;
if ( pad = = NULL )
return - EPIPE ;
/* FIMC.{N} subdevice */
sd = media_entity_to_v4l2_subdev ( pad - > entity ) ;
while ( 1 ) {
/* Retrieve format at the sink pad */
pad = & sd - > entity . pads [ 0 ] ;
if ( ! ( pad - > flags & MEDIA_PAD_FL_SINK ) )
break ;
/* Don't call FIMC subdev operation to avoid nested locking */
2012-04-21 01:57:25 +04:00
if ( sd = = & fimc - > vid_cap . subdev ) {
2011-08-25 03:35:30 +04:00
struct fimc_frame * ff = & vid_cap - > ctx - > s_frame ;
sink_fmt . format . width = ff - > f_width ;
sink_fmt . format . height = ff - > f_height ;
sink_fmt . format . code = ff - > fmt ? ff - > fmt - > mbus_code : 0 ;
} else {
sink_fmt . pad = pad - > index ;
sink_fmt . which = V4L2_SUBDEV_FORMAT_ACTIVE ;
ret = v4l2_subdev_call ( sd , pad , get_fmt , NULL , & sink_fmt ) ;
if ( ret < 0 & & ret ! = - ENOIOCTLCMD )
return - EPIPE ;
}
/* Retrieve format at the source pad */
pad = media_entity_remote_source ( pad ) ;
if ( pad = = NULL | |
media_entity_type ( pad - > entity ) ! = MEDIA_ENT_T_V4L2_SUBDEV )
break ;
sd = media_entity_to_v4l2_subdev ( pad - > entity ) ;
src_fmt . pad = pad - > index ;
src_fmt . which = V4L2_SUBDEV_FORMAT_ACTIVE ;
ret = v4l2_subdev_call ( sd , pad , get_fmt , NULL , & src_fmt ) ;
if ( ret < 0 & & ret ! = - ENOIOCTLCMD )
return - EPIPE ;
if ( src_fmt . format . width ! = sink_fmt . format . width | |
src_fmt . format . height ! = sink_fmt . format . height | |
src_fmt . format . code ! = sink_fmt . format . code )
return - EPIPE ;
}
return 0 ;
}
2010-10-07 17:06:16 +04:00
static int fimc_cap_streamon ( struct file * file , void * priv ,
2010-12-01 16:14:59 +03:00
enum v4l2_buf_type type )
2010-10-07 17:06:16 +04:00
{
2011-06-10 22:36:50 +04:00
struct fimc_dev * fimc = video_drvdata ( file ) ;
2011-08-26 21:51:00 +04:00
struct fimc_pipeline * p = & fimc - > pipeline ;
2011-08-25 03:35:30 +04:00
int ret ;
2010-10-07 17:06:16 +04:00
2011-08-26 21:51:00 +04:00
if ( fimc_capture_active ( fimc ) )
2010-12-01 16:25:18 +03:00
return - EBUSY ;
2010-10-07 17:06:16 +04:00
2011-08-26 21:51:00 +04:00
media_entity_pipeline_start ( & p - > sensor - > entity , p - > pipe ) ;
2010-10-07 17:06:16 +04:00
2011-08-25 03:35:30 +04:00
if ( fimc - > vid_cap . user_subdev_api ) {
ret = fimc_pipeline_validate ( fimc ) ;
if ( ret )
return ret ;
}
2010-12-01 16:25:18 +03:00
return vb2_streamon ( & fimc - > vid_cap . vbq , type ) ;
2010-10-07 17:06:16 +04:00
}
static int fimc_cap_streamoff ( struct file * file , void * priv ,
2010-12-01 16:25:18 +03:00
enum v4l2_buf_type type )
2010-10-07 17:06:16 +04:00
{
2011-06-10 22:36:50 +04:00
struct fimc_dev * fimc = video_drvdata ( file ) ;
2011-08-26 21:51:00 +04:00
struct v4l2_subdev * sd = fimc - > pipeline . sensor ;
int ret ;
2010-10-07 17:06:16 +04:00
2011-08-26 21:51:00 +04:00
ret = vb2_streamoff ( & fimc - > vid_cap . vbq , type ) ;
if ( ret = = 0 )
media_entity_pipeline_stop ( & sd - > entity ) ;
return ret ;
2010-10-07 17:06:16 +04:00
}
static int fimc_cap_reqbufs ( struct file * file , void * priv ,
2010-12-08 20:05:08 +03:00
struct v4l2_requestbuffers * reqbufs )
2010-10-07 17:06:16 +04:00
{
2011-06-10 22:36:50 +04:00
struct fimc_dev * fimc = video_drvdata ( file ) ;
int ret = vb2_reqbufs ( & fimc - > vid_cap . vbq , reqbufs ) ;
2010-10-07 17:06:16 +04:00
if ( ! ret )
2011-06-10 22:36:50 +04:00
fimc - > vid_cap . reqbufs_count = reqbufs - > count ;
2010-10-07 17:06:16 +04:00
return ret ;
}
static int fimc_cap_querybuf ( struct file * file , void * priv ,
struct v4l2_buffer * buf )
{
2011-06-10 22:36:50 +04:00
struct fimc_dev * fimc = video_drvdata ( file ) ;
2010-10-07 17:06:16 +04:00
2011-06-10 22:36:50 +04:00
return vb2_querybuf ( & fimc - > vid_cap . vbq , buf ) ;
2010-10-07 17:06:16 +04:00
}
static int fimc_cap_qbuf ( struct file * file , void * priv ,
struct v4l2_buffer * buf )
{
2011-06-10 22:36:50 +04:00
struct fimc_dev * fimc = video_drvdata ( file ) ;
return vb2_qbuf ( & fimc - > vid_cap . vbq , buf ) ;
2010-10-07 17:06:16 +04:00
}
static int fimc_cap_dqbuf ( struct file * file , void * priv ,
struct v4l2_buffer * buf )
{
2011-06-10 22:36:50 +04:00
struct fimc_dev * fimc = video_drvdata ( file ) ;
return vb2_dqbuf ( & fimc - > vid_cap . vbq , buf , file - > f_flags & O_NONBLOCK ) ;
2010-10-07 17:06:16 +04:00
}
2012-01-30 21:56:59 +04:00
static int fimc_cap_create_bufs ( struct file * file , void * priv ,
struct v4l2_create_buffers * create )
{
struct fimc_dev * fimc = video_drvdata ( file ) ;
return vb2_create_bufs ( & fimc - > vid_cap . vbq , create ) ;
}
static int fimc_cap_prepare_buf ( struct file * file , void * priv ,
struct v4l2_buffer * b )
{
struct fimc_dev * fimc = video_drvdata ( file ) ;
return vb2_prepare_buf ( & fimc - > vid_cap . vbq , b ) ;
}
2012-02-05 00:03:54 +04:00
static int fimc_cap_g_selection ( struct file * file , void * fh ,
struct v4l2_selection * s )
2010-11-25 17:01:51 +03:00
{
2011-06-10 22:36:50 +04:00
struct fimc_dev * fimc = video_drvdata ( file ) ;
2012-02-05 00:03:54 +04:00
struct fimc_ctx * ctx = fimc - > vid_cap . ctx ;
struct fimc_frame * f = & ctx - > s_frame ;
2010-11-25 17:01:51 +03:00
2012-02-05 00:03:54 +04:00
if ( s - > type ! = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE )
2010-11-25 17:01:51 +03:00
return - EINVAL ;
2012-02-05 00:03:54 +04:00
switch ( s - > target ) {
case V4L2_SEL_TGT_COMPOSE_DEFAULT :
case V4L2_SEL_TGT_COMPOSE_BOUNDS :
f = & ctx - > d_frame ;
case V4L2_SEL_TGT_CROP_BOUNDS :
case V4L2_SEL_TGT_CROP_DEFAULT :
s - > r . left = 0 ;
s - > r . top = 0 ;
s - > r . width = f - > o_width ;
s - > r . height = f - > o_height ;
return 0 ;
2010-11-25 17:01:51 +03:00
2012-02-05 00:03:54 +04:00
case V4L2_SEL_TGT_COMPOSE_ACTIVE :
f = & ctx - > d_frame ;
case V4L2_SEL_TGT_CROP_ACTIVE :
s - > r . left = f - > offs_h ;
s - > r . top = f - > offs_v ;
s - > r . width = f - > width ;
s - > r . height = f - > height ;
return 0 ;
}
return - EINVAL ;
2010-11-25 17:01:51 +03:00
}
2012-02-05 00:03:54 +04:00
/* Return 1 if rectangle a is enclosed in rectangle b, or 0 otherwise. */
int enclosed_rectangle ( struct v4l2_rect * a , struct v4l2_rect * b )
2010-11-25 17:01:51 +03:00
{
2012-02-05 00:03:54 +04:00
if ( a - > left < b - > left | | a - > top < b - > top )
return 0 ;
if ( a - > left + a - > width > b - > left + b - > width )
return 0 ;
if ( a - > top + a - > height > b - > top + b - > height )
return 0 ;
2010-11-25 17:01:51 +03:00
2012-02-05 00:03:54 +04:00
return 1 ;
2010-11-25 17:01:51 +03:00
}
2012-02-05 00:03:54 +04:00
static int fimc_cap_s_selection ( struct file * file , void * fh ,
struct v4l2_selection * s )
2010-10-07 17:06:16 +04:00
{
2011-06-10 22:36:50 +04:00
struct fimc_dev * fimc = video_drvdata ( file ) ;
struct fimc_ctx * ctx = fimc - > vid_cap . ctx ;
2012-02-05 00:03:54 +04:00
struct v4l2_rect rect = s - > r ;
struct fimc_frame * f ;
2011-08-25 03:35:30 +04:00
unsigned long flags ;
2012-02-05 00:03:54 +04:00
unsigned int pad ;
if ( s - > type ! = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE )
return - EINVAL ;
switch ( s - > target ) {
case V4L2_SEL_TGT_COMPOSE_DEFAULT :
case V4L2_SEL_TGT_COMPOSE_BOUNDS :
case V4L2_SEL_TGT_COMPOSE_ACTIVE :
f = & ctx - > d_frame ;
pad = FIMC_SD_PAD_SOURCE ;
break ;
case V4L2_SEL_TGT_CROP_BOUNDS :
case V4L2_SEL_TGT_CROP_DEFAULT :
case V4L2_SEL_TGT_CROP_ACTIVE :
f = & ctx - > s_frame ;
pad = FIMC_SD_PAD_SINK ;
break ;
default :
return - EINVAL ;
}
fimc_capture_try_crop ( ctx , & rect , pad ) ;
if ( s - > flags & V4L2_SEL_FLAG_LE & &
! enclosed_rectangle ( & rect , & s - > r ) )
return - ERANGE ;
2010-10-07 17:06:16 +04:00
2012-02-05 00:03:54 +04:00
if ( s - > flags & V4L2_SEL_FLAG_GE & &
! enclosed_rectangle ( & s - > r , & rect ) )
return - ERANGE ;
2010-10-07 17:06:16 +04:00
2012-02-05 00:03:54 +04:00
s - > r = rect ;
2011-08-25 03:35:30 +04:00
spin_lock_irqsave ( & fimc - > slock , flags ) ;
2012-02-05 00:03:54 +04:00
set_frame_crop ( f , s - > r . left , s - > r . top , s - > r . width ,
s - > r . height ) ;
2011-08-25 03:35:30 +04:00
spin_unlock_irqrestore ( & fimc - > slock , flags ) ;
2010-12-01 16:25:18 +03:00
2012-02-05 00:03:54 +04:00
set_bit ( ST_CAPT_APPLY_CFG , & fimc - > state ) ;
2010-12-01 16:25:18 +03:00
return 0 ;
2010-10-07 17:06:16 +04:00
}
static const struct v4l2_ioctl_ops fimc_capture_ioctl_ops = {
. vidioc_querycap = fimc_vidioc_querycap_capture ,
2011-06-13 18:09:40 +04:00
. vidioc_enum_fmt_vid_cap_mplane = fimc_cap_enum_fmt_mplane ,
2011-06-10 22:36:50 +04:00
. vidioc_try_fmt_vid_cap_mplane = fimc_cap_try_fmt_mplane ,
2010-12-08 20:05:08 +03:00
. vidioc_s_fmt_vid_cap_mplane = fimc_cap_s_fmt_mplane ,
2011-06-10 22:36:50 +04:00
. vidioc_g_fmt_vid_cap_mplane = fimc_cap_g_fmt_mplane ,
2010-10-07 17:06:16 +04:00
. vidioc_reqbufs = fimc_cap_reqbufs ,
. vidioc_querybuf = fimc_cap_querybuf ,
. vidioc_qbuf = fimc_cap_qbuf ,
. vidioc_dqbuf = fimc_cap_dqbuf ,
2012-01-30 21:56:59 +04:00
. vidioc_prepare_buf = fimc_cap_prepare_buf ,
. vidioc_create_bufs = fimc_cap_create_bufs ,
2010-10-07 17:06:16 +04:00
. vidioc_streamon = fimc_cap_streamon ,
. vidioc_streamoff = fimc_cap_streamoff ,
2012-02-05 00:03:54 +04:00
. vidioc_g_selection = fimc_cap_g_selection ,
. vidioc_s_selection = fimc_cap_s_selection ,
2010-10-07 17:06:16 +04:00
. vidioc_enum_input = fimc_cap_enum_input ,
. vidioc_s_input = fimc_cap_s_input ,
. vidioc_g_input = fimc_cap_g_input ,
} ;
2011-08-25 03:35:30 +04:00
/* Capture subdev media entity operations */
2011-08-25 02:28:18 +04:00
static int fimc_link_setup ( struct media_entity * entity ,
const struct media_pad * local ,
const struct media_pad * remote , u32 flags )
{
2011-08-25 03:35:30 +04:00
struct v4l2_subdev * sd = media_entity_to_v4l2_subdev ( entity ) ;
struct fimc_dev * fimc = v4l2_get_subdevdata ( sd ) ;
if ( media_entity_type ( remote - > entity ) ! = MEDIA_ENT_T_V4L2_SUBDEV )
return - EINVAL ;
2011-08-25 02:28:18 +04:00
if ( WARN_ON ( fimc = = NULL ) )
return 0 ;
dbg ( " %s --> %s, flags: 0x%x. input: 0x%x " ,
local - > entity - > name , remote - > entity - > name , flags ,
fimc - > vid_cap . input ) ;
if ( flags & MEDIA_LNK_FL_ENABLED ) {
if ( fimc - > vid_cap . input ! = 0 )
return - EBUSY ;
fimc - > vid_cap . input = sd - > grp_id ;
return 0 ;
}
fimc - > vid_cap . input = 0 ;
return 0 ;
}
2011-08-25 03:35:30 +04:00
static const struct media_entity_operations fimc_sd_media_ops = {
2011-08-25 02:28:18 +04:00
. link_setup = fimc_link_setup ,
} ;
2011-06-10 22:36:58 +04:00
/**
* fimc_sensor_notify - v4l2_device notification from a sensor subdev
* @ sd : pointer to a subdev generating the notification
* @ notification : the notification type , must be S5P_FIMC_TX_END_NOTIFY
* @ arg : pointer to an u32 type integer that stores the frame payload value
*
* The End Of Frame notification sent by sensor subdev in its still capture
* mode . If there is only a single VSYNC generated by the sensor at the
* beginning of a frame transmission , FIMC does not issue the LastIrq
* ( end of frame ) interrupt . And this notification is used to complete the
* frame capture and returning a buffer to user - space . Subdev drivers should
* call this notification from their last ' End of frame capture ' interrupt .
*/
void fimc_sensor_notify ( struct v4l2_subdev * sd , unsigned int notification ,
void * arg )
{
struct fimc_sensor_info * sensor ;
struct fimc_vid_buffer * buf ;
struct fimc_md * fmd ;
struct fimc_dev * fimc ;
unsigned long flags ;
if ( sd = = NULL )
return ;
sensor = v4l2_get_subdev_hostdata ( sd ) ;
fmd = entity_to_fimc_mdev ( & sd - > entity ) ;
spin_lock_irqsave ( & fmd - > slock , flags ) ;
fimc = sensor ? sensor - > host : NULL ;
if ( fimc & & arg & & notification = = S5P_FIMC_TX_END_NOTIFY & &
test_bit ( ST_CAPT_PEND , & fimc - > state ) ) {
unsigned long irq_flags ;
spin_lock_irqsave ( & fimc - > slock , irq_flags ) ;
if ( ! list_empty ( & fimc - > vid_cap . active_buf_q ) ) {
buf = list_entry ( fimc - > vid_cap . active_buf_q . next ,
struct fimc_vid_buffer , list ) ;
vb2_set_plane_payload ( & buf - > vb , 0 , * ( ( u32 * ) arg ) ) ;
}
2012-05-08 22:51:24 +04:00
fimc_capture_irq_handler ( fimc , 1 ) ;
2011-06-10 22:36:58 +04:00
fimc_deactivate_capture ( fimc ) ;
spin_unlock_irqrestore ( & fimc - > slock , irq_flags ) ;
}
spin_unlock_irqrestore ( & fmd - > slock , flags ) ;
}
2011-08-25 03:35:30 +04:00
static int fimc_subdev_enum_mbus_code ( struct v4l2_subdev * sd ,
struct v4l2_subdev_fh * fh ,
struct v4l2_subdev_mbus_code_enum * code )
{
struct fimc_fmt * fmt ;
fmt = fimc_find_format ( NULL , NULL , FMT_FLAGS_CAM , code - > index ) ;
if ( ! fmt )
return - EINVAL ;
code - > code = fmt - > mbus_code ;
return 0 ;
}
static int fimc_subdev_get_fmt ( struct v4l2_subdev * sd ,
struct v4l2_subdev_fh * fh ,
struct v4l2_subdev_format * fmt )
{
struct fimc_dev * fimc = v4l2_get_subdevdata ( sd ) ;
struct fimc_ctx * ctx = fimc - > vid_cap . ctx ;
struct v4l2_mbus_framefmt * mf ;
struct fimc_frame * ff ;
if ( fmt - > which = = V4L2_SUBDEV_FORMAT_TRY ) {
mf = v4l2_subdev_get_try_format ( fh , fmt - > pad ) ;
fmt - > format = * mf ;
return 0 ;
}
mf = & fmt - > format ;
mf - > colorspace = V4L2_COLORSPACE_JPEG ;
ff = fmt - > pad = = FIMC_SD_PAD_SINK ? & ctx - > s_frame : & ctx - > d_frame ;
mutex_lock ( & fimc - > lock ) ;
/* The pixel code is same on both input and output pad */
if ( ! WARN_ON ( ctx - > s_frame . fmt = = NULL ) )
mf - > code = ctx - > s_frame . fmt - > mbus_code ;
mf - > width = ff - > f_width ;
mf - > height = ff - > f_height ;
mutex_unlock ( & fimc - > lock ) ;
return 0 ;
}
static int fimc_subdev_set_fmt ( struct v4l2_subdev * sd ,
struct v4l2_subdev_fh * fh ,
struct v4l2_subdev_format * fmt )
{
struct fimc_dev * fimc = v4l2_get_subdevdata ( sd ) ;
struct v4l2_mbus_framefmt * mf = & fmt - > format ;
struct fimc_ctx * ctx = fimc - > vid_cap . ctx ;
struct fimc_frame * ff ;
struct fimc_fmt * ffmt ;
dbg ( " pad%d: code: 0x%x, %dx%d " ,
fmt - > pad , mf - > code , mf - > width , mf - > height ) ;
if ( fmt - > pad = = FIMC_SD_PAD_SOURCE & &
vb2_is_busy ( & fimc - > vid_cap . vbq ) )
return - EBUSY ;
mutex_lock ( & fimc - > lock ) ;
ffmt = fimc_capture_try_format ( ctx , & mf - > width , & mf - > height ,
& mf - > code , NULL , fmt - > pad ) ;
mutex_unlock ( & fimc - > lock ) ;
mf - > colorspace = V4L2_COLORSPACE_JPEG ;
if ( fmt - > which = = V4L2_SUBDEV_FORMAT_TRY ) {
mf = v4l2_subdev_get_try_format ( fh , fmt - > pad ) ;
* mf = fmt - > format ;
return 0 ;
}
2011-12-01 21:02:24 +04:00
/* Update RGB Alpha control state and value range */
fimc_alpha_ctrl_update ( ctx ) ;
2011-08-26 21:57:06 +04:00
fimc_capture_mark_jpeg_xfer ( ctx , fimc_fmt_is_jpeg ( ffmt - > color ) ) ;
2011-08-25 03:35:30 +04:00
ff = fmt - > pad = = FIMC_SD_PAD_SINK ?
& ctx - > s_frame : & ctx - > d_frame ;
mutex_lock ( & fimc - > lock ) ;
set_frame_bounds ( ff , mf - > width , mf - > height ) ;
2011-12-08 13:56:22 +04:00
fimc - > vid_cap . mf = * mf ;
2011-08-25 03:35:30 +04:00
ff - > fmt = ffmt ;
/* Reset the crop rectangle if required. */
if ( ! ( fmt - > pad = = FIMC_SD_PAD_SOURCE & & ( ctx - > state & FIMC_DST_CROP ) ) )
set_frame_crop ( ff , 0 , 0 , mf - > width , mf - > height ) ;
if ( fmt - > pad = = FIMC_SD_PAD_SINK )
ctx - > state & = ~ FIMC_DST_CROP ;
mutex_unlock ( & fimc - > lock ) ;
return 0 ;
}
static int fimc_subdev_get_crop ( struct v4l2_subdev * sd ,
struct v4l2_subdev_fh * fh ,
struct v4l2_subdev_crop * crop )
{
struct fimc_dev * fimc = v4l2_get_subdevdata ( sd ) ;
struct fimc_ctx * ctx = fimc - > vid_cap . ctx ;
struct v4l2_rect * r = & crop - > rect ;
struct fimc_frame * ff ;
if ( crop - > which = = V4L2_SUBDEV_FORMAT_TRY ) {
crop - > rect = * v4l2_subdev_get_try_crop ( fh , crop - > pad ) ;
return 0 ;
}
ff = crop - > pad = = FIMC_SD_PAD_SINK ?
& ctx - > s_frame : & ctx - > d_frame ;
mutex_lock ( & fimc - > lock ) ;
r - > left = ff - > offs_h ;
r - > top = ff - > offs_v ;
r - > width = ff - > width ;
r - > height = ff - > height ;
mutex_unlock ( & fimc - > lock ) ;
dbg ( " ff:%p, pad%d: l:%d, t:%d, %dx%d, f_w: %d, f_h: %d " ,
ff , crop - > pad , r - > left , r - > top , r - > width , r - > height ,
ff - > f_width , ff - > f_height ) ;
return 0 ;
}
static int fimc_subdev_set_crop ( struct v4l2_subdev * sd ,
struct v4l2_subdev_fh * fh ,
struct v4l2_subdev_crop * crop )
{
struct fimc_dev * fimc = v4l2_get_subdevdata ( sd ) ;
struct fimc_ctx * ctx = fimc - > vid_cap . ctx ;
struct v4l2_rect * r = & crop - > rect ;
struct fimc_frame * ff ;
unsigned long flags ;
dbg ( " (%d,%d)/%dx%d " , r - > left , r - > top , r - > width , r - > height ) ;
ff = crop - > pad = = FIMC_SD_PAD_SOURCE ?
& ctx - > d_frame : & ctx - > s_frame ;
mutex_lock ( & fimc - > lock ) ;
fimc_capture_try_crop ( ctx , r , crop - > pad ) ;
if ( crop - > which = = V4L2_SUBDEV_FORMAT_TRY ) {
2012-04-22 01:46:30 +04:00
mutex_unlock ( & fimc - > lock ) ;
2011-08-25 03:35:30 +04:00
* v4l2_subdev_get_try_crop ( fh , crop - > pad ) = * r ;
return 0 ;
}
spin_lock_irqsave ( & fimc - > slock , flags ) ;
set_frame_crop ( ff , r - > left , r - > top , r - > width , r - > height ) ;
if ( crop - > pad = = FIMC_SD_PAD_SOURCE )
ctx - > state | = FIMC_DST_CROP ;
set_bit ( ST_CAPT_APPLY_CFG , & fimc - > state ) ;
spin_unlock_irqrestore ( & fimc - > slock , flags ) ;
dbg ( " pad%d: (%d,%d)/%dx%d " , crop - > pad , r - > left , r - > top ,
r - > width , r - > height ) ;
mutex_unlock ( & fimc - > lock ) ;
return 0 ;
}
static struct v4l2_subdev_pad_ops fimc_subdev_pad_ops = {
. enum_mbus_code = fimc_subdev_enum_mbus_code ,
. get_fmt = fimc_subdev_get_fmt ,
. set_fmt = fimc_subdev_set_fmt ,
. get_crop = fimc_subdev_get_crop ,
. set_crop = fimc_subdev_set_crop ,
} ;
static struct v4l2_subdev_ops fimc_subdev_ops = {
. pad = & fimc_subdev_pad_ops ,
} ;
/* Set default format at the sensor and host interface */
static int fimc_capture_set_default_format ( struct fimc_dev * fimc )
{
struct v4l2_format fmt = {
. type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE ,
. fmt . pix_mp = {
. width = 640 ,
. height = 480 ,
. pixelformat = V4L2_PIX_FMT_YUYV ,
. field = V4L2_FIELD_NONE ,
. colorspace = V4L2_COLORSPACE_JPEG ,
} ,
} ;
return fimc_capture_set_format ( fimc , & fmt ) ;
}
2010-12-08 20:05:08 +03:00
/* fimc->lock must be already initialized */
2012-04-21 01:57:25 +04:00
static int fimc_register_capture_device ( struct fimc_dev * fimc ,
2011-06-10 22:36:48 +04:00
struct v4l2_device * v4l2_dev )
2010-10-07 17:06:16 +04:00
{
struct video_device * vfd ;
struct fimc_vid_cap * vid_cap ;
struct fimc_ctx * ctx ;
2010-12-01 16:14:59 +03:00
struct vb2_queue * q ;
2011-06-10 22:36:48 +04:00
int ret = - ENOMEM ;
2010-10-07 17:06:16 +04:00
ctx = kzalloc ( sizeof * ctx , GFP_KERNEL ) ;
if ( ! ctx )
return - ENOMEM ;
ctx - > fimc_dev = fimc ;
ctx - > in_path = FIMC_CAMERA ;
ctx - > out_path = FIMC_DMA ;
ctx - > state = FIMC_CTX_CAP ;
2011-08-25 03:35:30 +04:00
ctx - > s_frame . fmt = fimc_find_format ( NULL , NULL , FMT_FLAGS_CAM , 0 ) ;
2012-04-21 01:57:25 +04:00
ctx - > d_frame . fmt = ctx - > s_frame . fmt ;
2010-10-07 17:06:16 +04:00
vfd = video_device_alloc ( ) ;
if ( ! vfd ) {
v4l2_err ( v4l2_dev , " Failed to allocate video device \n " ) ;
2011-06-10 22:36:48 +04:00
goto err_vd_alloc ;
2010-10-07 17:06:16 +04:00
}
2012-04-21 01:57:25 +04:00
snprintf ( vfd - > name , sizeof ( vfd - > name ) , " fimc.%d.capture " , fimc - > id ) ;
2010-10-07 17:06:16 +04:00
vfd - > fops = & fimc_capture_fops ;
vfd - > ioctl_ops = & fimc_capture_ioctl_ops ;
2011-07-27 01:08:21 +04:00
vfd - > v4l2_dev = v4l2_dev ;
2010-10-07 17:06:16 +04:00
vfd - > minor = - 1 ;
vfd - > release = video_device_release ;
2010-12-01 16:25:18 +03:00
vfd - > lock = & fimc - > lock ;
2012-05-10 11:57:22 +04:00
/* Locking in file operations other than ioctl should be done
by the driver , not the V4L2 core .
This driver needs auditing so that this flag can be removed . */
set_bit ( V4L2_FL_LOCK_ALL_FOPS , & vfd - > flags ) ;
2010-10-07 17:06:16 +04:00
video_set_drvdata ( vfd , fimc ) ;
vid_cap = & fimc - > vid_cap ;
vid_cap - > vfd = vfd ;
vid_cap - > active_buf_cnt = 0 ;
vid_cap - > reqbufs_count = 0 ;
vid_cap - > refcnt = 0 ;
INIT_LIST_HEAD ( & vid_cap - > pending_buf_q ) ;
INIT_LIST_HEAD ( & vid_cap - > active_buf_q ) ;
vid_cap - > ctx = ctx ;
2010-12-01 16:14:59 +03:00
q = & fimc - > vid_cap . vbq ;
memset ( q , 0 , sizeof ( * q ) ) ;
2010-12-08 20:05:08 +03:00
q - > type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE ;
2010-12-01 16:14:59 +03:00
q - > io_modes = VB2_MMAP | VB2_USERPTR ;
q - > drv_priv = fimc - > vid_cap . ctx ;
q - > ops = & fimc_capture_qops ;
q - > mem_ops = & vb2_dma_contig_memops ;
q - > buf_struct_size = sizeof ( struct fimc_vid_buffer ) ;
vb2_queue_init ( q ) ;
2010-10-07 17:06:16 +04:00
2012-04-21 01:57:25 +04:00
vid_cap - > vd_pad . flags = MEDIA_PAD_FL_SINK ;
ret = media_entity_init ( & vfd - > entity , 1 , & vid_cap - > vd_pad , 0 ) ;
2011-07-27 01:08:21 +04:00
if ( ret )
goto err_ent ;
2012-04-21 01:57:25 +04:00
ret = video_register_device ( vfd , VFL_TYPE_GRABBER , - 1 ) ;
2011-08-25 03:35:30 +04:00
if ( ret )
2012-04-21 01:57:25 +04:00
goto err_vd ;
v4l2_info ( v4l2_dev , " Registered %s as /dev/%s \n " ,
vfd - > name , video_device_node_name ( vfd ) ) ;
2011-07-27 01:08:21 +04:00
2011-08-25 02:25:10 +04:00
vfd - > ctrl_handler = & ctx - > ctrl_handler ;
2010-10-07 17:06:16 +04:00
return 0 ;
2012-04-21 01:57:25 +04:00
err_vd :
2011-08-25 03:35:30 +04:00
media_entity_cleanup ( & vfd - > entity ) ;
2011-07-27 01:08:21 +04:00
err_ent :
2010-10-07 17:06:16 +04:00
video_device_release ( vfd ) ;
2011-06-10 22:36:48 +04:00
err_vd_alloc :
2011-06-02 13:18:34 +04:00
kfree ( ctx ) ;
2010-10-07 17:06:16 +04:00
return ret ;
}
2012-04-21 01:57:25 +04:00
static int fimc_capture_subdev_registered ( struct v4l2_subdev * sd )
2010-10-07 17:06:16 +04:00
{
2012-04-21 01:57:25 +04:00
struct fimc_dev * fimc = v4l2_get_subdevdata ( sd ) ;
int ret ;
2010-10-07 17:06:16 +04:00
2012-04-21 01:57:25 +04:00
ret = fimc_register_m2m_device ( fimc , sd - > v4l2_dev ) ;
if ( ret )
return ret ;
ret = fimc_register_capture_device ( fimc , sd - > v4l2_dev ) ;
if ( ret )
fimc_unregister_m2m_device ( fimc ) ;
return ret ;
}
static void fimc_capture_subdev_unregistered ( struct v4l2_subdev * sd )
{
struct fimc_dev * fimc = v4l2_get_subdevdata ( sd ) ;
if ( fimc = = NULL )
return ;
fimc_unregister_m2m_device ( fimc ) ;
if ( fimc - > vid_cap . vfd ) {
media_entity_cleanup ( & fimc - > vid_cap . vfd - > entity ) ;
video_unregister_device ( fimc - > vid_cap . vfd ) ;
fimc - > vid_cap . vfd = NULL ;
2011-07-27 01:08:21 +04:00
}
2012-04-21 01:57:25 +04:00
2011-07-27 01:08:21 +04:00
kfree ( fimc - > vid_cap . ctx ) ;
2011-08-26 22:40:36 +04:00
fimc - > vid_cap . ctx = NULL ;
2010-10-07 17:06:16 +04:00
}
2012-04-21 01:57:25 +04:00
static const struct v4l2_subdev_internal_ops fimc_capture_sd_internal_ops = {
. registered = fimc_capture_subdev_registered ,
. unregistered = fimc_capture_subdev_unregistered ,
} ;
int fimc_initialize_capture_subdev ( struct fimc_dev * fimc )
{
struct v4l2_subdev * sd = & fimc - > vid_cap . subdev ;
int ret ;
v4l2_subdev_init ( sd , & fimc_subdev_ops ) ;
sd - > flags = V4L2_SUBDEV_FL_HAS_DEVNODE ;
snprintf ( sd - > name , sizeof ( sd - > name ) , " FIMC.%d " , fimc - > pdev - > id ) ;
fimc - > vid_cap . sd_pads [ FIMC_SD_PAD_SINK ] . flags = MEDIA_PAD_FL_SINK ;
fimc - > vid_cap . sd_pads [ FIMC_SD_PAD_SOURCE ] . flags = MEDIA_PAD_FL_SOURCE ;
ret = media_entity_init ( & sd - > entity , FIMC_SD_PADS_NUM ,
fimc - > vid_cap . sd_pads , 0 ) ;
if ( ret )
return ret ;
sd - > entity . ops = & fimc_sd_media_ops ;
sd - > internal_ops = & fimc_capture_sd_internal_ops ;
v4l2_set_subdevdata ( sd , fimc ) ;
return 0 ;
}
void fimc_unregister_capture_subdev ( struct fimc_dev * fimc )
{
struct v4l2_subdev * sd = & fimc - > vid_cap . subdev ;
v4l2_device_unregister_subdev ( sd ) ;
media_entity_cleanup ( & sd - > entity ) ;
v4l2_set_subdevdata ( sd , NULL ) ;
}