硬汉嵌入式论坛

 找回密码
 立即注册
查看: 10184|回复: 16
收起左侧

[摄像头] STM32H7的摄像头接口DCMI配套的函数HAL_DCMI_Start_DMA非常不错,支持大分辨率直接传输

  [复制链接]

1万

主题

7万

回帖

11万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
117512
QQ
发表于 2019-3-4 03:05:46 | 显示全部楼层 |阅读模式
函数HAL_DCMI_Start_DMA封装的通用定时器DMA1,DMA2。使用这个定时器的特点是限制单轮最大65535次传输,这样使用大分辨率摄像头非常不方便。

使用这个函数的话,解决了这个问题,程序里面做了一个处理,对于需要大于65536次传输的情况,使用了双缓冲,不断动态调整地址,直到一帧传输完毕后会使能DCMI的帧中断



  1. /**
  2.   * @brief  Enables DCMI DMA request and enables DCMI capture  
  3.   * @param  hdcmi:     pointer to a DCMI_HandleTypeDef structure that contains
  4.   *                    the configuration information for DCMI.
  5.   * @param  DCMI_Mode: DCMI capture mode snapshot or continuous grab.
  6.   * @param  pData:     The destination memory Buffer address (LCD Frame buffer).
  7.   * @param  Length:    The length of capture to be transferred.
  8.   * @retval HAL status
  9.   */
  10. HAL_StatusTypeDef HAL_DCMI_Start_DMA(DCMI_HandleTypeDef* hdcmi, uint32_t DCMI_Mode, uint32_t pData, uint32_t Length)
  11. {  
  12.   /* Initialize the second memory address */
  13.   uint32_t SecondMemAddress = 0;

  14.   /* Check function parameters */
  15.   assert_param(IS_DCMI_CAPTURE_MODE(DCMI_Mode));

  16.   /* Process Locked */
  17.   __HAL_LOCK(hdcmi);
  18.   
  19.   /* Lock the DCMI peripheral state */
  20.   hdcmi->State = HAL_DCMI_STATE_BUSY;

  21.   /* Enable DCMI by setting DCMIEN bit */
  22.   __HAL_DCMI_ENABLE(hdcmi);
  23.   
  24.   /* Configure the DCMI Mode */
  25.   hdcmi->Instance->CR &= ~(DCMI_CR_CM);
  26.   hdcmi->Instance->CR |=  (uint32_t)(DCMI_Mode);

  27.   /* Set the DMA memory0 conversion complete callback */
  28.   hdcmi->DMA_Handle->XferCpltCallback = DCMI_DMAXferCplt;

  29.   /* Set the DMA error callback */
  30.   hdcmi->DMA_Handle->XferErrorCallback = DCMI_DMAError;

  31.   /* Set the dma abort callback */
  32.   hdcmi->DMA_Handle->XferAbortCallback = NULL;

  33.   /* Reset transfer counters value */
  34.   hdcmi->XferCount = 0;
  35.   hdcmi->XferTransferNumber = 0;
  36.   
  37.   if(Length <= 0xFFFF)
  38.   {
  39.     /* Enable the DMA Stream */
  40.     HAL_DMA_Start_IT(hdcmi->DMA_Handle, (uint32_t)&hdcmi->Instance->DR, (uint32_t)pData, Length);
  41.   }
  42.   else /* DCMI_DOUBLE_BUFFER Mode */
  43.   {
  44.     /* Set the DMA memory1 conversion complete callback */
  45.     hdcmi->DMA_Handle->XferM1CpltCallback = DCMI_DMAXferCplt;

  46.     /* Initialize transfer parameters */
  47.     hdcmi->XferCount = 1;
  48.     hdcmi->XferSize = Length;
  49.     hdcmi->pBuffPtr = pData;
  50.       
  51.     /* Get the number of buffer */
  52.     while(hdcmi->XferSize > 0xFFFF)
  53.     {
  54.       hdcmi->XferSize = (hdcmi->XferSize/2);
  55.       hdcmi->XferCount = hdcmi->XferCount*2;
  56.     }

  57.     /* Update DCMI counter  and transfer number*/
  58.     hdcmi->XferCount = (hdcmi->XferCount - 2);
  59.     hdcmi->XferTransferNumber = hdcmi->XferCount;

  60.     /* Update second memory address */
  61.     SecondMemAddress = (uint32_t)(pData + (4*hdcmi->XferSize));

  62.     /* Start DMA multi buffer transfer */
  63.     HAL_DMAEx_MultiBufferStart_IT(hdcmi->DMA_Handle, (uint32_t)&hdcmi->Instance->DR, (uint32_t)pData, SecondMemAddress, hdcmi->XferSize);
  64.   }
  65.   
  66.   /* Enable Capture */
  67.   hdcmi->Instance->CR |= DCMI_CR_CAPTURE;

  68.   /* Release Lock */
  69.   __HAL_UNLOCK(hdcmi);

  70.   /* Return function status */
  71.   return HAL_OK;
  72. }
