找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 3816|回复: 1
打印 上一主题 下一主题
收起左侧

新人求助,关于pcf8951测电压(设置电压检测范围)并用lcd1602显示的问题

[复制链接]
跳转到指定楼层
楼主
ID:357135 发表于 2018-6-23 11:33 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
pcf8951得到的val没法和我设置的值比较大小该怎么修改。怎么用键盘设置检测的电压上下限范围?
#include <reg52.h>
#include <intrins.h>

#define LCD1602_DB P0
#define uint unsigned int
#define uchar unsigned char


sbit LCD1602_RS=P2^0;
sbit LCD1602_RW=P2^1;
sbit LCD1602_E=P2^2;

void LcdWaitReady()  //等待液晶准备好
{
    unsigned char sta;

    LCD1602_DB = 0xFF;
    LCD1602_RS = 0;
    LCD1602_RW = 1;
    do
    {
        LCD1602_E = 1;
        sta = LCD1602_DB; //读取状态字
        LCD1602_E = 0;
    } while (sta & 0x80); //bit7等于1表示液晶正忙,重复检测直到其等于0为止
}
void LcdWriteCmd(unsigned char cmd)  //写入命令函数
{
    LcdWaitReady();
    LCD1602_RS = 0;
    LCD1602_RW = 0;
    LCD1602_DB = cmd;
    LCD1602_E  = 1;
    LCD1602_E  = 0;
}
void LcdWriteDat(unsigned char dat)  //写入数据函数
{
    LcdWaitReady();
    LCD1602_RS = 1;
    LCD1602_RW = 0;
    LCD1602_DB = dat;
    LCD1602_E  = 1;
    LCD1602_E  = 0;
}
void LcdShowStr(unsigned char x, unsigned char y, const unsigned char *str)  //显示字符串,屏幕起始坐标(x,y),字符串指针str
{
    unsigned char addr;

    //由输入的显示坐标计算显示RAM的地址
    if (y == 0)
        addr = 0x00 + x; //第一行字符地址从0x00起始
    else
        addr = 0x40 + x; //第二行字符地址从0x40起始

    //由起始显示RAM地址连续写入字符串
    LcdWriteCmd(addr | 0x80); //写入起始地址
    while (*str != '\0')      //连续写入字符串数据,直到检测到结束符
    {
        LcdWriteDat(*str);
        str++;
    }
}


void LcdInit()  //液晶初始化函数
{

          LcdWriteCmd(0x38);  //16*2显示,5*7点阵,8位数据接口
         
    LcdWriteCmd(0x0C);  //显示器开,光标关闭
       
    LcdWriteCmd(0x06);  //文字不动,地址自动+1
          
    LcdWriteCmd(0x01);  //清屏
         
        }
/***********************I2C.c文件程序源代码*************************/
#include <reg52.h>
#include <intrins.h>

#define I2CDelay()  {_nop_();_nop_();_nop_();_nop_();}

sbit I2C_SCL = P3^7;
sbit I2C_SDA = P3^6;

void I2CStart()  //产生总线起始信号
{
    I2C_SDA = 1; //首先确保SDA、SCL都是高电平
    I2C_SCL = 1;
    I2CDelay();
    I2C_SDA = 0; //先拉低SDA
    I2CDelay();
    I2C_SCL = 0; //再拉低SCL
}
void I2CStop()   //产生总线停止信号
{
    I2C_SCL = 0; //首先确保SDA、SCL都是低电平
    I2C_SDA = 0;
    I2CDelay();
    I2C_SCL = 1; //先拉高SCL
    I2CDelay();
    I2C_SDA = 1; //再拉高SDA
    I2CDelay();
}
bit I2CWrite(unsigned char dat) //I2C总线写操作,待写入字节dat,返回值为应答状态
{
    bit ack;  //用于暂存应答位的值
    unsigned char mask;  //用于探测字节内某一位值的掩码变量

    for (mask=0x80; mask!=0; mask>>=1) //从高位到低位依次进行
    {
        if ((mask&dat) == 0)  //该位的值输出到SDA上
            I2C_SDA = 0;
        else
            I2C_SDA = 1;
        I2CDelay();
        I2C_SCL = 1;          //拉高SCL
        I2CDelay();
        I2C_SCL = 0;          //再拉低SCL,完成一个位周期
    }
    I2C_SDA = 1;   //8位数据发送完后,主机释放SDA,以检测从机应答
    I2CDelay();
    I2C_SCL = 1;   //拉高SCL
    ack = I2C_SDA; //读取此时的SDA值,即为从机的应答值
    I2CDelay();
    I2C_SCL = 0;   //再拉低SCL完成应答位,并保持住总线

    return (~ack); //应答值取反以符合通常的逻辑:0=不存在或忙或写入失败,1=存在且空闲或写入成功
}
unsigned char I2CReadNAK() //I2C总线读操作,并发送非应答信号,返回值为读到的字节
{
    unsigned char mask;
    unsigned char dat;

    I2C_SDA = 1;  //首先确保主机释放SDA
    for (mask=0x80; mask!=0; mask>>=1) //从高位到低位依次进行
    {
        I2CDelay();
        I2C_SCL = 1;      //拉高SCL
        if(I2C_SDA == 0)  //读取SDA的值
            dat &= ~mask; //为0时,dat中对应位清零
        else
            dat |= mask;  //为1时,dat中对应位置1
        I2CDelay();
        I2C_SCL = 0;      //再拉低SCL,以使从机发送出下一位
    }
    I2C_SDA = 1;   //8位数据发送完后,拉高SDA,发送非应答信号
    I2CDelay();
    I2C_SCL = 1;   //拉高SCL
    I2CDelay();
    I2C_SCL = 0;   //再拉低SCL完成非应答位,并保持住总线

    return dat;
}
unsigned char I2CReadACK() //I2C总线读操作,并发送应答信号,返回值为读到的字节
{
    unsigned char mask;
    unsigned char dat;

    I2C_SDA = 1;  //首先确保主机释放SDA
    for (mask=0x80; mask!=0; mask>>=1) //从高位到低位依次进行
    {
        I2CDelay();
        I2C_SCL = 1;      //拉高SCL
        if(I2C_SDA == 0)  //读取SDA的值
            dat &= ~mask; //为0时,dat中对应位清零
        else
            dat |= mask;  //为1时,dat中对应位置1
        I2CDelay();
        I2C_SCL = 0;      //再拉低SCL,以使从机发送出下一位
    }
    I2C_SDA = 0;   //8位数据发送完后,拉低SDA,发送应答信号
    I2CDelay();
    I2C_SCL = 1;   //拉高SCL
    I2CDelay();
    I2C_SCL = 0;   //再拉低SCL完成应答位,并保持住总线

    return dat;
}
/***********************main.c文件程序源代码*************************/
#include <reg52.h>

