Help us understand the problem. What is going on with this article?

M5StickCはMadgwickフィルタで倒立振子の夢を見るか?

こちらの情報は古いです

ソフトウェアについてはこちらの記事の方が新しいのでこちらを参照してください

BalaC(M5StickC内蔵倒立振子)をPID制御で立たせてみた件について(製品版に更新) - Qiita
https://qiita.com/coppercele/items/527228e3f08c53597bd1


倒立振子を作ろうと思ったきっかけはこちらの記事を読んだことでした

実質500円&100Stepで作る超簡単「 ゆるメカトロ的 M5StickC 倒立振子」 - Qiita
https://qiita.com/Google_Homer/items/d0cdb2975491a5ea8129

構成もシンプルで自分もチャレンジしてみようと思いました

姿勢の推定

参考記事ではMPU6886の出力をいろいろと計算して現在の角度を推定しています

が自分は全然わからないのでMadgwickフィルタを使います

実は倒立振子といえばカルマンフィルタっていうのは噂で聞いていたのですが、
ライブラリを使うにしても事前計算があったりして結構めんどくさいんですよね

Twitterでカルマンフィルタより新しいMadgwickフィルタってのがあると言う噂を聞いていたので使ってみました

Madgwickフィルタライブラリを使う

Arduino IDEのライブラリマネージャで検索するとすぐに出てきます

追記:M5StickCライブラリにMadgwickフィルタを使用してる関数があり
M5.MPU6886.getAhrsData()を使うことで外部ライブラリが不要になりました

使い方は

#include <M5StickC.h>

float roll, pitch, yaw;

void setup() {
  M5.begin();
  M5.MPU6886.Init();
}

void loop() {
  M5.MPU6886.getAhrsData(&pitch, &roll, &yaw);
  Serial.printf("%5.1f,%5.1f,%5.1f\n",pitch,roll,yaw);
  delay(10);
}

なんとこれだけです

マッジで?!という声が出ました(これが言いたかっただけ)

値が収束するのに10秒ほどかかりますが重力加速度で補正するらしくて何もせずに簡易の角度計が作れちゃいます

ただし6軸センサの値を利用しているので重力に鉛直方向を軸(下図のZ軸)にしての回転方向(ヨー)はドリフトしっぱなしであてになりません
今回はロール軸(X軸)だけを使うので問題はありません

image.png

※こちらから引用
カルマン・フィルタで M5stickC 傾斜計 ー倒立振子への道 1ー - 電子工作 - HomeMadeGarbage
https://homemadegarbage.com/bala01

PID制御

最初は基準の角度に対してのずれに応じてモーターのパワーを上げたらいいんじゃね?(これはP制御)
くらいに思っていたらそんなに甘くはなかったのでPID制御を実装します

参考にした記事の方ではMPU6886の値から角速度などを得てPID制御してました
今回はMadgwickフィルタを使っていて現在の角度しかわからないのでロール角のみを使った処理に書き直します

Arduinoで簡単なPID制御のプログラムを作ってみる – 自作のいろいろ
https://garchiving.com/pid-control-in-arduino/

こちらを参考にしてソースをアレンジします

dt = (micros() - preTime) / 1000000; // 処理時間を求める
  preTime = micros(); // 処理時間を記録

// 目標角度から現在の角度を引いて偏差を求める
// -90~90を取るので90で割って-1.0~1.0にする
  P  = (target - roll) / 90;
// 偏差を積分する
  I += P * dt;
// 偏差を微分する
  D  = (P - preP) / dt;
// 偏差を記録する
  preP = P;
// 積分部分が大きくなりすぎると出力が飽和するので大きくなり過ぎたら0に戻す(アンチワインドアップ)
  if (100 < abs(I * Ki)) {
    I = 0;
    led = !led;
    digitalWrite(GPIO_NUM_10, led ? LOW : HIGH);
    // anti windup
  }

// 出力を計算する
  power = Kp * P + Ki * I + Kd * D;
  power = constrain(power, -100, 100);
// モーターの出力を決定する(PWM8bitで0~255)
// 今使ってるギアボックスは100でやっと回り始めるのでオフセットを100にしている
  Duty = (int)(130 * abs(power / 100) + 100); 

  // +-30度を越えたら倒れたとみなす
  if (-30 < now && now < 30) {
    ledcWrite( MOTOR_PWM_F,(power < 0 ?    0 : Duty) );  
    ledcWrite( MOTOR_PWM_R,(power < 0 ? Duty :    0) ); 
//    Serial.printf("power= %3f\n", power);
  }
  else { 
    ledcWrite(MOTOR_PWM_F, 0);
    ledcWrite(MOTOR_PWM_R, 0);
    power = 0;
    I = 0;
  }