复制代码


DMA传输完成中断里面的关键地方:
  1. /**
  2.   * @}
  3.   */
  4. /* Private functions ---------------------------------------------------------*/
  5. /** @defgroup DCMI_Private_Functions DCMI Private Functions
  6.   * @{
  7.   */
  8.   /**
  9.   * @brief  DMA conversion complete callback.
  10.   * @param  hdma: pointer to a DMA_HandleTypeDef structure that contains
  11.   *                the configuration information for the specified DMA module.
  12.   * @retval None
  13.   */
  14. static void DCMI_DMAXferCplt(DMA_HandleTypeDef *hdma)
  15. {
  16.   uint32_t tmp = 0;

  17.   DCMI_HandleTypeDef* hdcmi = ( DCMI_HandleTypeDef* )((DMA_HandleTypeDef* )hdma)->Parent;

  18.   if(hdcmi->XferCount != 0)
  19.   {
  20.     /* Update memory 0 address location */
  21.     tmp = ((((DMA_Stream_TypeDef *)(hdcmi->DMA_Handle->Instance))->CR) & DMA_SxCR_CT);
  22.     if(((hdcmi->XferCount % 2) == 0) && (tmp != 0))
  23.     {
  24.       tmp = ((DMA_Stream_TypeDef *)(hdcmi->DMA_Handle->Instance))->M0AR;
  25.       HAL_DMAEx_ChangeMemory(hdcmi->DMA_Handle, (tmp + (8*hdcmi->XferSize)), MEMORY0);
  26.       hdcmi->XferCount--;
  27.     }
  28.     /* Update memory 1 address location */
  29.     else if((((DMA_Stream_TypeDef *)(hdcmi->DMA_Handle->Instance))->CR & DMA_SxCR_CT) == 0)
  30.     {
  31.       tmp = ((DMA_Stream_TypeDef *)(hdcmi->DMA_Handle->Instance))->M1AR;
  32.       HAL_DMAEx_ChangeMemory(hdcmi->DMA_Handle, (tmp + (8*hdcmi->XferSize)), MEMORY1);
  33.       hdcmi->XferCount--;
  34.     }
  35.   }
  36.   /* Update memory 0 address location */
  37.   else if((((DMA_Stream_TypeDef *)(hdcmi->DMA_Handle->Instance))->CR & DMA_SxCR_CT) != 0)
  38.   {
  39.     ((DMA_Stream_TypeDef *)(hdcmi->DMA_Handle->Instance))->M0AR = hdcmi->pBuffPtr;
  40.   }
  41.   /* Update memory 1 address location */
  42.   else if((((DMA_Stream_TypeDef *)(hdcmi->DMA_Handle->Instance))->CR & DMA_SxCR_CT) == 0)
  43.   {
  44.     tmp = hdcmi->pBuffPtr;
  45.     ((DMA_Stream_TypeDef *)(hdcmi->DMA_Handle->Instance))->M1AR = (tmp + (4*hdcmi->XferSize));
  46.     hdcmi->XferCount = hdcmi->XferTransferNumber;
  47.   }

  48.   /* Check if the frame is transferred */
  49.   if(hdcmi->XferCount == hdcmi->XferTransferNumber)
  50.   {
  51.     /* Enable the Frame interrupt */
  52.     __HAL_DCMI_ENABLE_IT(hdcmi, DCMI_IT_FRAME);
  53.    
  54.     /* When snapshot mode, set dcmi state to ready */
  55.     if((hdcmi->Instance->CR & DCMI_CR_CM) == DCMI_MODE_SNAPSHOT)
  56.     {  
  57.       hdcmi->State= HAL_DCMI_STATE_READY;
  58.     }
  59.   }  
  60. }
