| 网站首页 | 业界新闻 | 小组 | 威客 | 人才 | 下载频道 | 博客 | 代码贴 | 在线编程 | 编程论坛
欢迎加入我们,一同切磋技术
用户名:   
 
密 码:  
共有 2896 人关注过本帖
标题:哈哈!功夫不负有心人,我成功了.
只看楼主 加入收藏
bausrpgf999
Rank: 1
等 级:新手上路
帖 子:35
专家分:0
注 册:2008-7-1
收藏
 问题点数:0 回复次数:26 
哈哈!功夫不负有心人,我成功了.
功夫不负有心人啊,并口模仿IIC读字节程序,今天终于成功
了, 总结一下,发现的错误有两个,1是读之前数据端口没有初始化,2是关键的输入判断位弄错了.当然关于延时还是不能达到十分的准确,这是以后努力的目标.字节读成功后,字节写,IIC后缓冲读写,都将很容易实现.当然本程序并不完善.感谢来到论坛后给我过帮助的人,特别是RocyCarry版主,成功于你们同在....
/*----------------------------
24c02读程序,根据单片机程序VIIC_C51.C进行改写.
使用资源定义如下
PC并口SPP,EPP模式
数据端口位0(scl):    PC输出到I2C的SCL时钟信号
数据端口位1(sda):     PC输出到I2C的SDA数据信号
状态端口位4(iicstate):I2C输入到PCI/O0X379的数据信号
----------------------------*/

#include<dos.h>
#include<bios.h>
#include<process.h>
#include <stdio.h>
#include <conio.h>
#define  uchar unsigned char /*宏定义*/
#define  uint  unsigned int
#define  sla   0xa0
#define  _Nop()  delay(3)        /*定义空指令*/

 uchar i2cdata=0;                 /*全局变量 */
/*******************************************************************
                     起动总线函数               
函数原型: void  Start_I2c();  
功能:       启动I2C总线,即发送I2C起始条件.
  
********************************************************************/
void Start_I2c()
{
   outportb(0x378,i2cdata|=2);/*SDA=1;发送起始条件的数据信号*/
  _Nop();
   outportb(0x378,i2cdata|=1); /*SCL=1;     */
  _Nop();    /*起始条件建立时间大于4.7us,延时*/
  _Nop();
  _Nop();
  _Nop();
  _Nop();   
   outportb(0x378,i2cdata&=253);/*SDA=0;发送起始信号*/
  _Nop();    /* 起始条件锁定时间大于4μs*/
  _Nop();
  _Nop();
  _Nop();
  _Nop();      
   outportb(0x378,i2cdata&=0);/*SCL=0;钳住I2C总线,准备发送或接收数据 */
  _Nop();
  _Nop();
}




/*******************************************************************
                      结束总线函数               
函数原型: void  Stop_I2c();  
功能:       结束I2C总线,即发送I2C结束条件.
  
********************************************************************/
void Stop_I2c()
{
  outportb(0x378,i2cdata&=253);/*SDA=0;发送结束条件的数据信号*/
  _Nop();   /*发送结束条件的时钟信号*/
  outportb(0x378,i2cdata|=1);/*SCL=1;结束条件建立时间大于4μs*/
  _Nop();
  _Nop();
  _Nop();
  _Nop();
  _Nop();
   outportb(0x378,i2cdata|=2);/*SDA=1;发送I2C总线结束信号*/
  _Nop();
  _Nop();
  _Nop();
  _Nop();
}




