test morning
Dependencies: ISR_Mini-explorer mbed
Fork of roboticLab_withclass_3_July by
MiniExplorerCoimbra.cpp
- Committer:
- Ludwigfr
- Date:
- 2017-07-12
- Revision:
- 13:b0ddd71e8f08
- Parent:
- 12:1e80471c5c6c
- Child:
- 14:696187e74411
File content as of revision 13:b0ddd71e8f08:
#include "MiniExplorerCoimbra.hpp" #include "robot.h" #define PI 3.14159 MiniExplorerCoimbra::MiniExplorerCoimbra(float defaultXWorld, float defaultYWorld, float defaultThetaWorld, float widthRealMap, float heightRealMap):map(widthRealMap,heightRealMap,12,8),sonarLeft(10*PI/36,-4,4),sonarFront(0,0,5),sonarRight(-10*PI/36,4,4),covariancePositionEstimationK(3,3){ i2c1.frequency(100000); initRobot(); //Initializing the robot pc.baud(9600); // baud for the pc communication measure_always_on();//TODO check if needed this->setXYThetaAndXYThetaWorld(defaultXWorld,defaultYWorld,defaultThetaWorld); this->radiusWheels=3.25; this->distanceWheels=7.2; //go to point this->k_linear=10; this->k_angular=200; //go to point with angle this->khro=0.3; this->ka=0.8; this->kb=-0.36; this->kv=200;//velocity //go to line kh > 0, kd > 0 (200,5), dont turn to line enough, (10,10) turn on itself, this->kh=650; this->kd=10;//previous 5 this->speed=300; this->rangeForce=50; this->attractionConstantForce=1; //-90 ok with attraction impacted by distance this->repulsionConstantForce=-90; } void MiniExplorerCoimbra::setXYThetaAndXYThetaWorld(float defaultXWorld, float defaultYWorld, float defaultThetaWorld){ this->xWorld=defaultXWorld; this->yWorld=defaultYWorld; this->thetaWorld=defaultThetaWorld; X=defaultYWorld; Y=-defaultXWorld; if(defaultThetaWorld < -PI/2) theta=PI/2+PI-defaultThetaWorld; else theta=defaultThetaWorld-PI/2; } void MiniExplorerCoimbra::myOdometria(){ Odometria(); this->xWorld=-Y; this->yWorld=X; if(theta >PI/2) this->thetaWorld=-PI+(theta-PI/2); else this->thetaWorld=theta+PI/2; } void MiniExplorerCoimbra::go_to_point(float targetXWorld, float targetYWorld) { float angleError; //angle error float d; //distance from target float angularLeft, angularRight, linear, angular; int speed=300; do { //Updating X,Y and theta with the odometry values this->myOdometria(); //Computing angle error and distance towards the target value angleError = atan2((targetYWorld-this->yWorld),(targetXWorld-this->xWorld))-this->thetaWorld; if(angleError>PI) angleError=-(angleError-PI); else if(angleError<-PI) angleError=-(angleError+PI); pc.printf("\n\r error=%f",angleError); d=this->dist(this->xWorld, this->yWorld, targetXWorld, targetYWorld); pc.printf("\n\r dist=%f/n", d); //Computing linear and angular velocities linear=k_linear*d; angular=k_angular*angleError; angularLeft=(linear-0.5*this->distanceWheels*angular)/this->radiusWheels; angularRight=(linear+0.5*this->distanceWheels*angular)/this->radiusWheels; //Normalize speed for motors if(angularLeft>angularRight) { angularRight=speed*angularRight/angularLeft; angularLeft=speed; } else { angularLeft=speed*angularLeft/angularRight; angularRight=speed; } pc.printf("\n\r X=%f", this->xWorld); pc.printf("\n\r Y=%f", this->yWorld); pc.printf("\n\r theta=%f", this->thetaWorld); //Updating motor velocities if(angularLeft>0){ leftMotor(1,angularLeft); } else{ leftMotor(0,-angularLeft); } if(angularRight>0){ rightMotor(1,angularRight); } else{ rightMotor(0,-angularRight); } //wait(0.5); } while(d>1); //Stop at the end leftMotor(1,0); rightMotor(1,0); } void MiniExplorerCoimbra::test_procedure_lab2(int nbIteration){ for(int i=0;i<nbIteration;i++){ this->randomize_and_map(); //this->print_map_with_robot_position(); } while(1) this->print_map_with_robot_position(); } //generate a position randomly and makes the robot go there while updating the map void MiniExplorerCoimbra::randomize_and_map() { //TODO check that it's aurelien's work float movementOnX=rand()%(int)(this->map.widthRealMap); float movementOnY=rand()%(int)(this->map.heightRealMap); float targetXWorld = movementOnX; float targetYWorld = movementOnY; float targetAngleWorld = 2*((float)(rand()%31416)-15708)/10000.0; //target between (0,0) and (widthRealMap,heightRealMap) this->go_to_point_with_angle(targetXWorld, targetYWorld, targetAngleWorld); } void MiniExplorerCoimbra::test_sonars_and_map(int nbIteration){ float leftMm; float frontMm; float rightMm; this->myOdometria(); this->print_map_with_robot_position(); for(int i=0;i<nbIteration;i++){ leftMm = get_distance_left_sensor(); frontMm = get_distance_front_sensor(); rightMm = get_distance_right_sensor(); pc.printf("\n\r 1 leftMm= %f",leftMm); pc.printf("\n\r 1 frontMm= %f",frontMm); pc.printf("\n\r 1 rightMm= %f",rightMm); this->update_sonar_values(leftMm, frontMm, rightMm); this->print_map_with_robot_position(); wait(1); } } //generate a position randomly and makes the robot go there while updating the map //move of targetXWorld and targetYWorld ending in a targetAngleWorld void MiniExplorerCoimbra::go_to_point_with_angle(float targetXWorld, float targetYWorld, float targetAngleWorld) { bool keepGoing=true; float leftMm; float frontMm; float rightMm; float dt; Timer t; float distanceToTarget; do { //Timer stuff dt = t.read(); t.reset(); t.start(); //Updating X,Y and theta with the odometry values this->myOdometria(); leftMm = get_distance_left_sensor(); frontMm = get_distance_front_sensor(); rightMm = get_distance_right_sensor(); //if in dangerzone 150 mm if((frontMm < 150 && frontMm > 0)|| (leftMm <150 && leftMm > 0) || (rightMm <150 && rightMm > 0) ){ //stop motors leftMotor(1,0); rightMotor(1,0); //update the map this->update_sonar_values(leftMm, frontMm, rightMm); this->myOdometria(); keepGoing=false; this->do_half_flip(); }else{ //if not in danger zone continue as usual this->update_sonar_values(leftMm, frontMm, rightMm); //Updating motor velocities distanceToTarget=this->update_angular_speed_wheels_go_to_point_with_angle(targetXWorld,targetYWorld,targetAngleWorld,dt); //wait(0.2); //Timer stuff t.stop(); pc.printf("\n\rdist to target= %f",distanceToTarget); } } while((distanceToTarget>2 || (abs(targetAngleWorld-this->thetaWorld)>PI/3)) && keepGoing); //Stop at the end leftMotor(1,0); rightMotor(1,0); pc.printf("\r\nReached Target!"); } //move of targetXWorld and targetYWorld ending in a targetAngleWorld void MiniExplorerCoimbra::go_to_point_with_angle_first_lab(float targetXWorld, float targetYWorld, float targetAngleWorld) { float dt; Timer t; float distanceToTarget; do { //Timer stuff dt = t.read(); t.reset(); t.start(); //Updating X,Y and theta with the odometry values this->myOdometria(); //Updating motor velocities distanceToTarget=this->update_angular_speed_wheels_go_to_point_with_angle(targetXWorld,targetYWorld,targetAngleWorld,dt); //wait(0.2); //Timer stuff t.stop(); pc.printf("\n\rdist to target= %f",distanceToTarget); } while(distanceToTarget>1 || (abs(targetAngleWorld-this->thetaWorld)>0.1)); //Stop at the end leftMotor(1,0); rightMotor(1,0); pc.printf("\r\nReached Target!"); } float MiniExplorerCoimbra::update_angular_speed_wheels_go_to_point_with_angle(float targetXWorld, float targetYWorld, float targetAngleWorld, float dt){ //compute_angles_and_distance //atan2 take the deplacement on x and the deplacement on y as parameters float angleToPoint = atan2((targetYWorld-this->yWorld),(targetXWorld-this->xWorld))-this->thetaWorld; if(angleToPoint>PI) angleToPoint=-(angleToPoint-PI); else if(angleToPoint<-PI) angleToPoint=-(angleToPoint+PI); //rho is the distance to the point of arrival float rho = dist(targetXWorld,targetYWorld,this->xWorld,this->yWorld); float distanceToTarget = rho; //TODO check that float beta = targetAngleWorld-angleToPoint-this->thetaWorld; //Computing angle error and distance towards the target value rho += dt*(-this->khro*cos(angleToPoint)*rho); float temp = angleToPoint; angleToPoint += dt*(this->khro*sin(angleToPoint)-this->ka*angleToPoint-this->kb*beta); beta += dt*(-this->khro*sin(temp)); //Computing linear and angular velocities float linear; float angular; if(angleToPoint>=-1.5708 && angleToPoint<=1.5708){ linear=this->khro*rho; angular=this->ka*angleToPoint+this->kb*beta; } else{ linear=-this->khro*rho; angular=-this->ka*angleToPoint-this->kb*beta; } float angularLeft=(linear-0.5*this->distanceWheels*angular)/this->radiusWheels; float angularRight=(linear+0.5*this->distanceWheels*angular)/this->radiusWheels; //Slowing down at the end for more precision if (distanceToTarget<30) { this->speed = distanceToTarget*10; } //Normalize speed for motors if(angularLeft>angularRight) { angularRight=this->speed*angularRight/angularLeft; angularLeft=this->speed; } else { angularLeft=this->speed*angularLeft/angularRight; angularRight=this->speed; } //compute_linear_angular_velocities leftMotor(1,angularLeft); rightMotor(1,angularRight); return distanceToTarget; } void MiniExplorerCoimbra::update_sonar_values(float leftMm,float frontMm,float rightMm){ float xWorldCell; float yWorldCell; float probaLeft; float probaFront; float probaRight; float leftCm=leftMm/10; float frontCm=frontMm/10; float rightCm=rightMm/10; for(int i=0;i<this->map.nbCellWidth;i++){ for(int j=0;j<this->map.nbCellHeight;j++){ xWorldCell=this->map.cell_width_coordinate_to_world(i); yWorldCell=this->map.cell_height_coordinate_to_world(j); probaLeft=this->sonarLeft.compute_probability_t(leftCm,xWorldCell,yWorldCell,this->xWorld,this->yWorld,this->thetaWorld); probaFront=this->sonarFront.compute_probability_t(frontCm,xWorldCell,yWorldCell,this->xWorld,this->yWorld,this->thetaWorld); probaRight=this->sonarRight.compute_probability_t(rightCm,xWorldCell,yWorldCell,this->xWorld,this->yWorld,this->thetaWorld); /* pc.printf("\n\r leftCm= %f",leftCm); pc.printf("\n\r frontCm= %f",frontCm); pc.printf("\n\r rightCm= %f",rightCm); */ /* pc.printf("\n\r probaLeft= %f",probaLeft); pc.printf("\n\r probaFront= %f",probaFront); pc.printf("\n\r probaRight= %f",probaRight); if(probaLeft> 1 || probaLeft < 0 || probaFront> 1 || probaFront < 0 ||probaRight> 1 || probaRight < 0)){ pwm_buzzer.pulsewidth_us(250); wait_ms(50); pwm_buzzer.pulsewidth_us(0); wait(20); pwm_buzzer.pulsewidth_us(250); wait_ms(50); pwm_buzzer.pulsewidth_us(0); } */ this->map.update_cell_value(i,j,probaLeft); this->map.update_cell_value(i,j,probaFront); this->map.update_cell_value(i,j,probaRight); } } } void MiniExplorerCoimbra::do_half_flip(){ this->myOdometria(); float theta_plus_h_pi=theta+PI/2;//theta is between -PI and PI if(theta_plus_h_pi > PI) theta_plus_h_pi=-(2*PI-theta_plus_h_pi); leftMotor(0,100); rightMotor(1,100); while(abs(theta_plus_h_pi-theta)>0.05){ this->myOdometria(); // pc.printf("\n\r diff=%f", abs(theta_plus_pi-theta)); } leftMotor(1,0); rightMotor(1,0); } //Distance computation function float MiniExplorerCoimbra::dist(float x1, float y1, float x2, float y2){ return sqrt(pow(y2-y1,2) + pow(x2-x1,2)); } //use virtual force field void MiniExplorerCoimbra::try_to_reach_target(float targetXWorld,float targetYWorld){ //atan2 gives the angle beetween PI and -PI this->myOdometria(); /* float deplacementOnXWorld=targetXWorld-this->xWorld; float deplacementOnYWorld=targetYWorld-this->yWorld; */ //float angleToTarget=atan2(targetYWorld-this->yWorld,targetXWorld-this->xWorld); //pc.printf("\n angleToTarget=%f",angleToTarget); //turn_to_target(angleToTarget); //TODO IDEA check if maybe set a low max range //this->sonarLeft.setMaxRange(30); //this->sonarFront.setMaxRange(30); //this->sonarRight.setMaxRange(30); bool reached=false; int print=0; int printLimit=1000; while (!reached) { this->vff(&reached,targetXWorld,targetYWorld); //test_got_to_line(&reached); if(print==printLimit){ leftMotor(1,0); rightMotor(1,0); this->print_map_with_robot_position_and_target(targetXWorld,targetYWorld); print=0; }else print+=1; } //Stop at the end leftMotor(1,0); rightMotor(1,0); pc.printf("\r\n target reached"); //wait(3);// } void MiniExplorerCoimbra::vff(bool* reached, float targetXWorld, float targetYWorld){ float line_a; float line_b; float line_c; //Updating X,Y and theta with the odometry values float forceXWorld=0; float forceYWorld=0; //we update the odometrie this->myOdometria(); //we check the sensors float leftMm = get_distance_left_sensor(); float frontMm = get_distance_front_sensor(); float rightMm = get_distance_right_sensor(); //update the probabilities values this->update_sonar_values(leftMm, frontMm, rightMm); this->myOdometria(); //we compute the force on X and Y this->compute_forceX_and_forceY(&forceXWorld, &forceYWorld,targetXWorld,targetYWorld); //we compute a new ine this->calculate_line(forceXWorld, forceYWorld, &line_a,&line_b,&line_c); //Updating motor velocities //pc.printf("\r\nX=%f;Y=%f",xWorld,yWorld); //pc.printf("\r\n%f*x+%f*y+%f=0",line_a,line_b,line_c); this->go_to_line(line_a,line_b,line_c,targetXWorld,targetYWorld); //wait(0.1); this->myOdometria(); if(dist(this->xWorld,this->yWorld,targetXWorld,targetYWorld)<3) *reached=true; } /*angleToTarget is obtained through atan2 so it s: < 0 if the angle is bettween PI and 2pi on a trigo circle > 0 if it is between 0 and PI */ void MiniExplorerCoimbra::turn_to_target(float angleToTarget){ this->myOdometria(); if(angleToTarget!=0){ if(angleToTarget>0){ leftMotor(0,1); rightMotor(1,1); }else{ leftMotor(1,1); rightMotor(0,1); } while(abs(angleToTarget-this->thetaWorld)>0.05) this->myOdometria(); } leftMotor(1,0); rightMotor(1,0); } void MiniExplorerCoimbra::print_map_with_robot_position_and_target(float targetXWorld, float targetYWorld) { float currProba; float heightIndiceInOrthonormal; float widthIndiceInOrthonormal; float widthMalus=-(3*this->map.sizeCellWidth/2); float widthBonus=this->map.sizeCellWidth/2; float heightMalus=-(3*this->map.sizeCellHeight/2); float heightBonus=this->map.sizeCellHeight/2; pc.printf("\n\r"); for (int y = this->map.nbCellHeight -1; y>-1; y--) { for (int x= 0; x<this->map.nbCellWidth; x++) { heightIndiceInOrthonormal=this->map.cell_height_coordinate_to_world(y); widthIndiceInOrthonormal=this->map.cell_width_coordinate_to_world(x); if(this->yWorld >= (heightIndiceInOrthonormal+heightMalus) && this->yWorld <= (heightIndiceInOrthonormal+heightBonus) && this->xWorld >= (widthIndiceInOrthonormal+widthMalus) && this->xWorld <= (widthIndiceInOrthonormal+widthBonus)) pc.printf(" R "); else{ if(targetYWorld >= (heightIndiceInOrthonormal+heightMalus) && targetYWorld <= (heightIndiceInOrthonormal+heightBonus) && targetXWorld >= (widthIndiceInOrthonormal+widthMalus) && targetXWorld <= (widthIndiceInOrthonormal+widthBonus)) pc.printf(" T "); else{ currProba=this->map.log_to_proba(this->map.cellsLogValues[x][y]); if ( currProba < 0.5){ pc.printf(" "); //pc.printf("%f",currProba); }else{ if(currProba==0.5){ pc.printf(" . "); //pc.printf("%f",currProba); }else{ pc.printf(" X "); //pc.printf("%f",currProba); } } } } } pc.printf("\n\r"); } } void MiniExplorerCoimbra::print_map_with_robot_position(){ float currProba; float heightIndiceInOrthonormal; float widthIndiceInOrthonormal; float widthMalus=-(3*this->map.sizeCellWidth/2); float widthBonus=this->map.sizeCellWidth/2; float heightMalus=-(3*this->map.sizeCellHeight/2); float heightBonus=this->map.sizeCellHeight/2; pc.printf("\n\r"); for (int y = this->map.nbCellHeight -1; y>-1; y--) { for (int x= 0; x<this->map.nbCellWidth; x++) { heightIndiceInOrthonormal=this->map.cell_height_coordinate_to_world(y); widthIndiceInOrthonormal=this->map.cell_width_coordinate_to_world(x); if(this->yWorld >= (heightIndiceInOrthonormal+heightMalus) && this->yWorld <= (heightIndiceInOrthonormal+heightBonus) && this->xWorld >= (widthIndiceInOrthonormal+widthMalus) && this->xWorld <= (widthIndiceInOrthonormal+widthBonus)){ pc.printf(" R "); //pc.printf("%f",currProba); }else{ currProba=this->map.log_to_proba(this->map.cellsLogValues[x][y]); if ( currProba < 0.5){ pc.printf(" "); //pc.printf("%f",currProba); }else{ if(currProba==0.5){ pc.printf(" . "); //pc.printf("%f",currProba); }else{ pc.printf(" X "); //pc.printf("%f",currProba); } } } } pc.printf("\n\r"); } } //robotX and robotY are from this->myOdometria(), calculate line_a, line_b and line_c void MiniExplorerCoimbra::calculate_line(float forceX, float forceY, float *line_a, float *line_b, float *line_c){ /* in the teachers maths it is *line_a=forceY; *line_b=-forceX; because a*x+b*y+c=0 a impact the vertical and b the horizontal and he has to put them like this because Robot space: World space: ^ ^ |x |y <- R O -> y x but since our forceX, forceY are already computed in the orthonormal space I m not sure we need to */ //*line_a=forceX; //*line_b=forceY; *line_a=forceY; *line_b=-forceX; //because the line computed always pass by the robot center we dont need lince_c *line_c=forceX*this->yWorld+forceY*this->xWorld; //*line_c=0; } //compute the force on X and Y void MiniExplorerCoimbra::compute_forceX_and_forceY(float* forceXWorld, float* forceYWorld, float targetXWorld, float targetYWorld){ float forceRepulsionComputedX=0; float forceRepulsionComputedY=0; this->print_map_with_robot_position(); for(int i=0;i<this->map.nbCellWidth;i++){ //for each cell of the map we compute a force of repulsion for(int j=0;j<this->map.nbCellHeight;j++){ this->update_force(i,j,&forceRepulsionComputedX,&forceRepulsionComputedY); } } //update with attraction force *forceXWorld=forceRepulsionComputedX; *forceYWorld=forceRepulsionComputedY; //this->print_map_with_robot_position(); //pc.printf("\r\nForce X repul:%f",*forceXWorld); //pc.printf("\r\nForce Y repul:%f",*forceYWorld); //test without atraction being impacted by distance //*forceXWorld+=this->attractionConstantForce*(targetXWorld-this->xWorld); //*forceYWorld+=this->attractionConstantForce*(targetYWorld-this->yWorld); float distanceTargetRobot=sqrt(pow(targetXWorld-this->xWorld,2)+pow(targetYWorld-this->yWorld,2)); if(distanceTargetRobot != 0){ *forceXWorld+=this->attractionConstantForce*(targetXWorld-this->xWorld)/distanceTargetRobot; *forceYWorld+=this->attractionConstantForce*(targetYWorld-this->yWorld)/distanceTargetRobot; }else{ *forceXWorld+=this->attractionConstantForce*(targetXWorld-this->xWorld)/0.001; *forceYWorld+=this->attractionConstantForce*(targetYWorld-this->yWorld)/0.001; } //pc.printf("\r\nForce X after attract:%f",*forceXWorld); //pc.printf("\r\nForce Y after attract:%f",*forceYWorld); float amplitude=sqrt(pow(*forceXWorld,2)+pow(*forceYWorld,2)); if(amplitude!=0){//avoid division by 0 if forceX and forceY == 0 *forceXWorld=*forceXWorld/amplitude; *forceYWorld=*forceYWorld/amplitude; }else{ *forceXWorld=*forceXWorld/0.001; *forceYWorld=*forceYWorld/0.001; } } //for vff void MiniExplorerCoimbra::go_to_line(float line_a, float line_b, float line_c,float targetXWorld, float targetYWorld){ float lineAngle; float angleError; float linear; float angular; float d; //line angle is beetween pi/2 and -pi/2 if(line_b!=0){ lineAngle=atan(line_a/-line_b); } else{ lineAngle=0; } this->myOdometria(); //Computing angle error angleError = lineAngle-this->thetaWorld;//TODO that I m not sure if(angleError>PI) angleError=-(angleError-PI); else if(angleError<-PI) angleError=-(angleError+PI); //d=this->distFromLine(this->xWorld, this->yWorld, line_a, line_b, line_c);//this could be 0 d=0; //pc.printf("\r\ndistance from line:%f",d); //Calculating velocities linear= this->kv*(3.14); angular=-this->kd*d + this->kh*angleError; float angularLeft=(linear-0.5*this->distanceWheels*angular)/this->radiusWheels; float angularRight=(linear+0.5*this->distanceWheels*angular)/this->radiusWheels; //Normalize speed for motors if(abs(angularLeft)>abs(angularRight)) { angularRight=this->speed*abs(angularRight/angularLeft)*this->sign1(angularRight); angularLeft=this->speed*this->sign1(angularLeft); } else { angularLeft=this->speed*abs(angularLeft/angularRight)*this->sign1(angularLeft); angularRight=this->speed*this->sign1(angularRight); } pc.printf("\r\nd = %f", d); pc.printf("\r\nerror = %f, lineAngle=%f, robotAngle=%f\n", angleError,lineAngle,this->thetaWorld); leftMotor(this->sign2(angularLeft),abs(angularLeft)); rightMotor(this->sign2(angularRight),abs(angularRight)); } void MiniExplorerCoimbra::go_to_line_first_lab(float line_a, float line_b, float line_c){ float lineAngle; float angleError; float linear; float angular; float d; //line angle is beetween pi/2 and -pi/2 if(line_b!=0){ lineAngle=atan(line_a/-line_b); } else{ lineAngle=1.5708; } do{ this->myOdometria(); //Computing angle error pc.printf("\r\nline angle = %f", lineAngle); pc.printf("\r\nthetaWorld = %f", thetaWorld); angleError = lineAngle-this->thetaWorld;//TODO that I m not sure if(angleError>PI) angleError=-(angleError-PI); else if(angleError<-PI) angleError=-(angleError+PI); pc.printf("\r\nangleError = %f\n", angleError); d=this->distFromLine(xWorld, yWorld, line_a, line_b, line_c); pc.printf("\r\ndistance to line = %f", d); //Calculating velocities linear= this->kv*(3.14); angular=-this->kd*d + this->kh*angleError; float angularLeft=(linear-0.5*this->distanceWheels*angular)/this->radiusWheels; float angularRight=(linear+0.5*this->distanceWheels*angular)/this->radiusWheels; //Normalize speed for motors if(abs(angularLeft)>abs(angularRight)) { angularRight=this->speed*abs(angularRight/angularLeft)*this->sign1(angularRight); angularLeft=this->speed*this->sign1(angularLeft); } else { angularLeft=this->speed*abs(angularLeft/angularRight)*this->sign1(angularLeft); angularRight=this->speed*this->sign1(angularRight); } leftMotor(this->sign2(angularLeft),abs(angularLeft)); rightMotor(this->sign2(angularRight),abs(angularRight)); }while(1); } void MiniExplorerCoimbra::update_force(int widthIndice, int heightIndice, float* forceRepulsionComputedX, float* forceRepulsionComputedY ){ //get the coordonate of the map and the robot in the ortonormal space float xWorldCell=this->map.cell_width_coordinate_to_world(widthIndice); float yWorldCell=this->map.cell_height_coordinate_to_world(heightIndice); //compute the distance beetween the cell and the robot float distanceCellToRobot=sqrt(pow(xWorldCell-this->xWorld,2)+pow(yWorldCell-this->yWorld,2)); float probaCell; //check if the cell is in range //float anglePointToRobot=atan2(yWorldCell-this->yWorld,xWorldCell-this->xWorld);//like world system float temp1; float temp2; if(distanceCellToRobot <= this->rangeForce) { probaCell=this->map.get_proba_cell(widthIndice,heightIndice); //pc.printf("\r\nupdate force proba:%f",probaCell); temp1=this->repulsionConstantForce*probaCell/pow(distanceCellToRobot,2); temp2=(xWorldCell-this->xWorld)/distanceCellToRobot; *forceRepulsionComputedX+=temp1*temp2; temp2=(yWorldCell-this->yWorld)/distanceCellToRobot; *forceRepulsionComputedY+=temp1*temp2; } } //return 1 if positiv, -1 if negativ float MiniExplorerCoimbra::sign1(float value){ if(value>=0) return 1; else return -1; } //return 1 if positiv, 0 if negativ int MiniExplorerCoimbra::sign2(float value){ if(value>=0) return 1; else return 0; } float MiniExplorerCoimbra::distFromLine(float robot_x, float robot_y, float line_a, float line_b, float line_c){ return abs((line_a*robot_x+line_b*robot_y+line_c)/sqrt(line_a*line_a+line_b*line_b)); } //4th LAB //starting position lower left void MiniExplorerCoimbra::test_procedure_lab_4(float sizeX, float sizeY){ this->map.fill_map_with_kalman_knowledge(); this->go_to_point_kalman(this->xWorld+sizeX,this->yWorld+sizeY); } //move of targetXWorld and targetYWorld ending in a targetAngleWorld void MiniExplorerCoimbra::go_turn_kalman(float targetXWorld, float targetYWorld, float targetAngleWorld) { //make sure the target is correct if(targetAngleWorld > PI) targetAngleWorld=-2*PI+targetAngleWorld; if(targetAngleWorld < -PI) targetAngleWorld=2*PI+targetAngleWorld; float distanceToTarget=100; do { leftMotor(1,50); rightMotor(0,50); pc.printf("\r\n test lab 4: OdometriaKalmanFilter"); this->OdometriaKalmanFilter(); float distanceToTarget=this->dist(this->xWorld, this->yWorld, targetXWorld, targetYWorld); //pc.printf("\n\rdist to target= %f",distanceToTarget); } while(distanceToTarget>1 || (abs(targetAngleWorld-this->thetaWorld)>0.1)); //Stop at the end leftMotor(1,0); rightMotor(1,0); pc.printf("\r\nReached Target!"); } //move of targetXWorld and targetYWorld ending in a targetAngleWorld void MiniExplorerCoimbra::go_straight_kalman(float targetXWorld, float targetYWorld, float targetAngleWorld) { //make sure the target is correct if(targetAngleWorld > PI) targetAngleWorld=-2*PI+targetAngleWorld; if(targetAngleWorld < -PI) targetAngleWorld=2*PI+targetAngleWorld; float distanceToTarget=100;; do { leftMotor(1,400); rightMotor(1,400); this->OdometriaKalmanFilter(); float distanceToTarget=this->dist(this->xWorld, this->yWorld, targetXWorld, targetYWorld); pc.printf("\n\rdist to target= %f",distanceToTarget); } while(distanceToTarget>1 || (abs(targetAngleWorld-this->thetaWorld)>0.1)); //Stop at the end leftMotor(1,0); rightMotor(1,0); pc.printf("\r\nReached Target!"); } void MiniExplorerCoimbra::OdometriaKalmanFilter(){ float encoderRightFailureRate=1; float encoderLeftFailureRate=1; int skip=1;//1==skip Kalman correction, just use KINEMATICS //=====KINEMATICS=========================== //we have trouble reading the encoders, in many case we only get 0 float R_cmK; float L_cmK; /* long int ticks1d=R_encoder(); char data_r_2[5]; i2c1.start(); i2c1.write(0x6C); i2c1.write(0x0C); i2c1.read(0x6D,data_r_2,4,0); i2c1.stop(); short int val1=data_r_2[0]; short int val2=data_r_2[1]; val1=(val1&0xf)*256; long int final=(val2+val1); ticks1d=final; long int ticks1e=L_encoder(); char data_r_2_2[5]; i2c.start(); i2c.write(0x6C); i2c.write(0x0C); i2c.read(0x6D,data_r_2_2,4,0); i2c.stop(); val1=data_r_2_2[0]; val2=data_r_2_2[1]; val1=(val1&0xf)*256; final=(val2+val1); ticks1e= final; pc.printf("\r\nEncoders ticks1d:%d,ticks1e:%d",ticks1d,ticks1e); long int R_ticks=ticks1d - ticks2d; long int L_ticks=ticks1e - ticks2e; pc.printf("\r\nEncoders R_ticks:%d,L_ticks:%d",R_ticks,L_ticks); //robot.h ticks2d=ticks1d; ticks2e=ticks1e; R_cmK= (float)R_ticks*((3.25*3.1415)/4096); L_cmK= (float)L_ticks*((3.25*3.1415)/4096); */ //fill R_cmK and L_cmK with how much is wheel has moved (custom robot.h) OdometriaKalman(&R_cmK, &L_cmK); //Odometria(); R_cmK= encoderRightFailureRate*R_cmK; L_cmK= encoderLeftFailureRate*L_cmK; //float R_cmK=D_cm*encoderRightFailureRate; //float L_cmK=L_cm*encoderLeftFailureRate; pc.printf("\r\nEncoders R_cm:%f,L_cm:%f",R_cmK,L_cmK); float distanceMoved=(R_cmK+L_cmK)/2; float angleMoved=(R_cmK-L_cmK)/this->distanceWheels; angleMoved = atan2(sin(angleMoved), cos(angleMoved)); //this->thetaWorld to theta robot float currTheta; if(this->thetaWorld < -PI/2) currTheta=PI/2+PI-this->thetaWorld; else currTheta=this->thetaWorld-PI/2; currTheta=currTheta+angleMoved; currTheta = atan2(sin(currTheta), cos(currTheta)); //currTheta to world coordinates float newAngleThetaWorld; if(currTheta >PI/2) newAngleThetaWorld=-PI+(currTheta-PI/2); else newAngleThetaWorld=currTheta+PI/2; //compute difference with cur angle angleMoved = newAngleThetaWorld-this->thetaWorld; if(angleMoved>PI) angleMoved=-(angleMoved-PI); else if(angleMoved<-PI) angleMoved=-(angleMoved+PI); pc.printf("\r\ndistanceMoved:%f, angleMoved:%f, totalAngle:%f",distanceMoved,angleMoved,newAngleThetaWorld); //here testing with robot coordinates float distanceMovedX=distanceMoved*cos(this->thetaWorld+(R_cmK-L_cmK)/(2*this->distanceWheels)); float distanceMovedY=distanceMoved*sin(this->thetaWorld+(R_cmK-L_cmK)/(2*this->distanceWheels)); pc.printf("\r\ndistanceMovedX:%f, distanceMovedY:%f",distanceMovedX,distanceMovedY); //try with world coordinate system myMatrix poseKplus1K(3,1); poseKplus1K.data[0][0]=this->xWorld+distanceMovedX; poseKplus1K.data[1][0]=this->yWorld+distanceMovedY; //poseKplus1K.data[2][0]=this->thetaWorld+angleMoved; poseKplus1K.data[2][0]=newAngleThetaWorld; //note if the encoders are correct it should work pc.printf("\r\nWithout correction: X=%f, Y=%f, theta=%f",poseKplus1K.get(0,0),poseKplus1K.get(1,0),poseKplus1K.get(2,0)); //=====ERROR_MODEL=========================== //FP Matrix slide LocalizationKALMAN myMatrix Fp(3,3); Fp.data[0][0]=1; Fp.data[1][1]=1; Fp.data[2][2]=1; Fp.data[0][2]=-1*distanceMoved*sin(this->thetaWorld+(angleMoved/2)); Fp.data[1][2]=distanceMoved*cos(this->thetaWorld+(angleMoved/2)); myMatrix FpTranspose(3,3); FpTranspose.fillWithTranspose(Fp); //Frl matrix slide LocalizationKALMAN myMatrix Frl(3,2); Frl.data[0][0]=0.5*cos(this->thetaWorld+(angleMoved/2))-(distanceMoved/(2*this->distanceWheels))*sin(this->thetaWorld+(angleMoved/2)); Frl.data[0][1]=0.5*cos(this->thetaWorld+(angleMoved/2))+(distanceMoved/(2*this->distanceWheels))*sin(this->thetaWorld+(angleMoved/2)); Frl.data[1][0]=0.5*sin(this->thetaWorld+(angleMoved/2))+(distanceMoved/(2*this->distanceWheels))*cos(this->thetaWorld+(angleMoved/2)); Frl.data[1][1]=0.5*sin(this->thetaWorld+(angleMoved/2))-(distanceMoved/(2*this->distanceWheels))*cos(this->thetaWorld+(angleMoved/2)); Frl.data[2][0]=(1/this->distanceWheels); Frl.data[2][1]=-(1/this->distanceWheels) ; myMatrix FrlTranspose(2,3); FrlTranspose.fillWithTranspose(Frl); //error constants... float kr=1; float kl=1; myMatrix covar(2,2); covar.data[0][0]=kr*abs(R_cmK); covar.data[1][1]=kl*abs(L_cmK); //uncertainty positionx, and theta at //1,1,1 myMatrix Q(3,3); Q.data[0][0]=1; Q.data[1][1]=2; Q.data[2][2]=0.01; //new covariance= Fp*old covariance*FpTranspose +Frl*covar*FrlTranspose +Q slide LocalizationKALMAN myMatrix covariancePositionEstimationKplus1K(3,3); myMatrix temp1(3,3); temp1.fillByMultiplication(Fp,this->covariancePositionEstimationK);//Fp*old covariance myMatrix temp2(3,3); temp2.fillByMultiplication(temp1,FpTranspose);//Fp*old covariance*FpTranspose temp1.fillWithZeroes(); myMatrix temp3(3,2); temp3.fillByMultiplication(Frl,covar);//Frl*covar temp1.fillByMultiplication(temp3,FrlTranspose);//Frl*covar*FrlTranspose covariancePositionEstimationKplus1K.addition(temp2);//Fp*old covariance*FpTranspose covariancePositionEstimationKplus1K.addition(temp1);//Fp*old covariance*FpTranspose +Frl*covar*FrlTranspose covariancePositionEstimationKplus1K.addition(Q);//Fp*old covariance*FpTranspose +Frl*covar*FrlTranspose +Q if(skip==1){ this->covariancePositionEstimationK.fillByCopy(covariancePositionEstimationKplus1K); //pc.printf("\r\nposeKplus1Kplus1 X=%f, Y=%f, theta=%f",poseKplus1Kplus1.data[0][0],poseKplus1Kplus1.data[1][0],poseKplus1Kplus1.data[2][0]); //update pose this->xWorld=poseKplus1K.data[0][0]; this->yWorld=poseKplus1K.data[1][0]; this->thetaWorld=poseKplus1K.data[2][0]; pc.printf("\r\nWithout Kalman X=%f, Y=%f, theta=%f",xWorld,yWorld,thetaWorld); }else{ //=====OBSERVATION===== //get the estimated measure we should have according to our knowledge of the map and the previously computed localisation float leftCmEstimated=this->sonarLeft.maxRange; float frontCmEstimated=this->sonarFront.maxRange; float rightCmEstimated=this->sonarRight.maxRange; float xWorldCell; float yWorldCell; float currDistance; float xClosestCellLeft; float yClosestCellLeft; float xClosestCellFront; float yClosestCellFront; float xClosestCellRight; float yClosestCellRight; //note: sonar.isInRange already incorpore the sonar position and angle relative to the robot //note curr change to isInRange: xs-xcell and not xcell-xs //get theorical distance to sonar for(int i=0;i<this->map.nbCellWidth;i++){ for(int j=0;j<this->map.nbCellHeight;j++){ //check if occupied, if not discard if(this->map.get_proba_cell(i,j)>0.5){ //check if in sonar range xWorldCell=this->map.cell_width_coordinate_to_world(i); yWorldCell=this->map.cell_height_coordinate_to_world(j); //check left currDistance=this->sonarLeft.isInRange(xWorldCell,yWorldCell,poseKplus1K.data[0][0],poseKplus1K.data[1][0],poseKplus1K.data[2][0]); if((currDistance < this->sonarLeft.maxRange) && currDistance > -1){ //check if distance is lower than previous, update lowest if so if(currDistance < leftCmEstimated){ leftCmEstimated= currDistance; xClosestCellLeft=xWorldCell; yClosestCellLeft=yWorldCell; } } //check front currDistance=this->sonarFront.isInRange(xWorldCell,yWorldCell,poseKplus1K.data[0][0],poseKplus1K.data[1][0],poseKplus1K.data[2][0]); if((currDistance < this->sonarFront.maxRange) && currDistance > -1){ //check if distance is lower than previous, update lowest if so if(currDistance < frontCmEstimated){ frontCmEstimated= currDistance; xClosestCellFront=xWorldCell; yClosestCellFront=yWorldCell; } } //check right currDistance=this->sonarRight.isInRange(xWorldCell,yWorldCell,poseKplus1K.data[0][0],poseKplus1K.data[1][0],poseKplus1K.data[2][0]); if((currDistance < this->sonarRight.maxRange) && currDistance > -1){ //check if distance is lower than previous, update lowest if so if(currDistance < rightCmEstimated){ rightCmEstimated= currDistance; xClosestCellRight=xWorldCell; yClosestCellRight=yWorldCell; } } } } } //check measurements from sonars, see if they passed the validation gate float leftCm = get_distance_left_sensor()/10; float frontCm = get_distance_front_sensor()/10; float rightCm = get_distance_right_sensor()/10; //if superior to sonar range, put the value to sonar max range if(leftCm > this->sonarLeft.maxRange) leftCm=this->sonarLeft.maxRange; if(frontCm > this->sonarFront.maxRange) frontCm=this->sonarFront.maxRange; if(rightCm > this->sonarRight.maxRange) rightCm=this->sonarRight.maxRange; pc.printf("\r\ndistance sonars read Lcm=%f, FCm=%f, RCm=%f",leftCm,frontCm,rightCm); pc.printf("\r\ndistance sonars estimated ELcm=%f, EFCm=%f, ERCm=%f",leftCmEstimated,frontCmEstimated,rightCmEstimated); //======INNOVATION======== //compute jacobian of observation myMatrix jacobianOfObservationLeft(1,3); myMatrix jacobianOfObservationRight(1,3); myMatrix jacobianOfObservationFront(1,3); //(5) //note: for cos/sin having abs(angles) > PI is not an issue float xSonarLeft=poseKplus1K.data[0][0]+this->sonarLeft.distanceX*cos(poseKplus1K.data[0][2])-this->sonarLeft.distanceY*sin(poseKplus1K.data[0][2]); float ySonarLeft=poseKplus1K.data[1][0]+this->sonarLeft.distanceX*sin(poseKplus1K.data[0][2])-this->sonarLeft.distanceY*cos(poseKplus1K.data[0][2]); float angleSonarLeft=poseKplus1K.data[0][2]+this->sonarLeft.angleFromCenter; if(angleSonarLeft > PI) angleSonarLeft=angleSonarLeft-2*PI; if(angleSonarLeft < -PI) angleSonarLeft=angleSonarLeft+2*PI; float xSonarFront=poseKplus1K.data[0][0]+this->sonarFront.distanceX*cos(poseKplus1K.data[0][2])-this->sonarFront.distanceY*sin(poseKplus1K.data[0][2]); float ySonarFront=poseKplus1K.data[1][0]+this->sonarFront.distanceX*sin(poseKplus1K.data[0][2])-this->sonarFront.distanceY*cos(poseKplus1K.data[0][2]); float angleSonarFront=poseKplus1K.data[0][2]+this->sonarFront.angleFromCenter; if(angleSonarFront > PI) angleSonarFront=angleSonarFront-2*PI; if(angleSonarFront < -PI) angleSonarFront=angleSonarFront+2*PI; float xSonarRight=poseKplus1K.data[0][0]+this->sonarRight.distanceX*cos(poseKplus1K.data[0][2])-this->sonarRight.distanceY*sin(poseKplus1K.data[0][2]); float ySonarRight=poseKplus1K.data[1][0]+this->sonarRight.distanceX*sin(poseKplus1K.data[0][2])-this->sonarRight.distanceY*cos(poseKplus1K.data[0][2]); float angleSonarRight=poseKplus1K.data[0][2]+this->sonarRight.angleFromCenter; if(angleSonarRight > PI) angleSonarRight=angleSonarRight-2*PI; if(angleSonarRight < -PI) angleSonarRight=angleSonarRight+2*PI; //note: last line divided by d, [0][2] all angles are theta^ teacher's advice, error in paper //left jacobianOfObservationLeft.data[0][0]=(xSonarLeft-xClosestCellLeft)/leftCmEstimated; jacobianOfObservationLeft.data[0][1]=(ySonarLeft-yClosestCellLeft)/leftCmEstimated; jacobianOfObservationLeft.data[0][2]=(1/leftCmEstimated)*(xClosestCellLeft-xSonarLeft)*(this->sonarLeft.distanceX*sin(angleSonarLeft)+this->sonarLeft.distanceY*cos(angleSonarLeft))+(yClosestCellLeft-ySonarLeft)*(-this->sonarLeft.distanceX*cos(angleSonarLeft)+this->sonarLeft.distanceY*sin(angleSonarLeft)); //front jacobianOfObservationFront.data[0][0]=(xSonarFront-xClosestCellFront)/frontCmEstimated; jacobianOfObservationFront.data[0][1]=(ySonarFront-yClosestCellFront)/frontCmEstimated; jacobianOfObservationFront.data[0][2]=(1/frontCmEstimated)*(xClosestCellFront-xSonarFront)*(this->sonarFront.distanceX*sin(angleSonarFront)+this->sonarFront.distanceY*cos(angleSonarFront))+(yClosestCellFront-ySonarFront)*(-this->sonarFront.distanceX*cos(angleSonarFront)+this->sonarFront.distanceY*sin(angleSonarFront)); //right jacobianOfObservationRight.data[0][0]=(xSonarRight-xClosestCellRight)/rightCmEstimated; jacobianOfObservationRight.data[0][1]=(ySonarRight-yClosestCellRight)/rightCmEstimated; jacobianOfObservationRight.data[0][2]=(xClosestCellRight-xSonarRight)*(xSonarRight*sin(poseKplus1K.data[2][0])+ySonarRight*cos(poseKplus1K.data[2][0]))+(yClosestCellRight-ySonarRight)*(-xSonarRight*cos(poseKplus1K.data[2][0])+ySonarRight*sin(poseKplus1K.data[2][0])); jacobianOfObservationRight.data[0][2]=(1/rightCmEstimated)*(xClosestCellRight-xSonarRight)*(this->sonarRight.distanceX*sin(angleSonarRight)+this->sonarRight.distanceY*cos(angleSonarRight))+(yClosestCellRight-ySonarRight)*(-this->sonarRight.distanceX*cos(angleSonarRight)+this->sonarRight.distanceY*sin(angleSonarRight)); myMatrix jacobianOfObservationRightTranspose(3,1); jacobianOfObservationRightTranspose.fillWithTranspose(jacobianOfObservationRight); myMatrix jacobianOfObservationFrontTranspose(3,1); jacobianOfObservationFrontTranspose.fillWithTranspose(jacobianOfObservationFront); myMatrix jacobianOfObservationLeftTranspose(3,1); jacobianOfObservationLeftTranspose.fillWithTranspose(jacobianOfObservationLeft); //error constants 0,0,0 sonars perfect; must be found by experimenting; gives mean and standart deviation //in centimeters float R_front=4; float R_left=4; float R_right=4; //R-> 4 in diagonal //for each sonar (concatenated covariance matrix of innovation) //equ (12), innovationCovariance =JacobianObservation*covariancePositionEstimationKplus1K*JacobianObservationTranspose+R myMatrix temp4(1,3); temp4.fillByMultiplication(jacobianOfObservationFront,covariancePositionEstimationKplus1K); myMatrix temp5(1,1); temp5.fillByMultiplication(temp4,jacobianOfObservationFrontTranspose); float innovationCovarianceFront=temp5.data[0][0]+R_front; temp4.fillWithZeroes(); temp5.fillWithZeroes(); temp4.fillByMultiplication(jacobianOfObservationLeft,covariancePositionEstimationKplus1K); temp5.fillByMultiplication(temp4,jacobianOfObservationLeftTranspose); float innovationCovarianceLeft=temp5.data[0][0]+R_left; temp4.fillWithZeroes(); temp5.fillWithZeroes(); temp4.fillByMultiplication(jacobianOfObservationRight,covariancePositionEstimationKplus1K); temp5.fillByMultiplication(temp4,jacobianOfObservationRightTranspose); float innovationCovarianceRight=temp5.data[0][0]+R_right; this->printMatrix(jacobianOfObservationLeft); this->printMatrix(jacobianOfObservationFront); this->printMatrix(jacobianOfObservationRight); pc.printf("\r\ninnovationCovariance L:%f, F:%f, R:%f",innovationCovarianceLeft,innovationCovarianceFront,innovationCovarianceRight); //get the innoncation: the value of the difference between the actual measure and what we anticipated float innovationLeft=leftCm-leftCmEstimated; float innovationFront=frontCm-frontCmEstimated; float innovationRight=rightCm-rightCmEstimated; //check if it pass the validation gate float gateScoreLeft=innovationLeft*innovationLeft/innovationCovarianceLeft; float gateScoreFront=innovationFront*innovationFront/innovationCovarianceFront; float gateScoreRight=innovationRight*innovationRight/innovationCovarianceRight; //5cm -> 25 int errorsquare=25;//constant error int nbPassed=0; if(gateScoreLeft <= errorsquare) nbPassed+=1; if(gateScoreFront <= errorsquare) nbPassed+=10; if(gateScoreRight <= errorsquare) nbPassed+=100; //for those who passed //compute composite innovation pc.printf("\r\nscore L:%f, F:%f, R:%f",gateScoreLeft,gateScoreFront,gateScoreRight); pc.printf("\r\nnbPassed=%d",nbPassed); //compositeMeasurementJacobian myMatrix compositeMeasurementJacobian3x3(3,3); myMatrix compositeMeasurementJacobian2x3(2,3); myMatrix compositeMeasurementJacobian1x3(1,3); myMatrix compositeMeasurementJacobian3x3Transpose(3,3); myMatrix compositeMeasurementJacobian2x3Transpose(3,2); myMatrix compositeMeasurementJacobian1x3Transpose(3,1); //compositeInnovation myMatrix compositeInnovation3x1(3,1); myMatrix compositeInnovation2x1(2,1); myMatrix compositeInnovation1x1(1,1); //compositeInnovationCovariance myMatrix compositeInnovationCovariance3x3(3,3); myMatrix compositeInnovationCovariance2x2(2,2); myMatrix compositeInnovationCovariance1x1(1,1); myMatrix tempCompositeInnovationCovariance3x3(3,3); myMatrix tempCompositeInnovationCovariance2x2(2,2); myMatrix compositeInnovationCovariance3x3Inverse(3,3); myMatrix compositeInnovationCovariance2x2Inverse(2,2); myMatrix compositeInnovationCovariance1x1Inverse(1,1); //Kalman Gain myMatrix kalmanGain3X3(3,3); myMatrix kalmanGain3X2(3,2); myMatrix kalmanGain3X1(3,1); myMatrix tempKalmanGain3X3(3,3); myMatrix tempKalmanGain3X2(3,2); myMatrix tempKalmanGain3X1(3,1); myMatrix kalmanGain3X3Transpose(3,3); myMatrix kalmanGain3X2Transpose(2,3); myMatrix kalmanGain3X1Transpose(1,3); //new pose estimation myMatrix poseKplus1Kplus1(3,1); poseKplus1Kplus1.fillByCopy(poseKplus1K); myMatrix tempPoseKplus1Kplus1(3,1); //covariancePositionEstimationKplus1Kplus1 myMatrix covariancePositionEstimationKplus1Kplus1(3,3); covariancePositionEstimationKplus1Kplus1.fillByCopy(covariancePositionEstimationKplus1K); myMatrix temp2CovariancePositionEstimationKplus1Kplus13x3(3,3); myMatrix tempCovariancePositionEstimationKplus1Kplus13x3(3,3); myMatrix tempCovariancePositionEstimationKplus1Kplus13x2(3,2); myMatrix tempCovariancePositionEstimationKplus1Kplus13x1(3,1); //we do not use the composite measurement jacobian (16), we directly use the values from the measurement jacobian (jacobianOfObservation) switch(nbPassed){ case 0://none //nothings happens it's okay pc.printf("\r\n000"); break; case 1://left //compute compositeMeasurementJacobian //here compositeMeasurementJacobian= jacobianOfObservationLeft compositeMeasurementJacobian1x3.fillByCopy(jacobianOfObservationLeft); pc.printf("\r\n1:calcul compositeMeasurementJacobian1x3 passed"); //get the compositeMeasurementJacobianTranspose compositeMeasurementJacobian1x3Transpose.fillWithTranspose(compositeMeasurementJacobian1x3); //compute compositeInnovation //here compositeInnovation=innovationLeft compositeInnovation1x1.data[0][0]=innovationLeft; pc.printf("\r\n1:calcul compositeInnovation1x1 passed"); //compute compositeInnovationCovariance=compositeMeasurementJacobian*covariancePositionEstimationKplus1K*compositeMeasurementJacobianTranpose //add the right R on the diagonal //here compositeInnovationCovariance=innovationCovarianceLeft compositeInnovationCovariance1x1.data[0][0]=innovationCovarianceLeft; pc.printf("\r\n1:calcul compositeInnovationCovariance1x1 passed"); //get the inverse of the compositeInnovationCovariance compositeInnovationCovariance1x1Inverse.data[0][0]=1/innovationCovarianceLeft; pc.printf("\r\n1:calcul compositeInnovationCovariance1x1Inverse passed"); //compute KalmanGain=covariancePositionEstimationKplus1K*compositeMeasurementJacobianTranspose*Inverse(compositeInnovationCovariance) tempKalmanGain3X1.fillByMultiplication(covariancePositionEstimationKplus1K,compositeMeasurementJacobian1x3Transpose); kalmanGain3X1.fillByMultiplication(tempKalmanGain3X1,compositeInnovationCovariance1x1Inverse); pc.printf("\r\n1:calcul kalmanGain3X1 passed"); //get the transpose of the kalman gain kalmanGain3X1Transpose.fillWithTranspose(kalmanGain3X1); //update pose estimation=old pose estimation + kalman gain*compositeInnovation tempPoseKplus1Kplus1.fillByMultiplication(kalmanGain3X1,compositeInnovation1x1); poseKplus1Kplus1.addition(tempPoseKplus1Kplus1); pc.printf("\r\n1:calcul poseKplus1Kplus1 passed"); //compute covariancePositionEstimationKplus1Kplus1=covariancePositionEstimationKplus1K-kalmanGain*compositeInnovationCovariance*KalmanGainTranspose tempCovariancePositionEstimationKplus1Kplus13x1.fillByMultiplication(kalmanGain3X1,compositeInnovationCovariance1x1); tempCovariancePositionEstimationKplus1Kplus13x3.fillByMultiplication(tempCovariancePositionEstimationKplus1Kplus13x1,kalmanGain3X1Transpose); covariancePositionEstimationKplus1Kplus1.subtraction(tempCovariancePositionEstimationKplus1Kplus13x3); pc.printf("\r\n1:calcul covariancePositionEstimationKplus1Kplus1 passed"); break; case 10://front //compute compositeMeasurementJacobian compositeMeasurementJacobian1x3.fillByCopy(jacobianOfObservationFront); pc.printf("\r\n10:calcul compositeMeasurementJacobian1x3 passed"); //get the compositeMeasurementJacobianTranspose compositeMeasurementJacobian1x3Transpose.fillWithTranspose(compositeMeasurementJacobian1x3); //compute compositeInnovation compositeInnovation1x1.data[0][0]=innovationFront; pc.printf("\r\n10:calcul compositeInnovation1x1 passed"); //compute compositeInnovationCovariance=compositeMeasurementJacobian*covariancePositionEstimationKplus1K*compositeMeasurementJacobianTranpose //add the right R on the diagonal compositeInnovationCovariance1x1.data[0][0]=innovationCovarianceFront; pc.printf("\r\n10:calcul compositeInnovationCovariance1x1 passed"); //get the inverse of the compositeInnovationCovariance compositeInnovationCovariance1x1Inverse.data[0][0]=1/innovationCovarianceFront; pc.printf("\r\n10:calcul compositeInnovationCovariance1x1Inverse passed"); //compute KalmanGain=covariancePositionEstimationKplus1K*compositeMeasurementJacobianTranspose*Inverse(compositeInnovationCovariance) tempKalmanGain3X1.fillByMultiplication(covariancePositionEstimationKplus1K,compositeMeasurementJacobian1x3Transpose); kalmanGain3X1.fillByMultiplication(tempKalmanGain3X1,compositeInnovationCovariance1x1Inverse); pc.printf("\r\n10:calcul kalmanGain3X1 passed"); //get the transpose of the kalman gain kalmanGain3X1Transpose.fillWithTranspose(kalmanGain3X1); //update pose estimation=old pose estimation + kalman gain*compositeInnovation tempPoseKplus1Kplus1.fillByMultiplication(kalmanGain3X1,compositeInnovation1x1); poseKplus1Kplus1.addition(tempPoseKplus1Kplus1); pc.printf("\r\n10:calcul poseKplus1Kplus1 passed"); //compute covariancePositionEstimationKplus1Kplus1=covariancePositionEstimationKplus1K-kalmanGain*compositeInnovationCovariance*KalmanGainTranspose tempCovariancePositionEstimationKplus1Kplus13x1.fillByMultiplication(kalmanGain3X1,compositeInnovationCovariance1x1); tempCovariancePositionEstimationKplus1Kplus13x3.fillByMultiplication(tempCovariancePositionEstimationKplus1Kplus13x1,kalmanGain3X1Transpose); covariancePositionEstimationKplus1Kplus1.subtraction(tempCovariancePositionEstimationKplus1Kplus13x3); pc.printf("\r\n10:calcul covariancePositionEstimationKplus1Kplus1 passed"); break; case 11://left and front //compute compositeMeasurementJacobian compositeMeasurementJacobian2x3.data[0][0]=jacobianOfObservationLeft.data[0][0]; compositeMeasurementJacobian2x3.data[0][1]=jacobianOfObservationLeft.data[0][1]; compositeMeasurementJacobian2x3.data[0][2]=jacobianOfObservationLeft.data[0][2]; compositeMeasurementJacobian2x3.data[1][0]=jacobianOfObservationFront.data[0][0]; compositeMeasurementJacobian2x3.data[1][1]=jacobianOfObservationFront.data[0][1]; compositeMeasurementJacobian2x3.data[1][2]=jacobianOfObservationFront.data[0][2]; pc.printf("\r\n11:calcul compositeMeasurementJacobian2x3 passed"); //get the compositeMeasurementJacobianTranspose compositeMeasurementJacobian2x3Transpose.fillWithTranspose(compositeMeasurementJacobian2x3); //compute compositeInnovation compositeInnovation2x1.data[0][0]=innovationLeft; compositeInnovation2x1.data[1][0]=innovationFront; pc.printf("\r\n11:calcul compositeInnovation2x1 passed"); //compute compositeInnovationCovariance=compositeMeasurementJacobian*covariancePositionEstimationKplus1K*compositeMeasurementJacobianTranpose+R tempCompositeInnovationCovariance2x2.fillByMultiplication(compositeMeasurementJacobian2x3,covariancePositionEstimationKplus1K); compositeInnovationCovariance2x2.fillByMultiplication(tempCompositeInnovationCovariance2x2,compositeMeasurementJacobian2x3Transpose); //add the right R on the diagonal compositeInnovationCovariance2x2.data[0][0]+=R_left; compositeInnovationCovariance2x2.data[1][1]+=R_front; pc.printf("\r\n11:calcul compositeInnovationCovariance2x2 passed"); //get the inverse of the compositeInnovationCovariance compositeInnovationCovariance2x2Inverse.fillWithInverse(compositeInnovationCovariance2x2); pc.printf("\r\n11:calcul compositeInnovationCovariance2x2Inverse passed"); //compute KalmanGain=covariancePositionEstimationKplus1K*compositeMeasurementJacobianTranspose*Inverse(compositeInnovationCovariance) tempKalmanGain3X2.fillByMultiplication(covariancePositionEstimationKplus1K,compositeMeasurementJacobian2x3Transpose); kalmanGain3X2.fillByMultiplication(tempKalmanGain3X2,compositeInnovationCovariance2x2Inverse); pc.printf("\r\n11:calcul kalmanGain3X2 passed"); //get the transpose of the kalman gain kalmanGain3X2Transpose.fillWithTranspose(kalmanGain3X2); //update pose estimation=old pose estimation + kalman gain*compositeInnovation tempPoseKplus1Kplus1.fillByMultiplication(kalmanGain3X2,compositeInnovation2x1); poseKplus1Kplus1.addition(tempPoseKplus1Kplus1); pc.printf("\r\n11:calcul poseKplus1Kplus1 passed"); //compute covariancePositionEstimationKplus1Kplus1=covariancePositionEstimationKplus1K-kalmanGain*compositeInnovationCovariance*KalmanGainTranspose tempCovariancePositionEstimationKplus1Kplus13x2.fillByMultiplication(kalmanGain3X2,compositeInnovationCovariance2x2); tempCovariancePositionEstimationKplus1Kplus13x3.fillByMultiplication(tempCovariancePositionEstimationKplus1Kplus13x2,kalmanGain3X2Transpose); covariancePositionEstimationKplus1Kplus1.subtraction(tempCovariancePositionEstimationKplus1Kplus13x3); pc.printf("\r\n11:calcul covariancePositionEstimationKplus1Kplus1 passed"); break; case 100://right //compute compositeMeasurementJacobian compositeMeasurementJacobian1x3.fillByCopy(jacobianOfObservationRight); pc.printf("\r\n100:calcul compositeMeasurementJacobian1x3 passed"); //get the compositeMeasurementJacobianTranspose compositeMeasurementJacobian1x3Transpose.fillWithTranspose(compositeMeasurementJacobian1x3); //compute compositeInnovation compositeInnovation1x1.data[0][0]=innovationRight; //compute compositeInnovationCovariance=compositeMeasurementJacobian*covariancePositionEstimationKplus1K*compositeMeasurementJacobianTranpose //add the right R on the diagonal compositeInnovationCovariance1x1.data[0][0]=innovationCovarianceRight; //get the inverse of the compositeInnovationCovariance compositeInnovationCovariance1x1Inverse.data[0][0]=1/innovationCovarianceRight; pc.printf("\r\n100:calcul compositeInnovationCovariance1x1Inverse passed"); //compute KalmanGain=covariancePositionEstimationKplus1K*compositeMeasurementJacobianTranspose*Inverse(compositeInnovationCovariance) tempKalmanGain3X1.fillByMultiplication(covariancePositionEstimationKplus1K,compositeMeasurementJacobian1x3Transpose); kalmanGain3X1.fillByMultiplication(tempKalmanGain3X1,compositeInnovationCovariance1x1Inverse); pc.printf("\r\n100:calcul kalmanGain3X1 passed"); //get the transpose of the kalman gain kalmanGain3X1Transpose.fillWithTranspose(kalmanGain3X1); //update pose estimation=old pose estimation + kalman gain*compositeInnovation tempPoseKplus1Kplus1.fillByMultiplication(kalmanGain3X1,compositeInnovation1x1); poseKplus1Kplus1.addition(tempPoseKplus1Kplus1); pc.printf("\r\n100:calcul poseKplus1Kplus1 passed"); //compute covariancePositionEstimationKplus1Kplus1=covariancePositionEstimationKplus1K-kalmanGain*compositeInnovationCovariance*KalmanGainTranspose tempCovariancePositionEstimationKplus1Kplus13x1.fillByMultiplication(kalmanGain3X1,compositeInnovationCovariance1x1); tempCovariancePositionEstimationKplus1Kplus13x3.fillByMultiplication(tempCovariancePositionEstimationKplus1Kplus13x1,kalmanGain3X1Transpose); covariancePositionEstimationKplus1Kplus1.subtraction(tempCovariancePositionEstimationKplus1Kplus13x3); pc.printf("\r\n100:calcul covariancePositionEstimationKplus1Kplus1 passed"); break; case 101://left and right //compute compositeMeasurementJacobian compositeMeasurementJacobian2x3.data[0][0]=jacobianOfObservationLeft.data[0][0]; compositeMeasurementJacobian2x3.data[0][1]=jacobianOfObservationLeft.data[0][1]; compositeMeasurementJacobian2x3.data[0][2]=jacobianOfObservationLeft.data[0][2]; compositeMeasurementJacobian2x3.data[1][0]=jacobianOfObservationRight.data[0][0]; compositeMeasurementJacobian2x3.data[1][1]=jacobianOfObservationRight.data[0][1]; compositeMeasurementJacobian2x3.data[1][2]=jacobianOfObservationRight.data[0][2]; pc.printf("\r\n101:calcul compositeMeasurementJacobian2x3 passed"); //get the compositeMeasurementJacobianTranspose compositeMeasurementJacobian2x3Transpose.fillWithTranspose(compositeMeasurementJacobian2x3); //compute compositeInnovation compositeInnovation2x1.data[0][0]=innovationLeft; compositeInnovation2x1.data[1][0]=innovationRight; //compute compositeInnovationCovariance=compositeMeasurementJacobian*covariancePositionEstimationKplus1K*compositeMeasurementJacobianTranpose+R tempCompositeInnovationCovariance2x2.fillByMultiplication(compositeMeasurementJacobian2x3,covariancePositionEstimationKplus1K); compositeInnovationCovariance2x2.fillByMultiplication(tempCompositeInnovationCovariance2x2,compositeMeasurementJacobian2x3Transpose); //add the right R on the diagonal compositeInnovationCovariance2x2.data[0][0]+=R_left; compositeInnovationCovariance2x2.data[1][1]+=R_right; pc.printf("\r\n101:calcul compositeInnovationCovariance2x2 passed"); //get the inverse of the compositeInnovationCovariance compositeInnovationCovariance2x2Inverse.fillWithInverse(compositeInnovationCovariance2x2); pc.printf("\r\n101:calcul compositeInnovationCovariance2x2Inverse passed"); //compute KalmanGain=covariancePositionEstimationKplus1K*compositeMeasurementJacobianTranspose*Inverse(compositeInnovationCovariance) tempKalmanGain3X2.fillByMultiplication(covariancePositionEstimationKplus1K,compositeMeasurementJacobian2x3Transpose); kalmanGain3X2.fillByMultiplication(tempKalmanGain3X2,compositeInnovationCovariance2x2Inverse); pc.printf("\r\n101:calcul kalmanGain3X2 passed"); //get the transpose of the kalman gain kalmanGain3X2Transpose.fillWithTranspose(kalmanGain3X2); //update pose estimation=old pose estimation + kalman gain*compositeInnovation tempPoseKplus1Kplus1.fillByMultiplication(kalmanGain3X2,compositeInnovation2x1); poseKplus1Kplus1.addition(tempPoseKplus1Kplus1); pc.printf("\r\n101:calcul poseKplus1Kplus1 passed"); //compute covariancePositionEstimationKplus1Kplus1=covariancePositionEstimationKplus1K-kalmanGain*compositeInnovationCovariance*KalmanGainTranspose tempCovariancePositionEstimationKplus1Kplus13x2.fillByMultiplication(kalmanGain3X2,compositeInnovationCovariance2x2); tempCovariancePositionEstimationKplus1Kplus13x3.fillByMultiplication(tempCovariancePositionEstimationKplus1Kplus13x2,kalmanGain3X2Transpose); covariancePositionEstimationKplus1Kplus1.subtraction(tempCovariancePositionEstimationKplus1Kplus13x3); pc.printf("\r\n101:calcul covariancePositionEstimationKplus1Kplus1 passed"); break; case 110:// front and right //compute compositeMeasurementJacobian compositeMeasurementJacobian2x3.data[0][0]=jacobianOfObservationFront.data[0][0]; compositeMeasurementJacobian2x3.data[0][1]=jacobianOfObservationFront.data[0][1]; compositeMeasurementJacobian2x3.data[0][2]=jacobianOfObservationFront.data[0][2]; compositeMeasurementJacobian2x3.data[1][0]=jacobianOfObservationRight.data[0][0]; compositeMeasurementJacobian2x3.data[1][1]=jacobianOfObservationRight.data[0][1]; compositeMeasurementJacobian2x3.data[1][2]=jacobianOfObservationRight.data[0][2]; pc.printf("\r\n110:calcul compositeMeasurementJacobian2x3 passed"); //get the compositeMeasurementJacobianTranspose compositeMeasurementJacobian2x3Transpose.fillWithTranspose(compositeMeasurementJacobian2x3); //compute compositeInnovation compositeInnovation2x1.data[0][0]=innovationFront; compositeInnovation2x1.data[1][0]=innovationRight; //compute compositeInnovationCovariance=compositeMeasurementJacobian*covariancePositionEstimationKplus1K*compositeMeasurementJacobianTranpose+R tempCompositeInnovationCovariance2x2.fillByMultiplication(compositeMeasurementJacobian2x3,covariancePositionEstimationKplus1K); compositeInnovationCovariance2x2.fillByMultiplication(tempCompositeInnovationCovariance2x2,compositeMeasurementJacobian2x3Transpose); //add the right R on the diagonal compositeInnovationCovariance2x2.data[0][0]+=R_front; compositeInnovationCovariance2x2.data[1][1]+=R_right; pc.printf("\r\n110:calcul compositeInnovationCovariance2x2 passed"); //get the inverse of the compositeInnovationCovariance compositeInnovationCovariance2x2Inverse.fillWithInverse(compositeInnovationCovariance2x2); pc.printf("\r\n110:calcul compositeInnovationCovariance2x2Inverse passed"); //compute KalmanGain=covariancePositionEstimationKplus1K*compositeMeasurementJacobianTranspose*Inverse(compositeInnovationCovariance) tempKalmanGain3X2.fillByMultiplication(covariancePositionEstimationKplus1K,compositeMeasurementJacobian2x3Transpose); kalmanGain3X2.fillByMultiplication(tempKalmanGain3X2,compositeInnovationCovariance2x2Inverse); pc.printf("\r\n110:calcul kalmanGain3X2 passed"); //get the transpose of the kalman gain kalmanGain3X2Transpose.fillWithTranspose(kalmanGain3X2); //update pose estimation=old pose estimation + kalman gain*compositeInnovation tempPoseKplus1Kplus1.fillByMultiplication(kalmanGain3X2,compositeInnovation2x1); poseKplus1Kplus1.addition(tempPoseKplus1Kplus1); pc.printf("\r\n110:calcul poseKplus1Kplus1 passed"); //compute covariancePositionEstimationKplus1Kplus1=covariancePositionEstimationKplus1K-kalmanGain*compositeInnovationCovariance*KalmanGainTranspose tempCovariancePositionEstimationKplus1Kplus13x2.fillByMultiplication(kalmanGain3X2,compositeInnovationCovariance2x2); tempCovariancePositionEstimationKplus1Kplus13x3.fillByMultiplication(tempCovariancePositionEstimationKplus1Kplus13x2,kalmanGain3X2Transpose); covariancePositionEstimationKplus1Kplus1.subtraction(tempCovariancePositionEstimationKplus1Kplus13x3); pc.printf("\r\n110:calcul covariancePositionEstimationKplus1Kplus1 passed"); break; case 111://left front and right //compute compositeMeasurementJacobian compositeMeasurementJacobian3x3.data[0][0]=jacobianOfObservationLeft.data[0][0]; compositeMeasurementJacobian3x3.data[0][1]=jacobianOfObservationLeft.data[0][1]; compositeMeasurementJacobian3x3.data[0][2]=jacobianOfObservationLeft.data[0][2]; compositeMeasurementJacobian3x3.data[1][0]=jacobianOfObservationFront.data[0][0]; compositeMeasurementJacobian3x3.data[1][1]=jacobianOfObservationFront.data[0][1]; compositeMeasurementJacobian3x3.data[1][2]=jacobianOfObservationFront.data[0][2]; compositeMeasurementJacobian3x3.data[2][0]=jacobianOfObservationRight.data[0][0]; compositeMeasurementJacobian3x3.data[2][1]=jacobianOfObservationRight.data[0][1]; compositeMeasurementJacobian3x3.data[2][2]=jacobianOfObservationRight.data[0][2]; pc.printf("\r\n111:calcul compositeMeasurementJacobian3x3 passed"); //get the compositeMeasurementJacobianTranspose compositeMeasurementJacobian3x3Transpose.fillWithTranspose(compositeMeasurementJacobian3x3); //compute compositeInnovation compositeInnovation3x1.data[0][0]=innovationLeft; compositeInnovation3x1.data[1][0]=innovationFront; compositeInnovation3x1.data[2][0]=innovationRight; //compute compositeInnovationCovariance=compositeMeasurementJacobian*covariancePositionEstimationKplus1K*compositeMeasurementJacobianTranpose+R tempCompositeInnovationCovariance3x3.fillByMultiplication(compositeMeasurementJacobian3x3,covariancePositionEstimationKplus1K); compositeInnovationCovariance3x3.fillByMultiplication(tempCompositeInnovationCovariance3x3,compositeMeasurementJacobian3x3Transpose); //add the right R on the diagonal compositeInnovationCovariance3x3.data[0][0]+=R_left; compositeInnovationCovariance3x3.data[1][1]+=R_front; compositeInnovationCovariance3x3.data[2][2]+=R_right; pc.printf("\r\n111:calcul compositeInnovationCovariance3x3 passed"); //get the inverse of the compositeInnovationCovariance compositeInnovationCovariance3x3Inverse.fillWithInverse(compositeInnovationCovariance3x3); pc.printf("\r\n111:calcul compositeInnovationCovariance3x3Inverse passed"); //compute KalmanGain=covariancePositionEstimationKplus1K*compositeMeasurementJacobianTranspose*Inverse(compositeInnovationCovariance) tempKalmanGain3X3.fillByMultiplication(covariancePositionEstimationKplus1K,compositeMeasurementJacobian3x3Transpose); kalmanGain3X3.fillByMultiplication(tempKalmanGain3X3,compositeInnovationCovariance3x3Inverse); pc.printf("\r\n111:calcul kalmanGain3X3 passed"); //get the transpose of the kalman gain kalmanGain3X3Transpose.fillWithTranspose(kalmanGain3X3); //update pose estimation=old pose estimation + kalman gain*compositeInnovation tempPoseKplus1Kplus1.fillByMultiplication(kalmanGain3X3,compositeInnovation3x1); poseKplus1Kplus1.addition(tempPoseKplus1Kplus1); pc.printf("\r\n111:calcul poseKplus1Kplus1 passed"); //compute covariancePositionEstimationKplus1Kplus1=covariancePositionEstimationKplus1K-kalmanGain*compositeInnovationCovariance*KalmanGainTranspose temp2CovariancePositionEstimationKplus1Kplus13x3.fillByMultiplication(kalmanGain3X3,compositeInnovationCovariance3x3); tempCovariancePositionEstimationKplus1Kplus13x3.fillByMultiplication(temp2CovariancePositionEstimationKplus1Kplus13x3,kalmanGain3X3Transpose); covariancePositionEstimationKplus1Kplus1.subtraction(tempCovariancePositionEstimationKplus1Kplus13x3); pc.printf("\r\n111:calcul covariancePositionEstimationKplus1Kplus1 passed"); break; } //update covariancePositionEstimationK =covariancePositionEstimationKplus1Kplus1 this->covariancePositionEstimationK.fillByCopy(covariancePositionEstimationKplus1Kplus1); //pc.printf("\r\nposeKplus1Kplus1 X=%f, Y=%f, theta=%f",poseKplus1Kplus1.data[0][0],poseKplus1Kplus1.data[1][0],poseKplus1Kplus1.data[2][0]); //update pose this->xWorld=poseKplus1Kplus1.data[0][0]; this->yWorld=poseKplus1Kplus1.data[1][0]; this->thetaWorld=poseKplus1Kplus1.data[2][0]; pc.printf("\r\nWith Kalman X=%f, Y=%f, theta=%f",xWorld,yWorld,thetaWorld); //try with robot one /* X=xEstimatedKNext; Y=yEstimatedKNext; theta=thetaWorldEstimatedKNext; this->xWorld=-Y; this->yWorld=X; if(theta >PI/2) this->thetaWorld=-PI+(theta-PI/2); else this->thetaWorld=theta+PI/2; this->print_map_with_robot_position(); pc.printf("\n\rX= %f",this->xWorld); pc.printf("\n\rY= %f",this->yWorld); pc.printf("\n\rtheta= %f",this->thetaWorld); */ //update odometrie X Y theta... } } void MiniExplorerCoimbra::go_to_point_kalman(float targetXWorld, float targetYWorld) { float angleError; //angle error float d; //distance from target float angularLeft, angularRight, linear, angular; int speed=300; do { //Updating X,Y and theta with the odometry values this->OdometriaKalmanFilter(); //this->myOdometria(); //pc.printf("\r\nD_cm:%f",D_cm); //pc.printf("\r\nL_cm:%f",L_cm); //Computing angle error and distance towards the target value angleError = atan2((targetYWorld-this->yWorld),(targetXWorld-this->xWorld))-this->thetaWorld; if(angleError>PI) angleError=-(angleError-PI); else if(angleError<-PI) angleError=-(angleError+PI); //pc.printf("\n\r error=%f",angleError); d=this->dist(this->xWorld, this->yWorld, targetXWorld, targetYWorld); //pc.printf("\n\r dist=%f/n", d); //Computing linear and angular velocities linear=k_linear*d; angular=k_angular*angleError; angularLeft=(linear-0.5*this->distanceWheels*angular)/this->radiusWheels; angularRight=(linear+0.5*this->distanceWheels*angular)/this->radiusWheels; //Normalize speed for motors if(angularLeft>angularRight) { angularRight=speed*angularRight/angularLeft; angularLeft=speed; } else { angularLeft=speed*angularLeft/angularRight; angularRight=speed; } pc.printf("\n\r X=%f", this->xWorld); pc.printf("\n\r Y=%f", this->yWorld); pc.printf("\n\r theta=%f", this->thetaWorld); //Updating motor velocities if(angularLeft>0){ leftMotor(1,angularLeft); } else{ leftMotor(0,-angularLeft); } if(angularRight>0){ rightMotor(1,angularRight); } else{ rightMotor(0,-angularRight); } //wait(0.5); } while(d>5); //Stop at the end leftMotor(1,0); rightMotor(1,0); } void MiniExplorerCoimbra::printMatrix(myMatrix mat1){ for(int i = 0; i < mat1.nbRow; ++i) { for(int j = 0; j < mat1.nbColumn; ++j) { pc.printf("\r%f",mat1.data[i][j]); } pc.printf("\n"); } pc.printf("\r\n"); }