OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 15220|回复: 4

关于DMA1_Channelx的问题

[复制链接]

1

主题

3

帖子

0

精华

新手入门

积分
27
金钱
27
注册时间
2013-5-9
在线时间
0 小时
发表于 2013-5-12 16:20:50 | 显示全部楼层 |阅读模式
最近在学习ADC之DMA方式采样,DMA1_Channel开始选取的为Channel4,编译没错误,但是运行时adc始终猜不到数据。
最后查其他网友的代码,DMA1_Channel选取的channel1. 改完后,可以正常猜到数据。
实在很纳闷,其他设置都没有改动过。
还请改为大大帮忙指点指点,小弟谢谢先!!!

以下是代码:
***********************************************************************************************************

/* 头文件    ------------------------------------------------------------------*/

#include "stm32f10x_lib.h"
#include "stdio.h"

/* 自定义同义关键字    --------------------------------------------------------*/

/* 自定义参数宏        --------------------------------------------------------*/
#define ADC1_DR_Address ((u32)0x4001244C)
/* 自定义函数宏        --------------------------------------------------------*/

/* 自定义全局变量      --------------------------------------------------------*/
vu32 AD_Value;    
/* 自定义函数声明      --------------------------------------------------------*/

void RCC_Configuration(void);
void GPIO_Configuration(void);
void USART_Configuration(void);
void ADC_Configuration(void); 
void DMA_Configuration(void);



/*******************************************************************************
* 函数名   : main
* 函数描述     : 主函数
* 输入参数     : 无
* 输出结果     : 无
* 返回值       : 无
*******************************************************************************/

int main(void)
{
float VolValue = 0.00; /* 转换结果,双精度浮点数 */
u32 ticks = 0; /* ADC显示延时参数 */

  /* 设置系统时钟 */
  RCC_Configuration();

  /* 设置 GPIO 端口 */
  GPIO_Configuration();
   
    /* 设置 USART */
USART_Configuration();
DMA_Configuration();
/* 设置 ADC */
ADC_Configuration();


  printf("\r\n The AD_value is:-------------------------- \r\n"); 

while(1)
{
if (ticks++ >= 2000000) 
{
ticks = 0;
VolValue = 256 * AD_Value / 0X0FFF;
// USART_SendData(USART1, 0x0c);       //清屏
        //注意,USART_SendData函数不检查是否发送完成
        //等待发送完成
        while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET);
printf( "\r\nThe current VolValue = %.2fv\r\n", VolValue/100);
}

}
}



/*******************************************************************************
* 函数名 : RCC_Configuration
* 函数描述 : 设置系统各部分时钟
* 输入参数 : 无
* 输出结果 : 无
* 返回值   : 无
*******************************************************************************/

void RCC_Configuration(void)
{
/* 定义枚举类型变量 HSEStartUpStatus */
ErrorStatus HSEStartUpStatus;

  /* 复位系统时钟设置 */
  RCC_DeInit();     /* 开启 HSE */
  RCC_HSEConfig(RCC_HSE_ON); /* 等待 HSE 起振并稳定 */
  HSEStartUpStatus = RCC_WaitForHSEStartUp(); /* 判断 HSE 起是否振成功,是则进入if()内部 */
  if(HSEStartUpStatus == SUCCESS)
  {
    /* 选择 HCLK(AHB)时钟源为SYSCLK 1分频 */
    RCC_HCLKConfig(RCC_SYSCLK_Div1);     /* 选择 PCLK2 时钟源为 HCLK(AHB)2分频 */
    RCC_PCLK2Config(RCC_HCLK_Div2);     /* 选择 PCLK1 时钟源为 HCLK(AHB)1分频 */
    RCC_PCLK1Config(RCC_HCLK_Div1);     /* 设置 FLASH 延时周期数为2 */
    FLASH_SetLatency(FLASH_Latency_2);   /* 使能 FLASH 预取缓存 */
    FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);
    /* 选择锁相环(PLL)时钟源为 HSE 1分频,倍频数为9,则PLL输出频率为 8MHz * 9 = 72MHz */
    RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);     /* 使能 PLL */ 
    RCC_PLLCmd(ENABLE);   /* 等待 PLL 输出稳定 */
    while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET); /* 选择 SYSCLK 时钟源为 PLL */
    RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK); /* 等待 PLL 成为 SYSCLK 时钟源 */
    while(RCC_GetSYSCLKSource() != 0x08);
  }
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE); /* 使能各个用到的外设时钟 */
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 | RCC_APB2Periph_GPIOA | 
  RCC_APB2Periph_ADC1 | RCC_APB2Periph_GPIOB, ENABLE);
}