复制代码


回复

使用道具 举报

692

主题

3576

回帖

5677

积分

论坛元老

积分
5677
发表于 2019-3-4 09:11:38 | 显示全部楼层
研究的很深啦
回复

使用道具 举报

2

主题

112

回帖

118

积分

初级会员

积分
118
发表于 2019-3-4 10:24:45 | 显示全部楼层
谢谢分享!
回复

使用道具 举报

0

主题

1

回帖

1

积分

新手上路

积分
1
发表于 2022-5-10 17:06:27 | 显示全部楼层
楼主你好,我使用STM32H723的DCMI的HAL_DCMI_Start_DMA()函数,程序会跑到 hdcmi->DMA_Handle->XferErrorCallback = DCMI_DMAError;不知道是什么问题请指教
回复

使用道具 举报

1万

主题

7万

回帖

11万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
117512
QQ
 楼主| 发表于 2022-5-11 09:41:55 | 显示全部楼层
jgmms 发表于 2022-5-10 17:06
楼主你好,我使用STM32H723的DCMI的HAL_DCMI_Start_DMA()函数,程序会跑到 hdcmi->DMA_Handle->XferError ...

直接使用我们的代码,测试是否正常。
回复

使用道具 举报

0

主题

2

回帖

2

积分

新手上路

积分
2
发表于 2024-7-11 00:52:16 来自手机 | 显示全部楼层
快照模式,调用后,dam会lock,状态也是busy,导致无法再次调用,还没想明白
回复

使用道具 举报

1万

主题

7万

回帖

11万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
117512
QQ
 楼主| 发表于 2024-7-12 08:52:42 | 显示全部楼层
2288 发表于 2024-7-11 00:52
快照模式,调用后,dam会lock,状态也是busy,导致无法再次调用,还没想明白

出现这种情况,应该是之前的传输还没正常结束,又开启了新的传输。
回复

使用道具 举报

0

主题

2

回帖

2

积分

新手上路

积分
2
发表于 2024-7-13 14:59:52 | 显示全部楼层
eric2013 发表于 2024-7-12 08:52
出现这种情况,应该是之前的传输还没正常结束,又开启了新的传输。

后面发现dma结束后,dcmi会解锁,dma不会解锁,解锁就好了,不知道HAL这样设计的目的
回复

使用道具 举报

1万

主题

7万

回帖

11万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
117512
QQ
 楼主| 发表于 2024-7-14 09:18:03 | 显示全部楼层
2288 发表于 2024-7-13 14:59
后面发现dma结束后,dcmi会解锁,dma不会解锁,解锁就好了,不知道HAL这样设计的目的

V7-031有使用这个,可供参考

https://forum.anfulai.cn/forum.p ... &extra=page%3D1
回复

使用道具 举报

18

主题

230

回帖

284

积分

高级会员

积分
284
发表于 2024-7-15 11:31:40 | 显示全部楼层
謝謝大佬回復.
回复

使用道具 举报

1

主题

7

回帖

10

积分

新手上路

积分
10
发表于 2025-3-14 02:01:33 | 显示全部楼层
大佬你好,请问我调用HAL_DCMI_Start_DMA()函数,无法传递完整数据到数组,好像传到一半突然停止了,这是为什么,我用的是NUCLEO-H743ZI开发板
回复

使用道具 举报

1万

主题

7万

回帖

11万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
117512
QQ
 楼主| 发表于 2025-3-14 08:03:14 | 显示全部楼层
dongkechao 发表于 2025-3-14 02:01
大佬你好,请问我调用HAL_DCMI_Start_DMA()函数,无法传递完整数据到数组,好像传到一半突然停止了,这是为 ...

