Un gyroscope 3 axes

L’extension Wii Motion Plus est constituée de 2 paires de gyroscopes 2 axes, un seul axe étant utilisé sur un des deux. Au final, une fois le circuit imprimé extrait, on a un ensemble de gyroscopes 3 axes dans un espace réduit. Mais surtout, ils sont montés à plat sans circuit imprimé (PCB) additionnels et c’est très pratique.

Lorsqu’on veut exploiter la valeur d’un capteur brute, on utilise sa sortie analogique pour ensuite la convertir dans un format numérique exploitable par un programme. L’extension Wii Motion Plus intègre un convertisseur analogique numérique 14 bits.

Cette extension communique avec la manette sur un bus I2C en mode rapide à 400kbit/s. C’est intéressant car ce bus couplé au convertisseur analogique numérique intégré permet de réaliser les opérations de conversion qui ne seront pas à gérer par la suite par le microcontrôleur. Ce bus ne mobilise que 2 fils de données.

A ce prix là, on n’a pas les meilleures performances existantes d’un gyroscope moderne. Le bruit (signaux parasites en l’absence de mouvement) est assez important, mais il s’agit quand même de capteurs MEMS qui surpassent les gyroscopes  piezo qu’on trouve encore dans beaucoup de gyroscopes RC. Correctement filtré, le signal exploitable est plutôt précis.

Quelques liens:

Utilisation pour la stabilisation d'un tricopter : triwiicopter

http://wiibrew.org/wiki/Wiimote/Extension_Controllers#Wii_Motion_Plus

http://prt.fernuni-hagen.de/lehre/KURSE/PRT001/course_main_sh3d/node11.html


Le connecteur de la Wiimote

_____
| 135 |
| 246 |
|_---_|

 


le connecteur situé sur le circuit imprimé du Nunchuk et WII Motion Plus

1|_|_|_|_|_|_|6

 


 

 

