STM32F303RE 四个ADC同步规则采样

STM32F303RE 芯片有4个ADC, 采样频率据说能达到5Mbsps 已经算是非常高的了.
比较适合做采样

参考文章
STM32三个ADC同步规则采样

参考上面的配置文章. 经过不断的失败和重复尝试, 最后终于搞通了.
其中遇到了好几个错误, 也许是 STM32F303RE 与其它的不一样, 也许是我用的固件库库版本不一样… 我目前用的是
STM32Cube FW_F3 V1.11.3 这个版本的 Firmware Package, 我的开发工具是 STM32CubeIDE

第一步配置ADC1
在这里插入图片描述
在这里插入图片描述
配置ADC1 的 DMA 这里一定要选择 Word, 为啥一定要是Word?
DMA配置
我也是被这个问题困扰了好久. 经过我一番好找, 查边各种文章, 最后终于搞明白了.

答案是, 在同步ADC模式中, ADC1是主ADC, ADC2是从ADC, 当模式为 [同步规则模式] 时 数据是存放在 ADC1_2 的 CDR 寄存器中的.
在这里插入图片描述
我是怎么知道的? 这是花了好长时间从寄存器参考手册中查到的.

Dual ADC modes
In devices with two ADCs or more, dual ADC modes can be used (see Figure 97):
• ADC1 and ADC2 can be used together in dual mode (ADC1 is master)
• ADC3 and ADC4 can be used together in dual mode (ADC3 is master)

译文如下:
双ADC模式
在有两个或多个ADC的设备中,可以使用双ADC模式(见图97):
•ADC1和ADC2双模式同时使用(ADC1为主)
•ADC3和ADC4双模式同时使用(ADC3为主)

In dual ADC mode, the converted data of the master and slave ADC can be read in parallel,
by reading the ADC common data register (ADCx_CDR). The status bits can be also read in
parallel by reading the dual-mode status register (ADCx_CSR).

译文如下
在双ADC模式下,通过读取ADC公共数据寄存器(ADCx_CDR),可以并行读取主、从ADC的转换数据。状态位也可以通过读取双模式状态寄存器(ADCx_CSR)来并行读取。

这里额外插一句, 我觉得 , 学习STM32 需要掌握的知识除了c语言之外,还需要手边放一个 [寄存器级别的 参考手册], 这个参考手册是芯片设计人员写给程序开发人员看的, 里面非常非常详细的介绍了每个寄存器的作用, 甚至还写了使用的注意事项和方法. 能看懂这个手册基本上就会用stm32了… 看起来也不是很难. stm32 参考手册好像还有中文版. 看参考手册要注意芯片型号要对应, 别看了半天最后发现型号都不一样的. 这个手册是可以从http://www.st.com 官方下载的. 怎么下载就不展开讲了.

另外这里要分享一个好东西, STM32CubeIDE 有寄存器查看窗口, 好像Kiel 也有. 能够查看寄存器的当前值, 基本上再根据寄存器的参考手册就知道当前的机器内部运行情况 . 而不是瞎猜了.
在这里插入图片描述
回到正题.
配置ADC2, 基本上跟ADC1一致,有几个选型因为ADC1选好之后ADC2就不能选了, 有些已经默认选好了.
在这里插入图片描述
选好通道, 再添加DMA
在这里插入图片描述

下面就是主代码


#define MAX_REC_COUNT  100//可随根据实际使用情况进行调整
#define MAX_ADC_COUNT  2000//可随根据实际使用情况进行调整