你的程序怎么写的,方便的话,分享下代码配置看看
回复

使用道具 举报

1

主题

7

回帖

10

积分

新手上路

积分
10
发表于 2025-3-14 08:48:21 | 显示全部楼层
eric2013 发表于 2025-3-14 08:03
你的程序怎么写的,方便的话,分享下代码配置看看

我移植了正点原子的H743代码,就改了引脚

H743 OV5640.zip

2.44 MB, 下载次数: 1

回复

使用道具 举报

1

主题

7

回帖

10

积分

新手上路

积分
10
发表于 2025-3-14 09:04:22 | 显示全部楼层
/**
* @brief       DCMI底层初始化
* @param       无
* @retval      无
*/
void HAL_DCMI_MspInit(DCMI_HandleTypeDef* hdcmi)
{
    GPIO_InitTypeDef gpio_init_struct = {0};
    static DMA_HandleTypeDef dma_handle = {0};

    if (hdcmi == &g_atk_mc5640_dcmi_sta.dcmi)
    {
        /* 使能时钟 */
        ATK_MC5640_DCMI_CLK_ENABLE();
        ATK_MC5640_DCMI_DMA_CLK_ENABLE();
        ATK_MC5640_DCMI_VSYNC_GPIO_CLK_ENABLE();
        ATK_MC5640_DCMI_HSYNC_GPIO_CLK_ENABLE();
        ATK_MC5640_DCMI_PIXCLK_GPIO_CLK_ENABLE();
        ATK_MC5640_DCMI_D0_GPIO_CLK_ENABLE();
        ATK_MC5640_DCMI_D1_GPIO_CLK_ENABLE();
        ATK_MC5640_DCMI_D2_GPIO_CLK_ENABLE();
        ATK_MC5640_DCMI_D3_GPIO_CLK_ENABLE();
        ATK_MC5640_DCMI_D4_GPIO_CLK_ENABLE();
        ATK_MC5640_DCMI_D5_GPIO_CLK_ENABLE();
        ATK_MC5640_DCMI_D6_GPIO_CLK_ENABLE();
        ATK_MC5640_DCMI_D7_GPIO_CLK_ENABLE();

        /* 初始化VSYNC引脚 */
        gpio_init_struct.Pin        = ATK_MC5640_DCMI_VSYNC_GPIO_PIN;
        gpio_init_struct.Mode       = GPIO_MODE_AF_PP;
        gpio_init_struct.Pull       = GPIO_PULLUP;
        gpio_init_struct.Speed      = GPIO_SPEED_FREQ_HIGH;
        gpio_init_struct.Alternate  = ATK_MC5640_DCMI_VSYNC_GPIO_AF;
        HAL_GPIO_Init(ATK_MC5640_DCMI_VSYNC_GPIO_PORT, &gpio_init_struct);

        /* 初始化HSYNC引脚 */
        gpio_init_struct.Pin        = ATK_MC5640_DCMI_HSYNC_GPIO_PIN;
        gpio_init_struct.Mode       = GPIO_MODE_AF_PP;
        gpio_init_struct.Pull       = GPIO_PULLUP;
        gpio_init_struct.Speed      = GPIO_SPEED_FREQ_HIGH;
        gpio_init_struct.Alternate  = ATK_MC5640_DCMI_HSYNC_GPIO_AF;
        HAL_GPIO_Init(ATK_MC5640_DCMI_HSYNC_GPIO_PORT, &gpio_init_struct);

        /* 初始化PIXCLK引脚 */
        gpio_init_struct.Pin        = ATK_MC5640_DCMI_PIXCLK_GPIO_PIN;
        gpio_init_struct.Mode       = GPIO_MODE_AF_PP;
        gpio_init_struct.Pull       = GPIO_PULLUP;
        gpio_init_struct.Speed      = GPIO_SPEED_FREQ_HIGH;
        gpio_init_struct.Alternate  = ATK_MC5640_DCMI_PIXCLK_GPIO_AF;
        HAL_GPIO_Init(ATK_MC5640_DCMI_PIXCLK_GPIO_PORT, &gpio_init_struct);

        /* 初始化D0引脚 */
        gpio_init_struct.Pin        = ATK_MC5640_DCMI_D0_GPIO_PIN;
        gpio_init_struct.Mode       = GPIO_MODE_AF_PP;
        gpio_init_struct.Pull       = GPIO_PULLUP;
        gpio_init_struct.Speed      = GPIO_SPEED_FREQ_HIGH;
        gpio_init_struct.Alternate  = ATK_MC5640_DCMI_D0_GPIO_AF;
        HAL_GPIO_Init(ATK_MC5640_DCMI_D0_GPIO_PORT, &gpio_init_struct);

        /* 初始化D1引脚 */
        gpio_init_struct.Pin        = ATK_MC5640_DCMI_D1_GPIO_PIN;
        gpio_init_struct.Mode       = GPIO_MODE_AF_PP;
        gpio_init_struct.Pull       = GPIO_PULLUP;
        gpio_init_struct.Speed      = GPIO_SPEED_FREQ_HIGH;
        gpio_init_struct.Alternate  = ATK_MC5640_DCMI_D1_GPIO_AF;
        HAL_GPIO_Init(ATK_MC5640_DCMI_D1_GPIO_PORT, &gpio_init_struct);

        /* 初始化D2引脚 */
        gpio_init_struct.Pin        = ATK_MC5640_DCMI_D2_GPIO_PIN;
        gpio_init_struct.Mode       = GPIO_MODE_AF_PP;
        gpio_init_struct.Pull       = GPIO_PULLUP;
        gpio_init_struct.Speed      = GPIO_SPEED_FREQ_HIGH;
        gpio_init_struct.Alternate  = ATK_MC5640_DCMI_D2_GPIO_AF;
        HAL_GPIO_Init(ATK_MC5640_DCMI_D2_GPIO_PORT, &gpio_init_struct);

        /* 初始化D3引脚 */
        gpio_init_struct.Pin        = ATK_MC5640_DCMI_D3_GPIO_PIN;
        gpio_init_struct.Mode       = GPIO_MODE_AF_PP;
        gpio_init_struct.Pull       = GPIO_PULLUP;
        gpio_init_struct.Speed      = GPIO_SPEED_FREQ_HIGH;
        gpio_init_struct.Alternate  = ATK_MC5640_DCMI_D3_GPIO_AF;
        HAL_GPIO_Init(ATK_MC5640_DCMI_D3_GPIO_PORT, &gpio_init_struct);

        /* 初始化D4引脚 */
        gpio_init_struct.Pin        = ATK_MC5640_DCMI_D4_GPIO_PIN;
        gpio_init_struct.Mode       = GPIO_MODE_AF_PP;
        gpio_init_struct.Pull       = GPIO_PULLUP;
        gpio_init_struct.Speed      = GPIO_SPEED_FREQ_HIGH;
        gpio_init_struct.Alternate  = ATK_MC5640_DCMI_D4_GPIO_AF;
        HAL_GPIO_Init(ATK_MC5640_DCMI_D4_GPIO_PORT, &gpio_init_struct);

        /* 初始化D5引脚 */
        gpio_init_struct.Pin        = ATK_MC5640_DCMI_D5_GPIO_PIN;
        gpio_init_struct.Mode       = GPIO_MODE_AF_PP;
        gpio_init_struct.Pull       = GPIO_PULLUP;
        gpio_init_struct.Speed      = GPIO_SPEED_FREQ_HIGH;
        gpio_init_struct.Alternate  = ATK_MC5640_DCMI_D5_GPIO_AF;
        HAL_GPIO_Init(ATK_MC5640_DCMI_D5_GPIO_PORT, &gpio_init_struct);

        /* 初始化D6引脚 */
        gpio_init_struct.Pin        = ATK_MC5640_DCMI_D6_GPIO_PIN;
        gpio_init_struct.Mode       = GPIO_MODE_AF_PP;
        gpio_init_struct.Pull       = GPIO_PULLUP;
        gpio_init_struct.Speed      = GPIO_SPEED_FREQ_HIGH;
        gpio_init_struct.Alternate  = ATK_MC5640_DCMI_D6_GPIO_AF;
        HAL_GPIO_Init(ATK_MC5640_DCMI_D6_GPIO_PORT, &gpio_init_struct);

        /* 初始化D7引脚 */
        gpio_init_struct.Pin        = ATK_MC5640_DCMI_D7_GPIO_PIN;
        gpio_init_struct.Mode       = GPIO_MODE_AF_PP;
        gpio_init_struct.Pull       = GPIO_PULLUP;
        gpio_init_struct.Speed      = GPIO_SPEED_FREQ_HIGH;
        gpio_init_struct.Alternate  = ATK_MC5640_DCMI_D7_GPIO_AF;
        HAL_GPIO_Init(ATK_MC5640_DCMI_D7_GPIO_PORT, &gpio_init_struct);

        /* 初始化DMA */
        dma_handle.Instance                 = ATK_MC5640_DCMI_DMA_INTERFACE;
        dma_handle.Init.Request             = DMA_REQUEST_DCMI;
        dma_handle.Init.Direction           = DMA_PERIPH_TO_MEMORY;
        dma_handle.Init.PeriphInc           = DMA_PINC_DISABLE;
        dma_handle.Init.MemInc              = DMA_MINC_DISABLE;
        dma_handle.Init.PeriphDataAlignment = DMA_PDATAALIGN_WORD;
        dma_handle.Init.MemDataAlignment    = DMA_MDATAALIGN_HALFWORD;
        dma_handle.Init.Mode                = DMA_CIRCULAR;
        dma_handle.Init.Priority            = DMA_PRIORITY_HIGH;
        dma_handle.Init.FIFOMode            = DMA_FIFOMODE_ENABLE;
        dma_handle.Init.FIFOThreshold       = DMA_FIFO_THRESHOLD_HALFFULL;
        dma_handle.Init.MemBurst            = DMA_MBURST_SINGLE;
        dma_handle.Init.PeriphBurst         = DMA_PBURST_SINGLE;
        __HAL_LINKDMA(hdcmi, DMA_Handle, dma_handle);
        HAL_DMA_Init(&dma_handle);

        /* 配置DCMI中断 */
        HAL_NVIC_SetPriority(ATK_MC5640_DCMI_IRQn, 2, 2);
        HAL_NVIC_EnableIRQ(ATK_MC5640_DCMI_IRQn);

        /* 配置DMA中断 */
        HAL_NVIC_SetPriority(ATK_MC5640_DCMI_DMA_IRQn, 2, 3);
        HAL_NVIC_EnableIRQ(ATK_MC5640_DCMI_DMA_IRQn);
    }
}