/*******************************************************************
                 字节数据传送函数               
函数原型: void  SendByte(uchar c);
功能:  将数据c发送出去,可以是地址,也可以是数据,发完后等待应答,并对
     此状态位进行操作.(不应答或非应答都使ack=0 假)     
     发送数据正常,ack=1; ack=0表示被控器无应答或损坏。
********************************************************************/
void  SendByte(uchar c)
{
 uchar temp,temp2,BitCnt;
 
 for(BitCnt=0;BitCnt<8;BitCnt++)  /*要传送的数据长度为8位*/
    {
     if((c<<BitCnt)&0x80) outportb(0x378,i2cdata|=2);/*SDA=1;判断发送位*/
       else  outportb(0x378,i2cdata&=253);/*SDA=0;*/
     _Nop();
     outportb(0x378,i2cdata|=1);/*SCL=1;置时钟线为高,通知被控器开始接收数据位*/
      _Nop();
      _Nop();               /*保证时钟高电平周期大于4μs*/
      _Nop();
      _Nop();
      _Nop();         
      outportb(0x378,i2cdata&=254);/*SCL=0; */
    }
   
    _Nop();
    _Nop();
     outportb(0x378,i2cdata|=2);/*SDA=1;8位发送完后释放数据线,准备接收应答位*/
    _Nop();
    _Nop();   
     outportb(0x378,i2cdata|=1);/*SCL=1; */
    _Nop();
    _Nop();
    _Nop();
    _Nop();
    _Nop();
    temp=inportb(0x379);
    temp2=temp;                   /*(temp2&=16)==16)原来的这里出错了,现在更正了.*/
    if((temp2&=16)==16) printf("\nSlave ack error:temp=%x    temp2=%x",temp,temp2);
       else printf("\nSlave ack ok:temp=%x    temp2=%x",temp,temp2);/*判断是否接收到应答信号*/
     outportb(0x378,i2cdata&=254);  /*SCL=0; */
    _Nop();
    _Nop();
}






/*******************************************************************
                 字节数据传送函数               
函数原型: uchar  RcvByte();
功能:  用来接收从器件传来的数据,并判断总线错误(不发应答信号),
     发完后请用应答函数。  
********************************************************************/   
uchar  RcvByte()
{
  uchar retc,temp;
  uchar BitCnt;
  
  retc=0;
  outportb(0x378,i2cdata|=2);/*SDA=1;置数据线为输入方式*/
  for(BitCnt=0;BitCnt<8;BitCnt++)
      {
        _Nop();           
        outportb(0x378,i2cdata&=254);/*SCL=0;置时钟线为低,准备接收数据位*/
        _Nop();
        _Nop();         /*时钟低电平周期大于4.7μs*/
        _Nop();
        _Nop();
        _Nop();
          outportb(0x378,i2cdata|=1);/*SCL=1;置时钟线为高使数据线上数据有效*/
        _Nop();
        _Nop();
        retc=retc<<1;
        temp=inportb(0x379);     /*(temp2&=16)==16)原来的这里出错了,现在更正了.*/
        if((temp&=16)==16)retc=retc+1; /*读数据位,接收的数据位放入retc中 */
        _Nop();
        _Nop();
      }
    outportb(0x378,i2cdata&=254);/*SCL=0;*/
  _Nop();
  _Nop();
  return(retc);
}




/********************************************************************
                     应答子函数
原型:  void Ack_I2c(bit a);
 
功能:主控器进行应答信号,(可以是应答或非应答信号)
********************************************************************/
void Ack_I2c(uchar a)
{
  
  if(a==0) outportb(0x378,i2cdata&=253);/* SDA=0;在此发出应答或非应答信号 */
        else  outportb(0x378,i2cdata|=2);/* SDA=1 */
  _Nop();
  _Nop();
  _Nop();      
   outportb(0x378,i2cdata|=1);/* SCL=1; */
    _Nop();
    _Nop();              /* 时钟低电平周期大于4μs */
    _Nop();
    _Nop();
    _Nop();  
      outportb(0x378,i2cdata&=254);/* SCL=0;清时钟线,钳住I2C总线以便继续接收 */
    _Nop();
    _Nop();   
}




