[STM32F1] STM32的ADC配合DMA为什么只转换一半!

[复制链接]
 楼主| woshiaokeman 发表于 2014-4-20 09:38 | 显示全部楼层 |阅读模式
首先说明,我这里用的是ADC的通道14,DMA转换次数16,结果只转换了8次(一半),即:存放数组前8个元素有数据,后8个没数据。

代码如下:
主函数(查询方式,串口打印部分省略。。):
  1. #ifndef __ADC_DMA_H__
  2. #define __ADC_DMA_H__

  3. #include "stm32f10x.h"

  4. #define ADC_DMA_SIZE 16

  5. #define ADC_DR_Address ((uint32_t)0x4001244C)

  6. void ADC_Config(void);
  7. void ADC_DMA_Config(void);
  8. void ADC_DMA_Standard_Init(void);
  9. extern void ADC_DMA_Init(void);
  10. extern uint16_t *Read_Convert_Buf(uint16_t *dst);

  11. #endif

功能实现:
  1. void ADC_DMA_Standard_Init(void)
  2. {
  3. GPIO_InitTypeDef GPIO_InitStructure;
  4. ADC_InitTypeDef ADC_InitStructure;
  5. DMA_InitTypeDef DMA_InitStructure;

  6. /* Enable peripheral clocks --------------------------------------------------*/
  7. /* Enable DMA1 clock */
  8. RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);

  9. /* Enable ADC1 and GPIOC clock */
  10. RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 | RCC_APB2Periph_GPIOC, ENABLE);

  11. /* Configure PC.04 (ADC Channel14) as analog input -------------------------*/
  12. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;
  13. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
  14. GPIO_Init(GPIOC, &GPIO_InitStructure);

  15. DMA_DeInit(DMA1_Channel1);
  16. DMA_InitStructure.DMA_PeripheralBaseAddr = ADC_DR_Address;
  17. DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)ConvertBuf;
  18. DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
  19. DMA_InitStructure.DMA_BufferSize = ADC_DMA_SIZE;
  20. DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
  21. DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
  22. DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
  23. DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
  24. DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;
  25. DMA_InitStructure.DMA_Priority = DMA_Priority_High;
  26. DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
  27. DMA_Init(DMA1_Channel1, &DMA_InitStructure);

  28. /* Enable DMA1 channel1 */
  29. DMA_Cmd(DMA1_Channel1, ENABLE);

  30. /* ADC1 configuration ------------------------------------------------------*/
  31. ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
  32. ADC_InitStructure.ADC_ScanConvMode = DISABLE;
  33. ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
  34. ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
  35. ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
  36. ADC_InitStructure.ADC_NbrOfChannel = 1;
  37. ADC_Init(ADC1, &ADC_InitStructure);

  38. /* ADC1 regular channel14 configuration */
  39. ADC_RegularChannelConfig(ADC1, ADC_Channel_14, 1, ADC_SampleTime_55Cycles5);

  40. /* Enable ADC1 DMA */
  41. ADC_DMACmd(ADC1, ENABLE);

  42. /* Enable ADC1 */
  43. ADC_Cmd(ADC1, ENABLE);

  44. /* Enable ADC1 reset calibaration register */
  45. ADC_ResetCalibration(ADC1);
  46. /* Check the end of ADC1 reset calibration register */
  47. while(ADC_GetResetCalibrationStatus(ADC1));

  48. /* Start ADC1 calibaration */
  49. ADC_StartCalibration(ADC1);
  50. /* Check the end of ADC1 calibration */
  51. while(ADC_GetCalibrationStatus(ADC1));

  52. /* Start ADC1 Software Conversion */
  53. ADC_SoftwareStartConvCmd(ADC1, ENABLE);
  54. }

  55. void ADC_DMA_Init(void)
  56. {
  57. ADC_DMA_Standard_Init();

  58. printf("ADC_DMA_Init ok.\n");
  59. }

  60. uint16_t *Read_Convert_Buf(uint16_t *dst)
  61. {
  62. if (SET == DMA_GetFlagStatus(DMA1_IT_TC1))
  63. {
  64. ADC_Cmd(ADC1, DISABLE);
  65. ADC_DMACmd(ADC1, DISABLE);
  66. DMA_Cmd(DMA1_Channel1, DISABLE);
  67. memset(dst, 0, ADC_DMA_SIZE);
  68. memcpy(dst, ConvertBuf, ADC_DMA_SIZE);
  69. memset(ConvertBuf, 0, ADC_DMA_SIZE);
  70. DMA1_Channel1->CNDTR = ADC_DMA_SIZE;
  71. DMA_ClearFlag(DMA1_IT_TC1);
  72. //DMA_Cmd(DMA1_Channel1, ENABLE);
  73. return dst;
  74. }
  75. return NULL;
  76. }

有遇到过这种问题的朋友可以帮我解决一下么!?
再次说明一下打印出来的结果:
-0:988-
-1:988-
-2:988-
...
-7:988-
-8:0-
...
-15:0-
前8个元素数据正确,后8个元素没有数据!
airwill 发表于 2014-4-20 12:54 | 显示全部楼层
我看不是 ADC 的问题. 而是在这里.
memcpy(dst, ConvertBuf, ADC_DMA_SIZE);

ADC_DMA_SIZE =16, ADC 结果有 32 个字节, 而你的 memcpy 只传了 16 个字节!
 楼主| woshiaokeman 发表于 2014-4-20 17:43 | 显示全部楼层
airwill 发表于 2014-4-20 12:54
我看不是 ADC 的问题. 而是在这里.
memcpy(dst, ConvertBuf, ADC_DMA_SIZE);

感谢版主的指导,那么如果要实现uint16_t src[128]内容复制到uint16_t dst[128]应该用什么函数?还是自己重新写?
还有就是如果用memcpy实现的话复制的结果应该是memcpy(dst, src, sizeof(uint16_t)*数组长度)?
 楼主| woshiaokeman 发表于 2014-4-20 19:46 | 显示全部楼层
airwill 发表于 2014-4-20 12:54
我看不是 ADC 的问题. 而是在这里.
memcpy(dst, ConvertBuf, ADC_DMA_SIZE);

感谢版主,确实是这样~
搞定了,谢谢~
玄德 发表于 2014-4-20 20:25 | 显示全部楼层
版主真熟悉,佩服!
您需要登录后才可以回帖 登录 | 注册

本版积分规则

2

主题

58

帖子

0

粉丝
快速回复 在线客服 返回列表 返回顶部