Embedded Motion Control 2013 Group 3
Contact info
| Vissers, Yorrick (YO) | 0619897 | y.vissers@student.tue.nl | 
| Wanders, Matthijs (MA) | 0620608 | m.wanders@student.tue.nl | 
| Gruntjens, Koen (KO) | 0760934 | k.g.j.gruntjens@student.tue.nl | 
| Bouazzaoui, Hicham (HI) | 0831797 | h.e.bouazzaoui@student.tue.nl | 
| Zhu ,Yifan (YI) | 0828010 | y.zhu@student.tue.nl | 
Meeting hours
Mondays 11:00 --> 17:00 
Wednesdays 8:45 --> 10:30 
Thursday 9:00 --> 10.00 Testing on Pico 
Meet with tutor: Mondays at 14:00 
Planning
Ma 09 sept:
- Finish installation of everything
- Go through ROS (beginner) and C++ manual
Wo 11 sept:
- Finish ROS, C++ manuals
- Start thinking about function architecture
Ma 16 sept:
- Design architecture
- Functionality division
- Divide programming tasks
Do 19 sept:
- Finish "state stop" (Koen)
- Finish "drive_parallel" (Matthijs, Yorrick)
- Creating a new "maze/corridor" in Gazebo (Yifan)
- Simulate and build the total code using Gazebo (Hicham)
- Testing with robot at 13:00-14:00
Vr 20 sept:
- Finish "crash_avoidance"
- Coding "gap_detection" (Yifan)
- Coding "dead_end_detection" (Matthijs, Yorrick)
- Coding "maze_finished" (Koen, Hicham)
Ma 23 sept:
- Finish "drive_parallel"
- Putting things together
- Testing with robot at 12:00-13:00 (Failed due to network down)
Unfortunately we encountered some major problems with the Pico robot due to a failing network. We discussed the approach for the corridor competition. At this point the robot is able to drive parallel through the corridor and can look for gaps either left or right. We a gap is reached the robot will make a smooth circle through the gap. This is all tested and simulated. For the corridor competition we will not check for death ends. There isn't enough time to implement this function before Wednesday 25 September. The corresponding actions such as "turn around" won't be finished either. Without these functions we should be alble to pass the corridor competition successfully.
Di 24 sept:
- Testing with robot at 13:00-14:00
- Finding proper parameters for each condition and state
Wo 25 sept:
- Finish clean_rotation
- Finish gap_handling
- Putting things together
- Corridor Challenge
Week 5:
- Design and simulate code
- Create a structure "laser_data" which processes the data of the laser. This contains the calculation of (Ma + Yo):
- Shortest distance to the wall
- Distance to the right wall with respect to Pico
- Distance to the left wall with respect to Pico
 
- Edit/improve state "drive_parallel" with a feedback controller according to the angle with respect to the right wall (Ma + Yo).
- Create the condition "dead_end". When a dead end is detected switch to the state "turn_around" (Yi + Ko).
- Edit/improve the condition "gap_detect_left/right". Make this condition more robust (Yi + Ko).
- Start researching the properties of the camera of Pico (Hi).
 
- Create a structure "laser_data" which processes the data of the laser. This contains the calculation of (Ma + Yo):
- Testing on Pico (Listed by priorities)
1) Test robustness of the feedback controller and gap_detection 
2) Tune the translational and rotational speed (faster) 
3) Test detecting dead ends 
4) Test strange intersections
Week 6:
- Design and simulate code
- Edit/improve state "drive_parallel" with a feedback controller according to the angle with respect to the right wall (Ma + Yo).
- Create the condition "finished". This is the condition when Pico exists the maze. Switch to state "finalize" (Yi + Ko).
- Improve the priorities list. Pico needs to make smart decisions when multiply conditions occur. Basically the priorities are listed as (Ma + Yo):
 
1) Turn right 
2) Go straight 
3) Turn left
- Continue researching the properties of the camera of Pico (Hi)
 
- Testing on Pico (Listed by priorities)
1) Test the state "turn_around" 
2) Test the condition "finished" 
3) Building a small maze containing a t-intersection and a dead end and test the priorities (Can also be done properly during simulation) 
Week 7:
- Design and simulate code
- Edit/improve the state "reset". This states handles the condition when a object is detected close to Pico.
- Imaging processing. Detect arrows on the wall (Hi + Yi).
 