powerが-100~100で決まるのでそれを%として使用してDutyを決定します

倒立振子を構築する

基本的にはこちらの記事と同じですが手元に安いギアボックスがあったのでそれを使っています
ただ遊びが大きいのでキッチリ立たせたいなら元記事の遊びが少ないギアードモータを使うべきです

実質500円&100Stepで作る超簡単「 ゆるメカトロ的 M5StickC 倒立振子」 - Qiita
https://qiita.com/Google_Homer/items/d0cdb2975491a5ea8129

Amazon | Yeeco DC 電動モーター 3-6V デュアルシャフトギアード TT 磁気ギアボックスエンジン
https://www.amazon.co.jp/dp/B07H4KXNP4/

ちなみに自分のはこういう感じになっています
3Dプリントした白い台座にL字ピンヘッダを入れてG0とG26からPWMを出力してモータードライバを動かしています

image.png

逆側のギアボックス側面にM5StickCを貼り付けてGroveポートからモータードライバに接続するのもアリだと思います(重心も下がるし)
最初はこの配置でやってました
image.png

個体差があるので同じKp,Ki,Kdでは立たないと思いますが試行錯誤してください

自分の理解に基づいたPIDパラメータの設定法をメモしておきます

Kp:全体的なモータのパワーに効く
  モーターの出力が足りなくて倒れるときはKpを上げる
Kpを上げると激しく往復するようになる(ハンチング)のでなるべく往復の周期を長くするようにする

Kd:傾斜角度が大きくなった時の反発力に効く
  大きくすると激しく往復運動をするようになるので程々の周期に落ち着くようにする

Kd:激しく動いたときのブレーキの役割をする
  Kpを上げて往復してるところに優しく効かせてやると往復が緩やかになってくる
  Kdも上げ過ぎると反発力が強くなりすぎて往復するのでバランスが大事

Ki:バランスが崩れて全体が平行移動しだしたときに効く
  全体が平行移動しだしたときに止める作用をする
  

#include <M5StickC.h>

#define MOTOR_PIN_F        26 // to DC Motor Driver FIN
#define MOTOR_PIN_R        0  // to DC Motor Driver RIN
#define MOTOR_PWM_F        0  // PWM CHANNEL
#define MOTOR_PWM_R        1  // PWM CHANNEL

float Kp = 142.0;
float Ki = 745.0;
float Kd = 2.0;
float target = -80.2;
float P, I, D, preP;

float power = 0;
float dt, preTime;

bool started = false;

Madgwick madgwick;
float roll, pitch, yaw;
float standard;
float now;

void drawScreen() {
    M5.Lcd.setCursor(0, 0);
    M5.Lcd.printf("S:%6.1f\n",target);
    M5.Lcd.printf("roll:%6.1f\n",roll);
    M5.Lcd.printf("now:%4.1f\n", target - roll);
    M5.Lcd.printf("Power:%5.1f\n", power);
    M5.Lcd.printf("Kp:%5.1f\n", Kp);
    M5.Lcd.printf("Ki:%5.1f\n", Ki);
    M5.Lcd.printf("Kd:%6.3f\n", Kd);
    M5.Lcd.printf("Bat:%5.1fV\n", M5.Axp.GetBatVoltage());
    M5.Lcd.printf("Cur:%5.1f\n", M5.Axp.GetBatCurrent());
}

void setup() {
  Serial.begin(151200);

  M5.begin();
  M5.Lcd.setRotation(2);
  M5.Lcd.fillScreen(BLACK);
  M5.Lcd.setTextSize(1);
  M5.Axp.ScreenBreath(8);
  M5.MPU6886.Init();

  pinMode(MOTOR_PIN_F, OUTPUT);
  pinMode(MOTOR_PIN_R, OUTPUT);
  ledcSetup(MOTOR_PWM_F, 312500, 8); //CHANNEL, FREQ, BIT
  ledcSetup(MOTOR_PWM_R, 312500, 8);
  ledcAttachPin(MOTOR_PIN_F, MOTOR_PWM_F);
  ledcAttachPin(MOTOR_PIN_R, MOTOR_PWM_R);
  pinMode(GPIO_NUM_10, OUTPUT);
  digitalWrite(GPIO_NUM_10, HIGH);

  drawScreen();
}


