OpenEdv-开源电子网

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

求助,看下程序的错误

[复制链接]

7

主题

14

帖子

0

精华

初级会员

Rank: 2

积分
84
金钱
84
注册时间
2016-1-22
在线时间
33 小时
发表于 2017-11-25 12:44:30 | 显示全部楼层 |阅读模式
10金钱
STM32103ZET6   我想要通过定时器触发ADC采样,定时器延时3.9ms.串口助手发送'B'到单片机,单片机将ADC数据以“#,#,高6位,低6位”的形式发送到串口助手,用定时发送'B',但是我接收到的都是乱码和'B',乱码的十六进制显示是FA,FF之类的。

下面是程序:

主函数:
     #define uchar unsigned char

extern __IO uint16_t ADC_ConvertedValue;
int main(void)
{
  char rec;
        u16 adcx , ECG;
        unsigned char *ECGH,*ECGL;

  SysTick_Init();//延时初始化
  USART1_Config();
  ADC1_Init();
        //NVIC_Configuration();

        while(1)
        {       
                if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)   //通过判断状态寄存器第五位RXNE是否为1判断是否接收到数据
                {
                        rec=USART_ReceiveData(USART1);//(USART1->DR);        //读取接收到的数据
                        if(rec=='B')
                        {
                                rec='0';
                                adcx=ADC_ConvertedValue;
                                //adcx=(float)ADC_ConvertedValue/4096*3.3;
                                ECG=adcx;
                                *ECGH=(uint8_t)((ECG >> 6)&0xf0);  // 取ADC高6位                                                 请问这样取对吗?
                                *ECGL=(uint8_t)(ECG&0x0f);                                        //取ADC低6位
                                send_data('#');
                                send_data('#');
                                send_char_String(ECGH);
                                send_char_String(ECGL);
                        }
                }
        }
}


ADC和定时器:
#include "adc.h"
#include "timer.h"
#define ADC1_DR_Address    ((u32)0x40012400+0x4c)

__IO uint16_t ADC_ConvertedValue;



/*
* 函数名:ADC1_GPIO_Config
* 描述  :使能ADC1和DMA1的时钟,初始化PC.01
* 输入  : 无
* 输出  :无
* 调用  :内部调用
*/
void ADC1_GPIO_Config(void)
{
        GPIO_InitTypeDef GPIO_InitStructure;
       
        /* Enable DMA clock */
        RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);  //开启DMA时钟
       
        /* Enable ADC1 and GPIOC clock */
        RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 | RCC_APB2Periph_GPIOC, ENABLE);//开启ADC时钟
       
        /* Configure PC.01  as analog input */
        GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
        GPIO_Init(GPIOC, &GPIO_InitStructure);                                // PC0,输入时不用设置速率
}


/* 函数名:ADC1_Mode_Config
* 描述  :配置ADC1的工作模式为MDA模式
* 输入  : 无
* 输出  :无
* 调用  :内部调用
*/
void ADC1_Mode_Config(void)
{
        DMA_InitTypeDef DMA_InitStructure;
        ADC_InitTypeDef ADC_InitStructure;
       
        /* DMA channel1 configuration */
        DMA_DeInit(DMA1_Channel1);
        DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address;         //ADC地址
        DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&ADC_ConvertedValue;//内存地址
        DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
        DMA_InitStructure.DMA_BufferSize = 100;   //缓存单元的个数
        DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;//外设地址固定
        DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;         //内存地址固定
        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_ITConfig(DMA1_Channel1,DMA_IT_TC,ENABLE);//启动中断标志
       
       
        /* ADC1 configuration */
       
        ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;        //独立ADC模式
        ADC_InitStructure.ADC_ScanConvMode = DISABLE ;             //禁止扫描模式,扫描模式用于多通道采集
        ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;        //开启连续转换模式,即不停地进行ADC转换
        ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_T2_CC2 ;        //使用外部触发转换
        //ADC_InitStructure.ADC_ExternalTrigConv =ADC_ExternalTrigConv_None;        //不使用外部触发转换
        ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;         //采集数据右对齐
        ADC_InitStructure.ADC_NbrOfChannel = 1;                          //要转换的通道数目1
        ADC_Init(ADC1, &ADC_InitStructure);
       
        /*配置ADC时钟,为PCLK2的8分频,即9Hz*/
        RCC_ADCCLKConfig(RCC_PCLK2_Div6);
        /*配置ADC1的通道11为55.        5个采样周期,序列为1 */
        ADC_RegularChannelConfig(ADC1, ADC_Channel_11, 1, ADC_SampleTime_239Cycles5);//239.5
       
        ADC_DMACmd(ADC1, ENABLE);     //使能DMA传输
        ADC_Cmd(ADC1, ENABLE);        //使能ADC外设

        ADC_ResetCalibration(ADC1);   //复位校准寄存器
        while(ADC_GetResetCalibrationStatus(ADC1));
        ADC_StartCalibration(ADC1);   //ADC校准
        while(ADC_GetCalibrationStatus(ADC1));
        DMA_Cmd(DMA1_Channel1, ENABLE);
  ADC_ExternalTrigConvCmd(ADC1, ENABLE);   //使用外部中断触发
        //ADC_SoftwareStartConvCmd(ADC1, ENABLE);
}

static void DMA_Interrupt_Init(void)      //配置DMA中断
{
       
  NVIC_InitTypeDef NVIC_InitStructure;
  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);
  NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel1_IRQn;
  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority =1;
  NVIC_InitStructure.NVIC_IRQChannelSubPriority =1;
  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
  NVIC_Init(&NVIC_InitStructure);
}