- Testing on Pico (Listed by priorities)
1) Build a maze and solve it without using camera. 
2) Test camera 
3) Test the improved states/conditions. See progress->week6->Testing 
Week 8:
- Design and simulate code
- Finalize the code
- Implementing the camera node
 
- Testing on Pico (Listed by priorities)
1) Build a maze and solve it using camera. 
2) Test camera 
- Administrative activities
- Fill in peer assessment (Sending to v.d. Molengraft by Koen)
- Upload a video of the Gazebo demo (Yifan)
- Updating the wiki:
 
| Condition/Node | State | Name | 
|---|---|---|
| crash_avoidance | state_reset | Yorrick & Matthijs | 
| maze_finished | state_finalize | Koen | 
| maze_exit | state_exit | Koen | 
| dead_end_detection | state_turn_around | Yifan | 
| gap_detection_right | state_gap_handling | Yifan | 
| gap_detection_left | state_gap_handling | Yifan | 
| go_straight_priority | state_drive_parallel | Yorrick & Matthijs | 
| Camera | / | Hicham | 
Progress
Week 1 
(All)
- Installed and setup all the software
- Went through the ros beginner tutorials and c++ tutorials
Week 2 
(All)
- Finishing up the tutorials that weren't yet completed
- Initial architecture design: condition and state functions.
Week 3 
(All)
- Continued architecture design, and begin of implementation.
- Simulation with drive safe example.
(MA & YO)
- Created state_drive_parallel function
(YI)
- Created custom mazes in Gazebo
(KO)
- Created state_stop function
Week 4 
Week 5 
(MA & YO)
- Removed scanning functionality from the states and conditions and put it in a seperate function that returns a struct "Laser_data" containing info about wall locations and scanner range/resolution etc. Did some tuning on the gap_detection algorithm to make it more robust and made some minor changes on the decision tree. (now only checks nect condition if previous one returns false). 
- Improved the decision tree to satisfy the required priorities for a wall follower strategy. The priorities for navigation are: 1) Take right gaps 2) Drive par 3) Take left turns. In order to only take left turns when there is nothing interesting ahead or to the right a new condition is introduced: go_straight_priority. When this condition returns false it is ok to take a left turn should there be a gap.
- Further improvements (w.r.t robustness) on gap_detection and dead_end_detection. Due to the new decision tree "number_of_spins" are now counted inside of the states (gap_handling), rather than inside the conditions.
Testing on Pico: (MA & YO & YI) : We learned that when there's a gap further ahead, Pico's laser doesn't see a clear jump in distance, but instead see several points (reflection?) within the gap. This is taken into account when designing for robustness.
- At the time of testing the priorities were not working as intended for the abovementioned reason, this has been improved later that day. Will be tested again next week.
- Speeds are tuned for now to x_vel = 0.2 in state_drive_par and z_rot = 0.15 in state_gap_handling and state_turn_around.
- Dead_end_detection worked as intended when there was a dead end, but sometimes also detected a dead end when there was none. This has been improved that same day. Also the distance when this condition checks in front has been improved. Will be tested again next week.
- state_turn_around was already finished ahead of scedule. Therefore it was already tested on Pico this week and is working as intended.
- feedback control in state_drive_parallel was not implemented yet, and therefore is also not tested and tuned. This will be done next testing session.
Week 6 
(KO)
- Modified the condition "maze_finished". Pico will first check in front is there is anything closer then 2 meter (variable). When this is not the case it will continue checking a range of approximately 150 degrees ahead for obstacles. When there is still nothing closer then 2 meters, Pico is done!
- When the condition "maze_finished" is satisfied Pico will be in the state "finalize". It will move with a constant translation velocity of 0.3 [m/s] until anyone hit the emergency button.
(HI & KO & YO & YI)
- Worked on improving gap handling. We implemented 'flexible' corners: when Pico starts to go around a corner, his current orientation is taken into account when determining the counter k that represents the total number of spins pico should rotate. In addition a constant tuning factor is included in k, which helps in simulation. This factor might be retuned when testing this friday.
(HI & YO)
- Worked on implementing proportional feedback in state_drive_parallel for the purpose of faster settling times (reach setpoint faster) and more stable behaviour while driving straight. We implemented a PD controller that is tuned to work well in the simulation and friday this will be re-tuned to work well on Pico.
(HI & YO)
- Created a new node for the camera functionality: camera.cpp. This node subscribes to the camera and calls "camera_controller.h" when data is published to the camera topic. The camera controller will process the camera data and detect arrows, which can be published by the camera node to our main node theseus. Hicham will work on processing the camera data and arrow detection.
- Note that CMakeLists will now also make an executable of camera.cpp. In addition manifest.xml now also depends on the opencv packages. In order to run both nodes use: rosrun theseus2 theseus camera
Testing on pico: (KO & YI & YO)
- Things which went good:
- Driving parallel to the right wall. We improved/tuned the control parameters on Pico.
- Turn around after detecting a dead end.
- Gap handling went good after tuning some parameters.
 
