OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 2683|回复: 3

求助STM32H743+mpu6050无法读取器件ID

[复制链接]

1

主题

7

帖子

0

精华

新手上路

积分
43
金钱
43
注册时间
2020-2-3
在线时间
15 小时
发表于 2020-7-10 14:34:24 | 显示全部楼层 |阅读模式
3金钱
我把f103驱动mpu6050的代码移植到H743上结果死活都找不到器件ID,球球大佬帮帮小弟吧
贴上源代码:
#include "mpu6050.h"
#include "sys.h"
#include "delay.h"
#include "usart.h"
//////////////////////////////////////////////////////////////////////////////////         
//本程序只供学习使用,未经作者许可,不得用于其它任何用途
//ALIENTEK NANO STM32F103开发板
//MPU6050 驱动代码          
//正点原子@ALIENTEK
//技术论坛:www.openedv.com
//创建日期:2018/7/28
//版本:V1.0
//版权所有,盗版必究。
//Copyright(C) 广州市星翼电子科技有限公司 2018-2028
//All rights reserved                                                                          
//////////////////////////////////////////////////////////////////////////////////
u8 MPU_Write_Byte(u8 reg,u8 data)         ;
//初始化MPU6050
//返回值:0,成功
//    其他,错误代码
u8 MPU_Init(void)
{
                 u8 res;
  IIC_Init();//初始化IIC总线
        delay_ms(500);
        MPU_Write_Byte(MPU_PWR_MGMT1_REG,0X80);        //复位MPU6050
        delay_ms(500);
        MPU_Write_Byte(MPU_PWR_MGMT1_REG,0X00);//唤醒MPU6050
       
        MPU_Set_Gyro_Fsr(3);                                        //陀螺仪传感器,±2000dps
        MPU_Set_Accel_Fsr(0);                                        //加速度传感器,±2g
        MPU_Set_Rate(50);                                                //设置采样率50Hz
        MPU_Write_Byte(MPU_INT_EN_REG,0X00);        //关闭所有中断
        MPU_Write_Byte(MPU_USER_CTRL_REG,0X00);        //I2C主模式关闭
        MPU_Write_Byte(MPU_FIFO_EN_REG,0X00);        //关闭FIFO
        MPU_Write_Byte(MPU_INTBP_CFG_REG,0X80);        //INT引脚低电平有效
        res=MPU_Read_Byte(MPU_DEVICE_ID_REG);
        if(res==MPU_ADDR)//器件ID正确
        {
                MPU_Write_Byte(MPU_PWR_MGMT1_REG,0X01);        //设置CLKSEL,PLL X轴为参考
                MPU_Write_Byte(MPU_PWR_MGMT2_REG,0X00);        //加速度与陀螺仪都工作
                MPU_Set_Rate(50);                                                //设置采样率为50Hz
        }else return 1;
        return 0;
}
//设置MPU6050陀螺仪传感器满量程范围
//fsr:0,±250dps;1,±500dps;2,±1000dps;3,±2000dps
//返回值:0,设置成功
//    其他,设置失败
u8 MPU_Set_Gyro_Fsr(u8 fsr)
{
        return MPU_Write_Byte(MPU_GYRO_CFG_REG,fsr<<3);//设置陀螺仪满量程范围  
}
//设置MPU6050加速度传感器满量程范围
//fsr:0,±2g;1,±4g;2,±8g;3,±16g
//返回值:0,设置成功
//    其他,设置失败
u8 MPU_Set_Accel_Fsr(u8 fsr)
{
        return MPU_Write_Byte(MPU_ACCEL_CFG_REG,fsr<<3);//设置加速度传感器满量程范围  
}
//设置MPU6050的数字低通滤波器
//lpf:数字低通滤波频率(Hz)
//返回值:0,设置成功
//    其他,设置失败
u8 MPU_Set_LPF(u16 lpf)
{
        u8 data=0;
        if(lpf>=188)data=1;
        else if(lpf>=98)data=2;
        else if(lpf>=42)data=3;
        else if(lpf>=20)data=4;
        else if(lpf>=10)data=5;
        else data=6;
        return MPU_Write_Byte(MPU_CFG_REG,data);//设置数字低通滤波器  
}
//设置MPU6050的采样率(假定Fs=1KHz)
//rate:4~1000(Hz)
//返回值:0,设置成功
//    其他,设置失败
u8 MPU_Set_Rate(u16 rate)
{
        u8 data;
        if(rate>1000)rate=1000;
        if(rate<4)rate=4;
        data=1000/rate-1;
        data=MPU_Write_Byte(MPU_SAMPLE_RATE_REG,data);        //设置数字低通滤波器
        return MPU_Set_LPF(rate/2);        //自动设置LPF为采样率的一半
}