volatile uint16_t Adc12_Values[MAX_ADC_COUNT*2]; // 这里是 uint16_t  类型的,下面用的时候是转换成 (uint32_t *) 的指针
volatile uint16_t Adc34_Values[MAX_ADC_COUNT*2]; // 这里是 uint16_t  类型的,下面用的时候是转换成 (uint32_t *) 的指针
int main(void)
{
  /* USER CODE BEGIN 1 */
  /* USER CODE END 1 */

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */
  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */
  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_DMA_Init();
  MX_USART2_UART_Init();
  MX_ADC1_Init();
  MX_ADC2_Init();
  MX_ADC3_Init();
  MX_ADC4_Init();
  MX_TIM4_Init();
  /* USER CODE BEGIN 2 */
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */

  //开启DMA接收命令
  //HAL_UART_Receive_DMA(&huart2, ReceiveBuffer, 10);

	HAL_ADCEx_Calibration_Start(&hadc1, ADC_SINGLE_ENDED); //校准
	HAL_ADCEx_Calibration_Start(&hadc2, ADC_SINGLE_ENDED); //校准
	HAL_ADCEx_Calibration_Start(&hadc3, ADC_SINGLE_ENDED); //校准
	HAL_ADCEx_Calibration_Start(&hadc4, ADC_SINGLE_ENDED); //校准


	// 启动adc dma模式下面这几句就够了参考  https://www.stm32cube.com/article/212,
	HAL_StatusTypeDef state = HAL_OK;

	//	HAL_ADC_Start(&hadc1); //不能启动主ADC, 如果启动了, 后面的 HAL_ADCEx_MultiModeStart_DMA 就不能正常启动了.
		HAL_ADC_Start(&hadc2);
	//	HAL_ADC_Start(&hadc3); //不能启动主ADC, 如果启动了, 后面的 HAL_ADCEx_MultiModeStart_DMA 就不能正常启动了.
		HAL_ADC_Start(&hadc4);//启动ADC

	 state = HAL_ADCEx_MultiModeStart_DMA(&hadc1, (uint32_t*)Adc12_Values, MAX_ADC_COUNT);//启动同步规则模式及DMA,4个ADC 
     state = HAL_ADCEx_MultiModeStart_DMA(&hadc3, (uint32_t*)Adc34_Values, MAX_ADC_COUNT);//启动同步规则模式及DMA,4个ADC 

//	 state = HAL_ADC_Start_DMA(&hadc1, (uint32_t*)Adc1_4_Values[0], MAX_ADC_COUNT);  //这个不需要, 也不行.
//	 state = HAL_ADC_Start_DMA(&hadc3, (uint32_t*)Adc1_4_Values[1], MAX_ADC_COUNT);  //这个不需要, 也不行.

// 这里调试 state 的状态可以方便知道前面的配置是否正常..如果不是HAL_OK可以调试跟踪进去,看看是哪一步代码出错了.
    if(state == HAL_OK){
			int a=0;
	}


  HAL_TIM_Base_Start(&htim4);//启动TIM2,用于触发ADC采样
 
  HAL_UARTEx_ReceiveToIdle_DMA(&huart2, ReceiveBuffer, MAX_REC_COUNT);


  while (1)
  {
             /* USER CODE END WHILE */
	        //HAL_UART_Transmit(&huart2, (uint8_t *)SendBuffer, sizeof(SendBuffer),10);
//	  	   if(DMA_Flag == 1)//AD转换完成标准
//	        {
//	             DMA_Flag = 0;
//	             HAL_TIM_Base_Stop(&htim4);
//	             HAL_ADCEx_MultiModeStop_DMA(&hadc1);
//	             HAL_ADC_Stop(&hadc1);//停止采样
//	             HAL_ADC_Stop(&hadc2);//停止采样
//	             HAL_ADC_Stop(&hadc3);//停止采样
//	             HAL_ADC_Stop(&hadc4);//停止采样
//
//				 for(uint8_t i=0;i<100;i++)
//				 {
						sprintf(str,"$,%d,%d,%d",Adc_Values[i][0],Adc_Values[i][1],Adc_Values[i][2]);
						HAL_UART_Transmit(&huart8, (uint8_t*)str,strlen(str),100);//串口发送到上位机
//				 }
				 HAL_ADC_Start(&hadc1);
//				 HAL_ADC_Start(&hadc2);
				 HAL_ADC_Start(&hadc3);
//				 HAL_ADC_Start(&hadc4);
//
//				 HAL_ADCEx_MultiModeStart_DMA(&hadc1,  (uint32_t*)Adc12_Values,  MAX_REC_COUNT/2);
//				 HAL_ADCEx_MultiModeStart_DMA(&hadc3,  (uint32_t*)Adc34_Values,  MAX_REC_COUNT/2);
//
//				 HAL_TIM_Base_Start(&htim4);//重新启动采样
//			//	  ————————————————
//			//	  版权声明:本文为CSDN博主「天外飞仙CUG」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
//			//	  原文链接:https://blog.csdn.net/zhang062061/article/details/113735874
//	        }

			  HAL_UART_Transmit_DMA(&huart2, (uint8_t *)SendBuffer, sizeof(SendBuffer));//DMA model
			 // HAL_UART_Transmit_DMA(&huart2, (uint8_t *)adc_vals, sizeof(SendBuffer));//DMA model

			  HAL_Delay(1000);

    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}

ADC时钟的配置, 需要自己根据项目需求更改合适的数值. 前面的ADC时钟源要选这个Timer… 注意Trigger Event Selection TRGO 要选择 Update Event

在这里插入图片描述

注意事项:
前面的参考文章中并未说明, HAL_ADCEx_MultiModeStart_DMA 要求, 主ADC不能先启动, 只能先启动从属ADC2和ADC4,

//				 HAL_ADC_Start(&hadc1);  // 这个不能启动, 否则下面的HAL_ADCEx_MultiModeStart_DMA(&hadc1, ..)  就不能正常启动.
				 HAL_ADC_Start(&hadc2);
//				 HAL_ADC_Start(&hadc3);  // 这个不能启动, 否则下面的HAL_ADCEx_MultiModeStart_DMA(&hadc3, ..)  就不能正常启动.
 			     HAL_ADC_Start(&hadc4);
 
 				 HAL_ADCEx_MultiModeStart_DMA(&hadc1,  (uint32_t*)Adc12_Values,  MAX_REC_COUNT/2);
 				 HAL_ADCEx_MultiModeStart_DMA(&hadc3,  (uint32_t*)Adc34_Values,  MAX_REC_COUNT/2);

我操这点功能搞了2天…

  • 6
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 8
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 8
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值