static  int Duty;
bool led = false;
void loop() {
  M5.update();
  if ( M5.BtnB.wasReleased() ) {
    // ボタンBを押すと再起動
    esp_restart();
  }

  if ( M5.BtnA.wasReleased() ) {
    // ボタンAを押して3秒後にモーター駆動開始
    Serial.println("BtnA.wasReleased() == TRUE");

    M5.Lcd.setCursor(0, 0);
    M5.Lcd.print("3");
    delay(1000);
    M5.Lcd.setCursor(0, 0);
    M5.Lcd.print("2");
    delay(1000);
    M5.Lcd.setCursor(0, 0);
    M5.Lcd.print("1");
    // ボタンを押したら直立させて待ち残り1秒になったら現在の角度を基準に設定する
    float temp = 0.0;
    for (int i = 0; i < 10; i++) {
      M5.MPU6886.getAhrsData(&pitch, &roll, &yaw);
      temp += roll;
      delay(10);
    }
    target = temp / 10.0;
    delay(800);
    M5.Lcd.setCursor(0, 0);
    M5.Lcd.print("0");
    preTime = micros();
    started = true;
  }
  M5.MPU6886.getAhrsData(&pitch, &roll, &yaw);

    Serial.printf("%8.1f,%8.1f,%8.1f\n",roll,pitch,yaw);
  drawScreen();
  delay(2);
  if (!started) {
    return;
  }
  now = target - roll;
dt = (micros() - preTime) / 1000000; // 処理時間を求める
  preTime = micros(); // 処理時間を記録

// 目標角度から現在の角度を引いて偏差を求める
// -90~90を取るので90で割って-1.0~1.0にする
  P  = (target - roll) / 90;
// 偏差を積分する
  I += P * dt;
// 偏差を微分する
  D  = (P - preP) / dt;
// 偏差を記録する
  preP = P;
// 積分部分が大きくなりすぎると出力が飽和するので大きくなり過ぎたら0に戻す(アンチワインドアップ)
  if (100 < abs(I * Ki)) {
    I = 0;
    led = !led;
    // アンチワインドアップされたらLEDを点灯する
    digitalWrite(GPIO_NUM_10, led ? LOW : HIGH);
    // anti windup
  }

// 出力を計算する
  power = Kp * P + Ki * I + Kd * D;
  power = constrain(power, -100, 100);
// モーターの出力を決定する(PWM8bitで0~255)
// 今使ってるギアボックスは100でやっと回り始めるのでオフセットを100にしている
  Duty = (int)(130 * abs(power / 100) + 100); 

  // +-30度を越えたら倒れたとみなす
  if (-30 < now && now < 30) {
    ledcWrite( MOTOR_PWM_F,(power < 0 ?    0 : Duty) );  
    ledcWrite( MOTOR_PWM_R,(power < 0 ? Duty :    0) ); 
//    Serial.printf("power= %3f\n", power);
  }
  else { 
    ledcWrite(MOTOR_PWM_F, 0);
    ledcWrite(MOTOR_PWM_R, 0);
    power = 0;
    I = 0;
  }
}

パラメータ調整前

パラメータ調整後

課題

・PIDパラメータの単位がおかしいのを直したい
・たまにMPU6886の値がぶっ壊れることがある(直立してるはずなのに70度を表示してるとか)
・さらなるPIDパラメータの追い込み

おまけ、というか今回のオチ

倒立振子を作った人は全員思ったと思います
PIDパラメータを変更するたびにスケッチ書き込むのめんどくさい!
と・・

私も思ったのでESP NOWを使ってリアルタイムでパラメーターを変更する奴を作りました

image.png

M5Stackの真ん中を押すと項目を移動し、左右ボタンで値をM5StickCに送信してパラメーターを変更します
モーター動作中でも変更できるので動きの変化がはっきりわかって面白いですよ

基本的にはESP32のMaster,SlaveのサンプルをコピペしてM5シリーズ固有の処理を追加したものです
書いた人はC++が不自由なので変なところがあったら直して使ってください

動いてるから良しとします

M5Stack側ソース

#include <M5Stack.h>
#include <esp_now.h>
#include <WiFi.h>