//得到温度值的100倍
short MPU_Get_Temperature(void)
{
    u8 buf[2];
    short raw;
    float temp;
        MPU_Read_Len(MPU_ADDR,MPU_TEMP_OUTH_REG,2,buf);
    raw=((u16)buf[0]<<8)|buf[1];  
    temp=36.53+((double)raw)/340;  
    return temp*100;
}
//得到陀螺仪值(原始值)
//gx,gy,gz:陀螺仪x,y,z轴的原始读数(带符号)
//返回值:0,成功
//    其他,错误代码
u8 MPU_Get_Gyroscope(short *gx,short *gy,short *gz)
{
    u8 buf[6],res;  
        res=MPU_Read_Len(MPU_ADDR,MPU_GYRO_XOUTH_REG,6,buf);
        if(res==0)
        {
                *gx=((u16)buf[0]<<8)|buf[1];  
                *gy=((u16)buf[2]<<8)|buf[3];  
                *gz=((u16)buf[4]<<8)|buf[5];
        }        
    return res;;
}
//得到加速度值(原始值)
//gx,gy,gz:陀螺仪x,y,z轴的原始读数(带符号)
//返回值:0,成功
//    其他,错误代码
u8 MPU_Get_Accelerometer(short *ax,short *ay,short *az)
{
    u8 buf[6],res;  
        res=MPU_Read_Len(MPU_ADDR,MPU_ACCEL_XOUTH_REG,6,buf);
        if(res==0)
        {
                *ax=((u16)buf[0]<<8)|buf[1];  
                *ay=((u16)buf[2]<<8)|buf[3];  
                *az=((u16)buf[4]<<8)|buf[5];
        }        
    return res;;
}
//IIC连续写
//addr:器件地址
//reg:寄存器地址
//len:写入长度
//buf:数据区
//返回值:0,正常
//    其他,错误代码
u8 MPU_Write_Len(u8 addr,u8 reg,u8 len,u8 *buf)
{
        u8 i;
    IIC_Start();
        IIC_Send_Byte((addr<<1)|0);//发送器件地址+写命令       
        if(IIC_Wait_Ack())        //等待应答
        {
                IIC_Stop();                 
                return 1;               
        }
    IIC_Send_Byte(reg);        //写寄存器地址
    IIC_Wait_Ack();                //等待应答
        for(i=0;i<len;i++)
        {
                IIC_Send_Byte(buf);        //发送数据
                if(IIC_Wait_Ack())                //等待ACK
                {
                        IIC_Stop();         
                        return 1;                 
                }               
        }   
   IIC_Stop();         
        return 0;       
}
//IIC连续读
//addr:器件地址
//reg:要读取的寄存器地址
//len:要读取的长度
//buf:读取到的数据存储区
//返回值:0,正常
//    其他,错误代码
u8 MPU_Read_Len(u8 addr,u8 reg,u8 len,u8 *buf)
{
        IIC_Start();
        IIC_Send_Byte((addr<<1)|0);//发送器件地址+写命令       
        if(IIC_Wait_Ack())        //等待应答
        {
                IIC_Stop();                 
                return 1;               
        }
    IIC_Send_Byte(reg);        //写寄存器地址
    IIC_Wait_Ack();                //等待应答
    IIC_Start();
        IIC_Send_Byte((addr<<1)|1);//发送器件地址+读命令       
    IIC_Wait_Ack();                //等待应答
        while(len)
        {
                if(len==1)*buf=IIC_Read_Byte(0);//读数据,发送nACK
                else *buf=IIC_Read_Byte(1);                //读数据,发送ACK  
                len--;
                buf++;
        }   
    IIC_Stop();        //产生一个停止条件
        return 0;       
}
//IIC写一个字节
//reg:寄存器地址
//data:数据
//返回值:0,正常
//    其他,错误代码
u8 MPU_Write_Byte(u8 reg,u8 data)                                  
{
    IIC_Start();
        IIC_Send_Byte((MPU_ADDR<<1)|0);//发送器件地址+写命令       
        if(IIC_Wait_Ack())        //等待应答
        {
                IIC_Stop();                 
                return 1;               
        }
    IIC_Send_Byte(reg);        //写寄存器地址
    IIC_Wait_Ack();                //等待应答
        IIC_Send_Byte(data);//发送数据
        if(IIC_Wait_Ack())        //等待ACK
        {
                IIC_Stop();         
                return 1;                 
        }                 
    IIC_Stop();         
        return 0;
}
//IIC读一个字节
//reg:寄存器地址
//返回值:读到的数据
u8 MPU_Read_Byte(u8 reg)
{
        u8 res;
    IIC_Start();
        IIC_Send_Byte((MPU_ADDR<<1)|0);//发送器件地址+写命令       
        IIC_Wait_Ack();                //等待应答
    IIC_Send_Byte(reg);        //写寄存器地址
    IIC_Wait_Ack();                //等待应答
    IIC_Start();
        IIC_Send_Byte((MPU_ADDR<<1)|1);//发送器件地址+读命令       
    IIC_Wait_Ack();                //等待应答
        res=IIC_Read_Byte(0);//读取数据,发送nACK
    IIC_Stop();                        //产生一个停止条件
        return res;               
}