bit flag300ms = 1;       //300ms定时标志
unsigned char T0RH = 0;  //T0重载值的高字节
unsigned char T0RL = 0;  //T0重载值的低字节

unsigned char GetADCValue(unsigned char chn);
void ValueToString(unsigned char *str, unsigned char val);
void ConfigTimer0(unsigned int ms);
extern void LcdInit();
extern void LcdShowStr(unsigned char x, unsigned char y, const unsigned char *str);
extern void I2CStart();
extern void I2CStop();
extern unsigned char I2CReadACK();
extern unsigned char I2CReadNAK();
extern bit I2CWrite(unsigned char dat);

void main ()
{
    unsigned char val;
    unsigned char str[10];

    EA = 1;           //开总中断
    ConfigTimer0(10); //配置T0定时10ms
    LcdInit();        //初始化液晶   
    LcdShowStr(2,0, "Volt   Volt");  //显示通道指示

    while(1)
    {
        if (flag300ms)
        {
            flag300ms = 0;
            //显示通道3的电压
            val = GetADCValue(3);     //获取ADC通道3的转换值
                                                ValueToString(str, val);  //转为字符串格式的电压值
                                        if(val<1.2)  
                LcdShowStr(2,1, "Low ");
        else if(val>=1.2&&val<=2.1)
        LcdShowStr(2, 1, str);    //显示到液晶上       
        else
                LcdShowStr(2,1, "High");
                        val = GetADCValue(0);     //获取ADC通道0的转换值
                                                ValueToString(str, val);  //转为字符串格式的电压值
        if(str[0]==str[5]||str[0]==str[6])
        LcdShowStr(9, 1, str);    //显示到液晶上       
        else
                LcdShowStr(9,1, "High");
  }
       
    }

        }
unsigned char GetADCValue(unsigned char chn)  //读取当前的ADC转换值,chn为ADC通道号0-3
{
    unsigned char val;

                    I2CStart();
            if (!I2CWrite(0x48<<1))   //寻址PCF8591,如未应答,则停止操作并返回0
    {
                I2CStop();
                        return 0;
            }
    I2CWrite(0x40|chn);       //写入控制字节,选择转换通道
    I2CStart();
    I2CWrite((0x48<<1)|0x01); //寻址PCF8591,指定后续为读操作   
            I2CReadACK();             //先空读一个字节,提供采样转换时间
            val = I2CReadNAK();       //读取刚刚转换完的值
            I2CStop();

    return val;
}
void ValueToString(unsigned char *str, unsigned char val)  //ADC转换值转为实际电压值的字符串形式
{

       
        val = (val*25) / 255;    //电压值=转换结果*2.5V/255
    str[0] = (val/10) + '0'; //整数位字符
    str[1] = '.';            //小数点
    str[2] = (val%10) + '0'; //小数位字符
    str[3] = 'V';            //电压单位
    str[4] = '\0';           //结束符
          str[5]='1';  //设置通道3电压上限
        str[6]='0';  //设置通道3电压下限
}

void ConfigTimer0(unsigned int ms)  //T0配置函数
{
    unsigned long tmp;

    tmp = 11059200 / 12;      //定时器计数频率
    tmp = (tmp * ms) / 1000;  //计算所需的计数值
    tmp = 65536 - tmp;        //计算定时器重载值
    tmp = tmp + 12;           //修正中断响应延时造成的误差

    T0RH = (unsigned char)(tmp >> 8);  //定时器重载值拆分为高低字节
    T0RL = (unsigned char)tmp;
    TMOD &= 0xF0;   //清零T0的控制位
    TMOD |= 0x01;   //配置T0为模式1
    TH0 = T0RH;     //加载T0重载值
    TL0 = T0RL;
    ET0 = 1;        //使能T0中断
    TR0 = 1;        //启动T0
}
void InterruptTimer0() interrupt 1  //T0中断服务函数
{
    static unsigned char tmr300ms = 0;

    TH0 = T0RH;  //定时器重新加载重载值
    TL0 = T0RL;
    tmr300ms++;
    if (tmr300ms >= 30)  //定时300ms
    {
        tmr300ms = 0;
        flag300ms = 1;
    }
}




分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏 分享淘帖 顶 踩
回复

使用道具 举报

沙发
ID:357135 发表于 2018-6-23 13:06 | 只看该作者
还有我仿真的时候lcd还能显示 烧录到开发板之后就不行了,用的是普中科技的开发板EM3 V3.0
回复

使用道具 举报

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

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

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