// Global copy of slave
esp_now_peer_info_t slave;
#define CHANNEL 3
#define PRINTSCANRESULTS 0
#define DELETEBEFOREPAIR 0

#define CHAR_X 10
#define CHAR_Y 18

float Kp = 3;
float Ki = 1;
float Kd = 200;
float target = -87.0;

int indexY = 0;

bool isPaired = false;

// Init ESP Now with fallback
void InitESPNow() {
  WiFi.disconnect();
  if (esp_now_init() == ESP_OK) {
    Serial.println("ESPNow Init Success");
  }
  else {
    Serial.println("ESPNow Init Failed");
    // Retry InitESPNow, add a counte and then restart?
    // InitESPNow();
    // or Simply Restart
    ESP.restart();
  }
}

// Scan for slaves in AP mode
void ScanForSlave() {
  int8_t scanResults = WiFi.scanNetworks();
  // reset on each scan
  bool slaveFound = 0;
  memset(&slave, 0, sizeof(slave));

  Serial.println("");
  if (scanResults == 0) {
    Serial.println("No WiFi devices in AP Mode found");
  } else {
    Serial.print("Found "); Serial.print(scanResults); Serial.println(" devices ");
    for (int i = 0; i < scanResults; ++i) {
      // Print SSID and RSSI for each device found
      String SSID = WiFi.SSID(i);
      int32_t RSSI = WiFi.RSSI(i);
      String BSSIDstr = WiFi.BSSIDstr(i);

      if (PRINTSCANRESULTS) {
        Serial.print(i + 1);
        Serial.print(": ");
        Serial.print(SSID);
        Serial.print(" (");
        Serial.print(RSSI);
        Serial.print(")");
        Serial.println("");
      }
      delay(10);
      // Check if the current device starts with `Slave`
      if (SSID.indexOf("Slave") == 0) {
        // SSID of interest
        Serial.println("Found a Slave.");
        Serial.print(i + 1); Serial.print(": "); Serial.print(SSID); Serial.print(" ["); Serial.print(BSSIDstr); Serial.print("]"); Serial.print(" ("); Serial.print(RSSI); Serial.print(")"); Serial.println("");
        // Get BSSID => Mac Address of the Slave
        int mac[6];
        if ( 6 == sscanf(BSSIDstr.c_str(), "%x:%x:%x:%x:%x:%x",  &mac[0], &mac[1], &mac[2], &mac[3], &mac[4], &mac[5] ) ) {
          for (int ii = 0; ii < 6; ++ii ) {
            slave.peer_addr[ii] = (uint8_t) mac[ii];
          }
        }

        slave.channel = CHANNEL; // pick a channel
        slave.encrypt = 0; // no encryption

        slaveFound = 1;
        // we are planning to have only one slave in this example;
        // Hence, break after we find one, to be a bit efficient
        break;
      }
    }
  }

  if (slaveFound) {
    Serial.println("Slave Found, processing..");
  } else {
    Serial.println("Slave Not Found, trying again.");
  }

  // clean up ram
  WiFi.scanDelete();
}

// Check if the slave is already paired with the master.
// If not, pair the slave with master
bool manageSlave() {
  if (slave.channel == CHANNEL) {
    if (DELETEBEFOREPAIR) {
      deletePeer();
    }

//    Serial.print("Slave Status: ");
    // check if the peer exists
    bool exists = esp_now_is_peer_exist(slave.peer_addr);
    if ( exists) {
      // Slave already paired.
//      Serial.println("Already Paired");
      return true;
    } else {
      // Slave not paired, attempt pair
      esp_err_t addStatus = esp_now_add_peer(&slave);
      if (addStatus == ESP_OK) {
        // Pair success
        Serial.println("Pair success");
        isPaired = true;
        drawScreen();
        return true;
      } else if (addStatus == ESP_ERR_ESPNOW_NOT_INIT) {
        // How did we get so far!!
        Serial.println("ESPNOW Not Init");
        return false;
      } else if (addStatus == ESP_ERR_ESPNOW_ARG) {
        Serial.println("Invalid Argument");
        return false;
      } else if (addStatus == ESP_ERR_ESPNOW_FULL) {
        Serial.println("Peer list full");
        return false;
      } else if (addStatus == ESP_ERR_ESPNOW_NO_MEM) {
        Serial.println("Out of memory");
        return false;
      } else if (addStatus == ESP_ERR_ESPNOW_EXIST) {
        Serial.println("Peer Exists");
        return true;
      } else {
        Serial.println("Not sure what happened");
        return false;
      }
    }
  } else {
    // No slave found to process
    Serial.println("No Slave found to process");
    return false;
  }
}

