-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathmain.cpp
220 lines (185 loc) · 7.46 KB
/
main.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
/**
* Hey there! Welcome to the Snowbots Software Challenge!
*
* As a prospective team member, we'd like to see you demonstrate your skills
* by programming our brand spankin' new Mars rover.
* The goal is to make the rover drive to a set of waypoints, read in from a
* text file. We'll provide a lot of the rough framework you'll need, including
* a _swanky_ simulator, we just need you to fill in a couple of functions
* below..........
*
*/
#include <iostream>
#include <stack>
#include <vector>
#include <cmath>
#include <math.h>
#include <fstream>
/** Constants **/
/** (you probably don't need to care about these, and should *NOT* change them) **/
// Length of the robot (used for kinematic model in simulator)
static const double ROBOT_LENGTH = 0.2;
// The time step that the simulator takes on each iteration
static const double SIM_TIME_STEP = 0.05;
// The tolerance we want to come within for each waypoint
static const double WAYPOINT_TOLERANCE = 0.2;
// The physical limit for the robot turning angle
static const double MAX_TURN_ANGLE = M_PI/3;
/** You CAN change these constants **/
// The name of the file to read the waypoints in from
static const char* WAYPOINTS_FILE_NAME = "waypoints.txt";
// The maximum number of iterations the simulator will run (presuming
// "waypoints" is never empty)
static const double MAX_SIM_ITERS = 1000;
// A 2D Point In Space (measured in meters)
struct Point {
double x;
double y;
};
// A message giving the status of the robot
struct StatusMsg {
// The current position of the robot
Point position;
// The current heading of the robot
double heading;
};
// A command we can send to the robot to control it
struct MovementMsg {
// An angle in radians, within [-pi/3, pi/3], with -pi/3 being turn right,
// pi/3 being turn left
double steering_angle;
// How fast the robot should move, in [-1, 1] with -1 being reverse,
// and 1 being forward. The value is in meters/second.
double throttle;
};
/** Test Macros **/
// Test that two values are exactly equal
#define TEST_EQ(EXPECTED, ACTUAL) \
if (EXPECTED != ACTUAL) { \
std::cout << "FAILURE: " << #ACTUAL << " gave " << ACTUAL << ", but expected " << EXPECTED << std::endl; \
}
// Test that two values are equal to within some tolerance
#define TEST_EQ_WITHIN_TOLERANCE(EXPECTED, ACTUAL, TOLERANCE) \
if (std::abs(EXPECTED - ACTUAL) > TOLERANCE) { \
std::cout << "FAILURE: " << #ACTUAL << " gave " << ACTUAL << ", but expected " << EXPECTED << " (with tolerance of " << TOLERANCE << ")" << std::endl; \
}
// Pre-declarations for functions
double distanceBetweenPoints(Point p1, Point p2);
double angleBetweenPoints(Point p1, Point p2);
MovementMsg computeNewMovement(StatusMsg status);
void runTests();
// Waypoints to navigate to
std::stack<Point> waypoints;
int main(int argc, char** argv) {
/** SETUP THE WAYPOINTS **/
// Read the waypoints in from a file
std::ifstream infile(WAYPOINTS_FILE_NAME);
double x, y;
while (infile >> x >> y){
waypoints.push({x,y});
}
// Flip the waypoints stack, since we want the first waypoint at the top
auto waypoints_copy = waypoints;
waypoints = {};
while (!waypoints_copy.empty()) {
Point waypoint = waypoints_copy.top();
waypoints_copy.pop();
waypoints.push(waypoint);
}
/** RUN THE SIMULATOR **/
// This is our crude "simulator". You don't need to touch this, though
// it might help to at least somewhat understand what we're doing.
StatusMsg curr_robot_status{ (Point){0,0}, 0 };
int curr_sim_iter = 0;
std::cout << "x, y, heading" << std::endl << "-------------" << std::endl;
while (!waypoints.empty() && curr_sim_iter < MAX_SIM_ITERS) {
curr_sim_iter++;
// Print Robot State
std::cout << curr_robot_status.position.x << ", "
<< curr_robot_status.position.y << ", "
<< curr_robot_status.heading << std::endl;
MovementMsg mv_msg = computeNewMovement(curr_robot_status);
// Constrain turn angle to physical limitations
mv_msg.steering_angle = std::min(std::max(-MAX_TURN_ANGLE, mv_msg.steering_angle), MAX_TURN_ANGLE);
// Update Robot State
// You don't need to understand this, but we're using a simple
// "tricycle" ackermann model here to update the robot state
double d_theta = (mv_msg.throttle / ROBOT_LENGTH) *
std::tan(mv_msg.steering_angle);
double d_x = mv_msg.throttle *
std::sin(curr_robot_status.heading + (d_theta * SIM_TIME_STEP)/2.0);
double d_y = mv_msg.throttle *
std::cos(curr_robot_status.heading + (d_theta * SIM_TIME_STEP)/2.0);
curr_robot_status.position.x += d_x * SIM_TIME_STEP;
curr_robot_status.position.y += d_y * SIM_TIME_STEP;
curr_robot_status.heading += d_theta * SIM_TIME_STEP;
}
if (waypoints.empty()) {
std::cout << "All waypoints reached!!" << std::endl;
}
/** RUN TESTS **/
runTests();
}
/* ------- YOUR CODE GOES BELOW HERE ------- */
void runTests() {
// HINT: Add tests for your functions here!!
std::cout << "-------------" << std::endl;
std::cout << "Failed Tests:" << std::endl;
std::cout << "-------------" << std::endl;
TEST_EQ(10, distanceBetweenPoints({0,0}, {10, 0}));
TEST_EQ(5, distanceBetweenPoints({0,0}, {3, 4}));
// We don't test for exact equality here because of floating point error
TEST_EQ_WITHIN_TOLERANCE(0, angleBetweenPoints({0,0}, {0, 10}), 0.0000001);
TEST_EQ_WITHIN_TOLERANCE(M_PI/2, angleBetweenPoints({0,0}, {10, 0}), 0.0000001);
std::cout << "-------------" << std::endl;
}
/**
* Calculates the distance between two given points
*
* @param p1 the first point
* @param p2 the second point
*
* @return the absolute distance between p1 and p2
*/
double distanceBetweenPoints(Point p1, Point p2) {
// TODO #1
}
/**
* Calculates the angle from the first point to the second point
*
* The angle is measured assuming +Y is 0 radians, +X is +PI/2 radians,
* -X is -PI/2 radians, -Y is -PI or +PI
*
* @param p1 the first point
* @param p2 the second point
*
* @return the angle between between p1 and p2 in *radians*
*/
double angleBetweenPoints(Point p1, Point p2) {
// TODO #2
}
/**
* Calculate a new MovementMsg based on the given StatusMsg
*
* This MovementMsg should move the Robot towards the next waypoint.
* If we're close enough (ie. within "WAYPOINT_TOLERANCE") to the waypoint,
* this will update "waypoints" so that we start moving towards the next one.
*
* @param status The current status of the robot
*
* @return A MovementMsg moving the robot towards the next waypoint
*/
MovementMsg computeNewMovement(StatusMsg status) {
// TODO #3
// NOTE: The "simulator" will stop once "waypoints" is empty
// HINT: You are allowed to change the waypoints file for testing (by
// modifying "WAYPOINTS_FILE_NAME" at the top of this file) (but we
// will be testing with our own secret waypoints!)
// HINT: You probably want to use the functions you filled in above....
// HINT: If you want to visualize what your robot is doing, you probably
// want to take the points that the simulator prints and put them
// into a scatter plot in excel (or equivalent). You may also want
// to change "MAX_SIM_ITERS" at the top of this file to limit the
// number of simulator iterations at first
return MovementMsg({3.14159/4,1});
}