| 먼저 코드 전문 첨부합니다.     #include <I2Cdev.h>#include <SoftwareSerial.h>
 #include <Servo.h>
 #include <Wire.h>
 // class default I2C address is 0x68// specific I2C addresses may be passed as a parameter here
 // AD0 low = 0x68 (default for InvenSense evaluation board)
 // AD0 high = 0x69
 const int MPU_addr = 0x68;
 #define INTERRUPT_PIN 2  // use pin 2 on Arduino Uno & most boards#define LED_PIN 13 // (Arduino is 13)
 // MPU control/status varsint16_t AcX, AcY, AcZ, Tmp, GyX, GyY, GyZ; //acc, tmp, gyro data
 SoftwareSerial bluetooth(4, 3);Servo left;
 Servo right;
 // ================================================================
 // ===                      INITIAL SETUP                       ===
 // ================================================================
 void setup(){
 mpuinit();
 calibAccelGyro();
 initDT();
 left.attach(5, 0, 2000);              //pin 5, range 0~2000
 right.attach(6, 0, 2000);             //pin 6, range 0~2000
 //Serial.setTimeout(50);                //50ms delayed
 Serial.begin(115200);
 bluetooth.begin(115200);
 Wire.begin();
   pinMode(INTERRUPT_PIN, INPUT);   // configure LED for outputpinMode(LED_PIN, OUTPUT);
 
 }
 // ================================================================
 // ===                    MAIN PROGRAM LOOP                     ===
 // ================================================================
 void loop(){
 static int cnt;
 cnt++;
 if(cnt%200 == 0)
 BTreceive();                   //receive maxoutput
 readAccelGyro();
 calcDT();                        //calc peroid time
 calcAccelYPR();                  //calc accel
 calcGyro();                      //calc gyro
 calccompliYPR();                 //using complimentary filter
 Validate();                      //validate data
 PWMtransmit();                   //transmit PWM to ESC
 }
 // ================================================================// ===                       SUBROUTINES                        ===
 // ================================================================
 void mpuinit()
 {
 Wire.begin();
 Wire.beginTransmission(MPU_addr); //comm I2C that having addr 0x68
 Wire.write(0x6B);
 Wire.write(0);
 Wire.endTransmission(true);
 }
 void readAccelGyro(){
 Wire.beginTransmission(MPU_addr); //I2C comm start
 Wire.write(0x3B);                 //save 0x3B
 Wire.endTransmission(false);
 //transmit restart msg (remains connection)
 Wire.requestFrom(MPU_addr, 14, true);
 //0x68 -> 0x3B ~ 48, total 14byte save
 AcX = Wire.read() << 8 | Wire.read();
 AcY = Wire.read() << 8 | Wire.read();
 AcZ = Wire.read() << 8 | Wire.read();
 Tmp = Wire.read() << 8 | Wire.read();
 GyX = Wire.read() << 8 | Wire.read();
 GyY = Wire.read() << 8 | Wire.read();
 GyZ = Wire.read() << 8 | Wire.read();
 }
 int PWM;float dt;
 float accel_angle_x, accel_angle_y, accel_angle_z;
 float gyro_angle_x, gyro_angle_y, gyro_angle_z;
 float compli_a_y;
 float baseAcX, baseAcY, baseAcZ;            //accel avg
 float baseGyX, baseGyY, baseGyZ;            //gyro avg
 void BTreceive(){
 /*receiving signal from master HC-06           */
 char buffer[3];                 //input data into array
 char i = 0;                     //index
   while (bluetooth.available()){
 buffer[i] = bluetooth.read(); //receiving data via bluetooth
 i++;                          //counting
 }
   PWM = atoi(buffer);         //char to int conversion/*initialize buffer & index                    */
 for (int a = 0; a < 4; a++)
 {
 buffer[a] = NULL;
 i = 0;
 }
 }
 void calibAccelGyro(){
 float sumAcX = 0, sumAcY = 0, sumAcZ = 0;
 float sumGyX = 0, sumGyY = 0, sumGyZ = 0;
   readAccelGyro();   //calc avgfor(int i=0; i<10; i++)
 {
 readAccelGyro();
 sumAcX += AcX; sumAcY += AcY; sumAcZ += AcZ;
 sumGyX += GyX; sumGyY += GyY; sumGyZ += GyZ;
 delay(100);
 }
 baseAcX = sumAcX / 10;
 baseAcY = sumAcY / 10;
 baseAcZ = sumAcZ / 10;
 baseGyX = sumGyX / 10;
 baseGyY = sumGyY / 10;
 baseGyZ = sumGyZ / 10;
 }
 unsigned long t_now;             //current measuring period timeunsigned long t_prev;            //prev measuring period time
 void initDT(){
 t_prev = millis();
 }
 void calcDT(){
 t_now = millis();
 dt = (t_now - t_prev) / 1000.0; //convert ms to sec
 t_prev = t_now;
 }
 void calcAccelYPR(){
 float accel_x, accel_y, accel_z;     //final compesated value
 float accel_xz, accel_yz;
 const float RADIANS_TO_DEGREES = 180/PI;
   accel_x = AcX - baseAcX;                 // ac_x now - avgaccel_y = AcY - baseAcY;
 accel_z = AcZ + (16384 - baseAcZ);       // make value 9.81m/s^2
   //calc angle by tiltingaccel_yz = sqrt(pow(accel_y, 2) + pow(accel_z, 2));
 accel_angle_y = atan(-accel_x / accel_yz)*RADIANS_TO_DEGREES;
   accel_xz = sqrt(pow(accel_x, 2) + pow(accel_z, 2));accel_angle_x = atan(accel_y / accel_xz)*RADIANS_TO_DEGREES;
   accel_angle_z = 0;}
 float gyro_x, gyro_y, gyro_z; //global var that saves gyro void calcGyro(){
 const float GYROXYZ_TO_DEGREES_PER_SED = 131;
 
 gyro_x = (GyX - baseGyX) / GYROXYZ_TO_DEGREES_PER_SED;
 gyro_y = (GyY - baseGyY) / GYROXYZ_TO_DEGREES_PER_SED;
 gyro_z = (GyZ - baseGyZ) / GYROXYZ_TO_DEGREES_PER_SED;
 }
 void calccompliYPR(){
 const float ALPHA = 0.96;
 float tmp_a_y;  //prev filtered value
   tmp_a_y = compli_a_y + gyro_y * dt;   compli_a_y = ALPHA * tmp_a_y + (1.0 - ALPHA) * accel_angle_y;}
 int roll_angle, yaw_gyro; void Validate(){
 yaw_gyro = abs(gyro_z);                    //absolute value
 roll_angle = compli_a_y;
   Serial.print("roll angle:\t");Serial.print(roll_angle);
 Serial.print("\tyaw gyro:\t");
 Serial.println(yaw_gyro);
 delay(5);
 }
 void PWMtransmit(){
 int turnover = 100;                        //stop standard gyro
 int minv = 1500;                           //stop PWM signal
 int maxoutput, biasoutput;
 int chkl, chkr;                            //removable
   maxoutput = map(PWM, 0, 255, minv, 2000);biasoutput = map(maxoutput, minv, 2000, minv, 1900);
   if (yaw_gyro < turnover){
 if (roll_angle < 0)
 {
 left.writeMicroseconds(maxoutput);
 right.writeMicroseconds(biasoutput);
 chkl = maxoutput;                      //removable
 chkr = biasoutput;                     //removable
 }
 else if (roll_angle > 0)
 {
 left.writeMicroseconds(biasoutput);
 right.writeMicroseconds(maxoutput);
 chkl = biasoutput;                     //removable
 chkr = maxoutput;                      //removable
 }
 }
 else
 {
 left.writeMicroseconds(minv);
 right.writeMicroseconds(minv);
 chkl = biasoutput;                       //removable
 chkr = biasoutput;                       //removable
 Serial.println("drift detected! stop for a 2s.");
 delay(2000);
 }
 //===================== removable ======================
 Serial.print("PWM : ");
 Serial.println(PWM);
 Serial.print("bias throttle : ");
 Serial.println(biasoutput);
 Serial.print("max throttle : ");
 Serial.println(maxoutput);
 Serial.print("left throttle : ");
 Serial.println(chkl);
 Serial.print("right throttle : ");
 Serial.println(chkr);
 }
   블루투스로 아날로그 입력값을 전달받아서, BLDC 모터를 제어하는 코드인데, 슬레이브 보드에 블루투스 값을 저장하는 버퍼만 코딩해서 작동할 때에는 안정적으로 값을 받는데, 이걸 서브루틴으로 빼서 코딩하면 값을 안정적으로 받지 못하더군요 (e.g 255 입력시 2, 5, 5 끊어서 저장). 이 서브루틴의 딜레이를 500ms 정도로 주면 해결되긴 하는데, 그러면 메인 루프의 연산속도가 느려져서 필터 연산에 차질이 생깁니다. 이 문제를 어떻게 해결해야 할까요? |