/*******************************************************************
                    向有子地址器件读取多字节数据函数               
函数原型: bit  ISendStr(uchar sla,uchar suba,ucahr *s,uchar no);  
功能:     从启动总线到发送地址,子地址,读数据,结束总线的全过程,从器件
          地址sla,子地址suba,读出的内容放入s指向的存储区,读no个字节。
           如果返回1表示操作成功,否则操作有误。
注意:    使用前必须已结束总线。
*******************************************************************
uchar IRcvStr(uchar sla,uchar suba,uchar *s,uchar no)  */
void main(){
   uchar s,y;
   uchar suba;
textcolor(2);
clrscr();
/****** read ***********************************/
printf("please inport one address:");
scanf("%x",&suba);
   outportb(0x378,0);          /*原来这里没有加数据初始化,数据端口初始化*/
   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(0);                /*发送就答位*/  
     /*s++;
    }                            */
   y=RcvByte();
    Ack_I2c(1);                 /*发送非应位*/
 Stop_I2c();                    /*结束总线*/
  /* return(1); */
 printf("\n%X address data:%X\n",suba,s);
 printf("%X address data:%X",suba,y);
 getch();
}






                        /*    完毕      */
搜索更多相关主题的帖子: 功夫 哈哈 单片机程序 有心人 资源 
2008-07-05 09:25
bausrpgf999
Rank: 1
等 级:新手上路
帖 子:35
专家分:0
注 册:2008-7-1
收藏
得分:0 
程序改得很乱,主要是心里比较乱,本人现价段主要探索PC与一些硬件的通信,如IIC芯片,时钟芯片,温度芯片,液晶器件.希望能和志同道合的朋友一起努力..

林子大了,嘛鸟都有了
            
2008-07-05 09:39
很远的那颗星
Rank: 2
等 级:新手上路
威 望:4
帖 子:544
专家分:0
注 册:2008-6-30
收藏
得分:0 
单片机?
前途无量...可惜我不会.

Fighting~~~~~~~~
2008-07-05 09:47
RockCarry
Rank: 16Rank: 16Rank: 16Rank: 16
等 级:版主
威 望:13
帖 子:662
专家分:58
注 册:2005-8-5
收藏
得分:0 
呵呵,恭喜了。你也满厉害的,我以前也尝试用 IO 口模拟 IIC,也是始终调不好,不过我的主芯片是 2440,这个可比单片机强大多了,所以后来我直接用它的 IIC Controller, 然后也成功了。

关键还是要理解 IIC 协议,弄清楚时序上的相互关系,等等各种相关俄概念。然后编写程序,耐心调试就可以了。IIC 总线在空闲时都是保持高电平,在数据传送时,当 SCL 保持高电平时,要求 SDA 在 SCL 的高电平期间保持稳定,也就是说发送方只有在 SCL 变低时才有改变 SDA 电平的机会,而接受方在等到 SCL 的高电平后,就可以放心的读取 SDA 的电平高低。然而有两种情况例外,那就是开始条件和起始条件。而所谓的 ACK 指的是发送方在发完一个字节后,释放 SDA 线,也就是发送方让 SDA 恢复高,之后发送方开始等待 ACK, 也就是等待接受方将 SDA 拉低。理解以上几点,也就是理解了 IIC 的本质。仔细体会 IIC 协议,会发现其巧妙之处。

通常情况下,SCL 由 master 控制,而 SDA 则是一个双向的数据线,因此在编程时需要注意切换 IO 的输入与输出属性。ACK 是指接受方将 SDA 拉低,如果总线上没有接器件,那是不可能发生的。

[[it] 本帖最后由 RockCarry 于 2008-7-5 09:56 编辑 [/it]]
2008-07-05 09:54
卧龙孔明
Rank: 9Rank: 9Rank: 9
等 级:贵宾
威 望:59
帖 子:3872
专家分:684
注 册:2006-10-13
收藏
得分:0 
恭喜楼主一下,虽然哦看不懂.

