Allspace 发表于 2024-2-14 09:59:42

目前看START命令的结果是对的,SCL和SDA引脚都从空闲状态(高电平)变成忙碌状态(低电平)。只是发送数据的命令不被I2C控制器执行。在这个步骤出问题,是不是不需要考虑硬件电路的问题?此时应该还不涉及从机的应答,应该是主机自身的问题,是吧?

神农鼎 发表于 2024-2-14 10:15:43

STC8G/STC8H的 I2C, 做的超级完美 !
咋整,也是你程序有细微不同的问题,用仿真功能,
单步跑下,一行行对比,看你程序错在哪,
和我们最新数据手册的程序有哪些不一样的地方

Allspace 发表于 2024-2-14 10:19:34

王昱顺 发表于 2024-2-14 09:58
考虑一下是否是从机未响应,之前自己做了一个小的IIC转UART通讯工具,使用官方库。
会发现如果IIC从机( ...

我的情况是主机在START命令后发送从机地址的时候,主机的发送完成不置位。个人理解,应该在主机成功发送从机地址后,等待从机的ACK时才需要从机响应。

神农鼎 发表于 2024-2-14 10:22:28

用 STC8H8K64U实验箱的测试程序测试下



深圳国芯人工智能有限公司-实验箱 (stcai.com)

Allspace 发表于 2024-2-14 10:23:07

神农鼎 发表于 2024-2-14 10:15
STC8G/STC8H的 I2C, 做的超级完美 !
咋整,也是你程序有细微不同的问题,用仿真功能,
单步跑下,一行行对 ...

看一下前面的仿真截图啊,那已经是示例程序了,有同样的问题。我倒是希望是我硬件电路有问题。但从分析上看,又感觉和硬件电路没啥关系。

神农鼎 发表于 2024-2-14 10:24:22


第28个演示程序,几万人测试通过了

神农鼎 发表于 2024-2-14 10:27:15

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

本例程基于STC8H8K64U为主控芯片的实验箱9进行编写测试,STC8G、STC8H系列芯片可通用参考.

用STC的MCU的IO方式驱动8位数码管.

使用Timer0的16位自动重装来产生1ms节拍,程序运行于这个节拍下,用户修改MCU主时钟频率时,自动定时于1ms.

通过硬件I2C接口读取AT24C02前8个字节数据,并在数码管显示.

将读取的数据加1后写回AT24C02前8个字节.

重新读取AT24C02前8个字节数据,并在数码管显示.

MCU上电后执行1次以上动作,可重复断电/上电测试AT24C02前8个字节的数据内容.

下载时, 选择时钟 24MHZ (用户可自行修改频率).

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

神农鼎 发表于 2024-2-14 10:28:21

/* --- STC 1T Series MCU Demo Programme -------------------------------*/
/* --- Mobile: (86)13922805190 ----------------------------------------*/
/* --- Fax: 86-0513-55012956,55012947,55012969 ------------------------*/
/* --- Tel: 86-0513-55012928,55012929,55012966 ------------------------*/
/* --- Web: www.STCAI.com ---------------------------------------------*/
/* --- BBS: www.STCAIMCU.com-----------------------------------------*/
/* 如果要在程序中使用此代码,请在程序中注明使用了STC的资料及程序            */
/*---------------------------------------------------------------------*/


/*************本程序功能说明**************

本例程基于STC8H8K64U为主控芯片的实验箱9进行编写测试,STC8G、STC8H系列芯片可通用参考.

用STC的MCU的IO方式驱动8位数码管.

使用Timer0的16位自动重装来产生1ms节拍,程序运行于这个节拍下,用户修改MCU主时钟频率时,自动定时于1ms.

通过硬件I2C接口读取AT24C02前8个字节数据,并在数码管显示.

将读取的数据加1后写回AT24C02前8个字节.

重新读取AT24C02前8个字节数据,并在数码管显示.

MCU上电后执行1次以上动作,可重复断电/上电测试AT24C02前8个字节的数据内容.

下载时, 选择时钟 24MHZ (用户可自行修改频率).

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

#include "stc8h.h"       //包含此头文件后,不需要再包含"reg51.h"头文件
#include "intrins.h"

#define   MAIN_Fosc       24000000L   //定义主时钟

typedef   unsigned char   u8;
typedef   unsigned int    u16;
typedef   unsigned long   u32;

sbit SDA = P2^4;
sbit SCL = P2^5;

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

#define DIS_DOT   0x20
#define DIS_BLACK   0x10
#define DIS_      0x11

#define SLAW    0xA0
#define SLAR    0xA1


/****************************** 用户定义宏 ***********************************/

#define Timer0_Reload   (65536UL -(MAIN_Fosc / 1000))       //Timer 0 中断频率, 1000次/秒

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


/*************本地常量声明    **************/
u8 code t_display[]={                     //标准字库
//   0    1    2    3    4    5    6    7    8    9    A    B    C    D    E    F
    0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F,0x77,0x7C,0x39,0x5E,0x79,0x71,
//black-   H    J    K    L    N    o   P    U   t    G    Q    r   M    y
    0x00,0x40,0x76,0x1E,0x70,0x38,0x37,0x5C,0x73,0x3E,0x78,0x3d,0x67,0x50,0x37,0x6e,
    0xBF,0x86,0xDB,0xCF,0xE6,0xED,0xFD,0x87,0xFF,0xEF,0x46};    //0. 1. 2. 3. 4. 5. 6. 7. 8. 9. -1

u8 code T_COM[]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80};      //位码

