求助,我根据论坛上的帖子修给了一下,用Timer2的CH2来触发ADC1,也就是图中的ICG,同时用DMA和串口通信,但不知道哪里出错了,串口助手就是没动静、
#include "stm32f10x_lib.h"
#include "sys.h"
#include "delay.h"
#include "led.h"
#define USART1_DR_Base 0x40013804
#define ADC1_DR_Address 0x4001244C //0x40012400+0x4C
#define KEY0 PAin(13) //PA13按键0
//#define DMA1_MEM_ADD (u32)ADC_Buf
//#define DMA1_MEM_SIZE (u32)sizeof(ADC_Buf)
u16 ADC_Buf[714];
u16 times=0;
void MYDMA_Config(DMA_Channel_TypeDef*DMA_CHx);//配置DMA1_CHx
void MYDMA_Enable(DMA_Channel_TypeDef*DMA_CHx);//使能DMA1_CHx
void MYDMA_GOnes(DMA_Channel_TypeDef*DMA_CHx); //执行一次DMA
void adc_init(void) ;
void USART_Initaize(u32 pclk2,u32 bound);
void Uart1_PutChar(u8 ch);
void Uart1_PutString(u8 *Buf, u8 Len);
void PWM_Init(u16 arr,u16 psc);
void PWM4_PULSE_Init(u16 arr1,u16 psc1,u16 arr2,u16 psc2);
void KEY_Init(void);
u8 KEY_Scan(void);
//主函数的内容:
int main(void)
{
u16 t;
u16 m;
u32 temp =0;
u16 adcx;
u8 table[4];
u8 *p;
p= table;
Stm32_Clock_Init(9);//72M
delay_init(72); //延时初始化
   WM_Init(18,1); //Timer3 CH3 PB0 主时钟
PWM4_PULSE_Init(1000,1440,12,125); // Timer4 CH1 时钟为100K SH PB6, Timer2 CH2 PA1 ICG
USART_Initaize(72,9600); //串口初始化
while (1)
{
t=KEY_Scan();//得到键值
if(m)
{
switch(m)
{
case 1:
adc_init();
break;
}
}
if(DMA1->ISR&(1<<1)) //传输完成了
{
for(t=0;t<715;t++)
{
temp = ADC_Buf[t];
temp = temp /5;
adcx=temp*330/4096;;
table[0]=adcx / 100 + 0x30;
table[1]='.';
table[2]=adcx % 100/10 + 0x30;
table[3]= adcx %10 + 0x30;
Uart1_PutString(p,4);
Uart1_PutString("\r\n",2);
temp =0;
DMA1->IFCR|=1<<1;
MYDMA_GOnes(DMA1_Channel1);
}
}
}
}
void adc_init(void)
{
RCC->APB2ENR|=1<<2; //使能PORTA口时钟
GPIOA->CRL&=0XFFFFFFF0;//PA.0 anolog输入
//通道10/11设置
RCC->APB2ENR|=1<<9; //ADC1时钟使能
__nop();
__nop();
RCC->APB2RSTR|=1<<9; //ADC1复位
RCC->APB2RSTR&=~(1<<9);//复位结束
RCC->CFGR|=2<<14; //SYSCLK/DIV2=72M/6=12Mhz ADCCLK12MHz
ADC1->CR1&=0XF0FFFF; //独立工作模式
ADC1->CR1|=1<<8; //扫描模式
ADC1->CR2|=0<<1; //单次转换模式
ADC1->CR2|=3<<17; //定时器2 CC2事件控制规则通道转换
ADC1->CR2|=1<<20; //用一个事件来触发
ADC1->CR2&=~(1<<11); //右对齐
ADC1->SQR1&=0xFFF0FFFF;//1个转换在规则序列中
ADC1->SQR3&=0XFFFFFFE0;//规则序列1=通道0
ADC1->SQR3|=0;
ADC1->SMPR2|=0X07; //通道0的转换时间为:239.5+12.5个ADC时钟周期
ADC1->CR2|=1<<0; //开启AD转换器,第一次唤醒AD转换器
ADC1->CR2|=1<<3; //使能复位校准
while(ADC1->CR2&1<<3); //等待校准结束
//该位由软件设置并由硬件清除。在校准寄存器被初始化后该位将被清除。
ADC1->CR2|=1<<2; //开启AD校准
while(ADC1->CR2&1<<2); //等待校准结束
//该位由软件设置以开始校准,并在校准结束时由硬件清除
ADC1->CR2|=1<<8; //开启ADC DMA转换
MYDMA_Config(DMA1_Channel1);//配置DMA通道1
MYDMA_Enable(DMA1_Channel1);//开启DMA通道1
ADC1->CR2|=1<<0; //开启AD转换
}
//获得ADC值
u16 get_adc(void)
{
ADC1->CR2|=1<<22; //启动规则转换通道
while(!(ADC1->SR&1<<1));//等待转换结束
return ADC1->DR; //返回adc值
}
//DMA1的各通道配置
//DMA_CHx  MA1的通道 参考手册
void MYDMA_Config(DMA_Channel_TypeDef*DMA_CHx)
{
RCC->AHBENR|=1<<0;//开启DMA1时钟
__nop(); //等待 DMA1 时钟稳定
__nop(); //经测试最少 2 个 nop
__nop();
DMA_CHx->CPAR=ADC1_DR_Address;//DMA1 外设地址 ADC1_DR_Address
DMA_CHx->CMAR=(u32)ADC_Buf ; //DMA1,ADC 存储器地址
DMA_CHx->CNDTR=(u32)sizeof(ADC_Buf) ; //DMA1,传输数据量
DMA_CHx->CCR=0X00000000;//复位
DMA_CHx->CCR|=0<<4; //从外设器件读数据
DMA_CHx->CCR|=0<<5; //非循环模式
DMA_CHx->CCR|=0<<6; //外设地址非增量模式
DMA_CHx->CCR|=1<<7; //存储器增量模式
DMA_CHx->CCR|=1<<8; //外设数据宽度为16位
DMA_CHx->CCR|=1<<10; //存储器数据宽度16位
DMA_CHx->CCR|=1<<13; //高优先级
DMA_CHx->CCR|=0<<14; //非存储器到存储器模式
}
//开启 DMA传输
void MYDMA_Enable(DMA_Channel_TypeDef*DMA_CHx)
{
DMA_CHx->CCR|=1<<0; //开启DMA传输
}
//开启一次DMA传输
//单次DMA转换之后,必须把DMA关闭,再次启动,才能实现第二次DMA传输!!!
void MYDMA_GOnes(DMA_Channel_TypeDef*DMA_CHx)
{
DMA_CHx->CCR&=~(1<<0); //关闭DMA传输
DMA_CHx->CNDTR=(u32)sizeof(ADC_Buf) ; //DMA1,传输数据量
DMA_CHx->CCR|=1<<0; //开启DMA传输
}
void USART_Initaize(u32 pclk2,u32 bound)
{
float temp;
u16 mantissa;
u16 fraction;
temp=(float)(pclk2*1000000)/(bound*16);//得到USARTDIV
mantissa=temp; //得到整数部分
fraction=(temp-mantissa)*16; //得到小数部分
mantissa<<=4;
mantissa+=fraction;
RCC->APB2ENR|=1<<2; //使能PORTA口时钟
RCC->APB2ENR|=1<<14; //使能串口时钟
GPIOA->CRH&=0XFFFFF00F;
GPIOA->CRH|=0X000008B0;//IO状态设置
RCC->APB2RSTR|=1<<14; //复位串口1
RCC->APB2RSTR&=~(1<<14);//停止复位
//波特率设置
USART1->BRR=mantissa; // 波特率设置
USART1->CR1|=0X200C; //1位停止,无校验位.
#ifdef EN_USART1_RX //如果使能了接收
//使能接收中断
USART1->CR1|=1<<8; //PE中断使能
USART1->CR1|=1<<5; //接收缓冲区非空中断使能
MY_NVIC_Init(3,3,USART1_IRQChannel,2);//组2,最低优先级
#endif
}
void Uart1_PutChar(u8 ch)
{
USART1->DR = (u8) ch;
while((USART1->SR&0X40)==0);//循环发送,直到发送完毕
}
void Uart1_PutString(u8 *Buf, u8 Len)
{
u8 i;
for(i= 0; i<Len; i++)
{
Uart1_PutChar(*Buf++);
}
}
//TIM3 PWM部分
//正点原子@ALIENTEK
//2010/6/2
//PWM输出初始化
//arr:自动重装值
//psc:时钟预分频数
void PWM_Init(u16 arr,u16 psc)
{
//此部分需手动修改IO口设置
RCC->APB1ENR|=1<<1; //TIM3时钟使能
RCC->APB2ENR|=1<<3; //使能PORTB时钟
// GPIOA->CRH&=0XFFFFFFF0;//PA8输出
// GPIOA->CRH|=0X00000004;//浮空输入
GPIOB->CRL&=0XFFFFFFF0;//PB0输出
GPIOB->CRL|=0X0000000B;//复用功能输出
GPIOB->ODR|=1;//PB0上拉
TIM3->ARR=arr;//设定计数器自动重装值
TIM3->  SC=psc;//预分频器
TIM3->CCMR2|=7<<4; //CH3 PWM2模式
TIM3->CCMR2|=1<<4; //CH3预装载使能
TIM3->CCER|=1<<8; //OC3 输出使能
TIM3->CCR3=0X0009; // 比较寄存器为值18,arr为18
TIM3->CR1=0x0080; //ARPE使能
TIM3->CR1|=0x01; //使能定时器3
}
void PWM4_PULSE_Init(u16 arr1,u16 psc1,u16 arr2,u16 psc2)
{
//此部分需手动修改IO口设置
RCC->APB1ENR|=1; //TIM2时钟使能
RCC->APB2ENR|=1<<2; //使能PORTA时钟
RCC->APB1ENR|=1<<2; //TIM4时钟使能
RCC->APB2ENR|=1<<3; //使能PORTB时钟
GPIOA->CRL&=0XFFFFFF0F;//PA1输出
GPIOA->CRL|=0X000000B0;//复用功能输出
GPIOA->ODR|=1<<1;//PA1上拉
GPIOB->CRL&=0XF0FFFFFF;//PB6输出
GPIOB->CRL|=0X0B000000;//复用功能输出
GPIOB->ODR|=1<<6;//PB6上拉
TIM2->ARR=arr1;//设定计数器自动重装值
TIM2->  SC=psc1;//预分频器
TIM4->ARR=arr2;//设定计数器自动重装值
TIM4->  SC=psc2;//预分频器不分频
TIM2->CCMR1|=7<<12; //CH2 PWM2模式
TIM2->CCMR1|=1<<11; //CH2预装载使能
TIM4->CCMR1|=7<<4; //CH1 PWM2模式
TIM4->CCMR1|=1<<3; //CH1预装载使能
TIM2->CCER|=1<<4; //OC2 输出使能
TIM4->CCER|=1; //OC1 输出使能
TIM2->CCR2=0X0002; //
TIM4->CCR1=0X0009;
TIM2->CR1=0x0080; //ARPE使能
TIM2->CR1|=0x01; //使能定时器2
TIM4->CR1=0x0080; //ARPE使能
TIM4->CR1|=0x01; //使能定时器4
}
void KEY_Init(void)
{
RCC->APB2ENR|=1<<2; //使能PORTA时钟
GPIOA->CRH&=0XFF0FFFFF;//PA13设置成输入
GPIOA->CRH|=0X00800000;
GPIOA->ODR|=1<<13; //PA13上拉
}
u8 KEY_Scan(void)
{
if(KEY0==1)
{
delay_ms(10);//去抖动
return 1;// 无按键按下
}else if(KEY0==0)
return 0;
}
还望大神们指点一下
|