void timer_init()
{
        TIM_TimeBaseInitTypeDef   TIM_TimeBaseStructure;
        TIM_OCInitTypeDef         TIM_OCInitStructure;
        RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);   //开启定时器时钟
       
        TIM_TimeBaseStructure.TIM_Period = 390;      //定时3.9ms
  TIM_TimeBaseStructure.TIM_Prescaler = 720-1;   //系统主频72M,这里分频720,相当于100K的定时器2时钟
        TIM_TimeBaseStructure.TIM_ClockDivision = 0x0;
        TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
        TIM_TimeBaseInit(TIM2, & TIM_TimeBaseStructure);
       
        TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;
        TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;//TIM_OutputState_Disable;
        TIM_OCInitStructure.TIM_Pulse = 200;
        TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_Low;//如果是PWM1要为Low,则为High
        TIM_OC2Init(TIM2, &TIM_OCInitStructure);
        TIM_Cmd(TIM2, ENABLE);
        TIM_InternalClockConfig(TIM2);
        TIM_OC2PreloadConfig(TIM2, TIM_OCPreload_Enable);
        TIM_UpdateDisableConfig(TIM2, DISABLE);
}


void ADC1_Init(void)
{
        ADC1_GPIO_Config();
        //NVIC_User_Init();   //中断配置函数
        ADC1_Mode_Config();
        DMA_Interrupt_Init();
        timer_init();
}



串口:
#include <stdio.h>
#include "stm32f10x.h"
#include "usart.h"


#ifdef __GNUC__
  /* With GCC/RAISONANCE, small printf (option LD Linker->Libraries->Small printf
     set to 'Yes') calls __io_putchar() */
  #define PUTCHAR_PROTOTYPE int __io_putchar(int ch)
#else
  #define PUTCHAR_PROTOTYPE int fputc(int ch, FILE *f)
#endif /* __GNUC__ */
       
PUTCHAR_PROTOTYPE
{
  /* Place your implementation of fputc here */
  /* e.g. write a character to the USART */
  USART_SendData(USART1, (unsigned char) ch);

  /* 循环等待直到发送结束*/
        while(!(USART1->SR&USART_FLAG_TXE));
  //while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET){}

  return ch;
}
void USART1_Config(void)
{
        GPIO_InitTypeDef GPIO_InitStructure;
        USART_InitTypeDef USART_InitStructure;
       
        /* config USART1 clock */
        RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 | RCC_APB2Periph_GPIOA, ENABLE);
       
        /* USART1 GPIO config */
        /* Configure USART1 Tx (PA.09) as alternate function push-pull */
        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);   
        /* Configure USART1 Rx (PA.10) as input floating */
        GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
        GPIO_Init(GPIOA, &GPIO_InitStructure);
          
        /* USART1 mode config */
        USART_InitStructure.USART_BaudRate =9600;
        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);
        USART_Cmd(USART1, ENABLE);
}

void send_data(unsigned char ascii_code)
{
  USART_SendData(USART1,ascii_code);
  while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET){}                                                               
}
void send_char(unsigned char c)//发送字符
{
  USART_SendData(USART1,c);
  while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET){}                                                               
}
void send_char_String(unsigned char *Str)//发送字符串
{
while(*Str)
  {
    send_char(*Str);
    Str++;
                while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET){}                                                               
        }
}


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

使用道具 举报

4

主题

380

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
3088
金钱
3088
注册时间
2015-10-17
在线时间
320 小时
发表于 2017-11-25 15:55:23 | 显示全部楼层
你去好好看看指针吧。
回复

使用道具 举报

7

主题

14

帖子

0

精华

初级会员

Rank: 2

积分
84
金钱
84
注册时间
2016-1-22
在线时间
33 小时
 楼主| 发表于 2017-11-25 17:05:26 | 显示全部楼层
hgr211 发表于 2017-11-25 15:55
你去好好看看指针吧。

指针错了的话,我把ADC屏蔽后只发#,为什么#也乱码
回复

使用道具 举报

4

主题

380

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
3088
金钱
3088
注册时间
2015-10-17
在线时间
320 小时
发表于 2017-11-27 13:43:56 | 显示全部楼层
你看都没人愿意回你,最基本的东西都自己不去了解,乱七八糟的,指针 *p,p代表什么意思,自己没点b数吗?对一个未知指针的地址值处理,真是脑袋坏掉了。发送字符串的函数,去发送一个未知的指针地址,真是无语了
回复

使用道具 举报

2

主题

91

帖子

0

精华

高级会员

Rank: 4

积分
509
金钱
509
注册时间
2016-11-16
在线时间
111 小时
发表于 2017-11-27 15:22:18 | 显示全部楼层
ECG=adcx;
                                *ECGH=(uint8_t)((ECG >> 6)&0xf0);  // 取ADC高6位                                                 请问这样取对吗?
                                *ECGL=(uint8_t)(ECG&0x0f);                                        //取ADC低6位
                                send_data('#');
                                send_data('#');
                                send_char_String(ECGH);
                                send_char_String(ECGL);

上面这一段是什么鬼。。。。
回复

使用道具 举报

56

主题

520

帖子

0

精华

高级会员

Rank: 4

积分
964
金钱
964
注册时间
2014-11-18
在线时间
160 小时
发表于 2017-11-27 17:33:12 | 显示全部楼层
ml598031805 发表于 2017-11-25 17:05
指针错了的话,我把ADC屏蔽后只发#,为什么#也乱码

指针那里确实有问题.    在单片机里面少用指针,除非你很清楚,因为很多应用都没必要.

用  数组就可以了.  
自己选择的路,成家前走完。
回复

使用道具 举报

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

本版积分规则



关闭

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

正点原子公众号

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

GMT+8, 2025-6-9 07:31

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

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