当我使用pulseIn()函数时,mpu6050传感器值变得不稳定

时间:2016-08-18 06:10:03

标签: arduino arduino-uno dmp

您好我正在使用arduino uno制作四轴飞行器(无人机)。

我有关于pulseIn功能的问题。

我发现dmp过滤器开源。我在循环函数中添加了4行pulseIn函数,以便从发送器接收器接收值。

当我在循环函数中添加这4行代码时,ypr [2]的值变得非常不稳定,如图片。

ypr [2]是由dmp代码frome mpu6050传感器制作的角度值。

当我删除4行(channel1 = pulseIn(7,HIGH); ....)时,不会出现此问题。并且ypr [2]的值是稳定的。

我尝试改变pulseIn功能的输入时间参数(channel1 = pulseIn(7,HIGH,10)或channel1 = pulseIn(7,HIGH,100)或channel1 = pulseIn(7,HIGH,1000)),但是这个问题不会消失。

如果(channel1 = pulseIn(7,HIGH,10)),ypr [2]的值是稳定的。但无法接收发射机接收器值.......

我该如何解决这个问题?请帮帮我T.T

当我添加4行pulsein函数时,我确认增加了代码更新时间。 (0.014s> 0.018s)但我不知道是否存在关系。

#include <Servo.h>
#include <Wire.h>
#include <I2Cdev.h>
#include <SPI.h>
#include <MPU6050_6Axis_MotionApps20.h>

MPU6050 mpu;                           // mpu interface object

bool dmpReady = false;  // set true if DMP init was successful
uint8_t mpuIntStatus;   // holds actual interrupt status byte from MPU
uint8_t devStatus;      // return status after each device operation (0 = success, !0 = error)
uint16_t packetSize;    // expected DMP packet size (default is 42 bytes)
uint16_t fifoCount;     // count of all bytes currently in FIFO
uint8_t fifoBuffer[64]; // FIFO storage buffer

double channel1 ;
double channel2 ;
double channel3 ;
double channel4 ;

Quaternion q;
VectorFloat gravity;

float ypr[3];
float yprLast[3];
int16_t gyro[3];

volatile bool mpuInterrupt = false;
void dmpDataReady() {
  mpuInterrupt = true;
}



////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

void dmpsetup() {
#if I2CDEV_IMPLEMENTATION == I2CDEV_ARDUINO_WIRE
  Wire.begin();
  TWBR = 12; // 400kHz I2C clock (200kHz if CPU is 8MHz)
#elif I2CDEV_IMPLEMENTATION == I2CDEV_BUILTIN_FASTWIRE
  Fastwire::setup(400, true);
#endif
  Serial.begin(115200);
  while (!Serial);
  Serial.println(F("Initializing I2C devices..."));
  mpu.initialize();
  Serial.println(F("Testing device connections..."));
  Serial.println(mpu.testConnection() ? F("MPU6050 connection successful") : F("MPU6050 connection failed"));
  Serial.println(F("Initializing DMP..."));
  devStatus = mpu.dmpInitialize();
  mpu.setXGyroOffset(33);
  mpu.setYGyroOffset(-13);
  mpu.setZGyroOffset(8);
  mpu.setZAccelOffset(1416);



  if (devStatus == 0) {
    Serial.println(F("Enabling DMP..."));
    mpu.setDMPEnabled(true);
    Serial.println(F("Enabling interrupt detection (Arduino external interrupt 0)..."));
    attachInterrupt(0, dmpDataReady, RISING);
    mpuIntStatus = mpu.getIntStatus();
    Serial.println(F("DMP ready! Waiting for first interrupt..."));
    dmpReady = true;
    packetSize = mpu.dmpGetFIFOPacketSize();
  }
  else {
    Serial.print(F("DMP Initialization failed (code "));
    Serial.print(devStatus);
    Serial.println(F(")"));
  }
}

void setup() {

  Serial.begin(115200);

  pinMode(7, INPUT);
  pinMode(8, INPUT);
  pinMode(12, INPUT);
  pinMode(13, INPUT);

  dmpsetup();

}

void dmploop() {
  if (!dmpReady) return;
  while (!mpuInterrupt && fifoCount < packetSize) {
  }
  mpuInterrupt = false;
  mpuIntStatus = mpu.getIntStatus();
  fifoCount = mpu.getFIFOCount();
  if ((mpuIntStatus & 0x10) || fifoCount == 1024) {
    mpu.resetFIFO();
    // Serial.println(F("FIFO overflow!"));
  } else if (mpuIntStatus & 0x02) {
    while (fifoCount < packetSize) fifoCount = mpu.getFIFOCount();
    mpu.getFIFOBytes(fifoBuffer, packetSize);
    fifoCount -= packetSize;
    mpu.dmpGetGyro(gyro, fifoBuffer);
    mpu.dmpGetQuaternion(&q, fifoBuffer);
    mpu.dmpGetGravity(&gravity, &q);
    mpu.dmpGetYawPitchRoll(ypr, &q, &gravity);
    ypr[1] = (ypr[1] * 180 / M_PI); //x
    ypr[2] = (ypr[2] * 180 / M_PI); //y
    ypr[0] = (ypr[0] * 180 / M_PI); //z
    //gyro[0],gyro[1],gyro[2]//x,y,z 각속도값
  }
}



void loop() {
  dmploop(); //refresh new angle datas from MPU6050

  channel1 = pulseIn(7, HIGH);
  channel2 = pulseIn(8, HIGH);
  channel3 = pulseIn(12, HIGH);
  channel4 = pulseIn(13, HIGH);

  Serial.print(-30);
  Serial.print(" ");
  Serial.print(30);
  Serial.print(" ");
  Serial.print(0);
  Serial.print(" ");
  Serial.println(ypr[2]);

}

只需4行

enter image description here

1 个答案:

答案 0 :(得分:1)

PulseIn(pin,HIGH)正在阻止呼叫,它等待LO-> HI转换,然后HI-> LO来测量它。我认为这会影响代码时间。

如果你不做非阻塞,你必须使用引脚更改中断(AVR的东西,它可能不会在arduino框架中实现)。但它似乎在PinChangeInt等一些库中实现。