素黑明 2014-03-27 08:32
浏览 1357

C51单片机程控滤波器的程序

#include

#include "i2c.h"
#include "delay.h"
#include "display.h"
#include "key.h"

#define AddWr 0x90

#define AddRd 0x91

sbit P00=P0^0;
sbit P01=P0^1;
sbit P02=P0^2;
sbit P03=P0^3;
sbit P04=P0^4;

extern bit ack;

unsigned char ReadADC(unsigned char Chl);

/*------------------------------------------------
读AD转值程序
输入参数 Chl 表示需要转换的通道,范围从0-3
返回值范围0-255

------------------------------------------------*/
unsigned char ReadADC(unsigned char Chl)
{
unsigned char Val;
Start_I2c();

SendByte(AddWr);

if(ack==0)return(0);

SendByte(0x40|Chl);

if(ack==0)return(0);
Start_I2c();
SendByte(AddWr+1);
if(ack==0)return(0);
Val=RcvByte();
NoAck_I2c();

Stop_I2c();

return(Val);
}

/*------------------------------------------------
主程序
------------------------------------------------*/
main()
{
unsigned char Chl;
unsigned char num=0;
P35=0;
Init_Timer0();

P3 = 0x0F; //按键初始化 0000 1111

while (1) //主循环
{

KeyScan();   /*按键控制电压输出0~2V变化 并在数码管上显示*/
DelayMs(100);
Display(0,4); 

num=ReadADC(0);/*AD转换 I/O口输出实时电压变化*/
DelayMs(100);
ShowData(ReadADC(Chl),TempData);
Display(0,4);

P00=SetDACOut(ReadADC(Chl));  /*5个I/O口分别输出5v电压*/
P01=SetDACOut(ReadADC(Chl));
P02=SetDACOut(ReadADC(Chl));
P03=SetDACOut(ReadADC(Chl));
P04=SetDACOut(ReadADC(Chl));

}
}

#include "i2c.h"
#include "delay.h"

#define Nop() _nop() //定义空指令

bit ack; //应答标志位

sbit SDA=P3^6;
sbit SCL=P3^7;

/*------------------------------------------------
启动总线
------------------------------------------------*/
void Start_I2c()
{
SDA=1; //发送起始条件的数据信号——SCL为高电平期间,SDA由高电平变为低电平
_Nop();
SCL=1;
_Nop(); //起始条件建立时间大于4.7us,延时
_Nop();
_Nop();
_Nop();
_Nop();

SDA=0; //发送起始信号
_Nop(); //起始条件锁定时间大于4μ
_Nop();
_Nop();
_Nop();
_Nop();

SCL=0; //钳住I2C总线,准备发送或接收数据
_Nop();
_Nop();
}
/*------------------------------------------------
结束总线
------------------------------------------------*/
void Stop_I2c()
{
SDA=0; //发送结束条件的数据信号——SCL为高电平期间,SDA由低电平变为高电平
_Nop(); //发送结束条件的时钟信号
SCL=1; //结束条件建立时间大于4μ
_Nop();
_Nop();
_Nop();
_Nop();
_Nop();
SDA=1; //发送I2C总线结束信号
_Nop();
_Nop();
_Nop();
_Nop();
}

/*----------------------------------------------------------------
字节数据传送函数

函数原型: void SendByte(unsigned char c);
功能: 将数据c发送出去,可以是地址,也可以是数据,发完后等待应答,并对
此状态位进行操作.(不应答或非应答都使ack=0 假)

发送数据正常,ack=1; ack=0表示被控器无应答或损坏。
------------------------------------------------------------------*/
void SendByte(unsigned char c)
{
unsigned char BitCnt;

for(BitCnt=0;BitCnt<8;BitCnt++) //要传送的数据长度为8位
{
if((c<<BitCnt)&0x80)SDA=1; //判断发送位&是二进制运算符,c左移后和1000000相与,
else SDA=0; //例如C=10101010,那么 1 0 1 0 1 0 1 0&1 0 0 0 0 0 0 0=1 0 0 0 0 0 0 0 此时if语句判断为真,则sda =1;发送出去

_Nop();//接着执行循环,C在左移一位后为 0 1 0 1 0 1 0 0和0x80相与结果为 0 0 0 0 0 0 0 0 if语句判断为0,则sda=0 发送出去。
SCL=1; //置时钟线为高,通知被控器开始接收数据位
_Nop();
_Nop(); //保证时钟高电平周期大于4μ
_Nop();
_Nop();
_Nop();

SCL=0;
}

_Nop();
_Nop();
SDA=1;               //8位发送完后释放数据线,准备接收应答位
_Nop();
_Nop();   
SCL=1;
_Nop();
_Nop();
_Nop();
if(SDA==1)ack=0;     
   else ack=1;        //判断是否接收到应答信号
SCL=0;
_Nop();
_Nop();

}