- Things to do:
- Improve robustness of detecting gaps (either right and left).
- Improve exit the maze. It makes a corner to the right when he leaves the maze.
- Check this again next week when the other states/conditions are improved.
 
(YO)
- In a t-split where pico had an orientation that was slightly rotated to the right (negative theta) he detects the gap on the left first and started cornering to the left. Therefore we split the counter variable number_of_spins into number_of_spins_left and number_of_spins_right. This helps to prevent errors when go_straight priority is not working correctly, because the right cornering takes over.
- When testing on Pico we found that when taking a wide corner, after finishing the corner, he would detect another corner if pico was not yet in the next corridor. This is now prevented by a new function reset_corner that requires pico to see a small corridor part before allowing a new corner in the same direction.
Week 7 
(KO & YO & MA)
- Worked on a reset state, that is triggered after crash_avoidance. The reset state should help Pico to continue after a near crash, rather than the previous state stop, that would just end the fun.
(YO & MA)
- Worked again on robustness of dead_end_detection and gap_handling. Just trying to prevent as many possible problems.
(YI & HI)
- Worked on the camera functionality.
(KO)
- Started preparing the presentation.
(All)
- Merging the camera and theseus nodes
Testing on pico (All)
- Things that went well
- Reset corner works as intended and solves the double corner problem.
- Dead end detection seems to work well after tuning.
- After tuning, our crash_avoidance + state_reset combination works well in several tests: wall in front, wall to the side. But possibly if pico ends up in a corner, there may be a problem (test again next week).
- Gap handling and drive parallel worked fine.
 
- Things to improve
- Camera should work from a greater distance. We need to detect arrows sooner in order to make the correct decision.
- Perhaps if we detect an arrowshape ahead, we should lower the speed of drive_parallel? Yi & Hi should judge if this may help.
- Exit maze still didn't work as we designed it. Very low priority, since pico exits the maze fully which is acceptable in the challenge. With people standing around outside of the maze, the exit maze will nog work anyway.
 
Week 8 
Testing on Pico (All)
- The camera worked from a bigger distance. Our camera node detected the arrows in time.
- There was an error in the communication between our nodes. The bool arrow_left was not published to the topic. After a while we found out we used ros::Spin() instead of ros::SpinOnce(), which prevented communication. Unfortunately this cost us a lot of time during testing.
- When this was solved Pico reacted correctly on an arrow. But the reset condition for the arrow detection was very bad (onrobust). The idea was to publish a bool from theseus to the camera node when a corner is finished.
(All)
- Implemented a more robust reset condition for arrow detection. When a corner is finished the arrow detection is reset. Unfortunately there was no time to test this anymore.
- We decided it wasn't necessary to lower the speed if a corner is ahead, because we detect the arrows in time now.
(KO)
- Created and prepared the presentation. He practiced once for the group, and group gave feedback for improvement. In the end we wanted to emphasize two things in the presentations:
- Clear split between conditions (information processing) and conditions (how should pico actuate).
- Iterative approach of testing and tuning (and therefore seeing how each change to the code works).
 
PICO usefull info
- minimal angle = 2,35739 rad
- maximal angle = -2,35739 rad
- angle increment = 0,00436554 rad
- scan.range.size() = 1081
We have defined the following conventions:
- We use a right turning orthogonal base w.r.t Pico:
- the pos x-axis is pointing forward w.r.t Pico
- the pos y axis is pointing to the left w.r.t Pico
- the pos z axis is then pointing up w.r.t.Pico
- the pos z rotation is then CCW
 
Design approach
Our approach for this project is to start from a very simple working example. From this starting point an iterative approach of adding, extending or improving functionality and then testing and tuning the new functionality right away. The benefit of this approach is getting instant feedback of the functionality: do the new functions work as intended, are they tuned well, and do they work together with the other parts?
Strategy
For navigating through the maze we use a "wall follower" strategy. This means that the robot will always stick to the right wall and always find it's way to the exit. At a later stage we will extend this base strategy with extra features such as the camera to detect instructions. The main robot controller can be in six states according to the conditions. The conditions are:
- Dead_end_detection
- Input(s): (sensor_msg) LaserScan_data
- Output(s): (Bool) dead_end_detect
- The flag "dead_end_detect" is set when the robot measures a death end in front.
 