/**
* @brief       初始化ATK-MC5640模块DCMI接口
* @param       无
* @retval      无
*/
void atk_mc5640_dcmi_init(void)
{
    g_atk_mc5640_dcmi_sta.dcmi.Instance                 = ATK_MC5640_DCMI_INTERFACE;
    g_atk_mc5640_dcmi_sta.dcmi.Init.SynchroMode         = DCMI_SYNCHRO_HARDWARE;
    g_atk_mc5640_dcmi_sta.dcmi.Init.PCKPolarity         = DCMI_PCKPOLARITY_RISING;
    g_atk_mc5640_dcmi_sta.dcmi.Init.VSPolarity          = DCMI_VSPOLARITY_LOW;
    g_atk_mc5640_dcmi_sta.dcmi.Init.HSPolarity          = DCMI_HSPOLARITY_LOW;
    g_atk_mc5640_dcmi_sta.dcmi.Init.CaptureRate         = DCMI_CR_ALL_FRAME;
    g_atk_mc5640_dcmi_sta.dcmi.Init.ExtendedDataMode    = DCMI_EXTEND_DATA_8B;
    g_atk_mc5640_dcmi_sta.dcmi.Init.JPEGMode            = DCMI_JPEG_DISABLE;
    HAL_DCMI_Init(&g_atk_mc5640_dcmi_sta.dcmi);
}