/*******************************************************************************
* 函数名   : GPIO_Configuration
* 函数描述     : 设置各GPIO端口功能
* 输入参数     : 无
* 输出结果     : 无
* 返回值       : 无
*******************************************************************************/

void GPIO_Configuration(void)
{
/* 定义 GPIO 初始化结构体 GPIO_InitStructure */
  GPIO_InitTypeDef GPIO_InitStructure;

/* 设置 USART1 的Tx脚(PA.9)为第二功能推挽输出功能 */
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  GPIO_Init(GPIOA , &GPIO_InitStructure);
    
  /* 设置 USART1 的Rx脚(PA.10)为浮空输入脚 */
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
  GPIO_Init(GPIOA , &GPIO_InitStructure);

  /* 将 PB.0 设置为模拟输入脚 */
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
  GPIO_Init(GPIOB , &GPIO_InitStructure);
}



/*******************************************************************************
* 函数名   : ADC_Configuration
* 函数描述     : 初始化并启动ADC转换
* 输入参数     : 无
* 输出结果     : 无
* 返回值       : 无
*******************************************************************************/

void ADC_Configuration(void)
{
/* 定义 ADC 初始化结构体 ADC_InitStructure */
ADC_InitTypeDef ADC_InitStructure;
/* 配置ADC时钟分频 */
RCC_ADCCLKConfig(RCC_PCLK2_Div4);

/*
* 独立工作模式;
* 多通道扫描模式;
* 连续模数转换模式;
* 转换触发方式:转换由软件触发启动;
* ADC 数据右对齐 ;
* 进行规则转换的 ADC 通道的数目为1; 
*/
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
ADC_InitStructure.ADC_ScanConvMode = ENABLE;
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
ADC_InitStructure.ADC_NbrOfChannel = 1;
ADC_Init(ADC1, &ADC_InitStructure);
/* 设置 ADC1 使用8转换通道,转换顺序1,采样时间为 55.5 周期 ---channel10*/ 
ADC_RegularChannelConfig(ADC1, ADC_Channel_8, 1, ADC_SampleTime_55Cycles5);   /* 使能 ADC1 */   
ADC_Cmd(ADC1, ENABLE);

ADC_DMACmd(ADC1,ENABLE);   /* 复位 ADC1 的校准寄存器 */   
ADC_ResetCalibration(ADC1); /* 等待 ADC1 校准寄存器复位完成 */
while(ADC_GetResetCalibrationStatus(ADC1)); /* 开始 ADC1 校准 */
ADC_StartCalibration(ADC1); /* 等待 ADC1 校准完成 */
while(ADC_GetCalibrationStatus(ADC1)); /* 启动 ADC1 转换 */ 
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
}

