同指向双指针在STM32 DMA访问中的一种应用
这里的同指向双指针,是指定义了两个指针变量,而赋值了同样的地址。一般场景下,定义一个指针变量,根据要访问的对象(变量,结构体等)的变化,将其地址动态赋值给指针,再通过指针寻址访问,能满足大部分的应用场景,并不需要采用两个指针变量指向同一个地址。
问题的起源
在设计通过OV系列摄像头获取数据的编程中,通过STM32的DCMI接口(DMA方式)获得数据并由DMA放置到目的地址,目的地址是一个赋值了内存地址的指针。当获取了部分数据,需要将数据及时转存及处理,因为DMA一侧是循环进行,目的地址缓冲区放满会从头开始覆盖放置。如STM32H750 DCMI DMA使用:
uint32_t * dcmi_data_buff;
uint8_t * compute_buff;
dcmi_data_buff = 0x30000000;
compute_buff = 0x38000000;
dcmi_dma_status = HAL_DCMI_Start_DMA(&hdcmi, DCMI_MODE_CONTINUOUS, dcmi_data_buff, 2560*100/4);
memcpy(compute_buff, dcmi_data_buff, 2560);
在初始的设计时,用memcpy()函数进行数据区拷贝转存,dcmi_data_buff为源地址指针,然而此时MCU会跑飞。经过一些分析测试,发现另外定义一个指针,指向dcmi_data_buff相同的地址,然后在memcpy()函数里用这个指针作为拷贝的源地址,则能够正常工作。
原因分析
从MCU机制上分析, 在设置DCMI的DMA的目的地址时,所用的指针,已经被MCU"锁定”。在DMA运行期间,成为了不能访问的变量,所以另外的程序部分,要去访问这个指针,会产生”越界冲突“,但并不是这个指针里面放的32位地址对应的数据区域不可访问。因此当第二个指针里放置的地址相同,而通过第二个指针获得操作地址,则可以解决冲突问题。如:
uint32_t * dcmi_data_buff;
uint8_t* PY_DataW_Seg;
uint8_t * compute_buff;
dcmi_data_buff = 0x30000000;
PY_DataW_Seg = 0x30000000;
compute_buff = 0x38000000;
dcmi_dma_status = HAL_DCMI_Start_DMA(&hdcmi, DCMI_MODE_CONTINUOUS, dcmi_data_buff, 2560*100/4);
memcpy(compute_buff, PY_DataW_Seg, 2560);
–End–
|