2685|1

3

帖子

1

TA的资源

一粒金砂(初级)

楼主
 

LM3S811开发板加入ADC测量后,1602显示异常 [复制链接]

用LM3S811开发板做了一个电源开关控制程序,LCD1602显示程序运行时间及当前的状态,这部分可以正常运行,但是后面在程序中加了ADC功能,用于检测继电器动作是否正常,加入ADC后,LCD1602中显示运行状态这一行出现了异常,表现为第一位显示字符一直在变化。其它字符应该变化时无刷新,DEBUG调试时,单步运行则显示正常,全速运行则出现问题,监视字符串内容,也确实是第一位字符一直在变化,2-4位内容为0D,其它字符正常,但显示无刷新。请各大神帮忙看看是哪里出问题了。代码如下:
#include
#include
#define ADCSequEnable ADCSequenceEnable
#define ADCSequDisable ADCSequenceDisable
#define ADCSequConfig ADCSequenceConfigure
#define ADCSequStepConfig ADCSequenceStepConfigure
#define ADCSequDataGet ADCSequenceDataGet
tBoolean ADC_EndFlag = false; // 定义ADC转换结束的标志

#define  SCPEN         SysCtlPeripheralEnable         //使能外设语句太长

#define  RS               GPIO_PORTB_BASE,GPIO_PIN_0  //PB0为LCD命令口
#define  RW               GPIO_PORTB_BASE,GPIO_PIN_1  //PB1为LCD命令口
#define  EN               GPIO_PORTB_BASE,GPIO_PIN_2  //PB2为LCD命令口
#define  GPIO_WR    GPIOPinWrite     //简化库函数名
#define  GPIO_RD    GPIOPinRead      //简化库函数名
#define  GPIO_IN      GPIOPinTypeGPIOInput //简化库函数名,周立功中文库函数说明书中居然没有对应的说明
#define  GPIO_OUT   GPIOPinTypeGPIOOutput//简化库函数名,周立功中文库函数说明书中居然没有对应的说明
#define  LCD_DATA   GPIO_PORTD_BASE//lcd数据口
#define  CHAOSHENG_T        GPIO_PIN_3//PB3超声驱动,DYP-ME007是超声模块
#define  CHAOSHENG_R        GPIO_PIN_4//PB4超声状态接收  
#define  LED                    GPIO_PIN_0//PE0计数器捕获指示,每次捕获都取反
#define  RY                      GPIO_PIN_1//PE1继电器捕获指示,每次捕获都取反   
#define  LCD_BUSY                GPIO_PIN_7//PD7数据口第七位测LCD忙
#define  PWM_OUT                GPIO_PIN_5//PC5即CCP1=PWM输出   
#define  COUNT_IN                GPIO_PIN_6//PC6即CCP3=计数输入
#define  PWM_PORT              GPIO_PORTC_BASE
#define  COUNT_PORT           GPIO_PORTC_BASE
#define  LCD_PORT                GPIO_PORTB_BASE//PB0\PB1\PB2给液晶用,PB5未用,PB7给JTAG用,尽量避免作IO
#define  LED_PORT                GPIO_PORTE_BASE//PE0给蜂鸣器用,测试完成时提醒用
#define  RY_PORT                GPIO_PORTE_BASE//PE1给继电器用

#define  CHAOSHENG_PORT   GPIO_PORTB_BASE//PB3、PB4给超声用
/*=================================================================*/
unsigned long pinlv_vlue=1000;//产生的频率可在这里设置,1MHz误差10%,100KHz误差1.6%,10KHz误差0.1%,1KHz~92Hz没有误差,91以下显示0,没有深入研究。
unsigned long count_vlue=65535;//65535TIME1B计数器count从最大值开始向下计数
unsigned long systick_vlue=6000000;//6MHz晶振,1s
unsigned char systick_flag,echo_flag=0,count_flag;
unsigned char run_flag=0; //运行标志
unsigned long key_flag=0; //User按键按下次数计数
unsigned char string0[16]={"ACC:OFF  LOOP:01"};        //16个字符
unsigned char string2[16]={" T:             "};        //16个字符
unsigned long frequency,second=0;
unsigned long cycle=10;     //设定循环次数,实际循环次数+1
unsigned long T1=1;     // 140
unsigned long T2=2;     // 150
unsigned long T3=3;     //  210
unsigned long T4=4;     //  420
unsigned long T_t=5;     //480 试验周期总时间