void DMA_Configuration(void)
{
DMA_InitTypeDef DMA_InitStructure;
// DMA_DeInit(DMA1_Channel4);
DMA_DeInit(DMA1_Channel1); /*Set the register of DMA channel6 as default value*/
/* 外设地址:(u32)SRC_Const_Buffer;内存地址:(u32)DST_Buffer;外设作为数据传输的来源;
* DMA缓存大小:BufferSize;外设地址寄存器递增;内存地址寄存器递增;外设数据宽度为32位;
* 内存数据宽度为32位;CAN工作在正常缓存模式(本例中无用到);设置DMA通道优先级为高;
* DMA通道设置为内存到内存传输; */
DMA_InitStructure.DMA_PeripheralBaseAddr= ADC1_DR_Address;
DMA_InitStructure.DMA_MemoryBaseAddr=(u32)&AD_Value;
DMA_InitStructure.DMA_DIR=DMA_DIR_PeripheralSRC;
DMA_InitStructure.DMA_BufferSize=1;
DMA_InitStructure.DMA_PeripheralInc=DMA_PeripheralInc_Disable;
DMA_InitStructure.DMA_MemoryInc=DMA_MemoryInc_Disable;
DMA_InitStructure.DMA_PeripheralDataSize=DMA_PeripheralDataSize_HalfWord;
DMA_InitStructure.DMA_MemoryDataSize=DMA_MemoryDataSize_HalfWord;
DMA_InitStructure.DMA_Mode=DMA_Mode_Circular;
DMA_InitStructure.DMA_Priority=DMA_Priority_High;
DMA_InitStructure.DMA_M2M=DMA_M2M_Disable;
DMA_Init(DMA1_Channel1,&DMA_InitStructure);
DMA_Cmd(DMA1_Channel1,ENABLE);
// DMA_Init(DMA1_Channel4,&DMA_InitStructure);
// DMA_Cmd(DMA1_Channel4,ENABLE);
}

/*******************************************************************************
* 函数名   : USART_Configuration
* 函数描述     : 设置USART1
* 输入参数     : 无
* 输出结果     : 无
* 返回值       : 无
*******************************************************************************/

void USART_Configuration(void)
{
/* 定义 USART 初始化结构体 USART_InitStructure */
  USART_InitTypeDef USART_InitStructure;

/* 波特率为115200bps;
* 8位数据长度;
* 1个停止位,无校验;
* 禁用硬件流控制;
* 禁止USART时钟;
* 时钟极性低;
* 在第2个边沿捕获数据
* 最后一位数据的时钟脉冲不从 SCLK 输出; 
*/

USART_InitStructure.USART_BaudRate = 19200;
USART_InitStructure.USART_WordLength = USART_WordLength_8b;
USART_InitStructure.USART_StopBits = USART_StopBits_1;
USART_InitStructure.USART_Parity = USART_Parity_No ;
USART_InitStructure.USART_HardwareFlowControl =  USART_HardwareFlowControl_None;
USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
USART_Init(USART1 , &USART_InitStructure);
    
  /* 使能 USART1 */
  USART_Cmd(USART1 , ENABLE);
}


 
/*******************************************************************************
* 函数名   : fputc
* 函数描述     : 将printf函数重定位到USATR1
* 输入参数     : 无
* 输出结果     : 无
* 返回值 : 无
*******************************************************************************/

int fputc(int ch, FILE *f)
{
USART_SendData(USART1, (u8) ch);
while(USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
return ch;
}




正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

1

主题

3

帖子

0

精华

新手入门

积分
27
金钱
27
注册时间
2013-5-9
在线时间
0 小时
 楼主| 发表于 2013-5-12 16:48:03 | 显示全部楼层

不好意思,犯了个低级错误。:)
回复 支持 1 反对 0

使用道具 举报

5

主题

19

帖子

0

精华

初级会员

Rank: 2

积分
60
金钱
60
注册时间
2013-4-18
在线时间
0 小时
发表于 2013-5-12 20:06:46 | 显示全部楼层
是把ADC的通道和DMA弄混了
回复 支持 反对

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165540
金钱
165540
注册时间
2010-12-1
在线时间
2117 小时
发表于 2013-5-12 22:38:48 | 显示全部楼层
谢谢分享
回复 支持 反对

使用道具 举报

0

主题

14

帖子

0

精华

初级会员

Rank: 2

积分
134
金钱
134
注册时间
2018-6-20
在线时间
35 小时
发表于 2020-5-14 11:32:08 | 显示全部楼层
通道5的串口接收错误,应该对应的是串口1接收不是3,这个有点害人了
回复 支持 反对

使用道具 举报

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

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-6-28 06:04

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

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