#include "iic.h"
#include "delay.h"
//////////////////////////////////////////////////////////////////////////////////         
//本程序只供学习使用,未经作者许可,不得用于其它任何用途
//ALIENTEK STM32F7开发板
//IIC驱动代码          
//正点原子@ALIENTEK
//技术论坛:www.openedv.com
//创建日期:2015/12/28
//版本:V1.0
//版权所有,盗版必究。
//Copyright(C) 广州市星翼电子科技有限公司 2014-2024
//All rights reserved                                                                          
//////////////////////////////////////////////////////////////////////////////////        

//IIC初始化
void IIC_Init(void)
{
    GPIO_InitTypeDef GPIO_Initure;

    __HAL_RCC_GPIOH_CLK_ENABLE();   //使能GPIOH时钟

    //PH4,5初始化设置
    GPIO_Initure.Pin=GPIO_PIN_4|GPIO_PIN_5;
    GPIO_Initure.Mode=GPIO_MODE_OUTPUT_PP;  //推挽输出
    GPIO_Initure.Pull=GPIO_PULLUP;          //上拉
    GPIO_Initure.Speed=GPIO_SPEED_FREQ_VERY_HIGH;    //快速
          HAL_GPIO_WritePin(GPIOH,GPIO_PIN_4|GPIO_PIN_5,GPIO_PIN_SET);//PB4,PB5 输出高       
       
    HAL_GPIO_Init(GPIOH,&GPIO_Initure);

}