unsigned long ulVal[];          //电压变量
unsigned char cBuf[30];          //电压串口输出字符串
void
UARTSend(const unsigned char *pucBuffer, unsigned long ulCount);
unsigned adcSample(unsigned long ulValue[]);
void put(const char *x);
/*============================延时===============================*/
void delay(unsigned long delay_clok_1us)//1us延时
{
    while(delay_clok_1us)delay_clok_1us--;
}
/*=======================测试LCD忙碌状态=========================*/
unsigned char LCD_check_busy(void)//测试LCD忙碌状态,返回字节型
{
    unsigned char busy;
    GPIO_WR(RS,0);         //RS=0
    GPIO_WR(RW,2);         //RW=1
    GPIO_WR(EN,4);         //EN=1
    GPIO_IN(LCD_DATA , 0XFF);           //设置数据口为输入        
    busy=GPIO_RD(LCD_DATA,LCD_BUSY)&0x80;//读取第八位忙闲标志        
    GPIO_OUT(LCD_DATA,0XFF);             //重新设置数据口为输出
    GPIO_WR(EN, 0);      //EN=0   
    return busy;           //返回检测信号
}
/*================为液晶LCD1602服务的函数===========================*/
void lcd_write(unsigned  char cd,unsigned  char temp)//写入到LCD
{   SysCtlDelay(1*(TheSysClock/3000/2)); //约1ms   
    while(LCD_check_busy())        //cheker busy         加入后重新上电不工作
    SysCtlDelay(1*(TheSysClock/3000/2)); //约1ms
        if(cd)GPIO_WR(RS, 1);   //当写数据时使RS=1
    else  GPIO_WR(RS, 0);   //当为写指令时RS=0      
    SysCtlDelay(1*(TheSysClock/3000/2)); //约1ms
        GPIO_WR(RW, 0);
    SysCtlDelay(1*(TheSysClock/3000/2)); //约1ms
        GPIO_WR(EN, 0);
    SysCtlDelay(1*(TheSysClock/3000/2)); //约1ms
        GPIO_WR(LCD_DATA,0xFF,temp);//送数据到LCD
    SysCtlDelay(1*(TheSysClock/3000/2)); //约1ms
        GPIO_WR(EN, 4);        // delay(4);//此延时必须要,不然不能显示
    SysCtlDelay(1*(TheSysClock/3000/2)); //约1ms
        GPIO_WR(EN, 0);      
}
/*================为液晶LCD1602服务的函数===========================*/
void lcd_strwdat( unsigned char x,unsigned char y,unsigned char *str)//准备显示内容
{            
    if(x<16)
        {if(y==0)
           x=0x80+x;
             else
           x=0xc0+x;
     lcd_write(0,x);
        }         //显示位置( x, y);
    while(*str!='\0')
        lcd_write(1,*str++);
}

/*===============User按键中断处理函数===============================*/
void GPIO_Port_C_ISR(void)
{
//unsigned char ucVal;
unsigned long ulStatus;
ulStatus=GPIOPinIntStatus(GPIO_PORTC_BASE,true);// get gpio c interrupt status
GPIOPinIntClear(GPIO_PORTC_BASE,ulStatus); // 清除中断状态
if(ulStatus&GPIO_PIN_4)//如果KEY的中断状态有效
        {
   //ucVal = GPIOPinRead(GPIO_PORTD_BASE, GPIO_PIN_0); // 翻转LED
   //GPIOPinWrite(GPIO_PORTD_BASE,GPIO_PIN_0,~ucVal);
   key_flag++; //User键按下次数计数
  if(key_flag%2==0)
    run_flag=0;
   else
            run_flag=1;
            //lcd_write(0,0x02); delay(20);  //清屏
   SysCtlDelay(10*(SysCtlClockGet()/3000)); //延时约10ms,消除按键抖动
   while(GPIOPinRead(GPIO_PORTC_BASE,GPIO_PIN_4)==0x00);//等待KEY抬起
   SysCtlDelay(10*(SysCtlClockGet()/3000)); //延时约10ms,消除松键抖动

  }
}