void deletePeer() {
  esp_err_t delStatus = esp_now_del_peer(slave.peer_addr);
  Serial.print("Slave Delete Status: ");
  if (delStatus == ESP_OK) {
    // Delete success
    Serial.println("Success");
  } else if (delStatus == ESP_ERR_ESPNOW_NOT_INIT) {
    // How did we get so far!!
    Serial.println("ESPNOW Not Init");
  } else if (delStatus == ESP_ERR_ESPNOW_ARG) {
    Serial.println("Invalid Argument");
  } else if (delStatus == ESP_ERR_ESPNOW_NOT_FOUND) {
    Serial.println("Peer not found.");
  } else {
    Serial.println("Not sure what happened");
  }
}

// callback when data is sent from Master to Slave
void OnDataSent(const uint8_t *mac_addr, esp_now_send_status_t status) {
  char macStr[18];
  snprintf(macStr, sizeof(macStr), "%02x:%02x:%02x:%02x:%02x:%02x",
           mac_addr[0], mac_addr[1], mac_addr[2], mac_addr[3], mac_addr[4], mac_addr[5]);
  Serial.print("Last Packet Sent to: "); Serial.println(macStr);
  Serial.print("Last Packet Send Status: "); Serial.println(status == ESP_NOW_SEND_SUCCESS ? "Delivery Success" : "Delivery Fail");
}

void setup() {
  Serial.begin(115200);
  //Set device in STA mode to begin with
  WiFi.mode(WIFI_STA);
  Serial.println("ESPNow/Basic/Master Example");
  // This is the mac address of the Master in Station Mode
  Serial.print("STA MAC: "); Serial.println(WiFi.macAddress());
  // Init ESPNow with a fallback logic
  InitESPNow();
  // Once ESPNow is successfully Init, we will register for Send CB to
  // get the status of Trasnmitted packet
  esp_now_register_send_cb(OnDataSent);
  ScanForSlave();
  pinMode(GPIO_NUM_10, OUTPUT); // internal LED
  digitalWrite(GPIO_NUM_10, HIGH);
  M5.begin();
  M5.Lcd.setRotation(1);
  M5.Lcd.fillScreen(BLACK);
  M5.Lcd.setTextSize(2);
  M5.Lcd.setBrightness(64);
  M5.Lcd.fillScreen(BLACK);
  File file2 = SD.open("/param.txt", FILE_READ);
  String str = "";

  while (file2.available()) {
      str += char(file2.read());
  }

  if (str.length() != 0) {
    int start = 0;
    target = str.substring(start,str.indexOf(",")).toFloat();
    start = str.indexOf(",") + 1;

    Kp = str.substring(start,str.indexOf(",",start)).toFloat();
    start = str.indexOf(",", start) + 1;
    Ki = str.substring(start,str.indexOf(",",start)).toFloat();
    start = str.indexOf(",", start) + 1;
    Kd = str.substring(start).toFloat();
    Serial.printf("%.1f,%.1f,%.1f,%.1f\n",target,Kp,Ki,Kd);
    file2.close();
  }

  drawScreen();    
}

void drawScreen() {
  M5.Lcd.fillScreen(BLACK);

  M5.Lcd.drawString(">",0,indexY * CHAR_Y);
  M5.Lcd.setCursor(3 * CHAR_X,0);
  M5.Lcd.printf("Target:%4.1f",target);
  M5.Lcd.setCursor(3 * CHAR_X,CHAR_Y *1);
  M5.Lcd.printf("Kp:%4.1f",Kp);
  M5.Lcd.setCursor(3 * CHAR_X,CHAR_Y *2);
  M5.Lcd.printf("Ki:%4.1f",Ki);
  M5.Lcd.setCursor(3 * CHAR_X,CHAR_Y *3);
  M5.Lcd.printf("Kd:%4.1f",Kd);
  M5.Lcd.setCursor(0,240 - CHAR_Y * 2);
  M5.Lcd.printf("%s",isPaired ? "CONNECTED" : "NOT CONNECTED");

  M5.Lcd.setCursor(0,240 - CHAR_Y);
  M5.Lcd.print("     -     down      +");
}