- Gap_detection (left and right are seperate but similar condition functions)
- Input(s): (sensor_msg) LaserScan_data
- Output(s): (Bool) gap_detect
- The flag "gap_detect" is set to true when the robot is located at the center of the gap. Only gaps on the right-hand side will be noticed.
 
- Crash_avoidance
- Input(s): (sensor_msg) LaserScan_data
- Output(s): (Bool) chrash_detect
- The flag "chrash_detect" is set when the robot measures an object closer then 30 centimeters over the entire laser range.
 
- Maze_finished
- Input(s): (sensor_msg) LaserScan_data
- Output(s): (Bool) finalize
- This flag will be set to true when the robot exits the maze and detect no object closer then 2 meters around it.
 
Within these functions a flag is set when the robot is in a certain condition. All based on the data of the laser scanner. 
Based on the flag which is set the robot encounters a certain condition and will be in the corresponding state.
The six states are: 
- state_drive_parallel: Keep the robot at a fixed distance (setpoint) from the right wall while driving.
- state_gap_handling: When arriving at the center of a gap (to the right). Rotate in place untill parallel with new right wall.
- state_stop: Stops the robot when necessary.
- state_finalize: When no longer between two walls and no maze in front, abort.
- state_turn_around: when at a dead end, turn around.
The state functions will be called according to the flag and determines the linear and rotational velocities. We will use one node that "spins" and subscribes to the laser data, and publishes the velocities.
The figure below depicts the simplified architecture of the program at this point.
Software architecture
- We use one package named Theseus2
- We use a node theseus that spins on 20 Hz and subscribes to pico/laser and pico/cam_data and publishes velocities. When pico/laser publishes something theseus_controller is called that contains our main functionality.
- theseus_controller is the controller that runs several steps:
1) Process the laserdata for future use.
2) Gather conditions (each conditions is explained elsewhere on the wiki)
3) Based on the conditions a state will be chosen and executed. In some situations (f.e gap_handling) the previous state is considered to determine the current state (Markov chain principle)
4) The chosen state will set the velocities that can be published by the node.
- datastructures.h: all global variables and datastructures are gathered here.
- send_velocities: this functions is used for theseus to publish the velocities.
- laser_processor: this function pre-processes the laserdata for future use.
- states: each contains the functionality of a state.
- conditions: each checks a certain condition to determine the right state.
We use a second node camera that spins on 5 Hz, subscribes to pico/camera and puslishes bool arrowleft to the topic pico/cam_data. Because of our right wall follower strategy, we only publish when a left arrow is detected. When pico/camera publishes something camera_controller is called which processes the images and when an arrow to the left is detected sets arrow_left to true.
Conditions/States
crash_avoidance: 
Pico is not allowed to hit the walls, so there when Pico gets too close to the wall, it should stop driving. This is carried out in the next loop:
- If the shortest distance from the wall to Pico is less than 25 cm, Pico return a stop command.
state_reset: 
This state is being called when Pico is attend to crash. Currently it will stop and do nothing. According to the planning we will modify this state in week 7. It should contain a smart algorithm which can offer new opportunities to continue his way through the maze.
=============================================================================================================================================================
maze_finished: 
When Pico exists the maze the condition "Finished" is reached. The following steps are carried out to check if Pico behaves in this condition:
- Read out the laser data from -135 to +135 degrees w.r.t the front of Pico
- Check if all distances are larger than 1.5 meters. When this is the case. Maze finished!
state_final: 
State Final is called when Pico satisfies the condition Finalize. This state will ensure that Pico will stop moving by sending 0 to the wheels for translation and rotation.
=============================================================================================================================================================
maze_exit: 
Pico is always looking for the exit of the maze. This will be done in two steps: 
1) Check if the shortest distance, within a range of +- 5 degrees w.r.t. the front of Pico, is larger then 1.5 meters 
2) Check if the shortest distance, within a range of 145 degrees w.r.t. the walls, is larger then 1.5 meters. 
When both steps are true, Pico is at the end of the maze a must switch to the state "exit". We use two steps just to reduce The steps are depicted in the figure below.
state_exit: 
This state ensures that Pico is driving straight out of the maze with a constant velocity. 
=============================================================================================================================================================
gap_detection_left: 
During the going straight state, if there exists a gap to the left side, the condition "Gap detection left" is reached. The following steps are carried out to check if Pico behaves in this condition:
- Pico will continue checking the laser date of left side between upper and lower index (variables) during going straight.
- If the distance within the upper and lower index is larger than the proudct of sensitivity (variable) and setpoint, a internal counter will start to count.
- As soon as the counter reaches 35, it's for sure that there is a gap to the left side. Pico will jump to state "gap handling" and start to turn.
gap_detection_right: 
This conditions is the same as gap detection left, but obviously for the right side.
state_gap_handling: 
When gap handling is entered, a 90 degrees rotation to the left or right (based on the gap_detection conditions) is made. The overall shortest distance of the laser is used as a radius for the corner. The relation: x_vel = radius * z_rot is used to make a nice corner. When Pico might come close to a wall the radius will be smaller, so also the corner Pico makes will be smaller, which helps in the situation of a small gap in a wide corridor. At the same time a minimum radius is set to ensure that pico will not make the corner too small.
When gap handling is entered, we consider Pico's current orientation (theta) and use this to enforce Pico to keep rotating for 90 degrees + theta. This helps to make corners a bit more flexible and ensures that pico will come out of gap_handling with a good orientation. This gives a good performance should several corners come after eachother.
=============================================================================================================================================================
go_straight_priority: 
This condition is necessary because of the right wall follower strategy: when there is a gap to the left, you might want to go straight instead if there is something interesting to the front or right side. In that case go straight priority will return true. Otherwise when there's nothing ahead or to the right (only walls), go straight priority returns false and it's ok to take a left turn.
state_drive_parallel: 
Drive parallel is the state that allows Pico to drive between two walls. In this state a setpoint is calculated as the middle of the closest point to the left and right of pico. On top of that we define the deviation from this setpoint as the error (positive error = right of setpoint), and the current rotation as theta (positive theta = CCW, parallel to wall theta = 0), both relative to closest point left and right. By feeding back the error and theta as the rotation velocity we keep Pico in the middle of corridors which has the benefit that per default it tries not to hit any obstacles. 
We do this by setting a constant translation velocity, and a PD controlled rotation velocity that feeds back both the error and theta (P-action) and their derivatives (D-action); z_rot = f(error,d_error, theta, d_theta) = kpe * error + kde * d_error - kpt * theta - kdt * d_theta
=============================================================================================================================================================
dead_end_detection: 
When Pico facing a dead end, the condition "Dead end" is reached. The following steps are carried out to check if Pico behaves in this condition:
- Check if the condition "Dead end" is active. If active, keeps Pico turning around until Pico facing backward to the dead end.
- If the condition "Dead end" is not active, read out the laser data from -5 to +5 degree w.r.t the Pico heading direction.
- Check if the distance from -5 to +5 degree are shorter than sensitivity times setpoint to see if there is a dead end in the front. When this is the case, set Boolean "dead_end_infront" to TRUE.
- When "dead_end_infront" is TRUE, calculate the differences of two nearby laser distance from -90 to +90 degree with respect to the robot.
- When all the difference is less than 0.2 meters, Pico reaches dead end, jump to state "Turn around".
Turn around: 
Pico is switched to this state when a dead end is detected. When Pico enters this state he will make a clean rotation on the spot of 180 degrees and is kept in this state untill finished rotating. Turning Pico is done by a simple open loop function. An angular velocity is send to the wheels and for a pre-calculated number of iterations corresponding to 180 degrees. We don't use any feedback to compensate slip and friction etc. In terms of accuracy this will not be the best solution but the small error due to slip and friction will be handled by the state drive_parallel which does contain a feedback controller.
=============================================================================================================================================================
Arrow detection: 
At a junction there is a possibility that an arrow is located at the wall. Pico can detect this arrow by using it's camera. 
When Pico detects the arrow it has to determine whether the arrows is point right or left and either turn right or left. 
...to be continued...
Corridor Competition
We succesfully passed the corridor competition with the 2nd fastest time. A video impression is shared here: Corridor competition Theseus
Maze Competition
The maze competition is held oktober 23rd. We expect to be ready to perform well in a wide range of possible mazes.
Presentation
We will soon start with preparing the presentation of our efforts.