Code source

  1.  /*Make Projects
  2.  Arduino Wii Motion Plus
  3.  This code is used for the Make Projects: Wii Motion Plus on the Arduino.
  4.  I could not verify the exact author as it looks to be a community driven effort. See this link for more details:
  5.  
  6.  http://www.arduino.cc/cgi-bin/yabb2/YaBB.pl?num=1248889032/0
  7.  This code is rather advanced so be warned up front. However if you just want to test out 2 servos and the wm+ then this
  8.  is a good start.
  9.  Trapezoid or Runge-kutta 4th order integration program of wm+
  10.  */
  11.  
  12.  #include <Streaming.h>
  13.  #include <Wire.h>
  14.  
  15.  byte data[6]; //six data bytes
  16.  int yaw, pitch, roll; //three axes
  17.  int yaw0, pitch0, roll0; //calibration zeroes
  18.  int time, last_time;
  19.  float delta_t;
  20.  int last_yaw[3], last_pitch[3], last_roll[3];
  21.  float yaw_deg, pitch_deg, roll_deg;
  22.  int yaw_deg2, pitch_deg2, roll_deg2;
  23.  int startTag=0xDEAD;
  24.  int accel_x_axis, accel_y_axis, accel_z_axis;
  25.  
  26.  #define steps_per_deg_slow 20
  27.  #define steps_per_deg_fast 4
  28.  
  29.  void wmpOn(){
  30.   Wire.beginTransmission(0x53); //WM+ starts out deactivated at address 0x53
  31.   Wire.send(0xfe); //send 0x04 to address 0xFE to activate WM+
  32.   Wire.send(0x04);
  33.   Wire.endTransmission(); //WM+ jumps to address 0x52 and is now active
  34.  }
  35.  
  36.  void wmpSendZero(){
  37.   Wire.beginTransmission(0x52); //now at address 0x52
  38.   Wire.send(0x00); //send zero to signal we want info
  39.   Wire.endTransmission();
  40.  }
  41.  
  42.  void calibrateZeroes(){
  43.   for (int i=0;i<10;i++){
  44.   wmpSendZero();
  45.   Wire.requestFrom(0x52,6);
  46.   for (int i=0;i<6;i++){
  47.   data[i]=Wire.receive();
  48.   }
  49.   yaw0+=(((data[3]>>2)<<8)+data[0])/10; //average 10 readings for each zero
  50.   pitch0+=(((data[4]>>2)<<8)+data[1])/10;
  51.   roll0+=(((data[5]>>2)<<8)+data[2])/10;
  52.   }
  53.   Serial.print("Yaw0:");
  54.   Serial.print(yaw0);
  55.   Serial.print(" Pitch0:");
  56.   Serial.print(pitch0);
  57.   Serial.print(" Roll0:");
  58.   Serial.println(roll0);
  59.  }
  60.  
  61.  void receiveData(){
  62.   wmpSendZero(); //send zero before each request (same as nunchuck)
  63.   Wire.requestFrom(0x52,6); //request the six bytes from the WM+
  64.   for (int i=0;i<6;i++){
  65.   data[i]=Wire.receive();
  66.   }
  67.   if(bitRead(data[3], 1)==1) yaw=(((data[3]>>2)<<8)+data[0]-yaw0)/steps_per_deg_slow; //see http://wiibrew.org/wiki/Wiimote/Extension_Controllers#Wii_Motion_Plus
  68.   else yaw=(((data[3]>>2)<<8)+data[0]-yaw0)/steps_per_deg_fast;
  69.   if(bitRead(data[3], 0)==1) pitch=(((data[4]>>2)<<8)+data[1]-pitch0)/steps_per_deg_slow; //for info on what each byte represents
  70.   else pitch=(((data[4]>>2)<<8)+data[1]-pitch0)/steps_per_deg_fast;
  71.   if(bitRead(data[4], 1)==1) roll=(((data[5]>>2)<<8)+data[2]-roll0)/steps_per_deg_slow;
  72.   else roll=(((data[5]>>2)<<8)+data[2]-roll0)/steps_per_deg_fast;
  73.  }
  74.  
  75.  float trapIntegrate(int y2, int y1, float deltax){
  76.   float area=0;
  77.   area=(y2+y1)/2*deltax/1000;
  78.   return area;
  79.  }
  80.  
  81.  float rk4Integrate(int y4, int y3, int y2, int y1, float deltax){
  82.   float area=0;
  83.   area=((y4+2*y3+2*y2+y1)/6)*deltax/1000;
  84.   return area;
  85.  }
  86.  
  87.  void setup()
  88.   {
  89.   Serial.begin(9600);
  90.   Serial.println("WM+");
  91.   Wire.begin();
  92.   wmpOn(); //turn WM+ on
  93.   calibrateZeroes(); //calibrate zeroes
  94.   delay(1000);
  95.   pinMode(4, INPUT);
  96.   }
  97.  
  98.  void loop(){
  99.   receiveData(); //receive data and calculate yaw pitch and roll
  100.   time=millis();
  101.   delta_t=(time-last_time);
  102.  
  103.   /* Runge-kutta 4th Order Integration */
  104.   yaw_deg+=rk4Integrate(yaw, last_yaw[0], last_yaw[1], last_yaw[2], delta_t);
  105.   pitch_deg+=rk4Integrate(pitch, last_pitch[0], last_pitch[1], last_pitch[2], delta_t);
  106.   roll_deg+=rk4Integrate(roll, last_roll[0], last_roll[1], last_roll[2], delta_t);
  107.   last_yaw[2]=last_yaw[1];
  108.   last_pitch[2]=last_pitch[1];
  109.   last_roll[2]=last_roll[1];
  110.   last_yaw[1]=last_yaw[0];
  111.   last_pitch[1]=last_pitch[0];
  112.   last_roll[1]=last_roll[0];
  113.   last_yaw[0]=yaw;
  114.   last_pitch[0]=pitch;
  115.   last_roll[0]=roll;
  116.   last_time=time;
  117.   /* Runge-kutta 4th Order Integration */
  118.  
  119.   if (digitalRead(4) == 1)
  120.   {
  121.   last_yaw[2]=0;
  122.   last_pitch[2]=0;
  123.   last_roll[2]=0;
  124.   last_yaw[1]=0;
  125.   last_pitch[1]=0;
  126.   last_roll[1]=0;
  127.   last_yaw[0]=0;
  128.   last_pitch[0]=0;
  129.   last_roll[0]=0;
  130.   yaw = 0;
  131.   pitch = 0;
  132.   roll = 0;
  133.   yaw_deg = 0;
  134.   pitch_deg = 0;
  135.   roll_deg = 0;
  136.   }
  137.  
  138.   Serial<<"A,"<<int(yaw_deg)<<","<<int(pitch_deg)<<","<<int(roll_deg)<<"\n";
  139.   delay(10);
  140.  }