uint8_t state = 0;


void loop() {
  M5.update();
  // In the loop we scan for slave
  // If Slave is found, it would be populate in `slave` variable
  // We will check if `slave` is defined and then we proceed further




  if (slave.channel == CHANNEL) {
    isPaired = manageSlave();
    if (isPaired) {
      if ( M5.BtnA.wasPressed() ) {

        switch (indexY) {
          case 0:
            target -= 0.1;
            drawScreen();
            break;
          case 1:
            Kp -= 0.1;
            if (Kp < 0) {
              Kp = 0;
            }
            drawScreen();
            break;
          case 2:
            Ki -= 0.1;
            if (Ki < 0) {
              Ki = 0;
            }
            drawScreen();
            break;
          case 3:
            Kd -= 0.1;
            if (Kd < 0) {
              Kd = 0;
            }
            drawScreen();
            break;
        }
        File file = SD.open("/param.txt",FILE_WRITE);
        file.printf("%.1f,%.1f,%.1f,%.1f",target,Kp,Ki,Kd);
        Serial.printf("Wrote %.1f,%.1f,%.1f,%.1f\n",target,Kp,Ki,Kd);
        file.close();
        Serial.print("Sending: ");

        String str = "";
        str += String(target);
        str += ",";
        str += String(Kp);
        str += ",";
        str += String(Ki);
        str += ",";
        str += String(Kd);
        Serial.printf("Send: %s\n",str.c_str());
        esp_err_t result;
        const uint8_t *peer_addr = slave.peer_addr;
              result = esp_now_send(peer_addr, (uint8_t*)str.c_str(), str.length());
        Serial.print("Send Status: ");
        if (result == ESP_OK) {
          Serial.println("Success");
        } else if (result == ESP_ERR_ESPNOW_NOT_INIT) {
          // How did we get so far!!
          Serial.println("ESPNOW not Init.");
        } else if (result == ESP_ERR_ESPNOW_ARG) {
          Serial.println("Invalid Argument");
        } else if (result == ESP_ERR_ESPNOW_INTERNAL) {
          Serial.println("Internal Error");
        } else if (result == ESP_ERR_ESPNOW_NO_MEM) {
          Serial.println("ESP_ERR_ESPNOW_NO_MEM");
        } else if (result == ESP_ERR_ESPNOW_NOT_FOUND) {
          Serial.println("Peer not found.");
        } else {
          Serial.println("Not sure what happened");
        }

      }
      if ( M5.BtnB.wasPressed() ) {
        indexY++;
        if (indexY == 4) {
          indexY = 0;
        }
        drawScreen();
      }
      if ( M5.BtnC.wasPressed() ) {
        switch (indexY) {
          case 0:
            target += 0.1;
            drawScreen();
            break;
          case 1:
            Kp += 0.1;
            drawScreen();
            break;
          case 2:
            Ki += 0.1;
            drawScreen();
            break;
          case 3:
            Kd += 0.1;
            drawScreen();
            break;
        }
        File file = SD.open("/param.txt",FILE_WRITE);
        file.printf("%.1f,%.1f,%.1f,%.1f",target,Kp,Ki,Kd);
        Serial.printf("Wrote %.1f,%.1f,%.1f,%.1f\n",target,Kp,Ki,Kd);
        file.close();
        Serial.print("Sending: ");

        String str = "";
        str += String(target);
        str += ",";
        str += String(Kp);
        str += ",";
        str += String(Ki);
        str += ",";
        str += String(Kd);
        Serial.printf("Send: %s\n",str.c_str());
        esp_err_t result;
        const uint8_t *peer_addr = slave.peer_addr;
              result = esp_now_send(peer_addr, (uint8_t*)str.c_str(), str.length());
        Serial.print("Send Status: ");
        if (result == ESP_OK) {
          Serial.println("Success");
        } else if (result == ESP_ERR_ESPNOW_NOT_INIT) {
          // How did we get so far!!
          Serial.println("ESPNOW not Init.");
        } else if (result == ESP_ERR_ESPNOW_ARG) {
          Serial.println("Invalid Argument");
        } else if (result == ESP_ERR_ESPNOW_INTERNAL) {
          Serial.println("Internal Error");
        } else if (result == ESP_ERR_ESPNOW_NO_MEM) {
          Serial.println("ESP_ERR_ESPNOW_NO_MEM");
        } else if (result == ESP_ERR_ESPNOW_NOT_FOUND) {
          Serial.println("Peer not found.");
        } else {
          Serial.println("Not sure what happened");
        }
      }

    } else {
      ScanForSlave();
    }
  }
  else {
    ScanForSlave();
  }
  delay(10);

  // wait for 3seconds to run the logic again
}