/*************本地变量声明    **************/

u8LED8;      //显示缓冲
u8display_index;//显示位索引
bit B_1ms;          //1ms标志

/*************本地函数声明    **************/
void WriteNbyte(u8 addr, u8 *p, u8 number);
void ReadNbyte( u8 addr, u8 *p, u8 number);
void delay_ms(u8 ms);

/**********************************************/
void main(void)
{
    u8i;
    u8tmp;

    P_SW2 |= 0x80; //扩展寄存器(XFR)访问使能

    P0M1 = 0x30;   P0M0 = 0x30;   //设置P0.4、P0.5为漏极开路(实验箱加了上拉电阻到3.3V)
    P1M1 = 0x30;   P1M0 = 0x30;   //设置P1.4、P1.5为漏极开路(实验箱加了上拉电阻到3.3V)
    P2M1 = 0x3c;   P2M0 = 0x3c;   //设置P2.2~P2.5为漏极开路(实验箱加了上拉电阻到3.3V)
    P3M1 = 0x50;   P3M0 = 0x50;   //设置P3.4、P3.6为漏极开路(实验箱加了上拉电阻到3.3V)
    P4M1 = 0x3c;   P4M0 = 0x3c;   //设置P4.2~P4.5为漏极开路(实验箱加了上拉电阻到3.3V)
    P5M1 = 0x0c;   P5M0 = 0x0c;   //设置P5.2、P5.3为漏极开路(实验箱加了上拉电阻到3.3V)
    P6M1 = 0xff;   P6M0 = 0xff;   //设置为漏极开路(实验箱加了上拉电阻到3.3V)
    P7M1 = 0x00;   P7M0 = 0x00;   //设置为准双向口

    display_index = 0;

    for(i=0; i<8; i++)LED8 = 0x10; //上电消隐

    AUXR = 0x80;    //Timer0 set as 1T, 16 bits timer auto-reload,
    TH0 = (u8)(Timer0_Reload / 256);
    TL0 = (u8)(Timer0_Reload % 256);
    ET0 = 1;    //Timer0 interrupt enable
    TR0 = 1;    //Tiner0 run

    P_SW2 |= 0x10;            //I2C选择P2.4,P2.5
    I2CCFG = 0xe0;            //使能I2C主机模式
    I2CMSST = 0x00;

    EA = 1;   //打开总中断

    ReadNbyte(0, tmp, 8);
    for(i=0; i<8; i++)LED8 = tmp & 0x0f;

    for(i=0; i<8; i++)
    {
      tmp += 1;
    }
    WriteNbyte(0, tmp, 8);
    delay_ms(250);
    delay_ms(250);

    ReadNbyte(0, tmp, 8);
    for(i=0; i<8; i++)LED8 = tmp & 0x0f;

    while(1);
}

