forked from anidev/612-code
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathencoder.cpp
152 lines (137 loc) · 5.21 KB
/
encoder.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
#include <cmath>
#include <RobotDrive.h>
#include "encoder.h"
#include "ports.h"
#include "trajectory.h"
#include "pid_controller.h"
#include "update.h"
// PID values from the almighty Wikipedia
// TODO retune PID for Suzie
const double DRIVETRAIN_PID_P = 0.24; // 0.025;
const double DRIVETRAIN_PID_I = 0.000; // 0.0005;
const double DRIVETRAIN_PID_D = 0;
const double DRIVETRAIN_PID_TOLERANCE = 0.75;
//const double DRIVETRAIN_TOLERANCE = 1.0;
//const double DRIVETRAIN_SPEED = 0.7;
// TODO Joystick + encoder pid driving at the same time
// TODO decide what is reverse and what isn't!
const int RESET = 1;
EncoderWheels * EncoderWheels::instance = NULL;
EncoderWheels::EncoderWheels(Encoder& el, Encoder& er, SpeedController& jag_fl, SpeedController& jag_fr, SpeedController& jag_rl, SpeedController& jag_rr) : encoder_left(el), encoder_right(er) {
enabled = false;
setpoint = 0.0;
encoder_left.SetDistancePerPulse(2 * pi * DRIVE_REDUCTION * WHEEL_RADIUS / 360 * -1.0);
encoder_right.SetDistancePerPulse(2 * pi * DRIVE_REDUCTION * WHEEL_RADIUS / 360);
encoder_left.SetPIDSourceParameter(Encoder::kDistance);
encoder_right.SetPIDSourceParameter(Encoder::kDistance);
drive_pid_obj_left = new drive_pid(jag_fl, jag_rl, drive_pid::SIDE_LEFT, true);
drive_pid_obj_right = new drive_pid(jag_fr, jag_rr, drive_pid::SIDE_RIGHT, false);
pid_left = new pid_controller(DRIVETRAIN_PID_P, DRIVETRAIN_PID_I, DRIVETRAIN_PID_D, &encoder_left, drive_pid_obj_left);
pid_right = new pid_controller(DRIVETRAIN_PID_P, DRIVETRAIN_PID_I, DRIVETRAIN_PID_D, &encoder_right, drive_pid_obj_right);
pid_left->SetTolerance(DRIVETRAIN_PID_TOLERANCE);
pid_right->SetTolerance(DRIVETRAIN_PID_TOLERANCE);
pid_left->SetOutputRange(-0.5, 0.5);
pid_right->SetOutputRange(-0.5, 0.5);
registry().register_func(EncoderWheels::update_help, (void*)this);
}
void EncoderWheels::update_help(void* obj) {
((EncoderWheels*)obj)->update();
}
void EncoderWheels::update() {
if(AtTarget()) {
std::printf("------------TARGET REACHED-----------------\n");
Disable();
}
if(left_joystick.GetRawButton(RESET)) {
std::printf("............RESET ENCODERS.............\n");
Disable();
Enable();
}
}
void EncoderWheels::Init(Encoder& el, Encoder& er, SpeedController& jag_fl, SpeedController& jag_fr, SpeedController& jag_rl, SpeedController& jag_rr) {
instance = new EncoderWheels(el, er, jag_fl, jag_fr, jag_rl, jag_rr);
//don't need to delete as this object will last the lifetime of the program,
//and the memory will be reclaimed when the process ends. Only call this
//once though or it will be VERY bad
}
EncoderWheels& EncoderWheels::GetInstance() {
//if you don't call init this will blow up in your face
return *instance;
}
double EncoderWheels::InchesToTicks(double inches, int axes=4) {
double ticks = inches*360;
ticks = ticks / (TICKS_PER_REV/(2*pi)) / DRIVE_REDUCTION / WHEEL_RADIUS;
ticks *= axes;
return ticks;
}
//there's a bool here but it will probably be more like a PID enable
void EncoderWheels::Enable() {
if(std::fabs(setpoint)>1.0) {
drive.SetSafetyEnabled(false);
pid_left->Enable();
pid_right->Enable();
}
encoder_left.Start();
encoder_left.Reset();
encoder_right.Start();
encoder_right.Reset();
enabled = true;
}
void EncoderWheels::Disable() {
pid_left->Disable();
pid_right->Disable();
drive.SetSafetyEnabled(true);
encoder_left.Stop();
encoder_right.Stop();
setpoint = 0.0;
enabled = false;
}
bool EncoderWheels::IsEnabled() {
return enabled;
}
void EncoderWheels::SetDistance(double distance) {
Disable();
distance /= 1.7;
setpoint = distance;
pid_left->SetSetpoint(setpoint);
std::printf("set pid left to %f\n", distance);
pid_right->SetSetpoint(setpoint);
std::printf("set pid right to %f\n", distance);
Enable();
/* if(AtTarget()) {
drive.Drive(0,0);
} else {
drive.Drive(DRIVETRAIN_SPEED,0);
}*/
}
bool EncoderWheels::AtTarget() {
bool TargetReached = pid_left->OnTarget() && pid_right->OnTarget();
return TargetReached;
// return std::fabs(GetCurDistance()-setpoint) > DRIVETRAIN_TOLERANCE;
}
double EncoderWheels::GetCurDistance(distance_side_t side) {
double distance_ret=0;
// int axes=4; // assume four, no way to find type of encoder right now
if(side & DISTANCE_LEFT) {
/* double left_ticks=encoder_left.Get()*1.0/axes;
std::printf("left_ticks = %f", left_ticks);
double left_distance=left_ticks * (TICKS_PER_REV/(2*pi)) * DRIVE_REDUCTION * WHEEL_RADIUS;
left_distance/=-360;
distance_ret+=left_distance;*/
distance_ret+=encoder_left.GetDistance()*1.7;
}
if(side & DISTANCE_RIGHT) {
/* double right_ticks=encoder_right.Get()*1.0/axes;
double right_distance=right_ticks * (TICKS_PER_REV/(2*pi)) * DRIVE_REDUCTION * WHEEL_RADIUS;
right_distance/=-360;
distance_ret+=right_distance;*/
distance_ret+=encoder_right.GetDistance()*1.7;
}
if(side == DISTANCE_AVG) {
distance_ret/=2.0;
}
return distance_ret;
}
double EncoderWheels::GetSetDistance() {
return setpoint*1.7;
}