/*====================定时器systick中断,每1秒===========================*/
void SysTick_ISR (void)//systick中断
{
     systick_flag=1;

}
/*==========================液晶始化init=======================*/
void LCD_UltraWave_init(void)//液晶初始化程序
{
    SCPEN(SYSCTL_PERIPH_GPIOB|SYSCTL_PERIPH_GPIOD);//端口使能声明
    //SysTickPeriodSet(systick_vlue);//设置SysTick初值,最大为16777216,用于计时
        SysTickPeriodSet(10000000);//设置SysTick初值,最大为16777216,用于计时
    SysTickEnable();     //启动SysTick
    SysTickIntEnable();  //systick系统定时器中断使能     
    GPIO_OUT(LCD_DATA,0xff);//设置LCD数据口(PD)全为输出
    GPIO_OUT(LCD_PORT,0x0f);//设置控制口(PB)低4位(LCD3位,超声模块1位)为输出
     SysCtlDelay(1*(TheSysClock/3000)); //约1ms   
        lcd_write(0,0x38); //delay(1);  //八位数据、双行显示、5X7点阵
     SysCtlDelay(1*(TheSysClock/3000)); //约1ms
        lcd_write(0,0x0C); //delay(1);  //显示开、关光标
    SysCtlDelay(1*(TheSysClock/3000)); //约1ms
        lcd_write(0,0x06); //delay(1);  //数据读、写操作后,地址指针AC自动增一
     SysCtlDelay(1*(TheSysClock/3000)); //约1ms
        lcd_write(0,0x01); //delay(1);  //清屏
}
/*==========================测试完成后显示及动作=======================*/
void test_finish(void)//测试完成后显示及动作程序
{  unsigned char string5[16]={"Wen Du Xun Huan "};        //16个字符
   unsigned char string6[16]={"TEST Finished   "};        //16个字符
   SysCtlDelay(100*(TheSysClock/3000)); //约1s
   //GPIO_WR(LED_PORT,LED,GPIO_RD(LED_PORT,LED)^LED);//翻转PE0,指示秒闪烁 (L7 GREEN LED)
   GPIO_WR(RY_PORT,RY,GPIO_RD(RY_PORT,RY)^RY);//翻转PE1,继电器动作
     while(1)
   {
            GPIO_WR(LED_PORT,LED,GPIO_RD(LED_PORT,LED)^LED);//翻转PE0,指示秒闪烁 (L7 GREEN LED) 可外加蜂鸣器用于声音提醒。
            lcd_strwdat(0,0,string5);    //显示第1行
     lcd_strwdat(0,1,string6);    //显示第2行
         lcd_write(0,0x08);                                           //字符闪烁
         SysCtlDelay(100*(TheSysClock/3000)); //约100ms
         lcd_write(0,0x0C);
         
   }

}

///*==========================串口初始化=======================*/  
void uartinit(void)
{
SysCtlPeripheralEnable(SYSCTL_PERIPH_GPIOA);
SysCtlPeripheralEnable(SYSCTL_PERIPH_UART0);
GPIOPinTypeUART(GPIO_PORTA_BASE,GPIO_PIN_0 | GPIO_PIN_1);
UARTConfigSet(UART0_BASE,115200,UART_CONFIG_WLEN_8 | UART_CONFIG_STOP_ONE | UART_CONFIG_PAR_NONE);
UARTEnable(UART0_BASE);
}
void put(const char *x)
{
while(*x!='\0')
{
  UARTCharPut(UART0_BASE,*(x++));
}
}

//*****************************************************************************
//
// The UART interrupt handler.
//
//*****************************************************************************
void
UARTIntHandler(void)
{
    unsigned long ulStatus;
    ulStatus = UARTIntStatus(UART0_BASE, true);
    UARTIntClear(UART0_BASE, ulStatus);
    while(UARTCharsAvail(UART0_BASE))
    {
        UARTCharPutNonBlocking(UART0_BASE, UARTCharGetNonBlocking(UART0_BASE));
    }
}


// ADC初始化
void adcInit(void)
{
        SysCtlPeripheralEnable(SYSCTL_PERIPH_ADC);        //使能ADC模块
        SysCtlADCSpeedSet(SYSCTL_ADCSPEED_125KSPS); //设置ADC采样速率
        ADCSequenceDisable(ADC_BASE, 0);         // 配置前先禁止采样序列
        ADCSequenceConfigure(ADC_BASE,0,ADC_TRIGGER_PROCESSOR,0);  
        //配置ADC采样序列的触发事件和优先级:ADC基址,采样序列编号,触发事件,采样优先级
        ADCSequenceStepConfigure(ADC_BASE, 0, 0, ADC_CTL_CH0);
        ADCSequenceStepConfigure(ADC_BASE, 0, 1, ADC_CTL_CH1 | ADC_CTL_IE | ADC_CTL_END);
        ADCIntEnable(ADC_BASE,0);        //使能ADC采样序列的中断
        IntEnable(INT_ADC0);                 // 使能ADC采样序列中断
        IntMasterEnable();                        // 使能处理器中断

        ADCSequenceEnable(ADC_BASE,0);                 // 使能一个ADC采样序列
}

