Linux下DMA添加两个channel,关于DMA1_Channelx的问题

最近在学习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;

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值