M5StickC側ソース

#include <M5StickC.h>
#include <MadgwickAHRS.h>
#include <esp_now.h>
#include <WiFi.h>

#define CHANNEL 1

#define MOTOR_PIN_F        26                      // to DC Motor Driver FIN
#define MOTOR_PIN_R        0                      // to DC Motor Driver RIN
#define MOTOR_PWM_F        0                       // PWM CHANNEL
#define MOTOR_PWM_R        1                       // PWM CHANNEL

float Kp = 30;
float Ki = 0.0;
float Kd = 0.0;
float target = -91.7;
float P, I, D, preP;

float power = 0;
float dt, preTime;

bool started = false;

Madgwick madgwick;
float roll, pitch, yaw;
float standard;
float now;

void InitESPNow() {
  WiFi.disconnect();
  if (esp_now_init() == ESP_OK) {
    Serial.println("ESPNow Init Success");
  }
  else {
    Serial.println("ESPNow Init Failed");
    // Retry InitESPNow, add a counte and then restart?
    // InitESPNow();
    // or Simply Restart
    ESP.restart();
  }
}

// config AP SSID
void configDeviceAP() {
  const char *SSID = "Slave_1";
  bool result = WiFi.softAP(SSID, "Slave_1_Password", CHANNEL, 0);
  if (!result) {
    Serial.println("AP Config failed.");
  } else {
    Serial.println("AP Config Success. Broadcasting with AP: " + String(SSID));
  }
}


void drawScreen() {
    M5.Lcd.setCursor(0, 0);
    M5.Lcd.printf("S:%6.1f\n",target);
    M5.Lcd.printf("roll:%6.1f\n",roll);
    M5.Lcd.printf("now:%4.1f\n", target - roll);
    M5.Lcd.printf("Power:%5.1f\n", power);
    M5.Lcd.printf("Kp:%5.1f\n", Kp);
    M5.Lcd.printf("Ki:%5.1f\n", Ki);
    M5.Lcd.printf("Kd:%6.3f\n", Kd);
    M5.Lcd.printf("Bat:%5.1fV\n", M5.Axp.GetBatVoltage());
    M5.Lcd.printf("Cur:%5.1f\n", M5.Axp.GetBatCurrent());
}




void OnDataRecv(const uint8_t *mac_addr, const uint8_t *data, int data_len) {
  char macStr[18];
  snprintf(macStr, sizeof(macStr), "%02x:%02x:%02x:%02x:%02x:%02x",
           mac_addr[0], mac_addr[1], mac_addr[2], mac_addr[3], mac_addr[4], mac_addr[5]);
  Serial.print("Last Packet Recv from: "); Serial.println(macStr);
  Serial.print("Last Packet Recv Data: "); Serial.println(*data);
  Serial.println("");

//  Serial.printf("Received: %s\n",(const char *)data);

  String str = "";
  for (int i = 0; i < data_len; i++) {
    str += String(((const char *)data)[i]);
  }

  Serial.printf("str: %s\n",str.c_str());

  int start = 0;
  target = str.substring(start,str.indexOf(",")).toFloat();
  start = str.indexOf(",") + 1;

  Kp = str.substring(start,str.indexOf(",",start)).toFloat();
  start = str.indexOf(",", start) + 1;
  Ki = str.substring(start,str.indexOf(",",start)).toFloat();
  start = str.indexOf(",", start) + 1;
  Kd = str.substring(start).toFloat();
  //    Serial.printf("t=%.1f, p=%.1f, i=%.1f, d=%.1f\n",t,p,i,d);
  Serial.printf("%.1f,%.1f,%.1f,%.1f\n",target,Kp,Ki,Kd);
}