My Blog: www.aiexp.info
虽然我的路是从这里开始的,但是这里不再是乐土.感谢曾经影响过,引导过,帮助过我的董凯,飞燕,leeco,starwing,Rockcarry,soft_wind等等等等.别了,BCCN.
2008-07-05 09:59
lnhaing
Rank: 1
等 级:新手上路
帖 子:111
专家分:0
注 册:2008-1-30
收藏
得分:0 
那我就介绍下:
/*I2C总线是由数据线SDA和时钟SCL构成的串行总线,可发送和接收数据.在 CPU与被控IC之间、IC与IC之间进行双向传送,最高传送速率400kbps.各种被控制电路均并联在这条总线上,但就像电话机一样只有拨通各自的号码才能工作,所以每个电路和模块都有唯一的地址,在信息的传输过程中,I2C总线上并接的每一模块电路既是主控器(或被控器),又是发送器(或接收器), 这取决于它所要完成的功能.CPU发出的控制信号分为地址码和控制量两部分,地址码用来选址,即接通需要控制的电路,确定控制的种类;控制量决定该调整的类别(如对比度、亮度等)及需要调整的量.这样,各控制电路虽然挂在同一条总线上,却彼此独立,互不相关.
   I2C总线在传送数据过程中共有三种类型信号, 它们分别是:开始信号、结束信号和应答信号.
   开始信号:SCL为高电平时,SDA由高电平向低电平跳变,开始传送数据.
   结束信号:SCL为低电平时,SDA由低电平向高电平跳变,结束传送数据.
   应答信号:接收数据的IC在接收到8bit数据后,向发送数据的IC发出特定的低电平脉冲,表示已收到数据.CPU向受控单元发出一个信号后,等待受控单元发出一个应答信号,CPU接收到应答信号后,根据实际情况作出是否继续传递信号的判断.若未收到应答信号,由判断为受控单元出现故障.
   目前有很多半导体集成电路上都集成了I2C接口.*/
--------------------------------------------------------
RocyCarry版主 and 楼主,是在嵌入式行业的吗?

[[it] 本帖最后由 lnhaing 于 2008-7-5 10:09 编辑 [/it]]

我来自偶然! bitter C
2008-07-05 10:07
flyue
Rank: 10Rank: 10Rank: 10
来 自:江南西道
等 级:贵宾
威 望:19
帖 子:3465
专家分:1563
注 册:2006-6-20
收藏
得分:0 
不懂。我只搞PC编程,那个单片机曾经也想学,可惜没机会啊。
我连单片机在哪里买都不知道。

天之道,损有余而补不足.人之道则不然,损不足以奉有余.孰能有余以奉天下,唯有道者.
2008-07-05 10:11
flyue
Rank: 10Rank: 10Rank: 10
来 自:江南西道
等 级:贵宾
威 望:19
帖 子:3465
专家分:1563
注 册:2006-6-20
收藏
得分:0 
_Nop();这个函数是哪里的啊?是不是跟汇编里的nop类似?

天之道,损有余而补不足.人之道则不然,损不足以奉有余.孰能有余以奉天下,唯有道者.
2008-07-05 10:16
lnhaing
Rank: 1
等 级:新手上路
帖 子:111
专家分:0
注 册:2008-1-30
收藏
得分:0 
LS说的是 _nop_();(空语句)
在#include <intrins.h>里面(以51为例)

我来自偶然! bitter C
2008-07-05 10:23
bausrpgf999
Rank: 1
等 级:新手上路
帖 子:35
专家分:0
注 册:2008-7-1
收藏
得分:0 
谢谢大家的支持..

林子大了,嘛鸟都有了
            
2008-07-05 10:41
快速回复:哈哈!功夫不负有心人,我成功了.
数据加载中...
 
   



关于我们 | 广告合作 | 编程中国 | 清除Cookies | TOP | 手机版

编程中国 版权所有,并保留所有权利。
Powered by Discuz, Processed in 0.025654 second(s), 7 queries.
Copyright©2004-2024, BCCN.NET, All Rights Reserved