[media] omap3isp: Mark next captured frame as faulty when an SBL overflow occurs
Instead of trying to propagate errors down the pipeline manually (and failing to do so properly in all cases), flag SBL errors in the pipeline to which the entity that triggered the error belongs, and use pipeline error flags to mark buffers as faulty when completing them. Signed-off-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com> Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
This commit is contained in:
parent
c3cd257402
commit
875e2e3edf
|
@ -410,6 +410,7 @@ static inline void isp_isr_dbg(struct isp_device *isp, u32 irqstatus)
|
||||||
static void isp_isr_sbl(struct isp_device *isp)
|
static void isp_isr_sbl(struct isp_device *isp)
|
||||||
{
|
{
|
||||||
struct device *dev = isp->dev;
|
struct device *dev = isp->dev;
|
||||||
|
struct isp_pipeline *pipe;
|
||||||
u32 sbl_pcr;
|
u32 sbl_pcr;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -423,27 +424,38 @@ static void isp_isr_sbl(struct isp_device *isp)
|
||||||
if (sbl_pcr)
|
if (sbl_pcr)
|
||||||
dev_dbg(dev, "SBL overflow (PCR = 0x%08x)\n", sbl_pcr);
|
dev_dbg(dev, "SBL overflow (PCR = 0x%08x)\n", sbl_pcr);
|
||||||
|
|
||||||
if (sbl_pcr & (ISPSBL_PCR_CCDC_WBL_OVF | ISPSBL_PCR_CSIA_WBL_OVF
|
if (sbl_pcr & ISPSBL_PCR_CSIB_WBL_OVF) {
|
||||||
| ISPSBL_PCR_CSIB_WBL_OVF)) {
|
pipe = to_isp_pipeline(&isp->isp_ccp2.subdev.entity);
|
||||||
isp->isp_ccdc.error = 1;
|
if (pipe != NULL)
|
||||||
if (isp->isp_ccdc.output & CCDC_OUTPUT_PREVIEW)
|
pipe->error = true;
|
||||||
isp->isp_prev.error = 1;
|
}
|
||||||
if (isp->isp_ccdc.output & CCDC_OUTPUT_RESIZER)
|
|
||||||
isp->isp_res.error = 1;
|
if (sbl_pcr & ISPSBL_PCR_CSIA_WBL_OVF) {
|
||||||
|
pipe = to_isp_pipeline(&isp->isp_csi2a.subdev.entity);
|
||||||
|
if (pipe != NULL)
|
||||||
|
pipe->error = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (sbl_pcr & ISPSBL_PCR_CCDC_WBL_OVF) {
|
||||||
|
pipe = to_isp_pipeline(&isp->isp_ccdc.subdev.entity);
|
||||||
|
if (pipe != NULL)
|
||||||
|
pipe->error = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sbl_pcr & ISPSBL_PCR_PRV_WBL_OVF) {
|
if (sbl_pcr & ISPSBL_PCR_PRV_WBL_OVF) {
|
||||||
isp->isp_prev.error = 1;
|
pipe = to_isp_pipeline(&isp->isp_prev.subdev.entity);
|
||||||
if (isp->isp_res.input == RESIZER_INPUT_VP &&
|
if (pipe != NULL)
|
||||||
!(isp->isp_ccdc.output & CCDC_OUTPUT_RESIZER))
|
pipe->error = true;
|
||||||
isp->isp_res.error = 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sbl_pcr & (ISPSBL_PCR_RSZ1_WBL_OVF
|
if (sbl_pcr & (ISPSBL_PCR_RSZ1_WBL_OVF
|
||||||
| ISPSBL_PCR_RSZ2_WBL_OVF
|
| ISPSBL_PCR_RSZ2_WBL_OVF
|
||||||
| ISPSBL_PCR_RSZ3_WBL_OVF
|
| ISPSBL_PCR_RSZ3_WBL_OVF
|
||||||
| ISPSBL_PCR_RSZ4_WBL_OVF))
|
| ISPSBL_PCR_RSZ4_WBL_OVF)) {
|
||||||
isp->isp_res.error = 1;
|
pipe = to_isp_pipeline(&isp->isp_res.subdev.entity);
|
||||||
|
if (pipe != NULL)
|
||||||
|
pipe->error = true;
|
||||||
|
}
|
||||||
|
|
||||||
if (sbl_pcr & ISPSBL_PCR_H3A_AF_WBL_OVF)
|
if (sbl_pcr & ISPSBL_PCR_H3A_AF_WBL_OVF)
|
||||||
omap3isp_stat_sbl_overflow(&isp->isp_af);
|
omap3isp_stat_sbl_overflow(&isp->isp_af);
|
||||||
|
@ -471,24 +483,17 @@ static irqreturn_t isp_isr(int irq, void *_isp)
|
||||||
IRQ0STATUS_HS_VS_IRQ;
|
IRQ0STATUS_HS_VS_IRQ;
|
||||||
struct isp_device *isp = _isp;
|
struct isp_device *isp = _isp;
|
||||||
u32 irqstatus;
|
u32 irqstatus;
|
||||||
int ret;
|
|
||||||
|
|
||||||
irqstatus = isp_reg_readl(isp, OMAP3_ISP_IOMEM_MAIN, ISP_IRQ0STATUS);
|
irqstatus = isp_reg_readl(isp, OMAP3_ISP_IOMEM_MAIN, ISP_IRQ0STATUS);
|
||||||
isp_reg_writel(isp, irqstatus, OMAP3_ISP_IOMEM_MAIN, ISP_IRQ0STATUS);
|
isp_reg_writel(isp, irqstatus, OMAP3_ISP_IOMEM_MAIN, ISP_IRQ0STATUS);
|
||||||
|
|
||||||
isp_isr_sbl(isp);
|
isp_isr_sbl(isp);
|
||||||
|
|
||||||
if (irqstatus & IRQ0STATUS_CSIA_IRQ) {
|
if (irqstatus & IRQ0STATUS_CSIA_IRQ)
|
||||||
ret = omap3isp_csi2_isr(&isp->isp_csi2a);
|
omap3isp_csi2_isr(&isp->isp_csi2a);
|
||||||
if (ret)
|
|
||||||
isp->isp_ccdc.error = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (irqstatus & IRQ0STATUS_CSIB_IRQ) {
|
if (irqstatus & IRQ0STATUS_CSIB_IRQ)
|
||||||
ret = omap3isp_ccp2_isr(&isp->isp_ccp2);
|
omap3isp_ccp2_isr(&isp->isp_ccp2);
|
||||||
if (ret)
|
|
||||||
isp->isp_ccdc.error = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (irqstatus & IRQ0STATUS_CCDC_VD0_IRQ) {
|
if (irqstatus & IRQ0STATUS_CCDC_VD0_IRQ) {
|
||||||
if (isp->isp_ccdc.output & CCDC_OUTPUT_PREVIEW)
|
if (isp->isp_ccdc.output & CCDC_OUTPUT_PREVIEW)
|
||||||
|
|
|
@ -1427,8 +1427,11 @@ static void ccdc_lsc_isr(struct isp_ccdc_device *ccdc, u32 events)
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
|
||||||
if (events & IRQ0STATUS_CCDC_LSC_PREF_ERR_IRQ) {
|
if (events & IRQ0STATUS_CCDC_LSC_PREF_ERR_IRQ) {
|
||||||
|
struct isp_pipeline *pipe =
|
||||||
|
to_isp_pipeline(&ccdc->subdev.entity);
|
||||||
|
|
||||||
ccdc_lsc_error_handler(ccdc);
|
ccdc_lsc_error_handler(ccdc);
|
||||||
ccdc->error = 1;
|
pipe->error = true;
|
||||||
dev_dbg(to_device(ccdc), "lsc prefetch error\n");
|
dev_dbg(to_device(ccdc), "lsc prefetch error\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1503,7 +1506,7 @@ static int ccdc_isr_buffer(struct isp_ccdc_device *ccdc)
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
buffer = omap3isp_video_buffer_next(&ccdc->video_out, ccdc->error);
|
buffer = omap3isp_video_buffer_next(&ccdc->video_out);
|
||||||
if (buffer != NULL) {
|
if (buffer != NULL) {
|
||||||
ccdc_set_outaddr(ccdc, buffer->isp_addr);
|
ccdc_set_outaddr(ccdc, buffer->isp_addr);
|
||||||
restart = 1;
|
restart = 1;
|
||||||
|
@ -1517,7 +1520,6 @@ static int ccdc_isr_buffer(struct isp_ccdc_device *ccdc)
|
||||||
ISP_PIPELINE_STREAM_SINGLESHOT);
|
ISP_PIPELINE_STREAM_SINGLESHOT);
|
||||||
|
|
||||||
done:
|
done:
|
||||||
ccdc->error = 0;
|
|
||||||
return restart;
|
return restart;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1743,7 +1745,6 @@ static int ccdc_set_stream(struct v4l2_subdev *sd, int enable)
|
||||||
*/
|
*/
|
||||||
ccdc_config_vp(ccdc);
|
ccdc_config_vp(ccdc);
|
||||||
ccdc_enable_vp(ccdc, 1);
|
ccdc_enable_vp(ccdc, 1);
|
||||||
ccdc->error = 0;
|
|
||||||
ccdc_print_status(ccdc);
|
ccdc_print_status(ccdc);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -150,7 +150,6 @@ struct ispccdc_lsc {
|
||||||
* @input: Active input
|
* @input: Active input
|
||||||
* @output: Active outputs
|
* @output: Active outputs
|
||||||
* @video_out: Output video node
|
* @video_out: Output video node
|
||||||
* @error: A hardware error occurred during capture
|
|
||||||
* @alaw: A-law compression enabled (1) or disabled (0)
|
* @alaw: A-law compression enabled (1) or disabled (0)
|
||||||
* @lpf: Low pass filter enabled (1) or disabled (0)
|
* @lpf: Low pass filter enabled (1) or disabled (0)
|
||||||
* @obclamp: Optical-black clamp enabled (1) or disabled (0)
|
* @obclamp: Optical-black clamp enabled (1) or disabled (0)
|
||||||
|
@ -178,7 +177,6 @@ struct isp_ccdc_device {
|
||||||
enum ccdc_input_entity input;
|
enum ccdc_input_entity input;
|
||||||
unsigned int output;
|
unsigned int output;
|
||||||
struct isp_video video_out;
|
struct isp_video video_out;
|
||||||
unsigned int error;
|
|
||||||
|
|
||||||
unsigned int alaw:1,
|
unsigned int alaw:1,
|
||||||
lpf:1,
|
lpf:1,
|
||||||
|
|
|
@ -556,7 +556,7 @@ static void ccp2_isr_buffer(struct isp_ccp2_device *ccp2)
|
||||||
struct isp_pipeline *pipe = to_isp_pipeline(&ccp2->subdev.entity);
|
struct isp_pipeline *pipe = to_isp_pipeline(&ccp2->subdev.entity);
|
||||||
struct isp_buffer *buffer;
|
struct isp_buffer *buffer;
|
||||||
|
|
||||||
buffer = omap3isp_video_buffer_next(&ccp2->video_in, ccp2->error);
|
buffer = omap3isp_video_buffer_next(&ccp2->video_in);
|
||||||
if (buffer != NULL)
|
if (buffer != NULL)
|
||||||
ccp2_set_inaddr(ccp2, buffer->isp_addr);
|
ccp2_set_inaddr(ccp2, buffer->isp_addr);
|
||||||
|
|
||||||
|
@ -567,8 +567,6 @@ static void ccp2_isr_buffer(struct isp_ccp2_device *ccp2)
|
||||||
omap3isp_pipeline_set_stream(pipe,
|
omap3isp_pipeline_set_stream(pipe,
|
||||||
ISP_PIPELINE_STREAM_SINGLESHOT);
|
ISP_PIPELINE_STREAM_SINGLESHOT);
|
||||||
}
|
}
|
||||||
|
|
||||||
ccp2->error = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -576,13 +574,11 @@ static void ccp2_isr_buffer(struct isp_ccp2_device *ccp2)
|
||||||
* @ccp2: Pointer to ISP CCP2 device
|
* @ccp2: Pointer to ISP CCP2 device
|
||||||
*
|
*
|
||||||
* This will handle the CCP2 interrupts
|
* This will handle the CCP2 interrupts
|
||||||
*
|
|
||||||
* Returns -EIO in case of error, or 0 on success.
|
|
||||||
*/
|
*/
|
||||||
int omap3isp_ccp2_isr(struct isp_ccp2_device *ccp2)
|
void omap3isp_ccp2_isr(struct isp_ccp2_device *ccp2)
|
||||||
{
|
{
|
||||||
|
struct isp_pipeline *pipe = to_isp_pipeline(&ccp2->subdev.entity);
|
||||||
struct isp_device *isp = to_isp_device(ccp2);
|
struct isp_device *isp = to_isp_device(ccp2);
|
||||||
int ret = 0;
|
|
||||||
static const u32 ISPCCP2_LC01_ERROR =
|
static const u32 ISPCCP2_LC01_ERROR =
|
||||||
ISPCCP2_LC01_IRQSTATUS_LC0_FIFO_OVF_IRQ |
|
ISPCCP2_LC01_IRQSTATUS_LC0_FIFO_OVF_IRQ |
|
||||||
ISPCCP2_LC01_IRQSTATUS_LC0_CRC_IRQ |
|
ISPCCP2_LC01_IRQSTATUS_LC0_CRC_IRQ |
|
||||||
|
@ -604,19 +600,18 @@ int omap3isp_ccp2_isr(struct isp_ccp2_device *ccp2)
|
||||||
ISPCCP2_LCM_IRQSTATUS);
|
ISPCCP2_LCM_IRQSTATUS);
|
||||||
/* Errors */
|
/* Errors */
|
||||||
if (lcx_irqstatus & ISPCCP2_LC01_ERROR) {
|
if (lcx_irqstatus & ISPCCP2_LC01_ERROR) {
|
||||||
ccp2->error = 1;
|
pipe->error = true;
|
||||||
dev_dbg(isp->dev, "CCP2 err:%x\n", lcx_irqstatus);
|
dev_dbg(isp->dev, "CCP2 err:%x\n", lcx_irqstatus);
|
||||||
return -EIO;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (lcm_irqstatus & ISPCCP2_LCM_IRQSTATUS_OCPERROR_IRQ) {
|
if (lcm_irqstatus & ISPCCP2_LCM_IRQSTATUS_OCPERROR_IRQ) {
|
||||||
ccp2->error = 1;
|
pipe->error = true;
|
||||||
dev_dbg(isp->dev, "CCP2 OCP err:%x\n", lcm_irqstatus);
|
dev_dbg(isp->dev, "CCP2 OCP err:%x\n", lcm_irqstatus);
|
||||||
ret = -EIO;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (omap3isp_module_sync_is_stopping(&ccp2->wait, &ccp2->stopping))
|
if (omap3isp_module_sync_is_stopping(&ccp2->wait, &ccp2->stopping))
|
||||||
return 0;
|
return;
|
||||||
|
|
||||||
/* Frame number propagation */
|
/* Frame number propagation */
|
||||||
if (lcx_irqstatus & ISPCCP2_LC01_IRQSTATUS_LC0_FS_IRQ) {
|
if (lcx_irqstatus & ISPCCP2_LC01_IRQSTATUS_LC0_FS_IRQ) {
|
||||||
|
@ -629,8 +624,6 @@ int omap3isp_ccp2_isr(struct isp_ccp2_device *ccp2)
|
||||||
/* Handle queued buffers on frame end interrupts */
|
/* Handle queued buffers on frame end interrupts */
|
||||||
if (lcm_irqstatus & ISPCCP2_LCM_IRQSTATUS_EOF_IRQ)
|
if (lcm_irqstatus & ISPCCP2_LCM_IRQSTATUS_EOF_IRQ)
|
||||||
ccp2_isr_buffer(ccp2);
|
ccp2_isr_buffer(ccp2);
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* -----------------------------------------------------------------------------
|
/* -----------------------------------------------------------------------------
|
||||||
|
@ -867,7 +860,6 @@ static int ccp2_s_stream(struct v4l2_subdev *sd, int enable)
|
||||||
if (enable == ISP_PIPELINE_STREAM_STOPPED)
|
if (enable == ISP_PIPELINE_STREAM_STOPPED)
|
||||||
return 0;
|
return 0;
|
||||||
atomic_set(&ccp2->stopping, 0);
|
atomic_set(&ccp2->stopping, 0);
|
||||||
ccp2->error = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
switch (enable) {
|
switch (enable) {
|
||||||
|
|
|
@ -82,7 +82,6 @@ struct isp_ccp2_device {
|
||||||
struct isp_video video_in;
|
struct isp_video video_in;
|
||||||
struct isp_csiphy *phy;
|
struct isp_csiphy *phy;
|
||||||
struct regulator *vdds_csib;
|
struct regulator *vdds_csib;
|
||||||
unsigned int error;
|
|
||||||
enum isp_pipeline_stream_state state;
|
enum isp_pipeline_stream_state state;
|
||||||
wait_queue_head_t wait;
|
wait_queue_head_t wait;
|
||||||
atomic_t stopping;
|
atomic_t stopping;
|
||||||
|
@ -94,6 +93,6 @@ void omap3isp_ccp2_cleanup(struct isp_device *isp);
|
||||||
int omap3isp_ccp2_register_entities(struct isp_ccp2_device *ccp2,
|
int omap3isp_ccp2_register_entities(struct isp_ccp2_device *ccp2,
|
||||||
struct v4l2_device *vdev);
|
struct v4l2_device *vdev);
|
||||||
void omap3isp_ccp2_unregister_entities(struct isp_ccp2_device *ccp2);
|
void omap3isp_ccp2_unregister_entities(struct isp_ccp2_device *ccp2);
|
||||||
int omap3isp_ccp2_isr(struct isp_ccp2_device *ccp2);
|
void omap3isp_ccp2_isr(struct isp_ccp2_device *ccp2);
|
||||||
|
|
||||||
#endif /* OMAP3_ISP_CCP2_H */
|
#endif /* OMAP3_ISP_CCP2_H */
|
||||||
|
|
|
@ -667,7 +667,7 @@ static void csi2_isr_buffer(struct isp_csi2_device *csi2)
|
||||||
|
|
||||||
csi2_ctx_enable(isp, csi2, 0, 0);
|
csi2_ctx_enable(isp, csi2, 0, 0);
|
||||||
|
|
||||||
buffer = omap3isp_video_buffer_next(&csi2->video_out, 0);
|
buffer = omap3isp_video_buffer_next(&csi2->video_out);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Let video queue operation restart engine if there is an underrun
|
* Let video queue operation restart engine if there is an underrun
|
||||||
|
@ -727,17 +727,15 @@ static void csi2_isr_ctx(struct isp_csi2_device *csi2,
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* omap3isp_csi2_isr - CSI2 interrupt handling.
|
* omap3isp_csi2_isr - CSI2 interrupt handling.
|
||||||
*
|
|
||||||
* Return -EIO on Transmission error
|
|
||||||
*/
|
*/
|
||||||
int omap3isp_csi2_isr(struct isp_csi2_device *csi2)
|
void omap3isp_csi2_isr(struct isp_csi2_device *csi2)
|
||||||
{
|
{
|
||||||
|
struct isp_pipeline *pipe = to_isp_pipeline(&csi2->subdev.entity);
|
||||||
u32 csi2_irqstatus, cpxio1_irqstatus;
|
u32 csi2_irqstatus, cpxio1_irqstatus;
|
||||||
struct isp_device *isp = csi2->isp;
|
struct isp_device *isp = csi2->isp;
|
||||||
int retval = 0;
|
|
||||||
|
|
||||||
if (!csi2->available)
|
if (!csi2->available)
|
||||||
return -ENODEV;
|
return;
|
||||||
|
|
||||||
csi2_irqstatus = isp_reg_readl(isp, csi2->regs1, ISPCSI2_IRQSTATUS);
|
csi2_irqstatus = isp_reg_readl(isp, csi2->regs1, ISPCSI2_IRQSTATUS);
|
||||||
isp_reg_writel(isp, csi2_irqstatus, csi2->regs1, ISPCSI2_IRQSTATUS);
|
isp_reg_writel(isp, csi2_irqstatus, csi2->regs1, ISPCSI2_IRQSTATUS);
|
||||||
|
@ -750,7 +748,7 @@ int omap3isp_csi2_isr(struct isp_csi2_device *csi2)
|
||||||
csi2->regs1, ISPCSI2_PHY_IRQSTATUS);
|
csi2->regs1, ISPCSI2_PHY_IRQSTATUS);
|
||||||
dev_dbg(isp->dev, "CSI2: ComplexIO Error IRQ "
|
dev_dbg(isp->dev, "CSI2: ComplexIO Error IRQ "
|
||||||
"%x\n", cpxio1_irqstatus);
|
"%x\n", cpxio1_irqstatus);
|
||||||
retval = -EIO;
|
pipe->error = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (csi2_irqstatus & (ISPCSI2_IRQSTATUS_OCP_ERR_IRQ |
|
if (csi2_irqstatus & (ISPCSI2_IRQSTATUS_OCP_ERR_IRQ |
|
||||||
|
@ -775,11 +773,11 @@ int omap3isp_csi2_isr(struct isp_csi2_device *csi2)
|
||||||
ISPCSI2_IRQSTATUS_COMPLEXIO2_ERR_IRQ) ? 1 : 0,
|
ISPCSI2_IRQSTATUS_COMPLEXIO2_ERR_IRQ) ? 1 : 0,
|
||||||
(csi2_irqstatus &
|
(csi2_irqstatus &
|
||||||
ISPCSI2_IRQSTATUS_FIFO_OVF_IRQ) ? 1 : 0);
|
ISPCSI2_IRQSTATUS_FIFO_OVF_IRQ) ? 1 : 0);
|
||||||
retval = -EIO;
|
pipe->error = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (omap3isp_module_sync_is_stopping(&csi2->wait, &csi2->stopping))
|
if (omap3isp_module_sync_is_stopping(&csi2->wait, &csi2->stopping))
|
||||||
return 0;
|
return;
|
||||||
|
|
||||||
/* Successful cases */
|
/* Successful cases */
|
||||||
if (csi2_irqstatus & ISPCSI2_IRQSTATUS_CONTEXT(0))
|
if (csi2_irqstatus & ISPCSI2_IRQSTATUS_CONTEXT(0))
|
||||||
|
@ -787,8 +785,6 @@ int omap3isp_csi2_isr(struct isp_csi2_device *csi2)
|
||||||
|
|
||||||
if (csi2_irqstatus & ISPCSI2_IRQSTATUS_ECC_CORRECTION_IRQ)
|
if (csi2_irqstatus & ISPCSI2_IRQSTATUS_ECC_CORRECTION_IRQ)
|
||||||
dev_dbg(isp->dev, "CSI2: ECC correction done\n");
|
dev_dbg(isp->dev, "CSI2: ECC correction done\n");
|
||||||
|
|
||||||
return retval;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* -----------------------------------------------------------------------------
|
/* -----------------------------------------------------------------------------
|
||||||
|
|
|
@ -156,7 +156,7 @@ struct isp_csi2_device {
|
||||||
atomic_t stopping;
|
atomic_t stopping;
|
||||||
};
|
};
|
||||||
|
|
||||||
int omap3isp_csi2_isr(struct isp_csi2_device *csi2);
|
void omap3isp_csi2_isr(struct isp_csi2_device *csi2);
|
||||||
int omap3isp_csi2_reset(struct isp_csi2_device *csi2);
|
int omap3isp_csi2_reset(struct isp_csi2_device *csi2);
|
||||||
int omap3isp_csi2_init(struct isp_device *isp);
|
int omap3isp_csi2_init(struct isp_device *isp);
|
||||||
void omap3isp_csi2_cleanup(struct isp_device *isp);
|
void omap3isp_csi2_cleanup(struct isp_device *isp);
|
||||||
|
|
|
@ -1404,16 +1404,14 @@ static void preview_isr_buffer(struct isp_prev_device *prev)
|
||||||
int restart = 0;
|
int restart = 0;
|
||||||
|
|
||||||
if (prev->input == PREVIEW_INPUT_MEMORY) {
|
if (prev->input == PREVIEW_INPUT_MEMORY) {
|
||||||
buffer = omap3isp_video_buffer_next(&prev->video_in,
|
buffer = omap3isp_video_buffer_next(&prev->video_in);
|
||||||
prev->error);
|
|
||||||
if (buffer != NULL)
|
if (buffer != NULL)
|
||||||
preview_set_inaddr(prev, buffer->isp_addr);
|
preview_set_inaddr(prev, buffer->isp_addr);
|
||||||
pipe->state |= ISP_PIPELINE_IDLE_INPUT;
|
pipe->state |= ISP_PIPELINE_IDLE_INPUT;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (prev->output & PREVIEW_OUTPUT_MEMORY) {
|
if (prev->output & PREVIEW_OUTPUT_MEMORY) {
|
||||||
buffer = omap3isp_video_buffer_next(&prev->video_out,
|
buffer = omap3isp_video_buffer_next(&prev->video_out);
|
||||||
prev->error);
|
|
||||||
if (buffer != NULL) {
|
if (buffer != NULL) {
|
||||||
preview_set_outaddr(prev, buffer->isp_addr);
|
preview_set_outaddr(prev, buffer->isp_addr);
|
||||||
restart = 1;
|
restart = 1;
|
||||||
|
@ -1440,8 +1438,6 @@ static void preview_isr_buffer(struct isp_prev_device *prev)
|
||||||
default:
|
default:
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
prev->error = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -1565,7 +1561,6 @@ static int preview_set_stream(struct v4l2_subdev *sd, int enable)
|
||||||
omap3isp_subclk_enable(isp, OMAP3_ISP_SUBCLK_PREVIEW);
|
omap3isp_subclk_enable(isp, OMAP3_ISP_SUBCLK_PREVIEW);
|
||||||
preview_configure(prev);
|
preview_configure(prev);
|
||||||
atomic_set(&prev->stopping, 0);
|
atomic_set(&prev->stopping, 0);
|
||||||
prev->error = 0;
|
|
||||||
preview_print_status(prev);
|
preview_print_status(prev);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -157,7 +157,6 @@ struct isptables_update {
|
||||||
* @output: Bitmask of the active output
|
* @output: Bitmask of the active output
|
||||||
* @video_in: Input video entity
|
* @video_in: Input video entity
|
||||||
* @video_out: Output video entity
|
* @video_out: Output video entity
|
||||||
* @error: A hardware error occurred during capture
|
|
||||||
* @params: Module configuration data
|
* @params: Module configuration data
|
||||||
* @shadow_update: If set, update the hardware configured in the next interrupt
|
* @shadow_update: If set, update the hardware configured in the next interrupt
|
||||||
* @underrun: Whether the preview entity has queued buffers on the output
|
* @underrun: Whether the preview entity has queued buffers on the output
|
||||||
|
@ -179,7 +178,6 @@ struct isp_prev_device {
|
||||||
unsigned int output;
|
unsigned int output;
|
||||||
struct isp_video video_in;
|
struct isp_video video_in;
|
||||||
struct isp_video video_out;
|
struct isp_video video_out;
|
||||||
unsigned int error;
|
|
||||||
|
|
||||||
struct prev_params params;
|
struct prev_params params;
|
||||||
unsigned int shadow_update:1;
|
unsigned int shadow_update:1;
|
||||||
|
|
|
@ -1038,7 +1038,7 @@ static void resizer_isr_buffer(struct isp_res_device *res)
|
||||||
/* Complete the output buffer and, if reading from memory, the input
|
/* Complete the output buffer and, if reading from memory, the input
|
||||||
* buffer.
|
* buffer.
|
||||||
*/
|
*/
|
||||||
buffer = omap3isp_video_buffer_next(&res->video_out, res->error);
|
buffer = omap3isp_video_buffer_next(&res->video_out);
|
||||||
if (buffer != NULL) {
|
if (buffer != NULL) {
|
||||||
resizer_set_outaddr(res, buffer->isp_addr);
|
resizer_set_outaddr(res, buffer->isp_addr);
|
||||||
restart = 1;
|
restart = 1;
|
||||||
|
@ -1047,7 +1047,7 @@ static void resizer_isr_buffer(struct isp_res_device *res)
|
||||||
pipe->state |= ISP_PIPELINE_IDLE_OUTPUT;
|
pipe->state |= ISP_PIPELINE_IDLE_OUTPUT;
|
||||||
|
|
||||||
if (res->input == RESIZER_INPUT_MEMORY) {
|
if (res->input == RESIZER_INPUT_MEMORY) {
|
||||||
buffer = omap3isp_video_buffer_next(&res->video_in, 0);
|
buffer = omap3isp_video_buffer_next(&res->video_in);
|
||||||
if (buffer != NULL)
|
if (buffer != NULL)
|
||||||
resizer_set_inaddr(res, buffer->isp_addr);
|
resizer_set_inaddr(res, buffer->isp_addr);
|
||||||
pipe->state |= ISP_PIPELINE_IDLE_INPUT;
|
pipe->state |= ISP_PIPELINE_IDLE_INPUT;
|
||||||
|
@ -1064,8 +1064,6 @@ static void resizer_isr_buffer(struct isp_res_device *res)
|
||||||
if (restart)
|
if (restart)
|
||||||
resizer_enable_oneshot(res);
|
resizer_enable_oneshot(res);
|
||||||
}
|
}
|
||||||
|
|
||||||
res->error = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -1154,7 +1152,6 @@ static int resizer_set_stream(struct v4l2_subdev *sd, int enable)
|
||||||
|
|
||||||
omap3isp_subclk_enable(isp, OMAP3_ISP_SUBCLK_RESIZER);
|
omap3isp_subclk_enable(isp, OMAP3_ISP_SUBCLK_RESIZER);
|
||||||
resizer_configure(res);
|
resizer_configure(res);
|
||||||
res->error = 0;
|
|
||||||
resizer_print_status(res);
|
resizer_print_status(res);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -107,7 +107,6 @@ struct isp_res_device {
|
||||||
enum resizer_input_entity input;
|
enum resizer_input_entity input;
|
||||||
struct isp_video video_in;
|
struct isp_video video_in;
|
||||||
struct isp_video video_out;
|
struct isp_video video_out;
|
||||||
unsigned int error;
|
|
||||||
|
|
||||||
u32 addr_base; /* stored source buffer address in memory mode */
|
u32 addr_base; /* stored source buffer address in memory mode */
|
||||||
u32 crop_offset; /* additional offset for crop in memory mode */
|
u32 crop_offset; /* additional offset for crop in memory mode */
|
||||||
|
|
|
@ -580,21 +580,20 @@ static const struct isp_video_queue_operations isp_video_queue_ops = {
|
||||||
/*
|
/*
|
||||||
* omap3isp_video_buffer_next - Complete the current buffer and return the next
|
* omap3isp_video_buffer_next - Complete the current buffer and return the next
|
||||||
* @video: ISP video object
|
* @video: ISP video object
|
||||||
* @error: Whether an error occurred during capture
|
|
||||||
*
|
*
|
||||||
* Remove the current video buffer from the DMA queue and fill its timestamp,
|
* Remove the current video buffer from the DMA queue and fill its timestamp,
|
||||||
* field count and state fields before waking up its completion handler.
|
* field count and state fields before waking up its completion handler.
|
||||||
*
|
*
|
||||||
* The buffer state is set to VIDEOBUF_DONE if no error occurred (@error is 0)
|
* For capture video nodes the buffer state is set to ISP_BUF_STATE_DONE if no
|
||||||
* or VIDEOBUF_ERROR otherwise (@error is non-zero).
|
* error has been flagged in the pipeline, or to ISP_BUF_STATE_ERROR otherwise.
|
||||||
|
* For video output nodes the buffer state is always set to ISP_BUF_STATE_DONE.
|
||||||
*
|
*
|
||||||
* The DMA queue is expected to contain at least one buffer.
|
* The DMA queue is expected to contain at least one buffer.
|
||||||
*
|
*
|
||||||
* Return a pointer to the next buffer in the DMA queue, or NULL if the queue is
|
* Return a pointer to the next buffer in the DMA queue, or NULL if the queue is
|
||||||
* empty.
|
* empty.
|
||||||
*/
|
*/
|
||||||
struct isp_buffer *omap3isp_video_buffer_next(struct isp_video *video,
|
struct isp_buffer *omap3isp_video_buffer_next(struct isp_video *video)
|
||||||
unsigned int error)
|
|
||||||
{
|
{
|
||||||
struct isp_pipeline *pipe = to_isp_pipeline(&video->video.entity);
|
struct isp_pipeline *pipe = to_isp_pipeline(&video->video.entity);
|
||||||
struct isp_video_queue *queue = video->queue;
|
struct isp_video_queue *queue = video->queue;
|
||||||
|
@ -629,7 +628,13 @@ struct isp_buffer *omap3isp_video_buffer_next(struct isp_video *video,
|
||||||
else
|
else
|
||||||
buf->vbuf.sequence = atomic_read(&pipe->frame_number);
|
buf->vbuf.sequence = atomic_read(&pipe->frame_number);
|
||||||
|
|
||||||
buf->state = error ? ISP_BUF_STATE_ERROR : ISP_BUF_STATE_DONE;
|
/* Report pipeline errors to userspace on the capture device side. */
|
||||||
|
if (queue->type == V4L2_BUF_TYPE_VIDEO_CAPTURE && pipe->error) {
|
||||||
|
buf->state = ISP_BUF_STATE_ERROR;
|
||||||
|
pipe->error = false;
|
||||||
|
} else {
|
||||||
|
buf->state = ISP_BUF_STATE_DONE;
|
||||||
|
}
|
||||||
|
|
||||||
wake_up(&buf->wait);
|
wake_up(&buf->wait);
|
||||||
|
|
||||||
|
@ -1015,6 +1020,8 @@ isp_video_streamon(struct file *file, void *fh, enum v4l2_buf_type type)
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
goto error;
|
goto error;
|
||||||
|
|
||||||
|
pipe->error = false;
|
||||||
|
|
||||||
spin_lock_irqsave(&pipe->lock, flags);
|
spin_lock_irqsave(&pipe->lock, flags);
|
||||||
pipe->state &= ~ISP_PIPELINE_STREAM;
|
pipe->state &= ~ISP_PIPELINE_STREAM;
|
||||||
pipe->state |= state;
|
pipe->state |= state;
|
||||||
|
|
|
@ -85,6 +85,10 @@ enum isp_pipeline_state {
|
||||||
ISP_PIPELINE_STREAM = 64,
|
ISP_PIPELINE_STREAM = 64,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* struct isp_pipeline - An ISP hardware pipeline
|
||||||
|
* @error: A hardware error occurred during capture
|
||||||
|
*/
|
||||||
struct isp_pipeline {
|
struct isp_pipeline {
|
||||||
struct media_pipeline pipe;
|
struct media_pipeline pipe;
|
||||||
spinlock_t lock; /* Pipeline state and queue flags */
|
spinlock_t lock; /* Pipeline state and queue flags */
|
||||||
|
@ -96,6 +100,7 @@ struct isp_pipeline {
|
||||||
unsigned int max_rate;
|
unsigned int max_rate;
|
||||||
atomic_t frame_number;
|
atomic_t frame_number;
|
||||||
bool do_propagation; /* of frame number */
|
bool do_propagation; /* of frame number */
|
||||||
|
bool error;
|
||||||
struct v4l2_fract max_timeperframe;
|
struct v4l2_fract max_timeperframe;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -194,8 +199,7 @@ void omap3isp_video_cleanup(struct isp_video *video);
|
||||||
int omap3isp_video_register(struct isp_video *video,
|
int omap3isp_video_register(struct isp_video *video,
|
||||||
struct v4l2_device *vdev);
|
struct v4l2_device *vdev);
|
||||||
void omap3isp_video_unregister(struct isp_video *video);
|
void omap3isp_video_unregister(struct isp_video *video);
|
||||||
struct isp_buffer *omap3isp_video_buffer_next(struct isp_video *video,
|
struct isp_buffer *omap3isp_video_buffer_next(struct isp_video *video);
|
||||||
unsigned int error);
|
|
||||||
void omap3isp_video_resume(struct isp_video *video, int continuous);
|
void omap3isp_video_resume(struct isp_video *video, int continuous);
|
||||||
struct media_pad *omap3isp_video_remote_pad(struct isp_video *video);
|
struct media_pad *omap3isp_video_remote_pad(struct isp_video *video);
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue