mirror of https://github.com/ArduPilot/ardupilot
1623 lines
54 KiB
Plaintext
1623 lines
54 KiB
Plaintext
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
|
|
|
/* This GCS protocol sends text-based information over the GCS port
|
|
*/
|
|
|
|
#if GCS_PROTOCOL == GCS_PROTOCOL_DEBUGTERMINAL
|
|
|
|
#define ERR(a) ((DEBUGTERMINAL_VERBOSE)>0?(PSTR(a)):(0))
|
|
|
|
void
|
|
GCS_DEBUGTERMINAL::update()
|
|
{
|
|
byte numc, i, c;
|
|
|
|
numc = _port->available();
|
|
for (i=0;i<numc;i++) {
|
|
c = _port->read();
|
|
processgcsinput(c);
|
|
}
|
|
}
|
|
|
|
void
|
|
GCS_DEBUGTERMINAL::processgcsinput(char c)
|
|
{
|
|
if (c==8) {
|
|
//Received a backspace character; move the buffer index backwards
|
|
if (bufferidx > 0) bufferidx--;
|
|
} else if (c==10) {
|
|
//Received a linefeed; do nothing
|
|
} else if (c==13) {
|
|
//Received a carriage return; process command
|
|
gcsinputbuffer[bufferidx] = 0;
|
|
run_debugt_command(gcsinputbuffer);
|
|
bufferidx = 0;
|
|
} else {
|
|
gcsinputbuffer[bufferidx++] = c;
|
|
if (bufferidx >= sizeof(gcsinputbuffer)) bufferidx = 0;
|
|
}
|
|
}
|
|
|
|
void
|
|
GCS_DEBUGTERMINAL::run_debugt_command(char *buf)
|
|
{
|
|
BetterStream *port = _port;
|
|
|
|
//*********** Ignore comments ***********
|
|
if (strmatch(buf, PSTR("//"))) {
|
|
|
|
//*********** Process 'show' commands ***********
|
|
} else if (strmatch(buf, PSTR("show "))) {
|
|
if (strmatch(buf+5, PSTR("heartbeat")))
|
|
report_heartbeat = 1;
|
|
else if (strmatch(buf+5, PSTR("attitude")))
|
|
report_attitude = 1;
|
|
else if (strmatch(buf+5, PSTR("location")))
|
|
report_location = 1;
|
|
else if (strmatch(buf+5, PSTR("command")))
|
|
report_command = 1;
|
|
else if (strmatch(buf+5, PSTR("cpuload")))
|
|
report_cpu_load = 1;
|
|
else if (strmatch(buf+5, PSTR("severity")))
|
|
report_severity = atoi(buf+14);
|
|
else
|
|
print_error(ERR("USAGE: show {heartbeat|attitude|location|command|severity <X>}"));
|
|
|
|
//*********** Process 'hide' commands ***********
|
|
} else if (strmatch(buf, PSTR("hide "))) {
|
|
if (strmatch(buf+5, PSTR("heartbeat")))
|
|
report_heartbeat = 0;
|
|
else if (strmatch(buf+5, PSTR("attitude")))
|
|
report_attitude = 0;
|
|
else if (strmatch(buf+5, PSTR("location")))
|
|
report_location = 0;
|
|
else if (strmatch(buf+5, PSTR("command")))
|
|
report_command = 0;
|
|
else if (strmatch(buf+5, PSTR("cpuload")))
|
|
report_cpu_load = 0;
|
|
else if (strmatch(buf+5, PSTR("all"))) {
|
|
report_heartbeat = 0;
|
|
report_attitude = 0;
|
|
report_location = 0;
|
|
report_command = 0;
|
|
report_cpu_load = 0;
|
|
} else
|
|
print_error(ERR("USAGE: hide {heartbeat|attitude|location|command|all}"));
|
|
|
|
//*********** Process 'copy' command ***********
|
|
} else if (strmatch(buf, PSTR("copy "))) {
|
|
//------- copy cmd <N1> <N2> -------
|
|
if (strmatch(buf+5, PSTR("cmd "))) {
|
|
unsigned char i = 9, index1, index2;
|
|
while (buf[i] != 0) {
|
|
i++;
|
|
if (buf[i] == ' ') break;
|
|
}
|
|
if (buf[i] == ' ') {
|
|
buf[i] = 0;
|
|
index1 = atoi(buf+9);
|
|
index2 = atoi(buf+i+1);
|
|
Location temp = get_wp_with_index(index1);
|
|
set_wp_with_index(temp, index2);
|
|
port->print_P(PSTR("Copied command index ")); port->print(index1,DEC); port->print_P(PSTR(" to ")); port->println(index2,DEC);
|
|
} else {
|
|
print_error(ERR("USAGE: copy cmd <srcindex> <targetindex>"));
|
|
}
|
|
}
|
|
|
|
//*********** Process 'echo' command ***********
|
|
} else if (strmatch(buf, PSTR("echo "))) {
|
|
port->println(buf+5);
|
|
|
|
//*********** Process 'groundstart' command ***********
|
|
} else if (strmatch(buf, PSTR("groundstart"))) {
|
|
startup_ground();
|
|
|
|
//*********** Process 'inithome' command ***********
|
|
} else if (strmatch(buf, PSTR("inithome"))) {
|
|
init_home();
|
|
port->println_P(PSTR("Home set."));
|
|
|
|
//------- k -? -------
|
|
} else if (strmatch(buf, PSTR("k -?"))) {
|
|
print_error(ERR("USAGE: {print|set} {k{p|i|d}|imax} {servoroll|servopitch|servorudder|navroll|navpitchasp|navpitchalt|throttlete|throttlealt}"));
|
|
print_error(ERR("USAGE: {print|set} kff {pitchcomp|ruddermix|pitchtothrottle}"));
|
|
|
|
//*********** Process 'print' commands ***********
|
|
} else if (strmatch(buf, PSTR("print "))) {
|
|
//------- print airspeedtrim -------
|
|
if (strmatch(buf+6, PSTR("airspeedtrim"))) {
|
|
port->printf_P("Trim airspeed = %f\n", (float)get(PARAM_TRIM_AIRSPEED)/100);
|
|
|
|
//------- print airspeednudge -------
|
|
} else if (strmatch(buf+6, PSTR("airspeednudge"))) {
|
|
port->printf_P("Airspeed nudge = %f\n", (float)airspeed_nudge/100);
|
|
|
|
//------- print altitude -------
|
|
} else if (strmatch(buf+6, PSTR("altitude"))) {
|
|
recalc_climb_rate();
|
|
port->printf_P(PSTR("Altitude:\n"
|
|
" Pressure: %.2fm\n"
|
|
" GPS: %.2fm\n"
|
|
" Mix ratio: %.3f\n"
|
|
" Mix: %.2fm\n"
|
|
" Above home: %.1fm\n"
|
|
" Climb rate: %.2fm/s\n"),
|
|
(float)press_alt / 100,
|
|
(float)gps.altitude / 100,
|
|
get(PARAM_ALT_MIX),
|
|
(((1.0 - get(PARAM_ALT_MIX)) * gps.altitude) + (get(PARAM_ALT_MIX) * press_alt)) / 100,
|
|
(float)get_altitude_above_home()/100,
|
|
(float)climb_rate/100);
|
|
|
|
//------- print attitude -------
|
|
} else if (strmatch(buf+6, PSTR("attitude"))) {
|
|
print_attitude();
|
|
|
|
//------- print commands[ <N1>-<N2>] -------
|
|
} else if (strmatch(buf+6, PSTR("commands"))) {
|
|
unsigned char dash, index1, index2;
|
|
for (dash=14; dash<sizeof(gcsinputbuffer); dash++) {
|
|
if (buf[dash] == 0) break;
|
|
if (buf[dash] == '-') break;
|
|
}
|
|
if (buf[dash] == 0) {
|
|
print_commands();
|
|
} else {
|
|
buf[dash] = 0;
|
|
index1 = atoi(buf+14);
|
|
index2 = atoi(buf+dash+1);
|
|
if (index2 < index1) index2 = get(PARAM_WP_TOTAL);
|
|
print_commands(index1, index2);
|
|
}
|
|
|
|
//------- print ctrlmode -------
|
|
} else if (strmatch(buf+6, PSTR("ctrlmode"))) {
|
|
print_control_mode();
|
|
|
|
//------- print curwaypts -------
|
|
} else if (strmatch(buf+6, PSTR("curwaypts"))) {
|
|
print_current_waypoints();
|
|
|
|
//------- print gains -------
|
|
} else if (strmatch(buf+6, PSTR("gains"))) {
|
|
print_gains();
|
|
|
|
//------- print flightmodes -------
|
|
} else if (strmatch(buf+6, PSTR("flightmodes"))) {
|
|
int i;
|
|
port->print_P(PSTR("EEPROM read: "));
|
|
for (i=0; i<6; i++) {
|
|
port->print_P(PSTR("Ch ")); port->print(i,DEC); port->print_P(PSTR(" = ")); port->print(get(uint8_param_t(PARAM_FLIGHT_MODE_1+i)),DEC); port->print_P(PSTR(", "));
|
|
}
|
|
port->println(" ");
|
|
|
|
//------- print holdalt -------
|
|
} else if (strmatch(buf+6, PSTR("holdalt"))) {
|
|
port->print_P(PSTR("Altitude above home set to ")); port->println(get(PARAM_ALT_HOLD_HOME),2);
|
|
|
|
//------- print imax {servoroll|servopitch|servorudder|navroll|navpitchasp|navpitchalt|throttlete|throttlealt} -------
|
|
} else if (strmatch(buf+6, PSTR("imax "))) {
|
|
int i;
|
|
unsigned char j;
|
|
if (get_pid_offset_name(buf+11, &i, &j)) {
|
|
port->print_P(PSTR("Integrator maximum for "));
|
|
port->print(buf+9);
|
|
port->print_P(PSTR(" = "));
|
|
port->println(pid_index[i]->imax(),DEC);
|
|
} else
|
|
print_gain_keyword_error();
|
|
|
|
//------- print kp {servoroll|servopitch|servorudder|navroll|navpitchasp|navpitchalt|throttlete|throttlealt} -------
|
|
} else if (strmatch(buf+6, PSTR("kp "))) {
|
|
int i;
|
|
unsigned char j;
|
|
if (get_pid_offset_name(buf+9, &i, &j)) {
|
|
port->print_P(PSTR("P gain for "));
|
|
port->print(buf+9);
|
|
port->print_P(PSTR(" = "));
|
|
port->println(pid_index[i]->kP(),DEC);
|
|
} else
|
|
print_gain_keyword_error();
|
|
|
|
//------- print ki {servoroll|servopitch|servorudder|navroll|navpitchasp|navpitchalt|throttlete|throttlealt} -------
|
|
} else if (strmatch(buf+6, PSTR("ki "))) {
|
|
int i;
|
|
unsigned char j;
|
|
if (get_pid_offset_name(buf+9, &i, &j)) {
|
|
port->print_P(PSTR("I gain for "));
|
|
port->print(buf+9);
|
|
port->print_P(PSTR(" = "));
|
|
port->println(pid_index[i]->kI(),DEC);
|
|
} else
|
|
print_gain_keyword_error();
|
|
|
|
//------- print kd {servoroll|servopitch|servorudder|navroll|navpitchasp|navpitchalt|throttlete|throttlealt} -------
|
|
} else if (strmatch(buf+6, PSTR("kd "))) {
|
|
int i;
|
|
unsigned char j;
|
|
if (get_pid_offset_name(buf+9, &i, &j)) {
|
|
port->print_P(PSTR("D gain for "));
|
|
port->print(buf+9);
|
|
port->print_P(PSTR(" = "));
|
|
port->println(pid_index[i]->kD(),DEC);
|
|
} else
|
|
print_gain_keyword_error();
|
|
|
|
//------- print kff {pitchcomp|ruddermix|pitchtothrottle} -------
|
|
} else if (strmatch(buf+6, PSTR("kff "))) {
|
|
if (strmatch(buf+10, PSTR("pitchcomp"))) {
|
|
port->print_P(PSTR("FF gain for pitchcomp = "));
|
|
port->println(get(PARAM_KFF_PTCHCOMP),DEC);
|
|
} else if (strmatch(buf+10, PSTR("ruddermix"))) {
|
|
port->print_P(PSTR("FF gain for ruddermix = "));
|
|
port->println(get(PARAM_KFF_RDDRMIX),DEC);
|
|
} else if (strmatch(buf+10, PSTR("pitchtothrottle"))) {
|
|
port->print_P(PSTR("FF gain for pitchtothrottle = "));
|
|
port->println(get(PARAM_KFF_PTCH2THR),DEC);
|
|
/*} else if (strmatch(buf+10, PSTR("throttletopitch"))) {
|
|
port->print_P(PSTR("FF gain for throttletopitch = "));
|
|
port->println(kff[CASE_T_TO_P],DEC);*/
|
|
} else
|
|
print_gain_keyword_error();
|
|
|
|
//------- print location -------
|
|
} else if (strmatch(buf+6, PSTR("location"))) {
|
|
print_position();
|
|
|
|
//------- print navrolllimit -------
|
|
} else if (strmatch(buf+6, PSTR("navrolllimit"))) {
|
|
port->print_P(PSTR("Nav roll limit = ")); port->println((float)get(PARAM_LIM_ROLL)/100,2);
|
|
|
|
//------- print navsettings -------
|
|
} else if (strmatch(buf+6, PSTR("navsettings"))) {
|
|
port->printf_P(PSTR("Navigation settings:\n"
|
|
#if AIRSPEED_SENSOR == ENABLED
|
|
" Cruise airspeed: %.2f\n"
|
|
#else
|
|
" Cruise throttle: %d\n"
|
|
#endif
|
|
" Hold altitude above home: %ld\n"
|
|
" Loiter radius: %d\n"
|
|
" Waypoint radius: %d\n"),
|
|
#if AIRSPEED_SENSOR == ENABLED
|
|
(float)get(PARAM_TRIM_AIRSPEED) / 100,
|
|
#else
|
|
get(PARAM_TRIM_THROTTLE),
|
|
#endif
|
|
get(PARAM_ALT_HOLD_HOME),
|
|
get(PARAM_LOITER_RADIUS),
|
|
get(PARAM_WP_RADIUS));
|
|
|
|
//------- print pitchmax -------
|
|
} else if (strmatch(buf+6, PSTR("pitchmax"))) {
|
|
port->print_P(PSTR("Maximum pitch = ")); port->println((float)get(PARAM_LIM_PITCH_MAX)/100,2);
|
|
|
|
//------- print pitchmin -------
|
|
} else if (strmatch(buf+6, PSTR("pitchmin"))) {
|
|
port->print_P(PSTR("Minimum pitch = ")); port->println((float)get(PARAM_LIM_PITCH_MIN)/100,2);
|
|
|
|
//------- print pitchtarget -------
|
|
} else if (strmatch(buf+6, PSTR("pitchtarget"))) {
|
|
port->print_P(PSTR("Target pitch = ")); port->println((float)get(PARAM_TRIM_PITCH)/100,2);
|
|
|
|
#if HIL_MODE != HIL_MODE_ATTITUDE
|
|
//------- print pressure -------
|
|
} else if (strmatch(buf+6, PSTR("pressure"))) {
|
|
port->println_P(PSTR("BMP085 pressure sensor:"));
|
|
port->print_P(PSTR(" Temperature: ")); port->println(pitot.Temp,DEC);
|
|
port->print_P(PSTR(" Pressure: ")); port->println(pitot.Press,DEC);
|
|
#endif
|
|
|
|
//------- print rlocation home -------
|
|
} else if (strmatch(buf+6, PSTR("rlocation home"))) {
|
|
print_rlocation(&home);
|
|
|
|
//------- print rlocation -------
|
|
//(implication is "relative to next waypoint")
|
|
} else if (strmatch(buf+6, PSTR("rlocation"))) {
|
|
print_rlocation(&next_WP);
|
|
|
|
//------- print speed -------
|
|
} else if (strmatch(buf+6, PSTR("speed"))) {
|
|
port->println_P(PSTR("Speed:"));
|
|
port->print_P(PSTR(" Ground: ")); port->println((float)gps.ground_speed/100.0,2);
|
|
#if AIRSPEED_SENSOR == ENABLED
|
|
port->print_P(PSTR(" Air: ")); port->println((float)airspeed/100.0,2);
|
|
port->print_P(PSTR(" Cruise: ")); port->println((float)get(PARAM_TRIM_AIRSPEED)/100.0,2);
|
|
#endif
|
|
|
|
//------- print throttlecruise -------
|
|
} else if (strmatch(buf+6, PSTR("throttlecruise"))) {
|
|
port->print_P(PSTR("Throttle cruise = ")); port->print(get(PARAM_TRIM_THROTTLE),DEC); port->println_P(PSTR("%"));
|
|
|
|
//------- print throttlemax -------
|
|
} else if (strmatch(buf+6, PSTR("throttlemax"))) {
|
|
port->print_P(PSTR("Throttle max = ")); port->print(get(PARAM_THR_MAX),DEC); port->println_P(PSTR("%"));
|
|
|
|
//------- print throttlemin -------
|
|
} else if (strmatch(buf+6, PSTR("throttlemin"))) {
|
|
port->print_P(PSTR("Throttle min = ")); port->print(get(PARAM_THR_MIN),DEC); port->println_P(PSTR("%"));
|
|
|
|
//------- print tuning -------
|
|
} else if (strmatch(buf+6, PSTR("tuning"))) {
|
|
print_tuning();
|
|
|
|
} else
|
|
print_error(ERR("USAGE: print {altitude|attitude|commands|ctrlmode|curwaypts|flightmodes|k -?|kp|ki|kd|kff|location|navsettings|pressure|rlocation [home]|speed|tuning|}"));
|
|
|
|
//*********** Process 'reset' commands ***********
|
|
} else if (strmatch(buf, PSTR("reset "))) {
|
|
//------- reset commands -------
|
|
if (strmatch(buf+6, PSTR("commands"))) {
|
|
reload_commands();
|
|
port->println_P(PSTR("Commands reloaded."));
|
|
} else
|
|
print_error(ERR("USAGE: reset commands"));
|
|
|
|
//*********** Process 'rtl' command ***********
|
|
} else if (strmatch(buf, PSTR("rtl"))) {
|
|
return_to_launch();
|
|
port->println_P(PSTR("Returning to launch..."));
|
|
|
|
//*********** Process 'set' commands ***********
|
|
} else if (strmatch(buf, PSTR("set "))) {
|
|
//------- set cmd -------
|
|
if (strmatch(buf+4, PSTR("cmd "))) {
|
|
process_set_cmd(buf+8, bufferidx-8);
|
|
|
|
//------- set cmdcount -------
|
|
} else if (strmatch(buf+4, PSTR("cmdcount "))) {
|
|
set(PARAM_WP_TOTAL, atoi(buf+13));
|
|
port->print_P(PSTR("PARAM_WP_TOTAL = ")); port->println(get(PARAM_WP_TOTAL),DEC);
|
|
|
|
//------- set cmdindex -------
|
|
} else if (strmatch(buf+4, PSTR("cmdindex "))) {
|
|
set(PARAM_WP_INDEX, atoi(buf+13));
|
|
decrement_WP_index();
|
|
next_command = get_wp_with_index(get(PARAM_WP_INDEX)+1);
|
|
port->print_P(PSTR("Command set to index ")); port->print(get(PARAM_WP_INDEX),DEC);
|
|
if (next_command.id >= 0x10 && next_command.id <= 0x1F) { //TODO: create a function the defines what type of command each command ID is
|
|
command_must_index = 0;
|
|
port->println_P(PSTR(" (must)"));
|
|
} else if (next_command.id >= 0x20 && next_command.id <= 0x2F) {
|
|
command_may_index = 0;
|
|
port->println_P(PSTR(" (may)"));
|
|
} else
|
|
port->println_P(PSTR(" (now)"));
|
|
|
|
next_command.id = CMD_BLANK;
|
|
if (get(PARAM_WP_INDEX) > get(PARAM_WP_TOTAL)) {
|
|
set(PARAM_WP_TOTAL, get(PARAM_WP_INDEX));
|
|
port->print_P(PSTR(" The total number of commands is now "));
|
|
port->println(get(PARAM_WP_TOTAL),DEC);
|
|
}
|
|
next_WP = current_loc;
|
|
update_commands();
|
|
|
|
//------- set cruise -------
|
|
} else if (strmatch(buf+4, PSTR("cruise "))) {
|
|
unsigned char j = 4+7;
|
|
#if AIRSPEED_SENSOR == 1
|
|
port->print_P(PSTR("airspeed_cruise changed from "));
|
|
port->print((float)get(PARAM_TRIM_AIRSPEED)/100,2);
|
|
port->print_P(PSTR(" to "));
|
|
set(PARAM_TRIM_AIRSPEED, (int)(readfloat(buf, &j)/100000));
|
|
set(PARAM_TRIM_AIRSPEED, constrain(get(PARAM_TRIM_AIRSPEED),0,9000)); //TODO: constrain minimum as stall speed, maximum as Vne
|
|
port->println(((float)get(PARAM_TRIM_AIRSPEED))/100,2);
|
|
#else
|
|
port->print_P(PSTR("throttle_cruise changed from "));
|
|
port->print(get(PARAM_TRIM_THROTTLE),DEC);
|
|
port->print_P(PSTR(" to "));
|
|
set(PARAM_TRIM_THROTTLE, constrain((int)(readfloat(buf, &j)/10000000),0,200));
|
|
port->println(get(PARAM_TRIM_THROTTLE),DEC);
|
|
#endif
|
|
//save_user_configs();
|
|
|
|
//------- set holdalt -------
|
|
} else if (strmatch(buf+4, PSTR("holdalt "))) {
|
|
int tempalt = atoi(buf+12)*100;
|
|
set(PARAM_ALT_HOLD_HOME, (float)tempalt); //save_alt_to_hold((int32_t)tempalt);
|
|
port->println_P(PSTR("Hold altitude above home set."));
|
|
|
|
//------- set imax {servoroll|servopitch|servorudder|navroll|navpitchasp|navpitchalt|throttlete|throttlealt} <X> -------
|
|
} else if (strmatch(buf+4, PSTR("imax "))) {
|
|
int i;
|
|
unsigned char j = 0;
|
|
if (get_pid_offset_name(buf+9, &i, &j)) {
|
|
buf[9+j] = 0;
|
|
port->print_P(PSTR("Integrator maximum for "));
|
|
port->print(buf+9);
|
|
port->print_P(PSTR(" changed from "));
|
|
port->print(pid_index[i]->imax(),DEC);
|
|
port->print_P(PSTR(" to "));
|
|
pid_index[i]->imax((float)atoi(buf+j));
|
|
pid_index[i]->save_gains();
|
|
port->println(pid_index[i]->imax(),DEC);
|
|
} else
|
|
print_error(ERR("ERROR: Did not recognize keyword; type set k -? for more information"));
|
|
|
|
//------- set kp {servoroll|servopitch|servorudder|navroll|navpitchasp|navpitchalt|throttlete|throttlealt} <X> -------
|
|
} else if (strmatch(buf+4, PSTR("kp "))) {
|
|
int i;
|
|
unsigned char j = 0;
|
|
if (get_pid_offset_name(buf+7, &i, &j)) {
|
|
buf[7+j] = 0;
|
|
port->print_P(PSTR("P gain for "));
|
|
port->print(buf+7);
|
|
port->print_P(PSTR(" changed from "));
|
|
port->print(pid_index[i]->kP(),DEC);
|
|
port->print_P(PSTR(" to "));
|
|
j += 7+1;
|
|
pid_index[i]->kP(((float)readfloat(buf, &j))/10000000);
|
|
pid_index[i]->save_gains();
|
|
port->println(pid_index[i]->kP(),DEC);
|
|
} else
|
|
print_gain_keyword_error();
|
|
|
|
//------- set ki {servoroll|servopitch|servorudder|navroll|navpitchasp|navpitchalt|throttlete|throttlealt} <X> -------
|
|
} else if (strmatch(buf+4, PSTR("ki "))) {
|
|
int i;
|
|
unsigned char j = 0;
|
|
if (get_pid_offset_name(buf+7, &i, &j)) {
|
|
buf[7+j] = 0;
|
|
port->print_P(PSTR("I gain for "));
|
|
port->print(buf+7);
|
|
port->print_P(PSTR(" changed from "));
|
|
port->print(pid_index[i]->kI(),DEC);
|
|
port->print_P(PSTR(" to "));
|
|
j += 7+1;
|
|
pid_index[i]->kI(((float)readfloat(buf, &j))/10000000);
|
|
pid_index[i]->save_gains();
|
|
port->println(pid_index[i]->kI(),DEC);
|
|
} else
|
|
print_gain_keyword_error();
|
|
|
|
//------- set kd {servoroll|servopitch|servorudder|navroll|navpitchasp|navpitchalt|throttlete|throttlealt} <X> -------
|
|
} else if (strmatch(buf+4, PSTR("kd "))) {
|
|
int i;
|
|
unsigned char j = 0;
|
|
if (get_pid_offset_name(buf+7, &i, &j)) {
|
|
buf[7+j] = 0;
|
|
port->print_P(PSTR("D gain for "));
|
|
port->print(buf+7);
|
|
port->print_P(PSTR(" changed from "));
|
|
port->print(pid_index[i]->kD(),DEC);
|
|
port->print_P(PSTR(" to "));
|
|
j += 7+1;
|
|
pid_index[i]->kD(((float)readfloat(buf, &j))/10000000);
|
|
pid_index[i]->save_gains();
|
|
port->println(pid_index[i]->kD(),DEC);
|
|
} else
|
|
print_gain_keyword_error();
|
|
|
|
//------- set kff {pitchcomp|ruddermix|pitchtothrottle} <X> -------
|
|
} else if (strmatch(buf+4, PSTR("kff "))) {
|
|
unsigned char j = 0;
|
|
if (strmatch(buf+8, PSTR("pitchcomp"))) {
|
|
buf[8+9] = 0;
|
|
port->print_P(PSTR("FF gain for "));
|
|
port->print(buf+8);
|
|
port->print_P(PSTR(" changed from "));
|
|
port->print(get(PARAM_KFF_PTCHCOMP),DEC);
|
|
port->print_P(PSTR(" to "));
|
|
j = 8+9+1;
|
|
set(PARAM_KFF_PTCHCOMP, ((float)readfloat(buf, &j))/10000000);
|
|
port->println(get(PARAM_KFF_PTCHCOMP),DEC);
|
|
} else if (strmatch(buf+8, PSTR("ruddermix"))) {
|
|
buf[8+9] = 0;
|
|
port->print_P(PSTR("FF gain for "));
|
|
port->print(buf+8);
|
|
port->print_P(PSTR(" changed from "));
|
|
port->print(get(PARAM_KFF_RDDRMIX),DEC);
|
|
port->print_P(PSTR(" to "));
|
|
j = 8+9+1;
|
|
set(PARAM_KFF_RDDRMIX, ((float)readfloat(buf, &j))/10000000);
|
|
port->println(get(PARAM_KFF_RDDRMIX),DEC);
|
|
} else if (strmatch(buf+8, PSTR("pitchtothrottle"))) {
|
|
buf[8+15] = 0;
|
|
port->print_P(PSTR("FF gain for "));
|
|
port->print(buf+8);
|
|
port->print_P(PSTR(" changed from "));
|
|
port->print(get(PARAM_KFF_PTCH2THR),DEC);
|
|
port->print_P(PSTR(" to "));
|
|
j = 8+15+1;
|
|
set(PARAM_KFF_PTCH2THR, ((float)readfloat(buf, &j))/10000000);
|
|
port->println(get(PARAM_KFF_PTCH2THR),DEC);
|
|
/*} else if (strmatch(buf+8, PSTR("throttletopitch"))) {
|
|
buf[8+15] = 0;
|
|
port->print_P(PSTR("FF gain for "));
|
|
port->print(buf+8);
|
|
port->print_P(PSTR(" changed from "));
|
|
port->print(kff[CASE_T_TO_P],DEC);
|
|
port->print_P(PSTR(" to "));
|
|
j = 8+15+1;
|
|
kff[CASE_T_TO_P] = ((float)readfloat(buf, &j))/10000000;
|
|
port->println(kff[CASE_T_TO_P],DEC);*/
|
|
} else
|
|
print_gain_keyword_error();
|
|
|
|
//------- set loiterradius -------
|
|
} else if (strmatch(buf+4, PSTR("loiterradius "))) {
|
|
set(PARAM_LOITER_RADIUS, atoi(buf+17));
|
|
port->print_P(PSTR("Set loiter radius to ")); port->print(get(PARAM_LOITER_RADIUS),DEC); port->println_P(PSTR(" meters"));
|
|
|
|
//------- set navrolllimit <X> -------
|
|
} else if (strmatch(buf+4, PSTR("navrolllimit "))) {
|
|
unsigned char j = 17;
|
|
port->print_P(PSTR("Nav roll limit changed from ")); port->print((float)get(PARAM_LIM_ROLL)/100,2);
|
|
port->print_P(PSTR(" to "));
|
|
set(PARAM_LIM_ROLL, readfloat(buf, &j)/100000);
|
|
port->println((float)get(PARAM_LIM_ROLL)/100,2);
|
|
|
|
//------- set pitchmax <X> -------
|
|
} else if (strmatch(buf+4, PSTR("pitchmax "))) {
|
|
unsigned char j = 13;
|
|
port->print_P(PSTR("Maximum pitch changed from ")); port->print((float)get(PARAM_LIM_PITCH_MAX)/100,2);
|
|
port->print_P(PSTR(" to "));
|
|
set(PARAM_LIM_PITCH_MAX, readfloat(buf, &j)/100000);
|
|
port->println((float)get(PARAM_LIM_PITCH_MAX)/100,2);
|
|
|
|
//------- set pitchmin <X> -------
|
|
} else if (strmatch(buf+4, PSTR("pitchmin "))) {
|
|
unsigned char j = 13;
|
|
port->print_P(PSTR("Minimum pitch changed from ")); port->print((float)get(PARAM_LIM_PITCH_MIN)/100,2);
|
|
port->print_P(PSTR(" to "));
|
|
set(PARAM_LIM_PITCH_MIN, readfloat(buf, &j)/100000);
|
|
port->println((float)get(PARAM_LIM_PITCH_MIN)/100,2);
|
|
|
|
//------- set pitchtarget <X> -------
|
|
} else if (strmatch(buf+4, PSTR("pitchtarget "))) {
|
|
unsigned char j = 16;
|
|
port->print_P(PSTR("Pitch target changed from ")); port->print((float)get(PARAM_TRIM_PITCH)/100,2);
|
|
port->print_P(PSTR(" to "));
|
|
set(PARAM_TRIM_PITCH, readfloat(buf, &j)/100000);
|
|
port->println((float)get(PARAM_TRIM_PITCH)/100,2);
|
|
|
|
//------- set rcin -------
|
|
} else if (strmatch(buf+4, PSTR("rcin"))) {
|
|
unsigned char index = buf[8]-'1';
|
|
if (index < 8) {
|
|
radio_in[index] = atoi(buf+9);
|
|
} else
|
|
print_error(ERR("USAGE: set rcin<N> <X>"));
|
|
|
|
//------- set rcout -------
|
|
} else if (strmatch(buf+4, PSTR("rcout"))) {
|
|
unsigned char index = buf[9]-'1';
|
|
if (index < 8) {
|
|
radio_out[index] = atoi(buf+10);
|
|
APM_RC.OutputCh(index, radio_out[index]);
|
|
} else
|
|
print_error(ERR("USAGE: set rcout<N> <X>"));
|
|
|
|
//------- set relay -------
|
|
} else if (strmatch(buf+4, PSTR("relay "))) {
|
|
unsigned char newvalue = atoi(buf+10);
|
|
if (newvalue == 0) {
|
|
relay_off();
|
|
port->println_P(PSTR("Relay turned off"));
|
|
} else if (newvalue == 1) {
|
|
relay_on();
|
|
port->println_P(PSTR("Relay turned on"));
|
|
} else {
|
|
relay_toggle();
|
|
port->println_P(PSTR("Relay toggled"));
|
|
}
|
|
|
|
//------- set throttlecruise <X> -------
|
|
} else if (strmatch(buf+4, PSTR("throttlecruise "))) {
|
|
port->print_P(PSTR("Throttle cruise changed from ")); port->print(get(PARAM_TRIM_THROTTLE),DEC); port->print_P(PSTR("% to "));
|
|
set(PARAM_TRIM_THROTTLE,atoi(buf+19));
|
|
port->print(get(PARAM_TRIM_THROTTLE),DEC); port->println_P(PSTR("%"));
|
|
|
|
//------- set throttlefailsafe <N> -------
|
|
} else if (strmatch(buf+4, PSTR("throttlefailsafe "))) {
|
|
set(PARAM_THR_FAILSAFE, atoi(buf+21));
|
|
/*if (get(PARAM_THROTTLE_FS) == 0)
|
|
throttle_FS_enabled = 0;
|
|
else
|
|
throttle_FS_enabled = 1;*/
|
|
//save_user_configs();
|
|
port->println_P(PSTR("Throttle failsafe adjusted."));
|
|
|
|
//------- set throttlefailsafeaction <N> -------
|
|
} else if (strmatch(buf+4, PSTR("throttlefailsafeaction "))) {
|
|
set(PARAM_THR_FS_ACTION, atoi(buf+27));
|
|
//save_user_configs();
|
|
|
|
//------- set throttlemax <X> -------
|
|
} else if (strmatch(buf+4, PSTR("throttlemax "))) {
|
|
port->print_P(PSTR("Throttle max changed from ")); port->print(get(PARAM_THR_MAX),DEC); port->print_P(PSTR("% to "));
|
|
set(PARAM_THR_MAX, atoi(buf+16));
|
|
port->print(get(PARAM_THR_MAX),DEC); port->println_P(PSTR("%"));
|
|
//save_user_configs();
|
|
|
|
//------- set throttlemin <X> -------
|
|
} else if (strmatch(buf+4, PSTR("throttlemin "))) {
|
|
port->print_P(PSTR("Throttle min changed from ")); port->print(get(PARAM_THR_MIN),DEC); port->print_P(PSTR("% to "));
|
|
set(PARAM_THR_MIN, atoi(buf+16));
|
|
port->print(get(PARAM_THR_MIN),DEC); port->println_P(PSTR("%"));
|
|
//save_user_configs();
|
|
|
|
//------- set wpradius -------
|
|
} else if (strmatch(buf+4, PSTR("wpradius "))) {
|
|
set(PARAM_WP_RADIUS, atoi(buf+13));
|
|
port->print_P(PSTR("Set waypoint radius to ")); port->print(get(PARAM_WP_RADIUS),DEC); port->println_P(PSTR(" meters"));
|
|
|
|
//------- set xtrackentryangle -------
|
|
} else if (strmatch(buf+4, PSTR("xtrackentryangle "))) {
|
|
unsigned char j = 21;
|
|
set(PARAM_XTRACK_ANGLE, readfloat(buf, &j)/100000);
|
|
port->print_P(PSTR("Crosstrack entry angle set to ")); port->println((float)get(PARAM_XTRACK_ANGLE)/100,2);
|
|
|
|
//------- set xtrackgain -------
|
|
} else if (strmatch(buf+4, PSTR("xtrackgain "))) {
|
|
unsigned char j = 15;
|
|
set(PARAM_XTRACK_GAIN, ((float)readfloat(buf, &j))/10000000);
|
|
port->print_P(PSTR("Crosstrack gain set to ")); port->println(get(PARAM_XTRACK_GAIN),2);
|
|
|
|
} else
|
|
print_error(ERR("USAGE: set {cmd|cmdcount|cmdindex|cruise|holdalt|kp|ki|kd|kff|loiterradius|rcin|rcout|wpradius}"));
|
|
|
|
//*********** Process 'status' commands ***********
|
|
} else if (strmatch(buf, PSTR("status "))) {
|
|
//------- status altitude -------
|
|
if (strmatch(buf+7, PSTR("altitude"))) {
|
|
port->printf_P(PSTR("Altitude:\n"
|
|
" altitude_error = %.2fm\n"
|
|
" target_altitude = %.2fm\n"
|
|
" next_WP.alt = %.2fm\n"
|
|
" wp_distance = %ldm\n"
|
|
" wp_totalDistance = %ldm\n"
|
|
" offset_altitude = %.2fm\n"),
|
|
(float)altitude_error/100,
|
|
(float)target_altitude/100,
|
|
(float)next_WP.alt/100,
|
|
wp_distance,
|
|
wp_totalDistance,
|
|
(float)offset_altitude/100);
|
|
|
|
//------- status climb -------
|
|
} else if (strmatch(buf+7, PSTR("climb"))) {
|
|
print_climb_debug_info();
|
|
|
|
//------- status control -------
|
|
} else if (strmatch(buf+7, PSTR("control"))) {
|
|
port->printf_P(PSTR("Control status:\n"
|
|
" Roll: nav= %.2f, servo_out= %.2f\n" // XXX float?
|
|
" Pitch: nav= %.2f, servo_out= %.2f\n" // XXX float?
|
|
" Throttle: nav= %d, servo_out= %d\n"),
|
|
(float)nav_roll / 100, (float)servo_out[CH_ROLL] / 100,
|
|
(float)nav_pitch / 100, (float)servo_out[CH_PITCH] / 100,
|
|
get(PARAM_TRIM_THROTTLE), servo_out[CH_THROTTLE]);
|
|
|
|
//------- status gps -------
|
|
} else if (strmatch(buf+7, PSTR("gps"))) {
|
|
port->printf_P(PSTR("GPS status:\n"
|
|
" Fix: %S (%d)\n"
|
|
" Satellites: %d\n"
|
|
" Fix count: %d\n"),
|
|
(gps.fix ? PSTR("Valid") : PSTR("INVALID")),
|
|
(int)gps.fix,
|
|
(int)gps.num_sats,
|
|
gps_fix_count);
|
|
|
|
//------- status landing -------
|
|
} else if (strmatch(buf+7, PSTR("landing"))) {
|
|
port->printf_P(PSTR("Landing status:"
|
|
" land_complete = %d\n"
|
|
" landing_pitch = %d\n"
|
|
" nav_pitch = %ld\n"
|
|
" airspeed_cruise = %d\n"
|
|
" throttle_cruise = %d\n"
|
|
" hold_course = %ld\n"
|
|
" nav_bearing = %ld\n"
|
|
" bearing_error = %ld\n"),
|
|
(int)land_complete,
|
|
landing_pitch,
|
|
nav_pitch,
|
|
get(PARAM_TRIM_AIRSPEED),
|
|
get(PARAM_TRIM_THROTTLE),
|
|
hold_course,
|
|
nav_bearing,
|
|
bearing_error);
|
|
|
|
//------- status loiter -------
|
|
} else if (strmatch(buf+7, PSTR("loiter"))) {
|
|
port->printf_P(PSTR("Loiter status:"
|
|
" Loiter radius: %d\n"
|
|
" Progress: %d/%d\n" // XXX original had non-ASCII units char
|
|
" Time: %ldms/%dms\n"),
|
|
get(PARAM_LOITER_RADIUS),
|
|
loiter_sum, loiter_total,
|
|
loiter_time, loiter_time_max);
|
|
|
|
//------- status navigation -------
|
|
} else if (strmatch(buf+7, PSTR("navigation"))) {
|
|
port->printf_P(PSTR("Navigation status:\n"
|
|
" From <%.6f, %.6f, %.1f>: "),
|
|
(float)prev_WP.lat/10000000.0,
|
|
(float)prev_WP.lng/10000000.0,
|
|
(float)prev_WP.alt/100.0);
|
|
print_rlocation(&prev_WP);
|
|
port->printf_P(PSTR(" At <%.6f, %.6f, %.1f>\n"),
|
|
(float)current_loc.lat/10000000.0,
|
|
(float)current_loc.lng/10000000.0,
|
|
(float)current_loc.alt/100.0);
|
|
port->printf_P(PSTR(" To <%.6f, %.6f, %.1f>: "),
|
|
(float)next_WP.lat/10000000.0,
|
|
(float)next_WP.lng/10000000.0,
|
|
(float)next_WP.alt/100.0);
|
|
print_rlocation(&next_WP);
|
|
port->printf_P(PSTR(" Distance: %.1f%% %ldm / %ldm; %.1f vertically\n"),
|
|
100.0*(float)(wp_totalDistance-wp_distance)/(float)wp_totalDistance,
|
|
wp_totalDistance-wp_distance,
|
|
wp_totalDistance,
|
|
(float)altitude_error/100.0);
|
|
port->printf_P(PSTR(" Nav bearing: %.2f; error = %.2f\n"),
|
|
(float)nav_bearing/100.0,
|
|
(float)bearing_error/100.0);
|
|
port->printf_P(PSTR(" Ground course: %.1f (current), %.1f (target)\n"),
|
|
(float)gps.ground_course/100.0,
|
|
(float)target_bearing/100.0);
|
|
if (hold_course >= 0) {
|
|
port->print_P(PSTR(" HOLD COURSE: ")); port->println(hold_course/100.0,2);
|
|
}
|
|
|
|
//------- status navpitch -------
|
|
} else if (strmatch(buf+7, PSTR("navpitch"))) {
|
|
#if AIRSPEED_SENSOR == ENABLED
|
|
port->printf_P(PSTR(">>> nav_pitch = PID[airspeed_error (%.2f) = airspeed_cruise (%.2f) - airspeed (%.2f)]\n"),
|
|
(float)airspeed_error / 100,
|
|
(float)get(PARAM_TRIM_AIRSPEED) / 100,
|
|
(float)airspeed / 100);
|
|
#else
|
|
port->printf_P(PSTR(">>> nav_pitch = PID[altitude_error (%ld) = target_altitude (%ld) - current_loc.alt (%ld)]\n"),
|
|
altitude_error,
|
|
target_altitude,
|
|
current_loc.alt);
|
|
#endif
|
|
port->printf_P(PSTR("nav_pitch (%.2f) - pitch_sensor (%.2f) + pitch_comp (%.2f) = %.2f\n"),
|
|
(float)nav_pitch / 100,
|
|
(float)dcm.pitch_sensor / 100,
|
|
fabs(dcm.roll_sensor * get(PARAM_KFF_PTCHCOMP)) / 100,
|
|
(float)(nav_pitch-dcm.pitch_sensor + fabs(dcm.roll_sensor * get(PARAM_KFF_PTCHCOMP))) / 100);
|
|
port->printf_P(PSTR("servo_out[CH_PITCH] (%.2f) = PID[nav_pitch + pitch_comp - pitch_sensor]"),
|
|
(float)servo_out[CH_PITCH] / 100);
|
|
|
|
//------- status navroll -------
|
|
} else if (strmatch(buf+7, PSTR("navroll"))) {
|
|
print_rlocation(&next_WP);
|
|
port->printf_P(PSTR("bearing_error (%ld) = nav_bearing (%ld) - yaw_sensor (%ld)\n"
|
|
"nav_roll (%ld) - roll_sensor (%ld) = %ld\n"
|
|
"servo_out[CH_ROLL] = %d\n"),
|
|
bearing_error, nav_bearing, dcm.yaw_sensor,
|
|
nav_roll, dcm.roll_sensor, nav_roll - dcm.roll_sensor,
|
|
servo_out[CH_ROLL]);
|
|
|
|
//------- status pid {servoroll|servopitch|servorudder|navroll|navpitchasp|navpitchalt|throttlete|throttlealt} -------
|
|
} else if (strmatch(buf+7, PSTR("pid "))) {
|
|
int i;
|
|
unsigned char j;
|
|
if (get_pid_offset_name(buf+11, &i, &j)) {
|
|
port->print(buf+11);
|
|
port->print_P(PSTR(": "));
|
|
display_PID = i; //The next time a PID calculation is performed on this channel, the print_PID routine in this GCS will be called
|
|
} else
|
|
print_error(ERR("ERROR: Did not recognize keyword"));
|
|
|
|
//------- status rcinputch -------
|
|
} else if (strmatch(buf+7, PSTR("rcinputch"))) {
|
|
unsigned char i;
|
|
port->println_P(PSTR("RC hardware input:"));
|
|
for (i=0; i<8; i++) {
|
|
port->print_P(PSTR(" Ch"));
|
|
port->print(i+1,DEC);
|
|
port->print_P(PSTR(": "));
|
|
port->println(APM_RC.InputCh(i));
|
|
}
|
|
|
|
//------- status rcin -------
|
|
} else if (strmatch(buf+7, PSTR("rcin"))) {
|
|
unsigned char i;
|
|
port->println_P(PSTR("RC software input:"));
|
|
for (i=0; i<8; i++) {
|
|
port->print_P(PSTR(" Ch"));
|
|
port->print(i+1,DEC);
|
|
port->print_P(PSTR(": "));
|
|
port->println(radio_in[i]);
|
|
}
|
|
|
|
//------- status rcout -------
|
|
} else if (strmatch(buf+7, PSTR("rcout"))) {
|
|
unsigned char i;
|
|
port->println_P(PSTR("RC software output:"));
|
|
for (i=0; i<8; i++) {
|
|
port->print_P(PSTR(" Ch"));
|
|
port->print(i+1,DEC);
|
|
port->print_P(PSTR(": "));
|
|
port->println(radio_out[i]);
|
|
}
|
|
|
|
//------- status rcpwm -------
|
|
} else if (strmatch(buf+7, PSTR("rcpwm"))) {
|
|
unsigned char i;
|
|
port->println_P(PSTR("RC hardware output:"));
|
|
for (i=0; i<8; i++) {
|
|
port->print_P(PSTR(" Ch"));
|
|
port->print(i+1,DEC);
|
|
port->print_P(PSTR(": "));
|
|
port->println(readOutputCh(i));
|
|
}
|
|
|
|
//------- status rctrim -------
|
|
} else if (strmatch(buf+7, PSTR("rctrim"))) {
|
|
unsigned char i;
|
|
port->println_P(PSTR("RC trim:"));
|
|
for (i=0; i<8; i++) {
|
|
port->print_P(PSTR(" Ch"));
|
|
port->print(i+1,DEC);
|
|
port->print_P(PSTR(": "));
|
|
port->println(radio_trim(i));
|
|
}
|
|
port->print_P(PSTR(" elevon1_trim = ")); port->println(elevon1_trim,DEC);
|
|
port->print_P(PSTR(" elevon2_trim = ")); port->println(elevon2_trim,DEC);
|
|
|
|
//------- status rc -------
|
|
} else if (strmatch(buf+7, PSTR("rc"))) {
|
|
unsigned char i;
|
|
port->println_P(PSTR("RC:\tCh\tHWin\tSWtrim\tSWin\tServo\tSWout\tHWout\t"));
|
|
for (i=0; i<8; i++) {
|
|
// XXX might benefit from field widths, since some terminals tab badly
|
|
port->printf_P(PSTR("\t%u\t%u\t%u\t%u\t%.2f\t%d\t%d\n"),
|
|
(unsigned int)(i + 1),
|
|
APM_RC.InputCh(i),
|
|
radio_trim(i),
|
|
radio_in[i],
|
|
(float)servo_out[i] / 100,
|
|
radio_out[i],
|
|
readOutputCh(i));
|
|
}
|
|
|
|
//------- status system -------
|
|
} else if (strmatch(buf+7, PSTR("system"))) {
|
|
port->printf_P(PSTR("System status:"
|
|
" Ground start: %S (%d)\n"
|
|
" Home: %SSET (%d)\n"),
|
|
(ground_start ? PSTR("YES") : PSTR("NO")), (int)ground_start,
|
|
(home_is_set ? PSTR("") : PSTR("NOT ")), (int)home_is_set);
|
|
|
|
//------- status takeoff -------
|
|
/*} else if (strmatch(buf+7, PSTR("takeoff"))) {
|
|
port->println_P(PSTR("Takeoff status:"));
|
|
port->print_P(PSTR(" takeoff in progress: "));
|
|
if (takeoff_in_progress)
|
|
port->println_P(PSTR("YES"));
|
|
else {
|
|
port->print_P(PSTR("NO; trigger = "));
|
|
port->print(toss_trigger,DEC);
|
|
port->print_P(PSTR(", current = "));
|
|
Vector3f temp = ardupilotDCM.get_accels();
|
|
port->println(abs(temp.y),1);
|
|
}
|
|
port->printf_P(PSTR(" takeoff_pitch = %.2f\n"
|
|
" scaler = %.3f\n"
|
|
" nav_pitch = %.2f\n"
|
|
" throttle = %d\n"
|
|
" hold_course = %.2f\n"
|
|
" nav_bearing = %ld\n"
|
|
" bearing_error = %ld\n"
|
|
" current_loc.alt = %ld\n"
|
|
" home.alt + takeoff_altitude = %ld"),
|
|
(float)takeoff_pitch / 100,
|
|
(float)airspeed / (float)airspeed_cruise,
|
|
(float)nav_pitch / 100,
|
|
servo_out[CH_THROTTLE],
|
|
(float)hold_course / 100,
|
|
nav_bearing,
|
|
bearing_error,
|
|
current_loc.alt,
|
|
home.alt + takeoff_altitude);*/
|
|
|
|
//------- status telemetry -------
|
|
} else if (strmatch(buf+7, PSTR("telemetry"))) {
|
|
port->printf_P(PSTR("Telemetry status:\n"
|
|
" %S heartbeat\n"
|
|
" %S location\n"
|
|
" %S attitude\n"
|
|
" %S command\n"
|
|
" Severity report level %d\n"),
|
|
(report_heartbeat ? PSTR("Show") : PSTR("Hide")),
|
|
(report_location ? PSTR("Show") : PSTR("Hide")),
|
|
(report_attitude ? PSTR("Show") : PSTR("Hide")),
|
|
(report_command ? PSTR("Show") : PSTR("Hide")),
|
|
(int)report_severity);
|
|
|
|
//------- status throttle -------
|
|
} else if (strmatch(buf+7, PSTR("throttle"))) {
|
|
#if AIRSPEED_SENSOR == ENABLED
|
|
port->printf_P(PSTR(">>> airspeed_energy_error (%ld) = 0.5 * (airspeed_cruise (%.2f)^2 - airspeed (%.2f)^2)\n"
|
|
"energy_error (%ld) = airspeed_energy_error (%ld) + altitude_error*0.098 (%ld)\n"
|
|
"servo_out[CH_THROTTLE] (%d) = PID[energy_error]\n"),
|
|
airspeed_energy_error, (float)get(PARAM_TRIM_AIRSPEED) / 100, (float)airspeed / 100,
|
|
energy_error, airspeed_energy_error, (long)((float)altitude_error * 0.098),
|
|
servo_out[CH_THROTTLE]);
|
|
#else
|
|
port->printf_P(PSTR("altitude_error (%ld) = target_altitude (%ld) - current_loc.alt (%ld)\n"
|
|
"servo_out[CH_THROTTLE] (%d) = PID[altitude_error]\n"),
|
|
altitude_error, target_altitude, current_loc.alt,
|
|
servo_out[CH_THROTTLE]);
|
|
#endif
|
|
|
|
//------- status waypoints -------
|
|
} else if (strmatch(buf+7, PSTR("waypoints"))) {
|
|
port->printf_P(PSTR("Waypoints status:\n"
|
|
" Distance: %ld/%ld\n"
|
|
" Index: %d/%d\n"
|
|
" Next: %d\n"),
|
|
wp_distance, wp_totalDistance,
|
|
(int)get(PARAM_WP_INDEX), (int)get(PARAM_WP_TOTAL),
|
|
(int)next_wp_index);
|
|
|
|
} else if (strmatch(buf+7, PSTR("james"))) {
|
|
port->println_P(PSTR("James is a monkey"));
|
|
} else {
|
|
print_error(ERR("USAGE: status {control|gps|landing|loiter|mixing|navigation|navpitch|navroll|rc|rcinputch|rcin|rcout|rcpwm|rctrim|system|takeoff|telemetry|throttle|waypoints}"));
|
|
}
|
|
|
|
} else {
|
|
print_error(ERR("USAGE: {echo <text>|groundstart|inithome|show|hide|print|status|set|reset|rtl}"));
|
|
print_error(ERR("Type <command> -? for specific usage guidance"));
|
|
}
|
|
}
|
|
|
|
/* strmatch compares two strings and returns 1 if they match and 0 if they don't
|
|
s2 must be stored in program memory (via PSTR) rather than RAM (like standard strings)
|
|
s1 must be at least as long as s2 for a valid match
|
|
if s1 is longer than s2, then only the beginning of s1 (the length of s2) must match s2 for a valid match */
|
|
int
|
|
GCS_DEBUGTERMINAL::strmatch(char* s1, const char* s2)
|
|
{
|
|
int i = 0;
|
|
char c1 = s1[0], c2 = pgm_read_byte(s2);
|
|
|
|
while (c1 != 0 && c2 != 0) {
|
|
if (c1 < c2)
|
|
return 0;
|
|
if (c1 > c2)
|
|
return 0;
|
|
i++;
|
|
c1 = s1[i];
|
|
c2 = pgm_read_byte(s2+i);
|
|
}
|
|
|
|
if (c2==0)
|
|
return 1;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
/* readfloat parses a string written as a float starting at the offset in *i and updates *i as it parses
|
|
numbers are multiplied by 10,000,000 and decimals beyond 7 places are discarded
|
|
parsing is terminated with either a space or a null, other characters are ignored */
|
|
long
|
|
GCS_DEBUGTERMINAL::readfloat(char* s, unsigned char* i)
|
|
{
|
|
long result = 0, multiplier = 0;
|
|
char c, neg = 0;
|
|
|
|
for (;;(*i)++) {
|
|
c = s[*i];
|
|
if (c == ' ')
|
|
break;
|
|
else if (c == 0)
|
|
break;
|
|
else if (c == '-')
|
|
neg = 1-neg;
|
|
else if (c == '.') {
|
|
result *= 10000000;
|
|
multiplier = 1000000;
|
|
} else if (c >= '0' && c <= '9') {
|
|
if (multiplier == 0)
|
|
result = (result*10) + c-'0';
|
|
else {
|
|
result += (c-'0')*multiplier;
|
|
multiplier /= 10;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (multiplier == 0)
|
|
result *= 10000000;
|
|
|
|
if (neg)
|
|
return -result;
|
|
else
|
|
return result;
|
|
}
|
|
|
|
/* process_set_cmd processing the parameters of a 'set cmd' command and takes the appropriate action
|
|
*buffer is the buffer containing the parameters of the command; it should start after the space after 'set cmd'
|
|
bufferlen is the length of the buffer; the routine will stop looking for parameters after the offset index reaches this value
|
|
*/
|
|
#define SETPARAM_NONE (0)
|
|
#define SETPARAM_ID (1)
|
|
#define SETPARAM_P1 (2)
|
|
#define SETPARAM_LAT (3)
|
|
#define SETPARAM_LNG (4)
|
|
#define SETPARAM_ALT (5)
|
|
#define SETPARAM_P2 (6)
|
|
#define SETPARAM_P3 (7)
|
|
#define SETPARAM_P4 (8)
|
|
|
|
void
|
|
GCS_DEBUGTERMINAL::process_set_cmd(char *buffer, int bufferlen)
|
|
{
|
|
BetterStream *port = _port;
|
|
unsigned char i, j, err=1, setparam=SETPARAM_NONE;
|
|
unsigned char cmdindex=0, p1=0, cmdid;
|
|
long lat, lng, alt;
|
|
Location temp;
|
|
|
|
//Parse the command index
|
|
for (i=0; i<bufferlen; i++)
|
|
if (buffer[i] >= '0' && buffer[i] <= '9')
|
|
cmdindex = (cmdindex*10) + buffer[i]-'0';
|
|
else
|
|
break;
|
|
|
|
if (buffer[i] == ' ') {
|
|
//Find the end of the command-type string
|
|
i++;
|
|
for (j=i; j<bufferlen; j++) {
|
|
if (buffer[j] == ' ' || buffer[j] == 0)
|
|
break;
|
|
}
|
|
if (buffer[j] == ' ') {
|
|
//Process the command-type string
|
|
buffer[j] = 0; //Null-terminate the command-type string so strmatch can figure out when to stop comparing, and so atoi can work
|
|
if (strmatch(buffer+i, PSTR("waypoint")))
|
|
cmdid = CMD_WAYPOINT;
|
|
else if (strmatch(buffer+i, PSTR("takeoff")))
|
|
cmdid = CMD_TAKEOFF;
|
|
else if (strmatch(buffer+i, PSTR("landoptions")))
|
|
cmdid = CMD_LAND_OPTIONS;
|
|
else if (strmatch(buffer+i, PSTR("land")))
|
|
cmdid = CMD_LAND;
|
|
else if (strmatch(buffer+i, PSTR("loiternturns")))
|
|
cmdid = CMD_LOITER_N_TURNS;
|
|
else if (strmatch(buffer+i, PSTR("loitertime")))
|
|
cmdid = CMD_LOITER_TIME;
|
|
else if (strmatch(buffer+i, PSTR("loiter")))
|
|
cmdid = CMD_LOITER;
|
|
else if (strmatch(buffer+i, PSTR("delay")))
|
|
cmdid = CMD_DELAY;
|
|
else if (strmatch(buffer+i, PSTR("resetindex")))
|
|
cmdid = CMD_RESET_INDEX;
|
|
else if (strmatch(buffer+i, PSTR("throttlecruise")))
|
|
cmdid = CMD_THROTTLE_CRUISE;
|
|
else if (strmatch(buffer+i, PSTR("resethome")))
|
|
cmdid = CMD_RESET_HOME;
|
|
else if (strmatch(buffer+i, PSTR("index")))
|
|
cmdid = CMD_INDEX;
|
|
else if (strmatch(buffer+i, PSTR("rtl")))
|
|
cmdid = CMD_RTL;
|
|
else if (strmatch(buffer+i, PSTR("relay")))
|
|
cmdid = CMD_RELAY;
|
|
else if (strmatch(buffer+i, PSTR("servo")))
|
|
cmdid = CMD_SERVO;
|
|
else if (strmatch(buffer+i, PSTR("id")))
|
|
setparam = SETPARAM_ID;
|
|
else if (strmatch(buffer+i, PSTR("p1")))
|
|
setparam = SETPARAM_P1;
|
|
else if (strmatch(buffer+i, PSTR("alt")))
|
|
setparam = SETPARAM_ALT;
|
|
else if (strmatch(buffer+i, PSTR("lat")))
|
|
setparam = SETPARAM_LAT;
|
|
else if (strmatch(buffer+i, PSTR("lng")))
|
|
setparam = SETPARAM_LNG;
|
|
else if (strmatch(buffer+i, PSTR("p2")))
|
|
setparam = SETPARAM_P2;
|
|
else if (strmatch(buffer+i, PSTR("p3")))
|
|
setparam = SETPARAM_P3;
|
|
else if (strmatch(buffer+i, PSTR("p4")))
|
|
setparam = SETPARAM_P4;
|
|
else
|
|
cmdid = atoi(buffer+i);
|
|
|
|
if (setparam > SETPARAM_NONE) {
|
|
//Process new parameter value
|
|
i = j+1;
|
|
if (setparam == SETPARAM_ALT || setparam == SETPARAM_LAT || setparam == SETPARAM_LNG) {
|
|
lat = readfloat(buffer, &i);
|
|
} else {
|
|
unsigned char k;
|
|
for (k=i; i<sizeof(gcsinputbuffer); k++)
|
|
if (buffer[k] != '-' && !(buffer[k] >= '0' && buffer[k] <= '9'))
|
|
break;
|
|
buffer[k] = 0;
|
|
lat = atol(buffer+i);
|
|
i = k;
|
|
}
|
|
|
|
temp = get_wp_with_index(cmdindex);
|
|
if (setparam == SETPARAM_ID)
|
|
temp.id = lat;
|
|
else if (setparam == SETPARAM_P1)
|
|
temp.p1 = lat;
|
|
else if (setparam == SETPARAM_ALT)
|
|
temp.alt = lat/100000;
|
|
else if (setparam == SETPARAM_LAT)
|
|
temp.lat = lat;
|
|
else if (setparam == SETPARAM_LNG)
|
|
temp.lng = lat;
|
|
else if (setparam == SETPARAM_P2)
|
|
temp.alt = lat;
|
|
else if (setparam == SETPARAM_P3)
|
|
temp.lat = lat;
|
|
else if (setparam == SETPARAM_P4)
|
|
temp.lng = lat;
|
|
cmdid = temp.id;
|
|
p1 = temp.p1;
|
|
lat = temp.lat;
|
|
lng = temp.lng;
|
|
alt = temp.alt;
|
|
err = 0;
|
|
} else {
|
|
//Process param 1
|
|
for (i=j+1; i<bufferlen; i++) {
|
|
if (buffer[i] >= '0' && buffer[i] <= '9')
|
|
p1 = (p1*10) + buffer[i]-'0';
|
|
else
|
|
break;
|
|
}
|
|
|
|
if (buffer[i] == ' ') {
|
|
//Process altitude
|
|
i++;
|
|
if (strmatch(buffer+i, PSTR("here"))) {
|
|
lat = gps.latitude;
|
|
lng = gps.longitude;
|
|
alt = get_altitude_above_home(); //GPS.altitude;
|
|
err = 0;
|
|
} else {
|
|
alt = readfloat(buffer, &i)/100000;
|
|
|
|
if (buffer[i] == ' ') {
|
|
//Process latitude
|
|
i++;
|
|
lat = readfloat(buffer, &i);
|
|
if (strmatch(buffer+i, PSTR("here"))) {
|
|
lat = gps.latitude;
|
|
lng = gps.longitude;
|
|
err = 0;
|
|
} else {
|
|
if (buffer[i] == ' ') {
|
|
//Process longitude
|
|
i++;
|
|
lng = readfloat(buffer, &i);
|
|
|
|
//TODO: add other command special cases here
|
|
if (cmdid == CMD_LAND_OPTIONS) {
|
|
temp.p1 = p1;
|
|
temp.alt = alt;
|
|
temp.lat = lat / 10000000;
|
|
temp.lng = lng / 10000000;
|
|
}
|
|
|
|
err = 0;
|
|
} else
|
|
print_error(ERR("Error processing set cmd: Could not find longitude parameter"));
|
|
}
|
|
} else
|
|
print_error(ERR("Error processing set cmd: Could not find latitude parameter"));
|
|
}
|
|
} else
|
|
print_error(ERR("Error processing set cmd: Could not find altitude parameter"));
|
|
}
|
|
} else
|
|
print_error(ERR("Error processing set cmd: Could not find command type"));
|
|
} else
|
|
print_error(ERR("Error processing set cmd: Could not find command index"));
|
|
|
|
if (err == 0) {
|
|
temp.id = cmdid;
|
|
temp.p1 = p1;
|
|
temp.lat = lat;
|
|
temp.lng = lng;
|
|
temp.alt = alt;
|
|
|
|
if (cmdindex >= get(PARAM_WP_TOTAL)) {
|
|
set(PARAM_WP_TOTAL, cmdindex+1);
|
|
}
|
|
set_wp_with_index(temp, cmdindex);
|
|
|
|
port->printf_P(PSTR("Set command %d to %d with p1=%d, lat=%ld, lng=%ld, alt=%ld\n"),
|
|
(int)cmdindex, (int)temp.id, (int)temp.p1, temp.lat, temp.lng, temp.alt);
|
|
}
|
|
}
|
|
|
|
/* get_pid_offset_name matches a string expressed in *buffer with a pid keyword and returns the k-array
|
|
gain offset in *offset, and the length of that string expression in *length. If a good keyword
|
|
match is found, 1 is returned; 0 otherwise
|
|
*/
|
|
int
|
|
GCS_DEBUGTERMINAL::get_pid_offset_name(char *buffer, int *offset, unsigned char *length)
|
|
{
|
|
if (strmatch(buffer, PSTR("servoroll"))) {
|
|
*length += 9; *offset = CASE_SERVO_ROLL;
|
|
} else if (strmatch(buffer, PSTR("servopitch"))) {
|
|
*length += 10; *offset = CASE_SERVO_PITCH;
|
|
} else if (strmatch(buffer, PSTR("servorudder"))) {
|
|
*length += 11; *offset = CASE_SERVO_RUDDER;
|
|
} else if (strmatch(buffer, PSTR("navroll"))) {
|
|
*length += 7; *offset = CASE_NAV_ROLL;
|
|
} else if (strmatch(buffer, PSTR("navpitchasp"))) {
|
|
*length += 11; *offset = CASE_NAV_PITCH_ASP;
|
|
} else if (strmatch(buffer, PSTR("navpitchalt"))) {
|
|
*length += 11; *offset = CASE_NAV_PITCH_ALT;
|
|
} else if (strmatch(buffer, PSTR("throttlete"))) {
|
|
*length += 10; *offset = CASE_TE_THROTTLE;
|
|
} else if (strmatch(buffer, PSTR("throttlealt"))) {
|
|
*length += 11; *offset = CASE_ALT_THROTTLE;
|
|
} else {
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
|
|
/* print_rlocation prints the relative location of the specified waypoint from the plane in easy-to-read cartesian format
|
|
*/
|
|
void
|
|
GCS_DEBUGTERMINAL::print_rlocation(Location *wp)
|
|
{
|
|
//float x, y;
|
|
//y = (float)(wp->lat - current_loc.lat) * 0.0111194927;
|
|
//x = (float)(wp->lng - current_loc.lng) * cos(radians(current_loc.lat)) * 0.0111194927;
|
|
BetterStream *port = _port;
|
|
int x, y;
|
|
y = (wp->lat - current_loc.lat) * 0.0111194927;
|
|
x = (wp->lng - current_loc.lng) * cos(radians(current_loc.lat)) * 0.0111194927;
|
|
port->printf_P(PSTR("dP = <%d%c, %d%c, %.1f> meters\n"),
|
|
abs(y), (y >= 0 ? 'N' : 'S'),
|
|
abs(x), (x >= 0 ? 'E' : 'W'),
|
|
(float)(wp->alt - current_loc.alt) / 100);
|
|
}
|
|
|
|
/* print_error prints an error message if the user sends an invalid command
|
|
*/
|
|
void
|
|
GCS_DEBUGTERMINAL::print_error(const char *msg)
|
|
{
|
|
BetterStream *port = _port;
|
|
|
|
if (msg == 0)
|
|
port->println_P(PSTR("ERROR: Invalid command"));
|
|
else
|
|
port->println_P(msg);
|
|
}
|
|
|
|
|
|
void
|
|
GCS_DEBUGTERMINAL::send_text(uint8_t severity, const char *str)
|
|
{
|
|
BetterStream *port = _port;
|
|
|
|
if(severity >= DEBUG_LEVEL){
|
|
port->print_P(PSTR("MSG: "));
|
|
port->println(str);
|
|
}
|
|
}
|
|
|
|
void
|
|
GCS_DEBUGTERMINAL::send_message(uint8_t id, uint32_t param)
|
|
{
|
|
switch(id) {
|
|
case MSG_MODE_CHANGE:
|
|
print_control_mode();
|
|
break;
|
|
|
|
case MSG_HEARTBEAT:
|
|
if (report_heartbeat)
|
|
print_control_mode();
|
|
break;
|
|
|
|
case MSG_ATTITUDE:
|
|
if (report_attitude)
|
|
print_attitude();
|
|
break;
|
|
|
|
case MSG_LOCATION:
|
|
if (report_location)
|
|
print_position();
|
|
if (first_location == 0) {
|
|
send_text(0,"First location received");
|
|
first_location = 1;
|
|
}
|
|
break;
|
|
|
|
case MSG_CPU_LOAD:
|
|
if (report_cpu_load) {
|
|
_port->printf_P(PSTR("MSG: load %ld%%\n"), param);
|
|
}
|
|
break;
|
|
|
|
case MSG_COMMAND_LIST:
|
|
struct Location cmd = get_wp_with_index(param);
|
|
print_command(&cmd, param);
|
|
break;
|
|
}
|
|
}
|
|
|
|
void
|
|
GCS_DEBUGTERMINAL::print_current_waypoints()
|
|
{
|
|
_port->printf_P(PSTR("Current waypoints:"
|
|
" Prev:\t%ld,\t%ld\t%ld\n"
|
|
" Cur: \t%ld,\t%ld\t%ld\n"
|
|
" Next:%d\t%ld,\t%ld\t%ld\n"),
|
|
prev_WP.lat, prev_WP.lng, prev_WP.alt,
|
|
current_loc.lat, current_loc.lng, current_loc.alt,
|
|
(int)get(PARAM_WP_INDEX), next_WP.lat, next_WP.lng, next_WP.alt);
|
|
}
|
|
|
|
void
|
|
GCS_DEBUGTERMINAL::print_position(void)
|
|
{
|
|
recalc_climb_rate();
|
|
_port->printf_P(PSTR("Pos: %ld, %ld, %ldcm, %ldcm/s GS, %d cm/s AS, %d cm above home, %d? climb, %ldm from wp, %d%% throttle, %ld alt err\n"),
|
|
current_loc.lat, current_loc.lng, current_loc.alt,
|
|
gps.ground_speed,
|
|
airspeed,
|
|
get_altitude_above_home(),
|
|
climb_rate,
|
|
wp_distance,
|
|
get(PARAM_TRIM_THROTTLE),
|
|
altitude_error);
|
|
}
|
|
|
|
void
|
|
GCS_DEBUGTERMINAL::print_attitude(void)
|
|
{
|
|
_port->printf_P(PSTR("Att: %u roll_in, %u pitch in, %u throttle_in, "
|
|
"%ld roll_sensor, %ld pitch_sensor, "
|
|
"%ld ground_course, %ld target_bearing, "
|
|
"%ld nav_roll, %d loiter_sum, "
|
|
"%d roll servo_out, %d pitch_servo_out\n"),
|
|
radio_in[CH_ROLL], radio_in[CH_PITCH], radio_in[CH_THROTTLE],
|
|
dcm.roll_sensor, dcm.pitch_sensor,
|
|
gps.ground_course, target_bearing,
|
|
nav_roll, loiter_sum,
|
|
servo_out[CH_ROLL], servo_out[CH_PITCH]);
|
|
}
|
|
|
|
// required by Groundstation to plot lateral tracking course
|
|
void
|
|
GCS_DEBUGTERMINAL::print_new_wp_info()
|
|
{
|
|
_port->printf_P(PSTR("New waypt (%d): %ld, %ld, %ldm -> "
|
|
"%ld, %ld, %ldm; "
|
|
"%ldm dist, %u roll trim, %u pitch trim\n"),
|
|
(int)get(PARAM_WP_INDEX),
|
|
prev_WP.lat, prev_WP.lng, prev_WP.alt,
|
|
next_WP.lat, next_WP.lng, next_WP.alt,
|
|
wp_totalDistance,
|
|
radio_trim(CH_ROLL), radio_trim(CH_PITCH));
|
|
}
|
|
|
|
void
|
|
GCS_DEBUGTERMINAL::print_control_mode(void)
|
|
{
|
|
BetterStream *port = _port;
|
|
|
|
switch (control_mode){
|
|
case MANUAL:
|
|
port->println_P(PSTR("##MANUAL"));
|
|
break;
|
|
case STABILIZE:
|
|
port->println_P(PSTR("##STABILIZE"));
|
|
break;
|
|
case FLY_BY_WIRE_A:
|
|
port->println_P(PSTR("##FBW A"));
|
|
break;
|
|
case FLY_BY_WIRE_B:
|
|
port->println_P(PSTR("##FBW B"));
|
|
break;
|
|
case AUTO:
|
|
port->println_P(PSTR("##AUTO"));
|
|
break;
|
|
case RTL:
|
|
port->println_P(PSTR("##RTL"));
|
|
break;
|
|
case LOITER:
|
|
port->println_P(PSTR("##LOITER"));
|
|
break;
|
|
case 98:
|
|
port->println_P(PSTR("##Air Start Complete"));
|
|
break;
|
|
case 99:
|
|
port->println_P(PSTR("##Ground Start Complete"));
|
|
break;
|
|
}
|
|
}
|
|
|
|
void
|
|
GCS_DEBUGTERMINAL::print_tuning(void)
|
|
{
|
|
_port->printf_P(PSTR("TUN:%d, %ld, %ld, %d, %ld, %ld\n"),
|
|
servo_out[CH_ROLL], nav_roll / 100, dcm.roll_sensor / 100,
|
|
servo_out[CH_PITCH], nav_pitch / 100, dcm.pitch_sensor / 100);
|
|
}
|
|
|
|
void
|
|
GCS_DEBUGTERMINAL::print_command_id(byte id)
|
|
{
|
|
BetterStream *port = _port;
|
|
|
|
switch (id) {
|
|
// Command IDs - Must
|
|
case CMD_BLANK: port->print_P(PSTR("CMD_BLANK")); break;
|
|
case CMD_WAYPOINT: port->print_P(PSTR("waypoint")); break;
|
|
case CMD_LOITER: port->print_P(PSTR("loiter")); break;
|
|
case CMD_LOITER_N_TURNS: port->print_P(PSTR("loiternturns")); break;
|
|
case CMD_LOITER_TIME: port->print_P(PSTR("loitertime")); break;
|
|
case CMD_RTL: port->print_P(PSTR("rtl")); break;
|
|
case CMD_LAND: port->print_P(PSTR("land")); break;
|
|
case CMD_TAKEOFF: port->print_P(PSTR("takeoff")); break;
|
|
|
|
// Command IDs - May
|
|
case CMD_DELAY: port->print_P(PSTR("delay")); break;
|
|
case CMD_CLIMB: port->print_P(PSTR("climb")); break;
|
|
case CMD_LAND_OPTIONS: port->print_P(PSTR("landoptions")); break;
|
|
|
|
// Command IDs - Now
|
|
case CMD_RESET_INDEX: port->print_P(PSTR("resetindex")); break;
|
|
case CMD_GOTO_INDEX: port->print_P(PSTR("CMD_GOTO_INDEX")); break;
|
|
case CMD_GETVAR_INDEX: port->print_P(PSTR("CMD_GETVAR_INDEX")); break;
|
|
case CMD_SENDVAR_INDEX: port->print_P(PSTR("CMD_SENDVAR_INDEX")); break;
|
|
case CMD_TELEMETRY: port->print_P(PSTR("CMD_TELEMETRY")); break;
|
|
|
|
case CMD_THROTTLE_CRUISE: port->print_P(PSTR("throttlecruise")); break;
|
|
case CMD_AIRSPEED_CRUISE: port->print_P(PSTR("CMD_AIRSPEED_CRUISE")); break;
|
|
case CMD_RESET_HOME: port->print_P(PSTR("resethome")); break;
|
|
|
|
case CMD_KP_GAIN: port->print_P(PSTR("CMD_KP_GAIN")); break;
|
|
case CMD_KI_GAIN: port->print_P(PSTR("CMD_KI_GAIN")); break;
|
|
case CMD_KD_GAIN: port->print_P(PSTR("CMD_KD_GAIN")); break;
|
|
case CMD_KI_MAX: port->print_P(PSTR("CMD_KI_MAX")); break;
|
|
case CMD_KFF_GAIN: port->print_P(PSTR("CMD_KFF_GAIN")); break;
|
|
|
|
case CMD_RADIO_TRIM: port->print_P(PSTR("CMD_RADIO_TRIM")); break;
|
|
case CMD_RADIO_MAX: port->print_P(PSTR("CMD_RADIO_MAX")); break;
|
|
case CMD_RADIO_MIN: port->print_P(PSTR("CMD_RADIO_MIN")); break;
|
|
case CMD_ELEVON_TRIM: port->print_P(PSTR("CMD_ELEVON_TRIM")); break;
|
|
|
|
case CMD_INDEX: port->print_P(PSTR("index")); break;
|
|
case CMD_REPEAT: port->print_P(PSTR("CMD_REPEAT")); break;
|
|
case CMD_RELAY: port->print_P(PSTR("relay")); break;
|
|
case CMD_SERVO: port->print_P(PSTR("servo")); break;
|
|
|
|
default: port->print(id,DEC);
|
|
}
|
|
}
|
|
|
|
void
|
|
GCS_DEBUGTERMINAL::print_command(struct Location *cmd,byte index)
|
|
{
|
|
BetterStream *port = _port;
|
|
|
|
port->print_P(PSTR(" command #: "));
|
|
port->print(index, DEC);
|
|
port->print_P(PSTR(" id: "));
|
|
print_command_id(cmd->id);
|
|
|
|
port->print_P(PSTR(" p1: "));
|
|
port->print(cmd->p1,DEC);
|
|
port->print_P(PSTR(" p2/alt: "));
|
|
port->print(cmd->alt,DEC);
|
|
port->print_P(PSTR(" p3/lat: "));
|
|
port->print(cmd->lat,DEC);
|
|
port->print_P(PSTR(" p4/lng: "));
|
|
port->println(cmd->lng,DEC);
|
|
}
|
|
|
|
void
|
|
GCS_DEBUGTERMINAL::print_commands()
|
|
{
|
|
print_commands(1, get(PARAM_WP_TOTAL));
|
|
}
|
|
|
|
void
|
|
GCS_DEBUGTERMINAL::print_commands(unsigned char i1, unsigned char i2)
|
|
{
|
|
BetterStream *port = _port;
|
|
|
|
port->println_P(PSTR("Commands in memory:"));
|
|
port->print_P(PSTR(" "));
|
|
port->print(get(PARAM_WP_TOTAL), DEC);
|
|
port->println_P(PSTR(" commands total"));
|
|
// create a location struct to hold the temp Waypoints for printing
|
|
//Location tmp;
|
|
port->println_P(PSTR(" Home: "));
|
|
struct Location cmd = get_wp_with_index(0);
|
|
print_command(&cmd, 0);
|
|
port->println_P(PSTR(" Commands: "));
|
|
|
|
for (int i=i1; i<=i2; i++){
|
|
cmd = get_wp_with_index(i);
|
|
print_command(&cmd, i);
|
|
delay(10);
|
|
}
|
|
}
|
|
|
|
void
|
|
GCS_DEBUGTERMINAL::print_gains()
|
|
{
|
|
BetterStream *port = _port;
|
|
unsigned char i;
|
|
|
|
port->println_P(PSTR("PID gains \tP \tI \tD \tIMax)"));
|
|
port->print_P(PSTR("Servo roll \t")); print_gain(CASE_SERVO_ROLL);
|
|
port->print_P(PSTR("Servo pitch \t")); print_gain(CASE_SERVO_PITCH);
|
|
port->print_P(PSTR("Servo yaw \t")); print_gain(CASE_SERVO_RUDDER);
|
|
port->print_P(PSTR("Nav roll \t")); print_gain(CASE_NAV_ROLL);
|
|
|
|
port->print_P(PSTR("Nav pitch \t"));
|
|
if (AIRSPEED_SENSOR)
|
|
print_gain(CASE_NAV_PITCH_ASP);
|
|
else
|
|
print_gain(CASE_NAV_PITCH_ALT);
|
|
|
|
port->print_P(PSTR("Nav throttle\t"));
|
|
if (AIRSPEED_SENSOR)
|
|
print_gain(CASE_TE_THROTTLE);
|
|
else
|
|
print_gain(CASE_ALT_THROTTLE);
|
|
|
|
port->println_P(PSTR("Feed-forward gains"));
|
|
port->print_P(PSTR("Pitch compensation\t")); port->println(get(PARAM_KFF_PTCHCOMP),3);
|
|
port->print_P(PSTR("Rudder mix \t")); port->println(get(PARAM_KFF_RDDRMIX),3);
|
|
port->print_P(PSTR("Pitch to throttle \t")); port->println(get(PARAM_KFF_PTCH2THR),3);
|
|
//port->print_P(PSTR("Throttle to pitch \t")); port->println(kff[CASE_T_TO_P],3);
|
|
}
|
|
|
|
void
|
|
GCS_DEBUGTERMINAL::print_gain(unsigned char g)
|
|
{
|
|
BetterStream *port = _port;
|
|
|
|
port->print(pid_index[g]->kP(),3);
|
|
port->print_P(PSTR("\t"));
|
|
port->print(pid_index[g]->kI(),3);
|
|
port->print_P(PSTR("\t"));
|
|
port->print(pid_index[g]->kD(),3);
|
|
port->print_P(PSTR("\t"));
|
|
port->println(pid_index[g]->imax(),DEC);
|
|
}
|
|
|
|
void
|
|
GCS_DEBUGTERMINAL::print_gain_keyword_error()
|
|
{
|
|
print_error(ERR("ERROR: Did not recognize keyword; type k -? for more information"));
|
|
}
|
|
|
|
void
|
|
GCS_DEBUGTERMINAL::print_PID(long PID_error, long dt, float scaler, float derivative, float integrator, float last_error)
|
|
{
|
|
BetterStream *port = _port;
|
|
|
|
port->print_P(PSTR("P = ")); port->print(pid_index[display_PID]->kP() * scaler * (float)PID_error,2);
|
|
port->print_P(PSTR(",\tI = ")); port->print(integrator,2);
|
|
port->print_P(PSTR(",\tD = ")); port->print(pid_index[display_PID]->kD() * scaler * derivative,2);
|
|
port->print_P(PSTR("\terrors = {")); port->print(PID_error,DEC);
|
|
port->print_P(PSTR(", ")); port->print(last_error,DEC);
|
|
port->print_P(PSTR("} with dt = ")); port->println(dt,DEC);
|
|
|
|
display_PID = -1;
|
|
}
|
|
|
|
#endif // GCS_PROTOCOL_DEBUGTERMINAL
|