// ADC采样
unsigned adcSample(unsigned long ulValue[])
{

        ADCProcessorTrigger(ADC_BASE, 0);         
        // 处理器触发采样序,调用ADCProcessorTrigger( )函数触发ADC采样

        while (!ADC_EndFlag)                          // 等待采样结束,加上后不能正确读出数据
        ADC_EndFlag = 1;                                 // 清除ADC采样结束标志
       
        ADCSequenceDataGet(ADC_BASE, 0, ulValue); // 读取ADC转换结果
        //return(ulValue);
}

// ADC采样序列0的中断
void ADC_Sequence_0_ISR(void)
{

unsigned long ulStatus;
        ulStatus = ADCIntStatus(ADC_BASE, 0, true);         // 读取中断状态
        ADCIntClear(ADC_BASE, 0);         // 清除中断状态,重要,等待下次AD中断
        if (ulStatus != 0)                         // 如果中断状态有效
        {
                ADC_EndFlag = 0;         // 置位ADC采样结束标志
        }
}

/*==========================主函数=================================*/
main(void) //
{       
    unsigned long x,w,v1,v2;
    unsigned long j,day,hour,mon,min;
    unsigned long loop=1;
        w=0;
        jtagWait();
        clockInit();
        uartinit();
        adcInit(); // ADC初始化
        SysCtlDelay(1*(TheSysClock/3000)); //1000倍时约1s
            SysCtlPeripheralEnable(SYSCTL_PERIPH_GPIOC|SYSCTL_PERIPH_GPIOE);                         // 使能KEY、LED所在的PC端口
         GPIOPinTypeGPIOOutput(GPIO_PORTE_BASE, GPIO_PIN_0 | GPIO_PIN_1);//配置PE0、PE1为输出
         GPIOPinIntEnable(GPIO_PORTC_BASE, GPIO_PIN_4);    //Enable GPIO C pin 4
         GPIOPinTypeGPIOInput(GPIO_PORTC_BASE, GPIO_PIN_4);// 设置KEY所在管脚PC4为输入
     GPIOIntTypeSet(GPIO_PORTC_BASE, GPIO_PIN_4, GPIO_LOW_LEVEL);//set interrupt type is low Falling edge
         IntEnable(INT_GPIOC);//enable GPIO C interrupt
          LCD_UltraWave_init();//液晶+超声初始化
         SysCtlDelay(1*(TheSysClock/3000)); //1000倍时约1s
         w=0;
    while(1)
      {

            if(run_flag==1)
                    {
                          
          /*===================以下为每一个秒钟到达是的事件处理======================*/
           if(systick_flag==1)//systick中断标志,表示又一秒到
            {
            systick_flag=0;      
       /*===============================以下处理时钟==============================*/
                 w++;
             second=(w-1)/5;         //秒数加1
            adcSample(ulVal);              // ADC采样
                    v1 = (ulVal[0] * 3000) / 1024; // 转换成电压值
                        v2 = (ulVal[1] * 3000) / 1024; // 转换成电压值
                        sprintf(cBuf, "ADC0 = %ld(mV)\r\n", v1); // 输出格式化
                        put(cBuf);
                        sprintf(cBuf, "ADC1 = %ld(mV)\r\n", v2); // 输出格式化
                        put(cBuf);                 
             day=second/(60*60*24);
             hour=second/3600-day*24;       //取出小时
                         mon=second/60-day*24*60-hour*60;   //取出分钟
                 min=second/60-(loop-1)*T_t;                   //计算分钟,用于判定动作及计算循环数。倍数为周期分钟数
                     j=second-mon*60-hour*60*60-day*24*60*60;
                         
                                x=3;                     //左边空3个,在液晶的第五、六位显示小时
             string2[x++]=day/10+0x30;  //写入天的十位
             string2[x++]=day%10+0x30;  //写入天的个位
             string2[x++]=':';          //写入":"
             string2[x++]=hour/10+0x30;  //写入小时的十位
             string2[x++]=hour%10+0x30;  //写入小时的个位
             string2[x++]=':';         //写入":"
             string2[x++]=mon/10+0x30;  //写入分钟的十位
             string2[x++]=mon%10+0x30;  //写入分钟的个位
             string2[x++]=':';           //写入":"
             string2[x++]=j/10+0x30;      //写入秒钟的十位
             string2[x++]=j%10+0x30;      //写入秒钟的个位
                               
                               if ((min==T1 ||min==T3) && j==0 && w%5==0)                                                 //140,210
                                       {
                                               GPIO_WR(LED_PORT,LED,GPIO_RD(LED_PORT,LED)^LED);//翻转PE0,指示秒闪烁 (L7 GREEN LED)
                                                GPIO_WR(RY_PORT,RY,GPIO_RD(RY_PORT,RY)^RY);//翻转PE1,继电器动作
                                                string0[4]='O';  //写入天的十位
                                               string0[5]='N';  //写入天的十位
                                                string0[6]=' ';  //写入天的十位
                                               }
                                       if ((min==T2 ||min==T4 )&& j==0 && w%5==0)                                   //150,420
                                               {       
                                                      
                                                  GPIO_WR(LED_PORT,LED,GPIO_RD(LED_PORT,LED)^LED);//翻转PE0,指示秒闪烁 (L7 GREEN LED)
                                                  GPIO_WR(RY_PORT,RY,GPIO_RD(RY_PORT,RY)^RY);//翻转PE1,继电器动作
                                                 string0[4]='O';  //写入天的十位
                                                 string0[5]='F';  //写入天的十位
                                                 string0[6]='F';  //写入天的十位
                                               }
                         if(GPIOPinRead(GPIO_PORTE_BASE,GPIO_PIN_1)==0x00);
                                                 {
                                                                               
                                                 }

                                                if (min==T_t && j==0 && w%2==0)                           //min=周期                         480
                                               {        //SysCtlDelay(1000*(TheSysClock/3000)); //约1000ms
                                                       min=0;
                                                       loop++;
                                                        if(loop==cycle)           //LOOP设定为需要循环数+1
                                                         {
                               test_finish();                                                                 
                                                         }
                                                 else
                                                         {
                                                                 string0[14]=loop/10+0x30;  //写入天的十位
                                                       string0[15]=loop%10+0x30;  //写入天的个位
                                                         }                                                                
                                               }            

             //if(string2="   00:01:30:00  " or string2)

          /*=====================以下驱动液晶显示==============================*/

                         lcd_strwdat(0,0,string0);    //显示第1行
                         SysCtlDelay(1*(TheSysClock/3000)); //约1ms                          
             lcd_strwdat(0,1,string2);    //显示第2行
             }
        }
        else
                {        if(key_flag==0)
                                  {
                           lcd_strwdat(0,0,"Wen Du Xun Huan");    //显示第1行
                           lcd_strwdat(0,1,"User Key To Run");    //显示第2行
                                  }
                                  else
                                  {
                                   while(run_flag==0)
                                   {
                                   lcd_strwdat(0,0,"Test Pause      ");    //显示第1行
                           lcd_strwdat(0,1,"User Key To Run ");    //显示第2行
                                   lcd_write(0,0x08);                                           //字符闪烁
                                   SysCtlDelay(100*(TheSysClock/3000)); //约100ms
                                   lcd_write(0,0x0C);
                                           }
                                  }
                }
     }
}
/*===================================================================*/


最新回复

把 ADC 的中断关闭了试一下。  详情 回复 发表于 2017-9-29 18:07
 
点赞 关注

回复
举报

918

帖子

0

TA的资源

纯净的硅(中级)

沙发
 
把 ADC 的中断关闭了试一下。
 
 

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

随便看看
查找数据手册?

EEWorld Datasheet 技术支持

相关文章 更多>>
关闭
站长推荐上一条 1/8 下一条

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

About Us 关于我们 客户服务 联系方式 器件索引 网站地图 最新更新 手机版

站点相关: 国产芯 安防电子 汽车电子 手机便携 工业控制 家用电子 医疗电子 测试测量 网络通信 物联网

北京市海淀区中关村大街18号B座15层1530室 电话:(010)82350740 邮编:100190

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2025 EEWORLD.com.cn, Inc. All rights reserved
快速回复 返回顶部 返回列表