/*******************************************

  • 设置DAC输出值
  • DA转换 * ********************************************/ bit SetDACOut(unsigned char Val) { Start_I2c(); //启动总线 SendByte(0x90); //发送器件地址 if(ack==0) return 1; SendByte(0x40); //写入控制字节 if(ack==0) return 1; SendByte(Val); if(ack==0) return 1; Stop_I2c(); //结束总线 return 0; }

/*----------------------------------------------------------------
字节数据传送函数

函数原型: unsigned char RcvByte();
功能: 用来接收从器件传来的数据,并判断总线错误(不发应答信号),
发完后请用应答函数。

------------------------------------------------------------------*/

unsigned char RcvByte()
{
unsigned char retc;
unsigned char BitCnt;

retc=0;
SDA=1; //置数据线为输入方式
for(BitCnt=0;BitCnt<8;BitCnt++)
{
_Nop();

SCL=0; //置时钟线为低,准备接收数据位
_Nop();
_Nop(); //时钟低电平周期大于4.7us
_Nop();
_Nop();
_Nop();
SCL=1; //置时钟线为高使数据线上数据有效
_Nop();
_Nop();
retc=retc<<1;
if(SDA==1)retc=retc+1; //读数据位,接收的数据位放入retc中
_Nop();
_Nop();
}
SCL=0;

_Nop();
_Nop();
return(retc);
}

/*----------------------------------------------------------------
应答子函数
原型: void Ack_I2c(void);

----------------------------------------------------------------*/
/*void Ack_I2c(void)
{

SDA=0;

_Nop();
_Nop();
_Nop();

SCL=1;
_Nop();
_Nop(); //时钟低电平周期大于4μ
_Nop();
_Nop();
_Nop();

SCL=0; //清时钟线,钳住I2C总线以便继续接收
_Nop();
_Nop();

}*/
/*----------------------------------------------------------------
非应答子函数
原型: void NoAck_I2c(void);

----------------------------------------------------------------*/
void NoAck_I2c(void)
{

SDA=1;
_Nop();
_Nop();
_Nop();

SCL=1;
_Nop();
_Nop(); //时钟低电平周期大于4μ
_Nop();
_Nop();
_Nop();

SCL=0; //清时钟线,钳住I2C总线以便继续接收
_Nop();
_Nop();

}

/*----------------------------------------------------------------
向无子地址器件发送字节数据函数

函数原型: bit ISendByte(unsigned char sla,ucahr c);

功能: 从启动总线到发送地址,数据,结束总线的全过程,从器件地址sla.
如果返回1表示操作成功,否则操作有误。
注意: 使用前必须已结束总线。
----------------------------------------------------------------*/
/*bit ISendByte(unsigned char sla,unsigned char c)
{
Start_I2c(); //启动总线
SendByte(sla); //发送器件地址
if(ack==0)return(0);
SendByte(c); //发送数据
if(ack==0)return(0);
Stop_I2c(); //结束总线
return(1);
}
*/

/*----------------------------------------------------------------
向有子地址器件发送多字节数据函数

函数原型: bit ISendStr(unsigned char sla,unsigned char suba,ucahr s,unsigned char no);

功能: 从启动总线到发送地址,子地址,数据,结束总线的全过程,从器件
地址sla,子地址suba,发送内容是s指向的内容,发送no个字节。
如果返回1表示操作成功,否则操作有误。
注意: 使用前必须已结束总线。
----------------------------------------------------------------
/
/*bit ISendStr(unsigned char sla,unsigned char suba,unsigned char *s,unsigned char no)
{
unsigned char i;
for(i=0;i<no;i++)
{
Start_I2c(); //启动总线
SendByte(sla); //发送器件地址
if(ack==0)return(0);
SendByte(suba); //发送器件子地址
if(ack==0)return(0);

 SendByte(*s);            //发送数据
   if(ack==0)return(0);
 Stop_I2c();                  //结束总线
 DelayMs(1);               //必须延时等待芯片内部自动处理数据完毕
 s++;
 suba++;
} 

return(1);
}
/
/
----------------------------------------------------------------
向无子地址器件读字节数据函数

函数原型: bit IRcvByte(unsigned char sla,ucahr c);

功能: 从启动总线到发送地址,读数据,结束总线的全过程,从器件地
址sla,返回值在c.
如果返回1表示操作成功,否则操作有误。
注意: 使用前必须已结束总线。
----------------------------------------------------------------
/
/*bit IRcvByte(unsigned char sla,unsigned char *c)
{
Start_I2c(); //启动总线
SendByte(sla+1); //发送器件地址
if(ack==0)return(0);
*c=RcvByte(); //读取数据
NoAck_I2c(); //发送非就答位
Stop_I2c(); //结束总线
return(1);
}

/
/
----------------------------------------------------------------
向有子地址器件读取多字节数据函数

函数原型: bit ISendStr(unsigned char sla,unsigned char suba,ucahr s,unsigned char no);

功能: 从启动总线到发送地址,子地址,读数据,结束总线的全过程,从器件
地址sla,子地址suba,读出的内容放入s指向的存储区,读no个字节。
如果返回1表示操作成功,否则操作有误。
注意: 使用前必须已结束总线。
----------------------------------------------------------------
/
/*bit IRcvStr(unsigned char sla,unsigned char suba,unsigned char *s,unsigned char no)
{
unsigned char i;

Start_I2c(); //启动总线
SendByte(sla); //发送器件地址
if(ack==0)return(0);
SendByte(suba); //发送器件子地址
if(ack==0)return(0);

Start_I2c();
SendByte(sla+1);
if(ack==0)return(0);

for(i=0;i<no-1;i++)
{
*s=RcvByte(); //发送数据
Ack_I2c(); //发送就答位
s++;
}
*s=RcvByte();
NoAck_I2c(); //发送非应位
Stop_I2c(); //结束总线
return(1);
}
*/

#include"key.h"
#include"delay.h"
#include"i2c.h"
#include"display.h"

sbit S17=P3^3; //独立按键
sbit S18=P3^2;
sbit S19=P3^1;
sbit S20=P3^0;

unsigned char TempData[8];

/*------------------------------------------------
按键扫描函数,返回扫描键值 程控放大倍数
------------------------------------------------*/
unsigned char KeyScan()
{

static unsigned char volt = 0;

if(!S20) //如果检测到低电平,说明按键按下
{ /*sbit S20=P3^0; 依次-6*/
DelayMs(10); //延时去抖,一般10-20ms
if(!S20) //再次确认按键是否按下,没有按下则退出
{
while(!S20);//如果确认按下按键等待按键释放,没有释放则一直等待
{
if(volt {
volt=volt+500;
SetDACOut(volt*0.51);//转换为AD输出值
TempData[8]=volt*0.51;
ShowData((unsigned char)(volt*0.51),TempData);
Display(0,4);
}
}
return (1);
}
}
if(!S19) //如果检测到低电平,说明按键按下
{ /*依次+6*/
DelayMs(10); //延时去抖,一般10-20ms
if(!S19) //再次确认按键是否按下,没有按下则退出
{
while(!S19);//如果确认按下按键等待按键释放,没有释放则一直等待
{ if(volt>0)
{
volt=volt-500;
SetDACOut(volt*0.51);
TempData[8]=volt*0.51;
ShowData((unsigned char)(volt*0.51),TempData);
Display(0,4);
}

}
return (2);
}
}
if(!S18) //如果检测到低电平,说明按键按下
{ /*乱跳????*/
DelayMs(10); //延时去抖,一般10-20ms
if(!S18) //再次确认按键是否按下,没有按下则退出
{
while(!S18);//如果确认按下按键等待按键释放,没有释放则一直等待
{
if(volt {
volt=volt+100;
SetDACOut(volt*0.51);
TempData[8]=volt*0.51;
ShowData((unsigned char)(volt*0.51),TempData);
Display(0,4);
}
}
return (3);
}
}
if(!S17) //如果检测到低电平,说明按键按下
{ /*乱跳????*/
DelayMs(10); //延时去抖,一般10-20ms
if(!S17) //再次确认按键是否按下,没有按下则退出
{
while(!S17);//如果确认按下按键等待按键释放,没有释放则一直等待
{
if(volt>0)
{
volt=volt-100;
SetDACOut(volt*0.51);
TempData[8]=volt*0.51;
ShowData((unsigned char)(volt*0.51),TempData);
Display(0,4);
}
}
return (4);
}
}
return 0;
}

下载程序到单片机后,数码管显示值自动跳转 且无规律 不知道哪里出错了?菜鸟求大神解答 万分感谢!

  • 写回答

0条回答 默认 最新

    报告相同问题?

    悬赏问题

    • ¥50 有数据,怎么建立模型求影响全要素生产率的因素
    • ¥50 有数据,怎么用matlab求全要素生产率
    • ¥15 TI的insta-spin例程
    • ¥15 完成下列问题完成下列问题
    • ¥15 C#算法问题, 不知道怎么处理这个数据的转换
    • ¥15 YoloV5 第三方库的版本对照问题
    • ¥15 请完成下列相关问题!
    • ¥15 drone 推送镜像时候 purge: true 推送完毕后没有删除对应的镜像,手动拷贝到服务器执行结果正确在样才能让指令自动执行成功删除对应镜像,如何解决?
    • ¥15 求daily translation(DT)偏差订正方法的代码
    • ¥15 js调用html页面需要隐藏某个按钮