/********************** 显示扫描函数 ************************/
void DisplayScan(void)
{   
    P7 = ~T_COM;
    P6 = ~t_display];
    if(++display_index >= 8)    display_index = 0;//8位结束回0
}

//========================================================================
// 函数: void delay_ms(u8 ms)
// 描述: 延时函数。
// 参数: ms,要延时的ms数, 这里只支持1~255ms. 自动适应主时钟.
// 返回: none.
// 版本: VER1.0
// 日期: 2021-3-9
// 备注:
//========================================================================
void delay_ms(u8 ms)
{
   u16 i;
   do{
          i = MAIN_Fosc / 10000;
          while(--i);   //10T per loop
   }while(--ms);
}

/********************** Timer0 1ms中断函数 ************************/
void timer0 (void) interrupt 1
{
    DisplayScan();//1ms扫描显示一位
    B_1ms = 1;      //1ms标志
}


void Wait()
{
    while (!(I2CMSST & 0x40));
    I2CMSST &= ~0x40;
}

void Start()
{
    I2CMSCR = 0x01;                         //发送START命令
    Wait();
}

void SendData(char dat)
{
    I2CTXD = dat;                           //写数据到数据缓冲区
    I2CMSCR = 0x02;                         //发送SEND命令
    Wait();
}

void RecvACK()
{
    I2CMSCR = 0x03;                         //发送读ACK命令
    Wait();
}

char RecvData()
{
    I2CMSCR = 0x04;                         //发送RECV命令
    Wait();
    return I2CRXD;
}

void SendACK()
{
    I2CMSST = 0x00;                         //设置ACK信号
    I2CMSCR = 0x05;                         //发送ACK命令
    Wait();
}

void SendNAK()
{
    I2CMSST = 0x01;                         //设置NAK信号
    I2CMSCR = 0x05;                         //发送ACK命令
    Wait();
}

void Stop()
{
    I2CMSCR = 0x06;                         //发送STOP命令
    Wait();
}

void WriteNbyte(u8 addr, u8 *p, u8 number)/*WordAddress,First Data Address,Byte lenth   */
{
    Start();                              //发送起始命令
    SendData(SLAW);                         //发送设备地址+写命令
    RecvACK();
    SendData(addr);                         //发送存储地址
    RecvACK();
    do
    {
      SendData(*p++);
      RecvACK();
    }
    while(--number);
    Stop();                                 //发送停止命令
}

void ReadNbyte(u8 addr, u8 *p, u8 number)   /*WordAddress,First Data Address,Byte lenth   */
{
    Start();                              //发送起始命令
    SendData(SLAW);                         //发送设备地址+写命令
    RecvACK();
    SendData(addr);                         //发送存储地址
    RecvACK();
    Start();                              //发送起始命令
    SendData(SLAR);                         //发送设备地址+读命令
    RecvACK();
    do
    {
      *p = RecvData();
      p++;
      if(number != 1) SendACK();          //send ACK
    }
    while(--number);
    SendNAK();                              //send no ACK      
    Stop();                                 //发送停止命令
}
/****************************/


神农鼎 发表于 2024-2-14 10:30:50

改下 I2C 在哪的设置,打开内部 4K 上拉电阻,或外部加上 4K/10K 上拉电阻,
你用这个程序来小改,如还调不通,只好请陈工放假期间来帮你了

神农鼎 发表于 2024-2-14 10:37:04

另外 8-Pin MCU STC8G1K08A-36I-SOP8,
有 T0/T1定时器,无T2定时器,
有 CCP0/CCP1/CCP2, 3通道 PWM定时器

页: 1 2 3 [4] 5
查看完整版本: 8H/8G 的I2C还需要外部上拉电阻吗?