2010-10-07 10:06:16 -03:00
/*
2011-04-28 09:06:19 -03:00
* Samsung S5P / EXYNOS4 SoC series camera interface ( camera capture ) driver
2010-10-07 10:06:16 -03:00
*
2012-04-25 06:55:42 -03:00
* Copyright ( C ) 2010 - 2012 Samsung Electronics Co . , Ltd .
* Sylwester Nawrocki < s . nawrocki @ samsung . com >
2010-10-07 10:06:16 -03: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 .
*/
# 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 06:25:32 -03:00
# include <linux/pm_runtime.h>
2010-10-07 10:06:16 -03: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>
2015-09-22 10:30:29 -03:00
# include <media/videobuf2-v4l2.h>
2010-12-01 10:14:59 -03:00
# include <media/videobuf2-dma-contig.h>
2010-10-07 10:06:16 -03:00
2013-05-31 11:37:19 -03:00
# include "common.h"
2010-10-07 10:06:16 -03:00
# include "fimc-core.h"
2012-05-02 06:14:49 -03:00
# include "fimc-reg.h"
2013-05-31 11:37:19 -03:00
# include "media-dev.h"
2010-10-07 10:06:16 -03:00
2012-04-27 09:33:23 -03:00
static int fimc_capture_hw_init ( struct fimc_dev * fimc )
2011-06-10 15:36:53 -03:00
{
2013-03-20 10:44:39 -03:00
struct fimc_source_info * si = & fimc - > vid_cap . source_config ;
2011-06-10 15:36:53 -03:00
struct fimc_ctx * ctx = fimc - > vid_cap . ctx ;
2013-03-20 10:44:39 -03:00
int ret ;
2011-06-10 15:36:53 -03:00
unsigned long flags ;
2013-03-20 10:44:39 -03:00
if ( ctx = = NULL | | ctx - > s_frame . fmt = = NULL )
2011-06-10 15:36:53 -03:00
return - EINVAL ;
2013-03-20 10:44:39 -03:00
if ( si - > fimc_bus_type = = FIMC_BUS_TYPE_ISP_WRITEBACK ) {
ret = fimc_hw_camblk_cfg_writeback ( fimc ) ;
if ( ret < 0 )
return ret ;
}
2011-06-10 15:36:53 -03:00
spin_lock_irqsave ( & fimc - > slock , flags ) ;
fimc_prepare_dma_offset ( ctx , & ctx - > d_frame ) ;
fimc_set_yuv_order ( ctx ) ;
2013-03-20 10:44:39 -03:00
fimc_hw_set_camera_polarity ( fimc , si ) ;
fimc_hw_set_camera_type ( fimc , si ) ;
fimc_hw_set_camera_source ( fimc , si ) ;
2011-06-10 15:36:53 -03:00
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 ) ;
2012-04-02 06:41:22 -03:00
fimc_hw_set_effect ( ctx ) ;
2011-06-10 15:36:53 -03:00
fimc_hw_set_output_path ( ctx ) ;
fimc_hw_set_out_dma ( ctx ) ;
2013-03-26 08:22:21 -03:00
if ( fimc - > drv_data - > alpha_color )
2011-12-01 14:02:24 -03:00
fimc_hw_set_rgb_alpha ( ctx ) ;
2011-08-24 20:35:30 -03:00
clear_bit ( ST_CAPT_APPLY_CFG , & fimc - > state ) ;
2011-06-10 15:36:53 -03:00
}
spin_unlock_irqrestore ( & fimc - > slock , flags ) ;
return ret ;
}
2012-04-27 09:33:23 -03:00
/*
* Reinitialize the driver so it is ready to start the streaming again .
* Set fimc - > state to indicate stream off and the hardware shut down state .
* If not suspending ( @ suspend is false ) , return any buffers to videobuf2 .
* Otherwise put any owned buffers onto the pending buffers queue , so they
* can be re - spun when the device is being resumed . Also perform FIMC
* software reset and disable streaming on the whole pipeline if required .
*/
2011-08-24 20:45:34 -03:00
static int fimc_capture_state_cleanup ( struct fimc_dev * fimc , bool suspend )
2010-10-07 10:06:16 -03:00
{
2011-08-29 08:51:49 -03:00
struct fimc_vid_cap * cap = & fimc - > vid_cap ;
2010-12-01 10:14:59 -03:00
struct fimc_vid_buffer * buf ;
2011-08-29 08:51:49 -03:00
unsigned long flags ;
2011-08-24 20:45:34 -03:00
bool streaming ;
2010-10-07 10:06:16 -03:00
spin_lock_irqsave ( & fimc - > slock , flags ) ;
2011-08-24 20:45:34 -03:00
streaming = fimc - > state & ( 1 < < ST_CAPT_ISP_STREAM ) ;
2010-10-07 10:06:16 -03:00
2011-08-24 20:45:34 -03:00
fimc - > state & = ~ ( 1 < < ST_CAPT_RUN | 1 < < ST_CAPT_SHUT |
1 < < ST_CAPT_STREAM | 1 < < ST_CAPT_ISP_STREAM ) ;
2012-03-16 11:47:51 -03:00
if ( suspend )
fimc - > state | = ( 1 < < ST_CAPT_SUSPENDED ) ;
else
2011-08-24 20:45:34 -03:00
fimc - > state & = ~ ( 1 < < ST_CAPT_PEND | 1 < < ST_CAPT_SUSPENDED ) ;
2010-12-01 10:14:59 -03:00
2011-08-24 20:45:34 -03:00
/* Release unused buffers */
while ( ! suspend & & ! list_empty ( & cap - > pending_buf_q ) ) {
2011-06-10 15:36:59 -03:00
buf = fimc_pending_queue_pop ( cap ) ;
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 10:30:30 -03:00
vb2_buffer_done ( & buf - > vb . vb2_buf , VB2_BUF_STATE_ERROR ) ;
2010-12-01 10:14:59 -03:00
}
2011-08-24 20:45:34 -03:00
/* If suspending put unused buffers onto pending queue */
2010-12-01 10:14:59 -03:00
while ( ! list_empty ( & cap - > active_buf_q ) ) {
2011-06-10 15:36:59 -03:00
buf = fimc_active_queue_pop ( cap ) ;
2011-08-24 20:45:34 -03:00
if ( suspend )
fimc_pending_queue_add ( cap , buf ) ;
else
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 10:30:30 -03:00
vb2_buffer_done ( & buf - > vb . vb2_buf , VB2_BUF_STATE_ERROR ) ;
2010-12-01 10:14:59 -03:00
}
2011-10-05 14:20:45 -03:00
fimc_hw_reset ( fimc ) ;
cap - > buf_index = 0 ;
2010-10-07 10:06:16 -03:00
spin_unlock_irqrestore ( & fimc - > slock , flags ) ;
2011-08-26 14:51:00 -03:00
2011-08-24 20:45:34 -03:00
if ( streaming )
2013-05-31 11:37:22 -03:00
return fimc_pipeline_call ( & cap - > ve , set_stream , 0 ) ;
2011-08-26 14:51:00 -03:00
else
return 0 ;
2011-08-29 08:51:49 -03:00
}
2011-08-24 20:45:34 -03:00
static int fimc_stop_capture ( struct fimc_dev * fimc , bool suspend )
2011-08-29 08:51:49 -03: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-24 20:45:34 -03:00
( 2 * HZ / 10 ) ) ; /* 200 ms */
2010-10-07 10:06:16 -03:00
2011-08-24 20:45:34 -03:00
return fimc_capture_state_cleanup ( fimc , suspend ) ;
2010-10-07 10:06:16 -03:00
}
2011-08-24 20:35:30 -03: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 15:51:24 -03:00
static int fimc_capture_config_update ( struct fimc_ctx * ctx )
2011-08-24 20:35:30 -03:00
{
struct fimc_dev * fimc = ctx - > fimc_dev ;
int ret ;
fimc_hw_set_camera_offset ( fimc , & ctx - > s_frame ) ;
2012-03-19 13:11:40 -03:00
2011-08-24 20:35:30 -03:00
ret = fimc_set_scaler_info ( ctx ) ;
2012-03-19 13:11:40 -03: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 ) ;
2012-04-02 06:41:22 -03:00
fimc_hw_set_effect ( ctx ) ;
2012-03-19 13:11:40 -03:00
fimc_prepare_dma_offset ( ctx , & ctx - > d_frame ) ;
fimc_hw_set_out_dma ( ctx ) ;
2013-03-26 08:22:21 -03:00
if ( fimc - > drv_data - > alpha_color )
2012-03-19 13:11:40 -03:00
fimc_hw_set_rgb_alpha ( ctx ) ;
clear_bit ( ST_CAPT_APPLY_CFG , & fimc - > state ) ;
2011-08-24 20:35:30 -03:00
return ret ;
}
2011-08-29 08:51:49 -03:00
2012-05-08 15:51:24 -03:00
void fimc_capture_irq_handler ( struct fimc_dev * fimc , int deq_buf )
{
struct fimc_vid_cap * cap = & fimc - > vid_cap ;
2013-05-31 11:37:22 -03:00
struct fimc_pipeline * p = to_fimc_pipeline ( cap - > ve . pipe ) ;
struct v4l2_subdev * csis = p - > subdevs [ IDX_CSIS ] ;
2012-09-24 11:08:45 -03:00
struct fimc_frame * f = & cap - > ctx - > d_frame ;
2012-05-08 15:51:24 -03:00
struct fimc_vid_buffer * v_buf ;
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 ) {
v_buf = fimc_active_queue_pop ( cap ) ;
2015-11-03 08:16:37 -02:00
v_buf - > vb . vb2_buf . timestamp = ktime_get_ns ( ) ;
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 10:30:30 -03:00
v_buf - > vb . sequence = cap - > frame_count + + ;
2012-05-08 15:51:24 -03:00
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 10:30:30 -03:00
vb2_buffer_done ( & v_buf - > vb . vb2_buf , VB2_BUF_STATE_DONE ) ;
2012-05-08 15:51:24 -03:00
}
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 ;
}
2012-09-24 11:08:45 -03:00
/*
* Set up a buffer at MIPI - CSIS if current image format
* requires the frame embedded data capture .
*/
if ( f - > fmt - > mdataplanes & & ! list_empty ( & cap - > active_buf_q ) ) {
unsigned int plane = ffs ( f - > fmt - > mdataplanes ) - 1 ;
unsigned int size = f - > payload [ plane ] ;
s32 index = fimc_hw_get_frame_index ( fimc ) ;
void * vaddr ;
list_for_each_entry ( v_buf , & cap - > active_buf_q , list ) {
if ( v_buf - > index ! = index )
continue ;
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 10:30:30 -03:00
vaddr = vb2_plane_vaddr ( & v_buf - > vb . vb2_buf , plane ) ;
2012-09-24 11:08:45 -03:00
v4l2_subdev_call ( csis , video , s_rx_buffer ,
vaddr , & size ) ;
break ;
}
}
2012-05-08 15:51:24 -03:00
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 ) ;
}
2012-04-27 09:33:23 -03:00
if ( test_bit ( ST_CAPT_APPLY_CFG , & fimc - > state ) )
fimc_capture_config_update ( cap - > ctx ) ;
2012-05-08 15:51:24 -03:00
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 08:51:49 -03:00
static int start_streaming ( struct vb2_queue * q , unsigned int count )
2010-12-01 10:14:59 -03:00
{
struct fimc_ctx * ctx = q - > drv_priv ;
struct fimc_dev * fimc = ctx - > fimc_dev ;
2011-06-10 15:36:53 -03:00
struct fimc_vid_cap * vid_cap = & fimc - > vid_cap ;
2011-08-29 08:51:49 -03:00
int min_bufs ;
2010-12-01 10:14:59 -03:00
int ret ;
2011-06-10 15:36:53 -03:00
vid_cap - > frame_count = 0 ;
2011-02-07 15:59:46 -03:00
2012-04-27 09:33:23 -03:00
ret = fimc_capture_hw_init ( fimc ) ;
if ( ret ) {
fimc_capture_state_cleanup ( fimc , false ) ;
return ret ;
}
2010-12-01 10:14:59 -03:00
set_bit ( ST_CAPT_PEND , & fimc - > state ) ;
2011-08-29 08:51:49 -03:00
min_bufs = fimc - > vid_cap . reqbufs_count > 1 ? 2 : 1 ;
2011-08-26 14:51:00 -03:00
if ( vid_cap - > active_buf_cnt > = min_bufs & &
! test_and_set_bit ( ST_CAPT_STREAM , & fimc - > state ) ) {
2011-08-29 08:51:49 -03:00
fimc_activate_capture ( ctx ) ;
2011-08-26 14:51:00 -03:00
if ( ! test_and_set_bit ( ST_CAPT_ISP_STREAM , & fimc - > state ) )
2013-05-31 11:37:22 -03:00
return fimc_pipeline_call ( & vid_cap - > ve , set_stream , 1 ) ;
2011-08-26 14:51:00 -03:00
}
2010-12-01 10:14:59 -03:00
return 0 ;
}
2014-04-17 02:47:21 -03:00
static void stop_streaming ( struct vb2_queue * q )
2010-12-01 10:14:59 -03:00
{
struct fimc_ctx * ctx = q - > drv_priv ;
struct fimc_dev * fimc = ctx - > fimc_dev ;
2011-02-23 08:24:33 -03:00
if ( ! fimc_capture_active ( fimc ) )
2014-04-17 02:47:21 -03:00
return ;
2010-12-01 10:14:59 -03:00
2014-04-17 02:47:21 -03:00
fimc_stop_capture ( fimc , false ) ;
2010-12-01 10:14:59 -03:00
}
2011-09-02 06:25:32 -03:00
int fimc_capture_suspend ( struct fimc_dev * fimc )
{
2011-08-24 20:45:34 -03:00
bool suspend = fimc_capture_busy ( fimc ) ;
int ret = fimc_stop_capture ( fimc , suspend ) ;
if ( ret )
return ret ;
2013-05-31 11:37:22 -03:00
return fimc_pipeline_call ( & fimc - > vid_cap . ve , close ) ;
2011-09-02 06:25:32 -03:00
}
2011-08-24 20:45:34 -03:00
static void buffer_queue ( struct vb2_buffer * vb ) ;
2011-09-02 06:25:32 -03:00
int fimc_capture_resume ( struct fimc_dev * fimc )
{
2011-08-24 20:45:34 -03:00
struct fimc_vid_cap * vid_cap = & fimc - > vid_cap ;
2013-05-31 11:37:18 -03:00
struct exynos_video_entity * ve = & vid_cap - > ve ;
2011-08-24 20:45:34 -03:00
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 ;
2013-05-31 11:37:22 -03:00
fimc_pipeline_call ( ve , open , & ve - > vdev . entity , false ) ;
2012-04-27 09:33:23 -03:00
fimc_capture_hw_init ( fimc ) ;
2011-08-24 20:45:34 -03:00
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 ) ;
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 10:30:30 -03:00
buffer_queue ( & buf - > vb . vb2_buf ) ;
2011-08-24 20:45:34 -03:00
}
2011-09-02 06:25:32 -03:00
return 0 ;
2011-08-24 20:45:34 -03:00
2011-09-02 06:25:32 -03:00
}
2015-10-28 00:50:37 -02:00
static int queue_setup ( struct vb2_queue * vq ,
2011-08-24 10:30:21 -03:00
unsigned int * num_buffers , unsigned int * num_planes ,
2016-04-15 09:15:05 -03:00
unsigned int sizes [ ] , struct device * alloc_devs [ ] )
2010-12-01 10:14:59 -03:00
{
struct fimc_ctx * ctx = vq - > drv_priv ;
2012-04-22 17:07:09 -03:00
struct fimc_frame * frame = & ctx - > d_frame ;
struct fimc_fmt * fmt = frame - > fmt ;
2015-10-28 00:50:37 -02:00
unsigned long wh = frame - > f_width * frame - > f_height ;
2010-12-08 14:05:08 -03:00
int i ;
2010-12-01 10:14:59 -03:00
2012-04-22 17:07:09 -03:00
if ( fmt = = NULL )
2010-12-01 10:14:59 -03:00
return - EINVAL ;
2015-10-28 00:50:37 -02:00
if ( * num_planes ) {
if ( * num_planes ! = fmt - > memplanes )
return - EINVAL ;
2016-02-16 07:30:19 -02:00
for ( i = 0 ; i < * num_planes ; i + + )
2015-10-28 00:50:37 -02:00
if ( sizes [ i ] < ( wh * fmt - > depth [ i ] ) / 8 )
return - EINVAL ;
return 0 ;
}
2010-12-08 14:05:08 -03:00
* num_planes = fmt - > memplanes ;
2010-12-01 10:14:59 -03:00
2010-12-08 14:05:08 -03:00
for ( i = 0 ; i < fmt - > memplanes ; i + + ) {
2012-04-22 17:07:09 -03:00
unsigned int size = ( wh * fmt - > depth [ i ] ) / 8 ;
2015-10-28 00:50:37 -02:00
if ( fimc_fmt_is_user_defined ( fmt - > color ) )
2012-09-24 11:08:45 -03:00
sizes [ i ] = frame - > payload [ i ] ;
2012-04-22 17:07:09 -03:00
else
2012-05-16 15:00:26 -03:00
sizes [ i ] = max_t ( u32 , size , frame - > payload [ i ] ) ;
2010-12-08 14:05:08 -03:00
}
2010-12-01 10:14:59 -03:00
2010-12-08 14:05:08 -03:00
return 0 ;
2010-12-01 10: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 14:51:00 -03:00
if ( ctx - > d_frame . fmt = = NULL )
2010-12-08 14:05:08 -03:00
return - EINVAL ;
2010-12-01 10:14:59 -03:00
2010-12-08 14:05:08 -03:00
for ( i = 0 ; i < ctx - > d_frame . fmt - > memplanes ; i + + ) {
2011-08-26 14:51:00 -03:00
unsigned long size = ctx - > d_frame . payload [ i ] ;
2010-12-01 10:14:59 -03:00
if ( vb2_plane_size ( vb , i ) < size ) {
2013-05-31 11:37:18 -03:00
v4l2_err ( & ctx - > fimc_dev - > vid_cap . ve . vdev ,
2011-06-10 15:36:48 -03:00
" User buffer too small (%ld < %ld) \n " ,
2010-12-01 10: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 )
{
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 10:30:30 -03:00
struct vb2_v4l2_buffer * vbuf = to_vb2_v4l2_buffer ( vb ) ;
2010-12-01 10:14:59 -03:00
struct fimc_vid_buffer * buf
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 10:30:30 -03:00
= container_of ( vbuf , struct fimc_vid_buffer , vb ) ;
2011-08-26 14:51:00 -03:00
struct fimc_ctx * ctx = vb2_get_drv_priv ( vb - > vb2_queue ) ;
struct fimc_dev * fimc = ctx - > fimc_dev ;
2010-12-01 10:14:59 -03:00
struct fimc_vid_cap * vid_cap = & fimc - > vid_cap ;
2013-05-31 11:37:18 -03:00
struct exynos_video_entity * ve = & vid_cap - > ve ;
2010-12-01 10:14:59 -03:00
unsigned long flags ;
2011-02-07 15:59:46 -03:00
int min_bufs ;
2010-12-01 10:14:59 -03:00
spin_lock_irqsave ( & fimc - > slock , flags ) ;
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 10:30:30 -03:00
fimc_prepare_addr ( ctx , & buf - > vb . vb2_buf , & ctx - > d_frame , & buf - > paddr ) ;
2011-02-07 15:59:46 -03:00
2011-08-24 20:45:34 -03: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 15: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 10:14:59 -03:00
2011-02-07 15:59:46 -03:00
fimc_hw_set_output_addr ( fimc , & buf - > paddr , buf_id ) ;
buf - > index = vid_cap - > buf_index ;
2011-06-10 15:36:59 -03:00
fimc_active_queue_add ( vid_cap , buf ) ;
2010-12-01 10:14:59 -03:00
2011-02-07 15: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 10:14:59 -03:00
}
2011-02-07 15:59:46 -03:00
min_bufs = vid_cap - > reqbufs_count > 1 ? 2 : 1 ;
2011-08-26 14:51:00 -03:00
2011-08-29 08:51:49 -03:00
if ( vb2_is_streaming ( & vid_cap - > vbq ) & &
vid_cap - > active_buf_cnt > = min_bufs & &
2011-08-26 14:51:00 -03:00
! test_and_set_bit ( ST_CAPT_STREAM , & fimc - > state ) ) {
2013-02-27 08:13:59 -03:00
int ret ;
2011-02-07 15:59:46 -03:00
fimc_activate_capture ( ctx ) ;
2011-08-26 14:51:00 -03:00
spin_unlock_irqrestore ( & fimc - > slock , flags ) ;
2011-02-07 15:59:46 -03:00
2013-02-27 08:13:59 -03:00
if ( test_and_set_bit ( ST_CAPT_ISP_STREAM , & fimc - > state ) )
return ;
2013-05-31 11:37:22 -03:00
ret = fimc_pipeline_call ( ve , set_stream , 1 ) ;
2013-02-27 08:13:59 -03:00
if ( ret < 0 )
2013-05-31 11:37:18 -03:00
v4l2_err ( & ve - > vdev , " stream on failed: %d \n " , ret ) ;
2011-08-26 14:51:00 -03:00
return ;
}
2010-12-01 10:14:59 -03:00
spin_unlock_irqrestore ( & fimc - > slock , flags ) ;
}
2016-09-08 20:59:10 -03:00
static const struct vb2_ops fimc_capture_qops = {
2010-12-01 10:14:59 -03:00
. queue_setup = queue_setup ,
. buf_prepare = buffer_prepare ,
. buf_queue = buffer_queue ,
2013-03-25 16:36:35 -03:00
. wait_prepare = vb2_ops_wait_prepare ,
. wait_finish = vb2_ops_wait_finish ,
2010-12-01 10:14:59 -03:00
. start_streaming = start_streaming ,
. stop_streaming = stop_streaming ,
} ;
2011-08-24 20:35:30 -03:00
static int fimc_capture_set_default_format ( struct fimc_dev * fimc ) ;
2010-10-07 10:06:16 -03:00
static int fimc_capture_open ( struct file * file )
{
struct fimc_dev * fimc = video_drvdata ( file ) ;
2013-05-31 11:37:19 -03:00
struct fimc_vid_cap * vc = & fimc - > vid_cap ;
struct exynos_video_entity * ve = & vc - > ve ;
2012-06-27 09:56:17 -03:00
int ret = - EBUSY ;
2010-10-07 10:06:16 -03:00
dbg ( " pid: %d, state: 0x%lx " , task_pid_nr ( current ) , fimc - > state ) ;
2012-12-06 10:26:19 -03:00
mutex_lock ( & fimc - > lock ) ;
2012-06-27 09:56:17 -03:00
2010-10-07 10:06:16 -03:00
if ( fimc_m2m_active ( fimc ) )
2012-06-27 09:56:17 -03:00
goto unlock ;
2010-10-07 10:06:16 -03:00
2011-08-24 20:45:34 -03:00
set_bit ( ST_CAPT_BUSY , & fimc - > state ) ;
2012-05-17 14:22:10 -03:00
ret = pm_runtime_get_sync ( & fimc - > pdev - > dev ) ;
if ( ret < 0 )
2012-06-27 09:56:17 -03:00
goto unlock ;
2011-08-26 14:51:00 -03:00
2012-05-17 14:22:10 -03:00
ret = v4l2_fh_open ( file ) ;
2012-06-27 09:56:17 -03:00
if ( ret ) {
2013-05-31 11:37:21 -03:00
pm_runtime_put_sync ( & fimc - > pdev - > dev ) ;
2012-06-27 09:56:17 -03:00
goto unlock ;
}
2012-05-17 14:22:10 -03:00
2013-03-25 16:36:35 -03:00
if ( v4l2_fh_is_singular_file ( file ) ) {
2013-05-31 11:37:20 -03:00
fimc_md_graph_lock ( ve ) ;
2013-05-31 11:37:22 -03:00
ret = fimc_pipeline_call ( ve , open , & ve - > vdev . entity , true ) ;
2013-05-31 11:37:19 -03:00
if ( ret = = 0 & & vc - > user_subdev_api & & vc - > inh_sensor_ctrls ) {
/*
* Recreate controls of the the video node to drop
* any controls inherited from the sensor subdev .
*/
fimc_ctrls_delete ( vc - > ctx ) ;
ret = fimc_ctrls_create ( vc - > ctx ) ;
if ( ret = = 0 )
vc - > inh_sensor_ctrls = false ;
}
2013-05-31 11:37:20 -03:00
if ( ret = = 0 )
ve - > vdev . entity . use_count + + ;
fimc_md_graph_unlock ( ve ) ;
2012-05-17 14:22:10 -03:00
2013-05-31 13:47:02 -03:00
if ( ret = = 0 )
ret = fimc_capture_set_default_format ( fimc ) ;
2012-06-27 09:56:17 -03:00
if ( ret < 0 ) {
clear_bit ( ST_CAPT_BUSY , & fimc - > state ) ;
pm_runtime_put_sync ( & fimc - > pdev - > dev ) ;
v4l2_fh_release ( file ) ;
}
}
unlock :
mutex_unlock ( & fimc - > lock ) ;
2011-08-24 19:25:10 -03:00
return ret ;
2010-10-07 10:06:16 -03:00
}
2013-03-25 16:36:35 -03:00
static int fimc_capture_release ( struct file * file )
2010-10-07 10:06:16 -03:00
{
struct fimc_dev * fimc = video_drvdata ( file ) ;
2013-03-25 16:50:50 -03:00
struct fimc_vid_cap * vc = & fimc - > vid_cap ;
2013-05-31 11:37:20 -03:00
bool close = v4l2_fh_is_singular_file ( file ) ;
2012-06-27 09:56:17 -03:00
int ret ;
2010-10-07 10:06:16 -03:00
dbg ( " pid: %d, state: 0x%lx " , task_pid_nr ( current ) , fimc - > state ) ;
2012-11-22 11:12:16 -03:00
mutex_lock ( & fimc - > lock ) ;
2012-06-27 09:56:17 -03:00
2013-05-31 11:37:20 -03:00
if ( close & & vc - > streaming ) {
media_entity_pipeline_stop ( & vc - > ve . vdev . entity ) ;
vc - > streaming = false ;
}
2013-11-25 05:49:02 -03:00
ret = _vb2_fop_release ( file , NULL ) ;
2013-05-31 11:37:20 -03:00
if ( close ) {
2011-08-24 20:45:34 -03:00
clear_bit ( ST_CAPT_BUSY , & fimc - > state ) ;
2013-05-31 11:37:22 -03:00
fimc_pipeline_call ( & vc - > ve , close ) ;
2011-08-24 20:45:34 -03:00
clear_bit ( ST_CAPT_SUSPENDED , & fimc - > state ) ;
2013-05-31 11:37:20 -03:00
fimc_md_graph_lock ( & vc - > ve ) ;
vc - > ve . vdev . entity . use_count - - ;
fimc_md_graph_unlock ( & vc - > ve ) ;
2010-10-07 10:06:16 -03:00
}
2013-05-31 11:37:21 -03:00
pm_runtime_put_sync ( & fimc - > pdev - > dev ) ;
2012-06-27 09:56:17 -03:00
mutex_unlock ( & fimc - > lock ) ;
return ret ;
2010-10-07 10:06:16 -03:00
}
static const struct v4l2_file_operations fimc_capture_fops = {
. owner = THIS_MODULE ,
. open = fimc_capture_open ,
2013-03-25 16:36:35 -03:00
. release = fimc_capture_release ,
. poll = vb2_fop_poll ,
2010-10-07 10:06:16 -03:00
. unlocked_ioctl = video_ioctl2 ,
2013-03-25 16:36:35 -03:00
. mmap = vb2_fop_mmap ,
2010-10-07 10:06:16 -03:00
} ;
2011-08-24 20:35:30 -03: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 ;
2012-08-02 10:27:46 -03:00
const struct fimc_variant * var = fimc - > variant ;
const struct fimc_pix_limit * pl = var - > pix_limit ;
2011-08-24 20:35:30 -03:00
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 ;
2012-09-24 11:08:45 -03:00
/* Conversion from/to JPEG or User Defined format is not supported */
2011-08-24 20:35:30 -03:00
if ( code & & ctx - > s_frame . fmt & & pad = = FIMC_SD_PAD_SOURCE & &
2012-09-24 11:08:45 -03:00
fimc_fmt_is_user_defined ( ctx - > s_frame . fmt - > color ) )
* code = ctx - > s_frame . fmt - > mbus_code ;
2011-08-24 20:35:30 -03:00
2013-03-20 10:44:39 -03:00
if ( fourcc & & * fourcc ! = V4L2_PIX_FMT_JPEG & & pad = = FIMC_SD_PAD_SOURCE )
2011-08-24 20:35:30 -03:00
mask | = FMT_FLAGS_M2M ;
2013-03-20 10:44:39 -03:00
if ( pad = = FIMC_SD_PAD_SINK_FIFO )
mask = FMT_FLAGS_WRITEBACK ;
2011-08-24 20:35:30 -03:00
ffmt = fimc_find_format ( fourcc , code , mask , 0 ) ;
if ( WARN_ON ( ! ffmt ) )
return NULL ;
2013-03-20 10:44:39 -03:00
2011-08-24 20:35:30 -03:00
if ( code )
* code = ffmt - > mbus_code ;
if ( fourcc )
* fourcc = ffmt - > fourcc ;
2013-03-20 10:44:39 -03:00
if ( pad ! = FIMC_SD_PAD_SOURCE ) {
2012-09-24 11:08:45 -03:00
max_w = fimc_fmt_is_user_defined ( ffmt - > color ) ?
2011-08-24 20:35:30 -03:00
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 ,
2012-09-24 11:08:45 -03:00
fimc_fmt_is_user_defined ( ffmt - > color ) ?
3 : 1 ,
2011-08-24 20:35:30 -03:00
0 ) ;
return ffmt ;
}
/* Can't scale or crop in transparent (JPEG) transfer mode */
2012-09-24 11:08:45 -03:00
if ( fimc_fmt_is_user_defined ( ffmt - > color ) ) {
2011-08-24 20:35:30 -03:00
* 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 ;
2012-04-29 17:46:03 -03:00
if ( ctx - > state & FIMC_COMPOSE ) {
min_w = dst - > offs_h + dst - > width ;
min_h = dst - > offs_v + dst - > height ;
} else {
min_w = var - > min_out_pixsize ;
min_h = var - > min_out_pixsize ;
}
2011-05-27 13:12:23 -03:00
if ( var - > min_vsize_align = = 1 & & ! rotation )
2011-08-24 20:35:30 -03: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 ;
}
2012-04-29 17:46:03 -03:00
static void fimc_capture_try_selection ( struct fimc_ctx * ctx ,
struct v4l2_rect * r ,
int target )
2011-08-24 20:35:30 -03:00
{
bool rotate = ctx - > rotation = = 90 | | ctx - > rotation = = 270 ;
struct fimc_dev * fimc = ctx - > fimc_dev ;
2012-08-02 10:27:46 -03:00
const struct fimc_variant * var = fimc - > variant ;
const struct fimc_pix_limit * pl = var - > pix_limit ;
2011-08-24 20:35:30 -03:00
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 */
2012-09-24 11:08:45 -03:00
if ( fimc_fmt_is_user_defined ( ctx - > d_frame . fmt - > color ) ) {
2011-08-24 20:35:30 -03:00
r - > width = sink - > f_width ;
r - > height = sink - > f_height ;
r - > left = r - > top = 0 ;
return ;
}
2012-05-20 11:17:12 -03:00
if ( target = = V4L2_SEL_TGT_COMPOSE ) {
2011-08-24 20:35:30 -03:00
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 ;
}
/*
2012-04-29 17:46:03 -03:00
* For the compose rectangle the following constraints must be met :
2011-08-24 20:35:30 -03:00
* - 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 ) ;
2012-04-29 17:46:03 -03:00
2012-05-20 11:17:12 -03:00
if ( target = = V4L2_SEL_TGT_COMPOSE ) {
2011-08-24 20:35:30 -03:00
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 ) ;
2012-04-29 17:46:03 -03:00
/* Adjust left/top if crop/compose rectangle is out of bounds */
2011-08-24 20:35:30 -03:00
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 ) ;
2012-04-29 17:46:03 -03:00
dbg ( " target %#x: (%d,%d)/%dx%d, sink fmt: %dx%d " ,
target , r - > left , r - > top , r - > width , r - > height ,
2011-08-24 20:35:30 -03:00
sink - > f_width , sink - > f_height ) ;
}
/*
* The video node ioctl operations
*/
2013-04-23 11:36:04 -03:00
static int fimc_cap_querycap ( struct file * file , void * priv ,
2010-10-07 10:06:16 -03:00
struct v4l2_capability * cap )
{
2011-06-10 15:36:50 -03:00
struct fimc_dev * fimc = video_drvdata ( file ) ;
2010-10-07 10:06:16 -03:00
2013-04-23 11:36:04 -03:00
__fimc_vidioc_querycap ( & fimc - > pdev - > dev , cap , V4L2_CAP_STREAMING |
V4L2_CAP_VIDEO_CAPTURE_MPLANE ) ;
2010-10-07 10:06:16 -03:00
return 0 ;
}
2011-06-13 11:09:40 -03: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 ;
2014-11-10 14:28:31 -03:00
if ( fmt - > fourcc = = MEDIA_BUS_FMT_JPEG_1X8 )
2011-06-13 11:09:40 -03:00
f - > flags | = V4L2_FMT_FLAG_COMPRESSED ;
return 0 ;
}
2012-12-03 06:24:32 -03:00
static struct media_entity * fimc_pipeline_get_head ( struct media_entity * me )
{
struct media_pad * pad = & me - > pads [ 0 ] ;
while ( ! ( pad - > flags & MEDIA_PAD_FL_SOURCE ) ) {
2013-06-03 05:16:13 -03:00
pad = media_entity_remote_pad ( pad ) ;
2012-12-03 06:24:32 -03:00
if ( ! pad )
break ;
me = pad - > entity ;
pad = & me - > pads [ 0 ] ;
}
return me ;
}
2011-08-24 20:35:30 -03: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 ;
2013-05-31 11:37:22 -03:00
struct fimc_pipeline * p = to_fimc_pipeline ( fimc - > vid_cap . ve . pipe ) ;
struct v4l2_subdev * sd = p - > subdevs [ IDX_SENSOR ] ;
2011-08-24 20:35:30 -03:00
struct v4l2_subdev_format sfmt ;
struct v4l2_mbus_framefmt * mf = & sfmt . format ;
2012-12-03 06:24:32 -03:00
struct media_entity * me ;
struct fimc_fmt * ffmt ;
struct media_pad * pad ;
int ret , i = 1 ;
u32 fcc ;
2011-08-24 20:35:30 -03:00
if ( WARN_ON ( ! sd | | ! tfmt ) )
return - EINVAL ;
2010-10-07 10:06:16 -03:00
2011-08-24 20:35:30 -03:00
memset ( & sfmt , 0 , sizeof ( sfmt ) ) ;
sfmt . format = * tfmt ;
sfmt . which = set ? V4L2_SUBDEV_FORMAT_ACTIVE : V4L2_SUBDEV_FORMAT_TRY ;
2012-12-03 06:24:32 -03:00
me = fimc_pipeline_get_head ( & sd - > entity ) ;
2011-08-24 20:35:30 -03:00
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 10:06:16 -03:00
2012-12-03 06:24:32 -03:00
/* set format on all pipeline subdevs */
while ( me ! = & fimc - > vid_cap . subdev . entity ) {
sd = media_entity_to_v4l2_subdev ( me ) ;
sfmt . pad = 0 ;
ret = v4l2_subdev_call ( sd , pad , set_fmt , NULL , & sfmt ) ;
if ( ret )
return ret ;
if ( me - > pads [ 0 ] . flags & MEDIA_PAD_FL_SINK ) {
sfmt . pad = me - > num_pads - 1 ;
mf - > code = tfmt - > code ;
ret = v4l2_subdev_call ( sd , pad , set_fmt , NULL ,
& sfmt ) ;
if ( ret )
return ret ;
}
2013-06-03 05:16:13 -03:00
pad = media_entity_remote_pad ( & me - > pads [ sfmt . pad ] ) ;
2012-12-03 06:24:32 -03:00
if ( ! pad )
return - EINVAL ;
me = pad - > entity ;
2011-08-24 20:35:30 -03:00
}
2010-10-07 10:06:16 -03:00
2012-12-03 06:24:32 -03:00
if ( mf - > code ! = tfmt - > code )
continue ;
fcc = ffmt - > fourcc ;
tfmt - > width = mf - > width ;
tfmt - > height = mf - > height ;
ffmt = fimc_capture_try_format ( ctx , & tfmt - > width , & tfmt - > height ,
2013-03-20 10:44:39 -03:00
NULL , & fcc , FIMC_SD_PAD_SINK_CAM ) ;
2012-12-03 06:24:32 -03:00
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 ;
if ( mf - > width ! = tfmt - > width | | mf - > height ! = tfmt - > height )
continue ;
tfmt - > code = mf - > code ;
break ;
2011-08-24 20:35:30 -03:00
}
2010-10-07 10:06:16 -03:00
2011-08-24 20:35:30 -03:00
if ( fmt_id & & ffmt )
* fmt_id = ffmt ;
* tfmt = * mf ;
2010-10-07 10:06:16 -03:00
2011-08-24 20:35:30 -03:00
return 0 ;
}
2010-10-07 10:06:16 -03:00
2012-09-24 11:08:45 -03:00
/**
* fimc_get_sensor_frame_desc - query the sensor for media bus frame parameters
* @ sensor : pointer to the sensor subdev
* @ plane_fmt : provides plane sizes corresponding to the frame layout entries
* @ try : true to set the frame parameters , false to query only
*
* This function is used by this driver only for compressed / blob data formats .
*/
static int fimc_get_sensor_frame_desc ( struct v4l2_subdev * sensor ,
struct v4l2_plane_pix_format * plane_fmt ,
unsigned int num_planes , bool try )
{
struct v4l2_mbus_frame_desc fd ;
int i , ret ;
2012-11-22 12:13:27 -03:00
int pad ;
2012-09-24 11:08:45 -03:00
for ( i = 0 ; i < num_planes ; i + + )
fd . entry [ i ] . length = plane_fmt [ i ] . sizeimage ;
2012-11-22 12:13:27 -03:00
pad = sensor - > entity . num_pads - 1 ;
2012-09-24 11:08:45 -03:00
if ( try )
2012-11-22 12:13:27 -03:00
ret = v4l2_subdev_call ( sensor , pad , set_frame_desc , pad , & fd ) ;
2012-09-24 11:08:45 -03:00
else
2012-11-22 12:13:27 -03:00
ret = v4l2_subdev_call ( sensor , pad , get_frame_desc , pad , & fd ) ;
2012-09-24 11:08:45 -03:00
if ( ret < 0 )
return ret ;
if ( num_planes ! = fd . num_entries )
return - EINVAL ;
for ( i = 0 ; i < num_planes ; i + + )
plane_fmt [ i ] . sizeimage = fd . entry [ i ] . length ;
if ( fd . entry [ 0 ] . length > FIMC_MAX_JPEG_BUF_SIZE ) {
v4l2_err ( sensor - > v4l2_dev , " Unsupported buffer size: %u \n " ,
fd . entry [ 0 ] . length ) ;
return - EINVAL ;
}
return 0 ;
}
2011-06-10 15:36:50 -03:00
static int fimc_cap_g_fmt_mplane ( struct file * file , void * fh ,
struct v4l2_format * f )
{
struct fimc_dev * fimc = video_drvdata ( file ) ;
2013-01-11 06:36:19 -03:00
__fimc_get_format ( & fimc - > vid_cap . ctx - > d_frame , f ) ;
return 0 ;
2011-06-10 15:36:50 -03:00
}
2013-05-31 13:47:02 -03:00
/*
* Try or set format on the fimc . X . capture video node and additionally
* on the whole pipeline if @ try is false .
* Locking : the caller must _not_ hold the graph mutex .
*/
static int __video_try_or_set_format ( struct fimc_dev * fimc ,
struct v4l2_format * f , bool try ,
struct fimc_fmt * * inp_fmt ,
struct fimc_fmt * * out_fmt )
2011-06-10 15:36:50 -03:00
{
2011-08-24 20:35:30 -03:00
struct v4l2_pix_format_mplane * pix = & f - > fmt . pix_mp ;
2013-05-31 13:47:02 -03:00
struct fimc_vid_cap * vc = & fimc - > vid_cap ;
struct exynos_video_entity * ve = & vc - > ve ;
struct fimc_ctx * ctx = vc - > ctx ;
unsigned int width = 0 , height = 0 ;
2012-12-06 10:26:19 -03:00
int ret = 0 ;
2013-05-31 13:47:02 -03:00
/* Pre-configure format at the camera input interface, for JPEG only */
2012-09-24 11:08:45 -03:00
if ( fimc_jpeg_fourcc ( pix - > pixelformat ) ) {
2011-08-24 20:35:30 -03:00
fimc_capture_try_format ( ctx , & pix - > width , & pix - > height ,
NULL , & pix - > pixelformat ,
2013-03-20 10:44:39 -03:00
FIMC_SD_PAD_SINK_CAM ) ;
2013-05-31 13:47:02 -03:00
if ( try ) {
width = pix - > width ;
height = pix - > height ;
} else {
ctx - > s_frame . f_width = pix - > width ;
ctx - > s_frame . f_height = pix - > height ;
}
2011-08-24 20:35:30 -03:00
}
2013-05-31 13:47:02 -03:00
/* Try the format at the scaler and the DMA output */
* out_fmt = fimc_capture_try_format ( ctx , & pix - > width , & pix - > height ,
NULL , & pix - > pixelformat ,
FIMC_SD_PAD_SOURCE ) ;
if ( * out_fmt = = NULL )
2013-05-31 11:37:20 -03:00
return - EINVAL ;
2011-08-24 20:35:30 -03:00
2013-05-31 13:47:02 -03:00
/* Restore image width/height for JPEG (no resizing supported). */
if ( try & & fimc_jpeg_fourcc ( pix - > pixelformat ) ) {
pix - > width = width ;
pix - > height = height ;
}
/* Try to match format at the host and the sensor */
if ( ! vc - > user_subdev_api ) {
struct v4l2_mbus_framefmt mbus_fmt ;
struct v4l2_mbus_framefmt * mf ;
mf = try ? & mbus_fmt : & fimc - > vid_cap . ci_fmt ;
mf - > code = ( * out_fmt ) - > mbus_code ;
mf - > width = pix - > width ;
mf - > height = pix - > height ;
2013-05-31 11:37:20 -03:00
fimc_md_graph_lock ( ve ) ;
2013-05-31 13:47:02 -03:00
ret = fimc_pipeline_try_format ( ctx , mf , inp_fmt , try ) ;
2013-05-31 11:37:20 -03:00
fimc_md_graph_unlock ( ve ) ;
2013-05-31 13:47:02 -03:00
if ( ret < 0 )
return ret ;
pix - > width = mf - > width ;
pix - > height = mf - > height ;
2011-08-24 20:35:30 -03:00
}
2011-06-10 15:36:50 -03:00
2013-05-31 13:47:02 -03:00
fimc_adjust_mplane_format ( * out_fmt , pix - > width , pix - > height , pix ) ;
if ( ( * out_fmt ) - > flags & FMT_FLAGS_COMPRESSED ) {
struct v4l2_subdev * sensor ;
2012-09-24 11:08:45 -03:00
2013-05-31 11:37:22 -03:00
fimc_md_graph_lock ( ve ) ;
sensor = __fimc_md_get_subdev ( ve - > pipe , IDX_SENSOR ) ;
if ( sensor )
fimc_get_sensor_frame_desc ( sensor , pix - > plane_fmt ,
2013-05-31 13:47:02 -03:00
( * out_fmt ) - > memplanes , try ) ;
2013-05-31 11:37:22 -03:00
else
ret = - EPIPE ;
fimc_md_graph_unlock ( ve ) ;
}
2012-09-24 11:08:45 -03:00
2012-12-06 10:26:19 -03:00
return ret ;
2011-06-10 15:36:50 -03:00
}
2013-05-31 13:47:02 -03:00
static int fimc_cap_try_fmt_mplane ( struct file * file , void * fh ,
struct v4l2_format * f )
{
struct fimc_dev * fimc = video_drvdata ( file ) ;
struct fimc_fmt * out_fmt = NULL , * inp_fmt = NULL ;
return __video_try_or_set_format ( fimc , f , true , & inp_fmt , & out_fmt ) ;
}
2012-09-24 11:08:45 -03:00
static void fimc_capture_mark_jpeg_xfer ( struct fimc_ctx * ctx ,
enum fimc_color_fmt color )
2011-08-26 14:57:06 -03:00
{
2012-09-24 11:08:45 -03:00
bool jpeg = fimc_fmt_is_user_defined ( color ) ;
2011-08-26 14:57:06 -03:00
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 ) ;
}
2012-12-06 10:26:19 -03:00
static int __fimc_capture_set_format ( struct fimc_dev * fimc ,
struct v4l2_format * f )
2010-10-07 10:06:16 -03:00
{
2013-05-31 13:47:02 -03:00
struct fimc_vid_cap * vc = & fimc - > vid_cap ;
struct fimc_ctx * ctx = vc - > ctx ;
2011-08-24 20:35:30 -03:00
struct v4l2_pix_format_mplane * pix = & f - > fmt . pix_mp ;
struct fimc_frame * ff = & ctx - > d_frame ;
2013-05-31 13:47:02 -03:00
struct fimc_fmt * inp_fmt = NULL ;
2011-08-24 20:35:30 -03:00
int ret , i ;
2010-10-07 10:06:16 -03:00
2011-08-24 20:35:30 -03:00
if ( vb2_is_busy ( & fimc - > vid_cap . vbq ) )
2010-12-08 14:05:08 -03:00
return - EBUSY ;
2010-10-07 10:06:16 -03:00
2013-05-31 13:47:02 -03:00
ret = __video_try_or_set_format ( fimc , f , false , & inp_fmt , & ff - > fmt ) ;
if ( ret < 0 )
return ret ;
2011-12-01 14:02:24 -03:00
/* Update RGB Alpha control state and value range */
fimc_alpha_ctrl_update ( ctx ) ;
2013-01-11 06:36:19 -03:00
for ( i = 0 ; i < ff - > fmt - > memplanes ; i + + ) {
ff - > bytesperline [ i ] = pix - > plane_fmt [ i ] . bytesperline ;
2012-05-16 15:00:26 -03:00
ff - > payload [ i ] = pix - > plane_fmt [ i ] . sizeimage ;
2013-01-11 06:36:19 -03:00
}
2011-08-24 20:35:30 -03:00
set_frame_bounds ( ff , pix - > width , pix - > height ) ;
/* Reset the composition rectangle if not yet configured */
2012-04-29 17:46:03 -03:00
if ( ! ( ctx - > state & FIMC_COMPOSE ) )
2011-08-24 20:35:30 -03:00
set_frame_crop ( ff , 0 , 0 , pix - > width , pix - > height ) ;
2012-09-24 11:08:45 -03:00
fimc_capture_mark_jpeg_xfer ( ctx , ff - > fmt - > color ) ;
2011-08-26 14:57:06 -03:00
2011-08-24 20:35:30 -03:00
/* Reset cropping and set format at the camera interface input */
2013-05-31 13:47:02 -03:00
if ( ! vc - > user_subdev_api ) {
ctx - > s_frame . fmt = inp_fmt ;
2011-08-24 20:35:30 -03:00
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 09:11:43 -03:00
}
2010-12-08 14:05:08 -03:00
2011-08-24 20:35:30 -03:00
return ret ;
}
2010-10-07 10:06:16 -03:00
2011-08-24 20:35:30 -03:00
static int fimc_cap_s_fmt_mplane ( struct file * file , void * priv ,
struct v4l2_format * f )
{
struct fimc_dev * fimc = video_drvdata ( file ) ;
2012-12-06 10:26:19 -03:00
2013-05-31 13:47:02 -03:00
return __fimc_capture_set_format ( fimc , f ) ;
2010-10-07 10:06:16 -03:00
}
static int fimc_cap_enum_input ( struct file * file , void * priv ,
2011-07-26 18:27:59 -03:00
struct v4l2_input * i )
2010-10-07 10:06:16 -03:00
{
2011-06-10 15:36:50 -03:00
struct fimc_dev * fimc = video_drvdata ( file ) ;
2013-05-31 11:37:22 -03:00
struct exynos_video_entity * ve = & fimc - > vid_cap . ve ;
struct v4l2_subdev * sd ;
2010-10-07 10:06:16 -03:00
2011-07-26 18:27:59 -03:00
if ( i - > index ! = 0 )
2010-10-07 10:06:16 -03:00
return - EINVAL ;
i - > type = V4L2_INPUT_TYPE_CAMERA ;
2013-05-31 11:37:22 -03:00
fimc_md_graph_lock ( ve ) ;
sd = __fimc_md_get_subdev ( ve - > pipe , IDX_SENSOR ) ;
fimc_md_graph_unlock ( ve ) ;
2011-08-26 14:51:00 -03:00
if ( sd )
strlcpy ( i - > name , sd - > name , sizeof ( i - > name ) ) ;
2013-05-31 11:37:22 -03:00
2010-10-07 10:06:16 -03:00
return 0 ;
}
2011-07-26 18:27:59 -03:00
static int fimc_cap_s_input ( struct file * file , void * priv , unsigned int i )
2010-10-07 10:06:16 -03:00
{
2011-07-26 18:27:59 -03:00
return i = = 0 ? i : - EINVAL ;
2010-10-07 10:06:16 -03:00
}
2011-07-26 18:27:59 -03:00
static int fimc_cap_g_input ( struct file * file , void * priv , unsigned int * i )
2010-10-07 10:06:16 -03:00
{
2011-07-26 18:27:59 -03:00
* i = 0 ;
2010-10-07 10:06:16 -03:00
return 0 ;
}
2011-08-24 20:35:30 -03: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 ;
2013-03-20 10:44:39 -03:00
struct fimc_vid_cap * vc = & fimc - > vid_cap ;
struct v4l2_subdev * sd = & vc - > subdev ;
2013-05-31 11:37:22 -03:00
struct fimc_pipeline * p = to_fimc_pipeline ( vc - > ve . pipe ) ;
2013-03-20 10:44:39 -03:00
struct media_pad * sink_pad , * src_pad ;
int i , ret ;
2011-08-24 20:35:30 -03:00
while ( 1 ) {
2013-03-20 10:44:39 -03:00
/*
* Find current entity sink pad and any remote sink pad linked
* to it . We stop if there is no sink pad in current entity or
* it is not linked to any other remote entity .
*/
src_pad = NULL ;
for ( i = 0 ; i < sd - > entity . num_pads ; i + + ) {
struct media_pad * p = & sd - > entity . pads [ i ] ;
if ( p - > flags & MEDIA_PAD_FL_SINK ) {
sink_pad = p ;
2013-06-03 05:16:13 -03:00
src_pad = media_entity_remote_pad ( sink_pad ) ;
2013-03-20 10:44:39 -03:00
if ( src_pad )
break ;
}
}
2015-05-07 22:12:32 -03:00
if ( ! src_pad | | ! is_media_entity_v4l2_subdev ( src_pad - > entity ) )
2011-08-24 20:35:30 -03:00
break ;
2013-03-20 10:44:39 -03:00
2011-08-24 20:35:30 -03:00
/* Don't call FIMC subdev operation to avoid nested locking */
2013-03-20 10:44:39 -03:00
if ( sd = = & vc - > subdev ) {
struct fimc_frame * ff = & vc - > ctx - > s_frame ;
2011-08-24 20:35:30 -03:00
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 {
2013-03-20 10:44:39 -03:00
sink_fmt . pad = sink_pad - > index ;
2011-08-24 20:35:30 -03:00
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 ;
}
2013-03-20 10:44:39 -03:00
/* Retrieve format at the source pad */
sd = media_entity_to_v4l2_subdev ( src_pad - > entity ) ;
src_fmt . pad = src_pad - > index ;
2011-08-24 20:35:30 -03:00
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 ;
2012-09-24 11:08:45 -03:00
2013-05-31 11:37:22 -03:00
if ( sd = = p - > subdevs [ IDX_SENSOR ] & &
2012-09-24 11:08:45 -03:00
fimc_user_defined_mbus_fmt ( src_fmt . format . code ) ) {
struct v4l2_plane_pix_format plane_fmt [ FIMC_MAX_PLANES ] ;
2013-03-20 10:44:39 -03:00
struct fimc_frame * frame = & vc - > ctx - > d_frame ;
2012-09-24 11:08:45 -03:00
unsigned int i ;
ret = fimc_get_sensor_frame_desc ( sd , plane_fmt ,
frame - > fmt - > memplanes ,
false ) ;
if ( ret < 0 )
return - EPIPE ;
for ( i = 0 ; i < frame - > fmt - > memplanes ; i + + )
if ( frame - > payload [ i ] < plane_fmt [ i ] . sizeimage )
return - EPIPE ;
}
2011-08-24 20:35:30 -03:00
}
return 0 ;
}
2010-10-07 10:06:16 -03:00
static int fimc_cap_streamon ( struct file * file , void * priv ,
2010-12-01 10:14:59 -03:00
enum v4l2_buf_type type )
2010-10-07 10:06:16 -03:00
{
2011-06-10 15:36:50 -03:00
struct fimc_dev * fimc = video_drvdata ( file ) ;
2013-03-20 10:35:43 -03:00
struct fimc_vid_cap * vc = & fimc - > vid_cap ;
2013-05-31 11:37:18 -03:00
struct media_entity * entity = & vc - > ve . vdev . entity ;
2013-03-20 10:44:39 -03:00
struct fimc_source_info * si = NULL ;
struct v4l2_subdev * sd ;
2011-08-24 20:35:30 -03:00
int ret ;
2010-10-07 10:06:16 -03:00
2011-08-26 14:51:00 -03:00
if ( fimc_capture_active ( fimc ) )
2010-12-01 10:25:18 -03:00
return - EBUSY ;
2010-10-07 10:06:16 -03:00
2013-05-31 11:37:22 -03:00
ret = media_entity_pipeline_start ( entity , & vc - > ve . pipe - > mp ) ;
2012-05-25 07:08:05 -03:00
if ( ret < 0 )
return ret ;
2010-10-07 10:06:16 -03:00
2013-05-31 11:37:22 -03:00
sd = __fimc_md_get_subdev ( vc - > ve . pipe , IDX_SENSOR ) ;
2013-03-20 10:44:39 -03:00
if ( sd )
si = v4l2_get_subdev_hostdata ( sd ) ;
if ( si = = NULL ) {
ret = - EPIPE ;
goto err_p_stop ;
}
/*
* Save configuration data related to currently attached image
* sensor or other data source , e . g . FIMC - IS .
*/
vc - > source_config = * si ;
if ( vc - > input = = GRP_ID_FIMC_IS )
vc - > source_config . fimc_bus_type = FIMC_BUS_TYPE_ISP_WRITEBACK ;
2013-03-20 10:35:43 -03:00
if ( vc - > user_subdev_api ) {
2011-08-24 20:35:30 -03:00
ret = fimc_pipeline_validate ( fimc ) ;
2013-03-20 10:35:43 -03:00
if ( ret < 0 )
goto err_p_stop ;
2011-08-24 20:35:30 -03:00
}
2013-03-20 10:35:43 -03:00
2013-03-25 16:36:35 -03:00
ret = vb2_ioctl_streamon ( file , priv , type ) ;
2013-03-25 16:50:50 -03:00
if ( ! ret ) {
vc - > streaming = true ;
2013-03-20 10:35:43 -03:00
return ret ;
2013-03-25 16:50:50 -03:00
}
2013-03-20 10:35:43 -03:00
err_p_stop :
media_entity_pipeline_stop ( entity ) ;
return ret ;
2010-10-07 10:06:16 -03:00
}
static int fimc_cap_streamoff ( struct file * file , void * priv ,
2010-12-01 10:25:18 -03:00
enum v4l2_buf_type type )
2010-10-07 10:06:16 -03:00
{
2011-06-10 15:36:50 -03:00
struct fimc_dev * fimc = video_drvdata ( file ) ;
2013-05-31 11:37:18 -03:00
struct fimc_vid_cap * vc = & fimc - > vid_cap ;
2011-08-26 14:51:00 -03:00
int ret ;
2010-10-07 10:06:16 -03:00
2013-03-25 16:36:35 -03:00
ret = vb2_ioctl_streamoff ( file , priv , type ) ;
2013-03-25 16:50:50 -03:00
if ( ret < 0 )
return ret ;
2013-03-20 10:35:43 -03:00
2013-05-31 11:37:18 -03:00
media_entity_pipeline_stop ( & vc - > ve . vdev . entity ) ;
vc - > streaming = false ;
2013-03-25 16:50:50 -03:00
return 0 ;
2010-10-07 10:06:16 -03:00
}
static int fimc_cap_reqbufs ( struct file * file , void * priv ,
2010-12-08 14:05:08 -03:00
struct v4l2_requestbuffers * reqbufs )
2010-10-07 10:06:16 -03:00
{
2011-06-10 15:36:50 -03:00
struct fimc_dev * fimc = video_drvdata ( file ) ;
2013-03-25 16:36:35 -03:00
int ret ;
ret = vb2_ioctl_reqbufs ( file , priv , reqbufs ) ;
2010-10-07 10:06:16 -03:00
if ( ! ret )
2011-06-10 15:36:50 -03:00
fimc - > vid_cap . reqbufs_count = reqbufs - > count ;
2012-01-30 14:56:59 -03:00
2013-03-25 16:36:35 -03:00
return ret ;
2012-01-30 14:56:59 -03:00
}
2012-02-04 17:03:54 -03:00
static int fimc_cap_g_selection ( struct file * file , void * fh ,
struct v4l2_selection * s )
2010-11-25 11:01:51 -03:00
{
2011-06-10 15:36:50 -03:00
struct fimc_dev * fimc = video_drvdata ( file ) ;
2012-02-04 17:03:54 -03:00
struct fimc_ctx * ctx = fimc - > vid_cap . ctx ;
struct fimc_frame * f = & ctx - > s_frame ;
2010-11-25 11:01:51 -03:00
2012-02-04 17:03:54 -03:00
if ( s - > type ! = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE )
2010-11-25 11:01:51 -03:00
return - EINVAL ;
2012-02-04 17:03:54 -03: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 11:01:51 -03:00
2012-05-20 11:17:12 -03:00
case V4L2_SEL_TGT_COMPOSE :
2012-02-04 17:03:54 -03:00
f = & ctx - > d_frame ;
2012-05-20 11:17:12 -03:00
case V4L2_SEL_TGT_CROP :
2012-02-04 17:03:54 -03:00
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 11:01:51 -03:00
}
2012-02-04 17:03:54 -03:00
/* Return 1 if rectangle a is enclosed in rectangle b, or 0 otherwise. */
2012-05-26 11:11:54 -03:00
static int enclosed_rectangle ( struct v4l2_rect * a , struct v4l2_rect * b )
2010-11-25 11:01:51 -03:00
{
2012-02-04 17:03:54 -03: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 11:01:51 -03:00
2012-02-04 17:03:54 -03:00
return 1 ;
2010-11-25 11:01:51 -03:00
}
2012-02-04 17:03:54 -03:00
static int fimc_cap_s_selection ( struct file * file , void * fh ,
struct v4l2_selection * s )
2010-10-07 10:06:16 -03:00
{
2011-06-10 15:36:50 -03:00
struct fimc_dev * fimc = video_drvdata ( file ) ;
struct fimc_ctx * ctx = fimc - > vid_cap . ctx ;
2012-02-04 17:03:54 -03:00
struct v4l2_rect rect = s - > r ;
struct fimc_frame * f ;
2011-08-24 20:35:30 -03:00
unsigned long flags ;
2012-02-04 17:03:54 -03:00
if ( s - > type ! = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE )
return - EINVAL ;
2012-05-20 11:17:12 -03:00
if ( s - > target = = V4L2_SEL_TGT_COMPOSE )
2012-02-04 17:03:54 -03:00
f = & ctx - > d_frame ;
2012-05-20 11:17:12 -03:00
else if ( s - > target = = V4L2_SEL_TGT_CROP )
2012-02-04 17:03:54 -03:00
f = & ctx - > s_frame ;
2012-04-29 17:46:03 -03:00
else
2012-02-04 17:03:54 -03:00
return - EINVAL ;
2012-04-29 17:46:03 -03:00
fimc_capture_try_selection ( ctx , & rect , s - > target ) ;
2012-02-04 17:03:54 -03:00
if ( s - > flags & V4L2_SEL_FLAG_LE & &
! enclosed_rectangle ( & rect , & s - > r ) )
return - ERANGE ;
2010-10-07 10:06:16 -03:00
2012-02-04 17:03:54 -03:00
if ( s - > flags & V4L2_SEL_FLAG_GE & &
! enclosed_rectangle ( & s - > r , & rect ) )
return - ERANGE ;
2010-10-07 10:06:16 -03:00
2012-02-04 17:03:54 -03:00
s - > r = rect ;
2011-08-24 20:35:30 -03:00
spin_lock_irqsave ( & fimc - > slock , flags ) ;
2012-02-04 17:03:54 -03:00
set_frame_crop ( f , s - > r . left , s - > r . top , s - > r . width ,
s - > r . height ) ;
2011-08-24 20:35:30 -03:00
spin_unlock_irqrestore ( & fimc - > slock , flags ) ;
2010-12-01 10:25:18 -03:00
2012-02-04 17:03:54 -03:00
set_bit ( ST_CAPT_APPLY_CFG , & fimc - > state ) ;
2010-12-01 10:25:18 -03:00
return 0 ;
2010-10-07 10:06:16 -03:00
}
static const struct v4l2_ioctl_ops fimc_capture_ioctl_ops = {
2013-04-23 11:36:04 -03:00
. vidioc_querycap = fimc_cap_querycap ,
2010-10-07 10:06:16 -03:00
2011-06-13 11:09:40 -03:00
. vidioc_enum_fmt_vid_cap_mplane = fimc_cap_enum_fmt_mplane ,
2011-06-10 15:36:50 -03:00
. vidioc_try_fmt_vid_cap_mplane = fimc_cap_try_fmt_mplane ,
2010-12-08 14:05:08 -03:00
. vidioc_s_fmt_vid_cap_mplane = fimc_cap_s_fmt_mplane ,
2011-06-10 15:36:50 -03:00
. vidioc_g_fmt_vid_cap_mplane = fimc_cap_g_fmt_mplane ,
2010-10-07 10:06:16 -03:00
. vidioc_reqbufs = fimc_cap_reqbufs ,
2013-03-25 16:36:35 -03:00
. vidioc_querybuf = vb2_ioctl_querybuf ,
. vidioc_qbuf = vb2_ioctl_qbuf ,
. vidioc_dqbuf = vb2_ioctl_dqbuf ,
. vidioc_expbuf = vb2_ioctl_expbuf ,
. vidioc_prepare_buf = vb2_ioctl_prepare_buf ,
. vidioc_create_bufs = vb2_ioctl_create_bufs ,
2012-01-30 14:56:59 -03:00
2010-10-07 10:06:16 -03:00
. vidioc_streamon = fimc_cap_streamon ,
. vidioc_streamoff = fimc_cap_streamoff ,
2012-02-04 17:03:54 -03:00
. vidioc_g_selection = fimc_cap_g_selection ,
. vidioc_s_selection = fimc_cap_s_selection ,
2010-10-07 10:06:16 -03:00
. vidioc_enum_input = fimc_cap_enum_input ,
. vidioc_s_input = fimc_cap_s_input ,
. vidioc_g_input = fimc_cap_g_input ,
} ;
2011-08-24 20:35:30 -03:00
/* Capture subdev media entity operations */
2011-08-24 19:28:18 -03:00
static int fimc_link_setup ( struct media_entity * entity ,
const struct media_pad * local ,
const struct media_pad * remote , u32 flags )
{
2011-08-24 20:35:30 -03:00
struct v4l2_subdev * sd = media_entity_to_v4l2_subdev ( entity ) ;
struct fimc_dev * fimc = v4l2_get_subdevdata ( sd ) ;
2013-05-31 11:37:19 -03:00
struct fimc_vid_cap * vc = & fimc - > vid_cap ;
struct v4l2_subdev * sensor ;
2011-08-24 20:35:30 -03:00
2015-05-07 22:12:32 -03:00
if ( ! is_media_entity_v4l2_subdev ( remote - > entity ) )
2011-08-24 20:35:30 -03:00
return - EINVAL ;
2011-08-24 19:28:18 -03: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 ) ;
2013-05-31 11:37:19 -03:00
if ( ! ( flags & MEDIA_LNK_FL_ENABLED ) ) {
fimc - > vid_cap . input = 0 ;
2011-08-24 19:28:18 -03:00
return 0 ;
}
2013-05-31 11:37:19 -03:00
if ( vc - > input ! = 0 )
return - EBUSY ;
vc - > input = sd - > grp_id ;
if ( vc - > user_subdev_api | | vc - > inh_sensor_ctrls )
return 0 ;
/* Inherit V4L2 controls from the image sensor subdev. */
sensor = fimc_find_remote_sensor ( & vc - > subdev . entity ) ;
if ( sensor = = NULL )
return 0 ;
return v4l2_ctrl_add_handler ( & vc - > ctx - > ctrls . handler ,
sensor - > ctrl_handler , NULL ) ;
2011-08-24 19:28:18 -03:00
}
2011-08-24 20:35:30 -03:00
static const struct media_entity_operations fimc_sd_media_ops = {
2011-08-24 19:28:18 -03:00
. link_setup = fimc_link_setup ,
} ;
2011-06-10 15:36:58 -03: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 )
{
2013-04-09 11:11:58 -03:00
struct fimc_source_info * si ;
2011-06-10 15:36:58 -03:00
struct fimc_vid_buffer * buf ;
struct fimc_md * fmd ;
struct fimc_dev * fimc ;
unsigned long flags ;
if ( sd = = NULL )
return ;
2013-04-09 11:11:58 -03:00
si = v4l2_get_subdev_hostdata ( sd ) ;
2011-06-10 15:36:58 -03:00
fmd = entity_to_fimc_mdev ( & sd - > entity ) ;
spin_lock_irqsave ( & fmd - > slock , flags ) ;
2013-04-09 11:11:58 -03:00
fimc = si ? source_to_sensor_info ( si ) - > host : NULL ;
2011-06-10 15:36:58 -03:00
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 ) ;
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 10:30:30 -03:00
vb2_set_plane_payload ( & buf - > vb . vb2_buf , 0 ,
* ( ( u32 * ) arg ) ) ;
2011-06-10 15:36:58 -03:00
}
2012-05-08 15:51:24 -03:00
fimc_capture_irq_handler ( fimc , 1 ) ;
2011-06-10 15:36:58 -03:00
fimc_deactivate_capture ( fimc ) ;
spin_unlock_irqrestore ( & fimc - > slock , irq_flags ) ;
}
spin_unlock_irqrestore ( & fmd - > slock , flags ) ;
}
2011-08-24 20:35:30 -03:00
static int fimc_subdev_enum_mbus_code ( struct v4l2_subdev * sd ,
2015-03-04 01:47:54 -08:00
struct v4l2_subdev_pad_config * cfg ,
2011-08-24 20:35:30 -03:00
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 ,
2015-03-04 01:47:54 -08:00
struct v4l2_subdev_pad_config * cfg ,
2011-08-24 20:35:30 -03:00
struct v4l2_subdev_format * fmt )
{
struct fimc_dev * fimc = v4l2_get_subdevdata ( sd ) ;
struct fimc_ctx * ctx = fimc - > vid_cap . ctx ;
2013-03-20 10:44:39 -03:00
struct fimc_frame * ff = & ctx - > s_frame ;
2011-08-24 20:35:30 -03:00
struct v4l2_mbus_framefmt * mf ;
if ( fmt - > which = = V4L2_SUBDEV_FORMAT_TRY ) {
2015-03-04 01:47:54 -08:00
mf = v4l2_subdev_get_try_format ( sd , cfg , fmt - > pad ) ;
2011-08-24 20:35:30 -03:00
fmt - > format = * mf ;
return 0 ;
}
2013-03-20 10:44:39 -03:00
mf = & fmt - > format ;
2011-08-24 20:35:30 -03:00
mutex_lock ( & fimc - > lock ) ;
2013-03-20 10:44:39 -03:00
switch ( fmt - > pad ) {
case FIMC_SD_PAD_SOURCE :
if ( ! WARN_ON ( ff - > fmt = = NULL ) )
mf - > code = ff - > fmt - > mbus_code ;
/* Sink pads crop rectangle size */
mf - > width = ff - > width ;
mf - > height = ff - > height ;
break ;
case FIMC_SD_PAD_SINK_FIFO :
* mf = fimc - > vid_cap . wb_fmt ;
break ;
case FIMC_SD_PAD_SINK_CAM :
default :
* mf = fimc - > vid_cap . ci_fmt ;
break ;
}
2011-08-24 20:35:30 -03:00
mutex_unlock ( & fimc - > lock ) ;
2013-03-20 10:44:39 -03:00
mf - > colorspace = V4L2_COLORSPACE_JPEG ;
2011-08-24 20:35:30 -03:00
return 0 ;
}
static int fimc_subdev_set_fmt ( struct v4l2_subdev * sd ,
2015-03-04 01:47:54 -08:00
struct v4l2_subdev_pad_config * cfg ,
2011-08-24 20:35:30 -03:00
struct v4l2_subdev_format * fmt )
{
struct fimc_dev * fimc = v4l2_get_subdevdata ( sd ) ;
struct v4l2_mbus_framefmt * mf = & fmt - > format ;
2013-03-20 10:44:39 -03:00
struct fimc_vid_cap * vc = & fimc - > vid_cap ;
struct fimc_ctx * ctx = vc - > ctx ;
2011-08-24 20:35:30 -03:00
struct fimc_frame * ff ;
struct fimc_fmt * ffmt ;
dbg ( " pad%d: code: 0x%x, %dx%d " ,
fmt - > pad , mf - > code , mf - > width , mf - > height ) ;
2013-03-20 10:44:39 -03:00
if ( fmt - > pad = = FIMC_SD_PAD_SOURCE & & vb2_is_busy ( & vc - > vbq ) )
2011-08-24 20:35:30 -03:00
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 ) {
2015-03-04 01:47:54 -08:00
mf = v4l2_subdev_get_try_format ( sd , cfg , fmt - > pad ) ;
2011-08-24 20:35:30 -03:00
* mf = fmt - > format ;
return 0 ;
}
2012-12-05 13:52:00 -03:00
/* There must be a bug in the driver if this happens */
if ( WARN_ON ( ffmt = = NULL ) )
return - EINVAL ;
2011-12-01 14:02:24 -03:00
/* Update RGB Alpha control state and value range */
fimc_alpha_ctrl_update ( ctx ) ;
2012-09-24 11:08:45 -03:00
fimc_capture_mark_jpeg_xfer ( ctx , ffmt - > color ) ;
2013-03-20 10:44:39 -03:00
if ( fmt - > pad = = FIMC_SD_PAD_SOURCE ) {
ff = & ctx - > d_frame ;
/* Sink pads crop rectangle size */
mf - > width = ctx - > s_frame . width ;
mf - > height = ctx - > s_frame . height ;
} else {
ff = & ctx - > s_frame ;
}
2011-08-24 20:35:30 -03:00
mutex_lock ( & fimc - > lock ) ;
set_frame_bounds ( ff , mf - > width , mf - > height ) ;
2013-03-20 10:44:39 -03:00
if ( fmt - > pad = = FIMC_SD_PAD_SINK_FIFO )
vc - > wb_fmt = * mf ;
else if ( fmt - > pad = = FIMC_SD_PAD_SINK_CAM )
vc - > ci_fmt = * mf ;
2011-08-24 20:35:30 -03:00
ff - > fmt = ffmt ;
/* Reset the crop rectangle if required. */
2012-04-29 17:46:03 -03:00
if ( ! ( fmt - > pad = = FIMC_SD_PAD_SOURCE & & ( ctx - > state & FIMC_COMPOSE ) ) )
2011-08-24 20:35:30 -03:00
set_frame_crop ( ff , 0 , 0 , mf - > width , mf - > height ) ;
2013-03-20 10:44:39 -03:00
if ( fmt - > pad ! = FIMC_SD_PAD_SOURCE )
2012-04-29 17:46:03 -03:00
ctx - > state & = ~ FIMC_COMPOSE ;
2013-03-20 10:44:39 -03:00
2011-08-24 20:35:30 -03:00
mutex_unlock ( & fimc - > lock ) ;
return 0 ;
}
2012-04-29 17:46:03 -03:00
static int fimc_subdev_get_selection ( struct v4l2_subdev * sd ,
2015-03-04 01:47:54 -08:00
struct v4l2_subdev_pad_config * cfg ,
2012-04-29 17:46:03 -03:00
struct v4l2_subdev_selection * sel )
2011-08-24 20:35:30 -03:00
{
struct fimc_dev * fimc = v4l2_get_subdevdata ( sd ) ;
struct fimc_ctx * ctx = fimc - > vid_cap . ctx ;
2012-04-29 17:46:03 -03:00
struct fimc_frame * f = & ctx - > s_frame ;
struct v4l2_rect * r = & sel - > r ;
struct v4l2_rect * try_sel ;
2013-03-20 10:44:39 -03:00
if ( sel - > pad = = FIMC_SD_PAD_SOURCE )
2012-04-29 17:46:03 -03:00
return - EINVAL ;
mutex_lock ( & fimc - > lock ) ;
2011-08-24 20:35:30 -03:00
2012-04-29 17:46:03 -03:00
switch ( sel - > target ) {
2012-05-18 09:31:18 -03:00
case V4L2_SEL_TGT_COMPOSE_BOUNDS :
2012-04-29 17:46:03 -03:00
f = & ctx - > d_frame ;
2012-05-18 09:31:18 -03:00
case V4L2_SEL_TGT_CROP_BOUNDS :
2012-04-29 17:46:03 -03:00
r - > width = f - > o_width ;
r - > height = f - > o_height ;
r - > left = 0 ;
r - > top = 0 ;
mutex_unlock ( & fimc - > lock ) ;
2011-08-24 20:35:30 -03:00
return 0 ;
2012-04-29 17:46:03 -03:00
2012-05-18 09:31:18 -03:00
case V4L2_SEL_TGT_CROP :
2015-03-04 01:47:54 -08:00
try_sel = v4l2_subdev_get_try_crop ( sd , cfg , sel - > pad ) ;
2012-04-29 17:46:03 -03:00
break ;
2012-05-18 09:31:18 -03:00
case V4L2_SEL_TGT_COMPOSE :
2015-03-04 01:47:54 -08:00
try_sel = v4l2_subdev_get_try_compose ( sd , cfg , sel - > pad ) ;
2012-04-29 17:46:03 -03:00
f = & ctx - > d_frame ;
break ;
default :
mutex_unlock ( & fimc - > lock ) ;
return - EINVAL ;
2011-08-24 20:35:30 -03:00
}
2012-04-29 17:46:03 -03:00
if ( sel - > which = = V4L2_SUBDEV_FORMAT_TRY ) {
sel - > r = * try_sel ;
} else {
r - > left = f - > offs_h ;
r - > top = f - > offs_v ;
r - > width = f - > width ;
r - > height = f - > height ;
}
2011-08-24 20:35:30 -03:00
2012-04-29 17:46:03 -03:00
dbg ( " target %#x: l:%d, t:%d, %dx%d, f_w: %d, f_h: %d " ,
sel - > pad , r - > left , r - > top , r - > width , r - > height ,
f - > f_width , f - > f_height ) ;
2011-08-24 20:35:30 -03:00
2012-04-29 17:46:03 -03:00
mutex_unlock ( & fimc - > lock ) ;
2011-08-24 20:35:30 -03:00
return 0 ;
}
2012-04-29 17:46:03 -03:00
static int fimc_subdev_set_selection ( struct v4l2_subdev * sd ,
2015-03-04 01:47:54 -08:00
struct v4l2_subdev_pad_config * cfg ,
2012-04-29 17:46:03 -03:00
struct v4l2_subdev_selection * sel )
2011-08-24 20:35:30 -03:00
{
struct fimc_dev * fimc = v4l2_get_subdevdata ( sd ) ;
struct fimc_ctx * ctx = fimc - > vid_cap . ctx ;
2012-04-29 17:46:03 -03:00
struct fimc_frame * f = & ctx - > s_frame ;
struct v4l2_rect * r = & sel - > r ;
struct v4l2_rect * try_sel ;
2011-08-24 20:35:30 -03:00
unsigned long flags ;
2013-03-20 10:44:39 -03:00
if ( sel - > pad = = FIMC_SD_PAD_SOURCE )
2012-04-29 17:46:03 -03:00
return - EINVAL ;
2011-08-24 20:35:30 -03:00
mutex_lock ( & fimc - > lock ) ;
2012-05-20 11:17:12 -03:00
fimc_capture_try_selection ( ctx , r , V4L2_SEL_TGT_CROP ) ;
2011-08-24 20:35:30 -03:00
2012-04-29 17:46:03 -03:00
switch ( sel - > target ) {
2012-05-18 09:31:18 -03:00
case V4L2_SEL_TGT_CROP :
2015-03-04 01:47:54 -08:00
try_sel = v4l2_subdev_get_try_crop ( sd , cfg , sel - > pad ) ;
2012-04-29 17:46:03 -03:00
break ;
2012-05-18 09:31:18 -03:00
case V4L2_SEL_TGT_COMPOSE :
2015-03-04 01:47:54 -08:00
try_sel = v4l2_subdev_get_try_compose ( sd , cfg , sel - > pad ) ;
2012-04-29 17:46:03 -03:00
f = & ctx - > d_frame ;
break ;
default :
mutex_unlock ( & fimc - > lock ) ;
return - EINVAL ;
2011-08-24 20:35:30 -03:00
}
2012-04-29 17:46:03 -03:00
if ( sel - > which = = V4L2_SUBDEV_FORMAT_TRY ) {
* try_sel = sel - > r ;
} else {
spin_lock_irqsave ( & fimc - > slock , flags ) ;
set_frame_crop ( f , r - > left , r - > top , r - > width , r - > height ) ;
set_bit ( ST_CAPT_APPLY_CFG , & fimc - > state ) ;
2012-05-18 09:31:18 -03:00
if ( sel - > target = = V4L2_SEL_TGT_COMPOSE )
2012-04-29 17:46:03 -03:00
ctx - > state | = FIMC_COMPOSE ;
2013-01-30 09:55:46 -03:00
spin_unlock_irqrestore ( & fimc - > slock , flags ) ;
2012-04-29 17:46:03 -03:00
}
2011-08-24 20:35:30 -03:00
2012-04-29 17:46:03 -03:00
dbg ( " target %#x: (%d,%d)/%dx%d " , sel - > target , r - > left , r - > top ,
2011-08-24 20:35:30 -03:00
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 ,
2012-04-29 17:46:03 -03:00
. get_selection = fimc_subdev_get_selection ,
. set_selection = fimc_subdev_set_selection ,
2011-08-24 20:35:30 -03:00
. get_fmt = fimc_subdev_get_fmt ,
. set_fmt = fimc_subdev_set_fmt ,
} ;
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 = {
2013-06-18 14:56:11 -03:00
. width = FIMC_DEFAULT_WIDTH ,
. height = FIMC_DEFAULT_HEIGHT ,
2011-08-24 20:35:30 -03:00
. pixelformat = V4L2_PIX_FMT_YUYV ,
. field = V4L2_FIELD_NONE ,
. colorspace = V4L2_COLORSPACE_JPEG ,
} ,
} ;
2012-12-06 10:26:19 -03:00
return __fimc_capture_set_format ( fimc , & fmt ) ;
2011-08-24 20:35:30 -03:00
}
2010-12-08 14:05:08 -03:00
/* fimc->lock must be already initialized */
2012-04-20 18:57:25 -03:00
static int fimc_register_capture_device ( struct fimc_dev * fimc ,
2011-06-10 15:36:48 -03:00
struct v4l2_device * v4l2_dev )
2010-10-07 10:06:16 -03:00
{
2013-05-31 11:37:18 -03:00
struct video_device * vfd = & fimc - > vid_cap . ve . vdev ;
2013-03-25 16:36:35 -03:00
struct vb2_queue * q = & fimc - > vid_cap . vbq ;
2010-10-07 10:06:16 -03:00
struct fimc_ctx * ctx ;
2013-03-25 16:36:35 -03:00
struct fimc_vid_cap * vid_cap ;
2013-06-18 14:56:11 -03:00
struct fimc_fmt * fmt ;
2011-06-10 15:36:48 -03:00
int ret = - ENOMEM ;
2010-10-07 10:06:16 -03:00
2012-08-17 03:28:27 -03:00
ctx = kzalloc ( sizeof ( * ctx ) , GFP_KERNEL ) ;
2010-10-07 10:06:16 -03:00
if ( ! ctx )
return - ENOMEM ;
ctx - > fimc_dev = fimc ;
2012-04-26 06:26:29 -03:00
ctx - > in_path = FIMC_IO_CAMERA ;
ctx - > out_path = FIMC_IO_DMA ;
2010-10-07 10:06:16 -03:00
ctx - > state = FIMC_CTX_CAP ;
2011-08-24 20:35:30 -03:00
ctx - > s_frame . fmt = fimc_find_format ( NULL , NULL , FMT_FLAGS_CAM , 0 ) ;
2012-04-20 18:57:25 -03:00
ctx - > d_frame . fmt = ctx - > s_frame . fmt ;
2010-10-07 10:06:16 -03:00
2012-07-26 07:15:42 -03:00
memset ( vfd , 0 , sizeof ( * vfd ) ) ;
2012-04-20 18:57:25 -03:00
snprintf ( vfd - > name , sizeof ( vfd - > name ) , " fimc.%d.capture " , fimc - > id ) ;
2010-10-07 10:06:16 -03:00
vfd - > fops = & fimc_capture_fops ;
vfd - > ioctl_ops = & fimc_capture_ioctl_ops ;
2011-07-26 18:08:21 -03:00
vfd - > v4l2_dev = v4l2_dev ;
2010-10-07 10:06:16 -03:00
vfd - > minor = - 1 ;
2012-07-26 07:15:42 -03:00
vfd - > release = video_device_release_empty ;
2013-03-25 16:36:35 -03:00
vfd - > queue = q ;
2010-12-01 10:25:18 -03:00
vfd - > lock = & fimc - > lock ;
2012-06-27 09:56:17 -03:00
2010-10-07 10:06:16 -03:00
video_set_drvdata ( vfd , fimc ) ;
vid_cap = & fimc - > vid_cap ;
vid_cap - > active_buf_cnt = 0 ;
2013-03-25 16:36:35 -03:00
vid_cap - > reqbufs_count = 0 ;
vid_cap - > ctx = ctx ;
2010-10-07 10:06:16 -03:00
INIT_LIST_HEAD ( & vid_cap - > pending_buf_q ) ;
INIT_LIST_HEAD ( & vid_cap - > active_buf_q ) ;
2010-12-01 10:14:59 -03:00
memset ( q , 0 , sizeof ( * q ) ) ;
2010-12-08 14:05:08 -03:00
q - > type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE ;
2012-06-14 10:37:49 -03:00
q - > io_modes = VB2_MMAP | VB2_USERPTR | VB2_DMABUF ;
2013-03-25 16:36:35 -03:00
q - > drv_priv = ctx ;
2010-12-01 10:14:59 -03:00
q - > ops = & fimc_capture_qops ;
q - > mem_ops = & vb2_dma_contig_memops ;
q - > buf_struct_size = sizeof ( struct fimc_vid_buffer ) ;
2014-02-25 19:12:19 -03:00
q - > timestamp_flags = V4L2_BUF_FLAG_TIMESTAMP_MONOTONIC ;
2013-03-25 16:36:35 -03:00
q - > lock = & fimc - > lock ;
2016-02-16 07:30:19 -02:00
q - > dev = & fimc - > pdev - > dev ;
2010-12-01 10:14:59 -03:00
2012-10-19 08:24:18 -03:00
ret = vb2_queue_init ( q ) ;
if ( ret )
2013-05-31 11:37:19 -03:00
goto err_free_ctx ;
2010-10-07 10:06:16 -03:00
2013-06-18 14:56:11 -03:00
/* Default format configuration */
fmt = fimc_find_format ( NULL , NULL , FMT_FLAGS_CAM , 0 ) ;
vid_cap - > ci_fmt . width = FIMC_DEFAULT_WIDTH ;
vid_cap - > ci_fmt . height = FIMC_DEFAULT_HEIGHT ;
vid_cap - > ci_fmt . code = fmt - > mbus_code ;
ctx - > s_frame . width = FIMC_DEFAULT_WIDTH ;
ctx - > s_frame . height = FIMC_DEFAULT_HEIGHT ;
ctx - > s_frame . fmt = fmt ;
fmt = fimc_find_format ( NULL , NULL , FMT_FLAGS_WRITEBACK , 0 ) ;
vid_cap - > wb_fmt = vid_cap - > ci_fmt ;
vid_cap - > wb_fmt . code = fmt - > mbus_code ;
2012-04-20 18:57:25 -03:00
vid_cap - > vd_pad . flags = MEDIA_PAD_FL_SINK ;
2015-12-11 07:44:40 -02:00
ret = media_entity_pads_init ( & vfd - > entity , 1 , & vid_cap - > vd_pad ) ;
2011-07-26 18:08:21 -03:00
if ( ret )
2013-05-31 11:37:19 -03:00
goto err_free_ctx ;
ret = fimc_ctrls_create ( ctx ) ;
if ( ret )
goto err_me_cleanup ;
2012-04-20 18:57:25 -03:00
ret = video_register_device ( vfd , VFL_TYPE_GRABBER , - 1 ) ;
2011-08-24 20:35:30 -03:00
if ( ret )
2013-05-31 11:37:19 -03:00
goto err_ctrl_free ;
2012-04-20 18:57:25 -03:00
v4l2_info ( v4l2_dev , " Registered %s as /dev/%s \n " ,
vfd - > name , video_device_node_name ( vfd ) ) ;
2011-07-26 18:08:21 -03:00
2012-04-02 06:41:22 -03:00
vfd - > ctrl_handler = & ctx - > ctrls . handler ;
2010-10-07 10:06:16 -03:00
return 0 ;
2013-05-31 11:37:19 -03:00
err_ctrl_free :
fimc_ctrls_delete ( ctx ) ;
err_me_cleanup :
2011-08-24 20:35:30 -03:00
media_entity_cleanup ( & vfd - > entity ) ;
2013-05-31 11:37:19 -03:00
err_free_ctx :
2011-06-02 06:18:34 -03:00
kfree ( ctx ) ;
2010-10-07 10:06:16 -03:00
return ret ;
}
2012-04-20 18:57:25 -03:00
static int fimc_capture_subdev_registered ( struct v4l2_subdev * sd )
2010-10-07 10:06:16 -03:00
{
2012-04-20 18:57:25 -03:00
struct fimc_dev * fimc = v4l2_get_subdevdata ( sd ) ;
int ret ;
2010-10-07 10:06:16 -03:00
2012-07-26 07:59:11 -03:00
if ( fimc = = NULL )
return - ENXIO ;
2012-04-20 18:57:25 -03:00
ret = fimc_register_m2m_device ( fimc , sd - > v4l2_dev ) ;
if ( ret )
return ret ;
2013-05-31 11:37:22 -03:00
fimc - > vid_cap . ve . pipe = v4l2_get_subdev_hostdata ( sd ) ;
2012-11-22 06:17:19 -03:00
2012-04-20 18:57:25 -03:00
ret = fimc_register_capture_device ( fimc , sd - > v4l2_dev ) ;
2012-11-22 06:17:19 -03:00
if ( ret ) {
2012-04-20 18:57:25 -03:00
fimc_unregister_m2m_device ( fimc ) ;
2013-05-31 11:37:22 -03:00
fimc - > vid_cap . ve . pipe = NULL ;
2012-11-22 06:17:19 -03:00
}
2012-04-20 18:57:25 -03:00
return ret ;
}
static void fimc_capture_subdev_unregistered ( struct v4l2_subdev * sd )
{
struct fimc_dev * fimc = v4l2_get_subdevdata ( sd ) ;
2013-05-31 11:37:18 -03:00
struct video_device * vdev ;
2012-04-20 18:57:25 -03:00
if ( fimc = = NULL )
return ;
2013-05-31 11:37:25 -03:00
mutex_lock ( & fimc - > lock ) ;
2012-04-20 18:57:25 -03:00
fimc_unregister_m2m_device ( fimc ) ;
2013-05-31 11:37:18 -03:00
vdev = & fimc - > vid_cap . ve . vdev ;
2012-04-20 18:57:25 -03:00
2013-05-31 11:37:18 -03:00
if ( video_is_registered ( vdev ) ) {
video_unregister_device ( vdev ) ;
media_entity_cleanup ( & vdev - > entity ) ;
2013-05-31 11:37:19 -03:00
fimc_ctrls_delete ( fimc - > vid_cap . ctx ) ;
2013-05-31 11:37:22 -03:00
fimc - > vid_cap . ve . pipe = NULL ;
2011-07-26 18:08:21 -03:00
}
kfree ( fimc - > vid_cap . ctx ) ;
2011-08-26 15:40:36 -03:00
fimc - > vid_cap . ctx = NULL ;
2013-05-31 11:37:25 -03:00
mutex_unlock ( & fimc - > lock ) ;
2010-10-07 10:06:16 -03:00
}
2012-04-20 18:57:25 -03: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 ) ;
2013-04-18 08:49:54 -03:00
sd - > flags | = V4L2_SUBDEV_FL_HAS_DEVNODE ;
2013-03-26 08:22:21 -03:00
snprintf ( sd - > name , sizeof ( sd - > name ) , " FIMC.%d " , fimc - > id ) ;
2012-04-20 18:57:25 -03:00
2013-03-20 10:44:39 -03:00
fimc - > vid_cap . sd_pads [ FIMC_SD_PAD_SINK_CAM ] . flags = MEDIA_PAD_FL_SINK ;
fimc - > vid_cap . sd_pads [ FIMC_SD_PAD_SINK_FIFO ] . flags = MEDIA_PAD_FL_SINK ;
2012-04-20 18:57:25 -03:00
fimc - > vid_cap . sd_pads [ FIMC_SD_PAD_SOURCE ] . flags = MEDIA_PAD_FL_SOURCE ;
2015-12-11 07:44:40 -02:00
ret = media_entity_pads_init ( & sd - > entity , FIMC_SD_PADS_NUM ,
2015-08-06 09:25:57 -03:00
fimc - > vid_cap . sd_pads ) ;
2012-04-20 18:57:25 -03:00
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 ) ;
}