//产生IIC起始信号
void IIC_Start(void)
{
        SDA_OUT();     //sda线输出
        IIC_SDA(1);                    
        IIC_SCL(1);
        delay_us(2);
        IIC_SDA(0);//START:when CLK is high,DATA change form high to low
        delay_us(2);
        IIC_SCL(0);//钳住I2C总线,准备发送或接收数据
}          
//产生IIC停止信号
void IIC_Stop(void)
{
        SDA_OUT();//sda线输出
        IIC_SCL(0);
        IIC_SDA(0);//STOP:when CLK is high DATA change form low to high
        delay_us(2);
        IIC_SCL(1);
        IIC_SDA(1);//发送I2C总线结束信号
        delay_us(2);                                                                  
}
//等待应答信号到来
//返回值:1,接收应答失败
//        0,接收应答成功
u8 IIC_Wait_Ack(void)
{
        u8 ucErrTime=0;
        SDA_IN();      //SDA设置为输入  
        IIC_SDA(1);delay_us(2);          
        IIC_SCL(1);delay_us(2);         
        while(READ_SDA)
        {
                ucErrTime++;
                if(ucErrTime>250)
                {
                        IIC_Stop();
                        return 1;
                }
        }
        IIC_SCL(0);//时钟输出0           
        return 0;  
}
//产生ACK应答
void IIC_Ack(void)
{
        IIC_SCL(0);
        SDA_OUT();
        IIC_SDA(0);
        delay_us(2);
        IIC_SCL(1);
        delay_us(2);
        IIC_SCL(0);
}
//不产生ACK应答                    
void IIC_NAck(void)
{
        IIC_SCL(0);
        SDA_OUT();
        IIC_SDA(1);
        delay_us(2);
        IIC_SCL(1);
        delay_us(2);
        IIC_SCL(0);
}                                                                              
//IIC发送一个字节
//返回从机有无应答
//1,有应答
//0,无应答                          
void IIC_Send_Byte(u8 txd)
{                        
    u8 t;   
        SDA_OUT();             
    IIC_SCL(0);//拉低时钟开始数据传输
    for(t=0;t<8;t++)
    {              
        IIC_SDA((txd&0x80)>>7);
        txd<<=1;           
                delay_us(2);   //对TEA5767这三个延时都是必须的
                IIC_SCL(1);
                delay_us(2);
                IIC_SCL(0);       
                delay_us(2);
    }         
}             
//读1个字节,ack=1时,发送ACK,ack=0,发送nACK   
u8 IIC_Read_Byte(unsigned char ack)
{
        unsigned char i,receive=0;
        SDA_IN();//SDA设置为输入
    for(i=0;i<8;i++ )
        {
        IIC_SCL(0);
        delay_us(2);
                IIC_SCL(1);
        receive<<=1;
        if(READ_SDA)receive++;   
                delay_us(2);
    }                                         
    if (!ack)
        IIC_NAck();//发送nACK
    else
        IIC_Ack(); //发送ACK   
    return receive;
}




正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

1

主题

7

帖子

0

精华

新手上路

积分
43
金钱
43
注册时间
2020-2-3
在线时间
15 小时
 楼主| 发表于 2020-7-10 16:04:46 | 显示全部楼层
回复

使用道具 举报

109

主题

5564

帖子

0

精华

资深版主

Rank: 8Rank: 8

积分
10572
金钱
10572
注册时间
2017-2-18
在线时间
1914 小时
发表于 2020-7-10 18:53:30 | 显示全部楼层
基于H743的工程来修改
修改的时候注意:
1.MPU_Init()对6050模块AD0引脚所接IO的初始化
2.修改MPU_IIC_Init()中对模块SDA 、SCL引脚的GPIO初始化部分
3.mpuiic.h中的这里也得修改
其他地方基本不用改了
1.JPG
回复

使用道具 举报

1

主题

7

帖子

0

精华

新手上路

积分
43
金钱
43
注册时间
2020-2-3
在线时间
15 小时
 楼主| 发表于 2020-7-10 19:53:22 | 显示全部楼层
peng1554 发表于 2020-7-10 18:53
基于H743的工程来修改
修改的时候注意:
1.MPU_Init()对6050模块AD0引脚所接IO的初始化

好的 谢谢 但是AD0引脚所接的不是vcc或者gnd嘛 初始化是什么意思
回复

使用道具 举报

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

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-6-18 06:27

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

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