/**
* @brief       开始ATK-MC5640模块DCMI接口DMA传输
* @param       dts_addr        : 帧数据的接收目的地址
*              meminc          : DMA_MINC_DISABLE: 帧数据接收的目的地址自动增加
*                                DMA_MINC_ENABLE : 帧数据接收的目的地址不自动增加
*              memdataalignment: DMA_MDATAALIGN_BYTE    : 帧数据接收缓冲的位宽为8比特
*                                DMA_MDATAALIGN_HALFWORD: 帧数据接收缓冲的位宽为16比特
*                                DMA_MDATAALIGN_WORD    : 帧数据接收缓冲的位宽为32比特
*              len             : 传输的帧数据大小
* @retval      无
*/
void atk_mc5640_dcmi_start(uint32_t dts_addr, uint32_t meminc, uint32_t memdataalignment, uint32_t len)
{
//    /* 根据需求重新配置DMA */
    g_atk_mc5640_dcmi_sta.dcmi.DMA_Handle->Init.MemInc = meminc;
    g_atk_mc5640_dcmi_sta.dcmi.DMA_Handle->Init.MemDataAlignment = memdataalignment;
    HAL_DMA_Init(g_atk_mc5640_dcmi_sta.dcmi.DMA_Handle);

    /* 清空帧接收完成标记
     * 使能DCMI帧接收中断
     */
    g_atk_mc5640_dcmi_sta.frame_sem = 0;
    __HAL_DCMI_ENABLE_IT(&g_atk_mc5640_dcmi_sta.dcmi, DCMI_IT_FRAME);
    HAL_DCMI_Start_DMA(&g_atk_mc5640_dcmi_sta.dcmi, DCMI_MODE_SNAPSHOT, dts_addr, len);

    /* 等待传输完成 */
    while (g_atk_mc5640_dcmi_sta.frame_sem == 0);
    HAL_DCMI_Stop(&g_atk_mc5640_dcmi_sta.dcmi);
}
这是我DCMI和DMA的配置,摄像头的寄存器配置都做好了,不知道为什么DMA传输会突然停止,数据寄存器也无法更新
屏幕截图 2025-03-14 004834.png
附上源码 H743 OV5640.zip (2.44 MB, 下载次数: 0)