void setup() {
  Serial.begin(151200);

  M5.begin();
  M5.Lcd.setRotation(2);
  M5.Lcd.fillScreen(BLACK);
  M5.Lcd.setTextSize(1);
  M5.Axp.ScreenBreath(8);
  M5.MPU6886.Init();
//  madgwick.begin(100);

  pinMode(MOTOR_PIN_F, OUTPUT);
  pinMode(MOTOR_PIN_R, OUTPUT);
  ledcSetup(MOTOR_PWM_F, 312500, 8); //CHANNEL, FREQ, BIT
  ledcSetup(MOTOR_PWM_R, 312500, 8);
  ledcAttachPin(MOTOR_PIN_F, MOTOR_PWM_F);
  ledcAttachPin(MOTOR_PIN_R, MOTOR_PWM_R);
  pinMode(GPIO_NUM_10, OUTPUT);
  digitalWrite(GPIO_NUM_10, HIGH);
  drawScreen();
    WiFi.mode(WIFI_AP);
  // configure device AP mode
  configDeviceAP();
  // This is the mac address of the Slave in AP Mode
  Serial.print("AP MAC: "); Serial.println(WiFi.softAPmacAddress());
  // Init ESPNow with a fallback logic
  InitESPNow();
  // Once ESPNow is successfully Init, we will register for recv CB to
  // get recv packer info.
  esp_now_register_recv_cb(OnDataRecv);

}

static  int Duty;
bool led = false;
void loop() {
  M5.update();
  if ( M5.BtnB.wasReleased() ) {
    // ボタンBを押すと再起動
    esp_restart();
  }

  if ( M5.BtnA.wasReleased() ) {
    // ボタンAを押して3秒後にモーター駆動開始
    Serial.println("BtnA.wasReleased() == TRUE");

    M5.Lcd.setCursor(0, 0);
    M5.Lcd.print("3");
    delay(1000);
    M5.Lcd.setCursor(0, 0);
    M5.Lcd.print("2");
    delay(1000);
    M5.Lcd.setCursor(0, 0);
    M5.Lcd.print("1");
    // ボタンを押したら直立させて待ち残り1秒になったら現在の角度を基準に設定する
    float temp = 0.0;
    for (int i = 0; i < 10; i++) {
      M5.MPU6886.getAhrsData(&pitch, &roll, &yaw);
      temp += roll;
      delay(10);
    }
    target = temp / 10.0;
    delay(800);
    M5.Lcd.setCursor(0, 0);
    M5.Lcd.print("0");
    preTime = micros();
    started = true;
  }
  M5.MPU6886.getAhrsData(&pitch, &roll, &yaw);

    Serial.printf("%8.1f,%8.1f,%8.1f\n",roll,pitch,yaw);
  drawScreen();
  delay(2);
  if (!started) {
    return;
  }
  now = target - roll;
dt = (micros() - preTime) / 1000000; // 処理時間を求める
  preTime = micros(); // 処理時間を記録

// 目標角度から現在の角度を引いて偏差を求める
// -90~90を取るので90で割って-1.0~1.0にする
  P  = (target - roll) / 90;
// 偏差を積分する
  I += P * dt;
// 偏差を微分する
  D  = (P - preP) / dt;
// 偏差を記録する
  preP = P;
// 積分部分が大きくなりすぎると出力が飽和するので大きくなり過ぎたら0に戻す(アンチワインドアップ)
  if (100 < abs(I * Ki)) {
    I = 0;
    led = !led;
    digitalWrite(GPIO_NUM_10, led ? LOW : HIGH);
    // anti windup
  }

// 出力を計算する
  power = Kp * P + Ki * I + Kd * D;
  power = constrain(power, -100, 100);
// モーターの出力を決定する(PWM8bitで0~255)
// 今使ってるギアボックスは100でやっと回り始めるのでオフセットを100にしている
  Duty = (int)(130 * abs(power / 100) + 100); 

  // +-30度を越えたら倒れたとみなす
  if (-30 < now && now < 30) {
    ledcWrite( MOTOR_PWM_F,(power < 0 ?    0 : Duty) );  
    ledcWrite( MOTOR_PWM_R,(power < 0 ? Duty :    0) ); 
//    Serial.printf("power= %3f\n", power);
  }
  else { 
    ledcWrite(MOTOR_PWM_F, 0);
    ledcWrite(MOTOR_PWM_R, 0);
    power = 0;
    I = 0;
  }
}
Why not register and get more from Qiita?
  1. We will deliver articles that match you
    By following users and tags, you can catch up information on technical fields that you are interested in as a whole
  2. you can read useful information later efficiently
    By "stocking" the articles you like, you can search right away