回复

使用道具 举报

1万

主题

7万

回帖

11万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
117512
QQ
 楼主| 发表于 2025-3-15 08:43:08 | 显示全部楼层
dongkechao 发表于 2025-3-14 09:04
/**
* @brief       DCMI底层初始化
* @param       无

你的这这个是配置DMA循环模式,传输完一帧后HAL_DCMI_Stop(&g_atk_mc5640_dcmi_sta.dcmi)终止了。

下次再用,再启动。
回复

使用道具 举报

0

主题

2

回帖

2

积分

新手上路

积分
2
发表于 6 天前 | 显示全部楼层
大佬你好,请问我启用dma双缓冲HAL_DMAEx_MultiBufferStart_IT(&hdma_dcmi,(uint32_t)&DCMI->DR,(uint32_t)dcmi_line_buf[0],(uint32_t)dcmi_line_buf[1],480);,但是目前传输完成中断只进入了不到一半大概才五百次,我看了行中断是1200次没有问题,一直不知道没找到是什么原因,卡了好几天了
回复

使用道具 举报

1万

主题

7万

回帖

11万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
117512
QQ
 楼主| 发表于 5 天前 | 显示全部楼层
suga116 发表于 2025-8-5 09:53
大佬你好,请问我启用dma双缓冲HAL_DMAEx_MultiBufferStart_IT(&hdma_dcmi,(uint32_t)&DCMI->DR,(uint32_t) ...

这个函数要是有问题的话,可以不使用这种方式的双缓存,使用半传输完成中断+传输完成中断实现双缓冲
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|小黑屋|Archiver|手机版|硬汉嵌入式论坛

GMT+8, 2025-8-11 20:58 , Processed in 0.050075 second(s), 27 queries .

Powered by Discuz! X3.4 Licensed

Copyright © 2001-2023, Tencent Cloud.

快速回复 返回顶部 返回列表