// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*- #if CLI_ENABLED == ENABLED // Functions called from the setup menu static int8_t setup_radio (uint8_t argc, const Menu::arg *argv); static int8_t setup_show (uint8_t argc, const Menu::arg *argv); static int8_t setup_factory (uint8_t argc, const Menu::arg *argv); static int8_t setup_level (uint8_t argc, const Menu::arg *argv); #if !defined( __AVR_ATmega1280__ ) static int8_t setup_accel_scale (uint8_t argc, const Menu::arg *argv); static int8_t setup_set (uint8_t argc, const Menu::arg *argv); #endif static int8_t setup_erase (uint8_t argc, const Menu::arg *argv); static int8_t setup_compass (uint8_t argc, const Menu::arg *argv); // Command/function table for the setup menu static const struct Menu::command setup_menu_commands[] PROGMEM = { // command function called // ======= =============== {"reset", setup_factory}, {"radio", setup_radio}, {"level", setup_level}, #if !defined( __AVR_ATmega1280__ ) {"accel", setup_accel_scale}, #endif {"compass", setup_compass}, {"show", setup_show}, #if !defined( __AVR_ATmega1280__ ) {"set", setup_set}, #endif {"erase", setup_erase}, }; // Create the setup menu object. MENU(setup_menu, "setup", setup_menu_commands); // Called from the top-level menu to run the setup menu. static int8_t setup_mode(uint8_t argc, const Menu::arg *argv) { // Give the user some guidance cliSerial->printf_P(PSTR("Setup Mode\n" "\n" "IMPORTANT: if you have not previously set this system up, use the\n" "'reset' command to initialize the EEPROM to sensible default values\n" "and then the 'radio' command to configure for your radio.\n" "\n")); // Run the setup menu. When the menu exits, we will return to the main menu. setup_menu.run(); return 0; } // Print the current configuration. // Called by the setup menu 'show' command. static int8_t setup_show(uint8_t argc, const Menu::arg *argv) { #if !defined( __AVR_ATmega1280__ ) AP_Param *param; ap_var_type type; //If a parameter name is given as an argument to show, print only that parameter if(argc>=2) { param=AP_Param::find(argv[1].str, &type); if(!param) { cliSerial->printf_P(PSTR("Parameter not found: '%s'\n"), argv[1]); return 0; } AP_Param::show(param, argv[1].str, type, cliSerial); return 0; } #endif AP_Param::show_all(cliSerial); return(0); } #if !defined( __AVR_ATmega1280__ ) //Set a parameter to a specified value. It will cast the value to the current type of the //parameter and make sure it fits in case of INT8 and INT16 static int8_t setup_set(uint8_t argc, const Menu::arg *argv) { int8_t value_int8; int16_t value_int16; AP_Param *param; enum ap_var_type p_type; if(argc!=3) { cliSerial->printf_P(PSTR("Invalid command. Usage: set \n")); return 0; } param = AP_Param::find(argv[1].str, &p_type); if(!param) { cliSerial->printf_P(PSTR("Param not found: %s\n"), argv[1].str); return 0; } switch(p_type) { case AP_PARAM_INT8: value_int8 = (int8_t)(argv[2].i); if(argv[2].i!=value_int8) { cliSerial->printf_P(PSTR("Value out of range for type INT8\n")); return 0; } ((AP_Int8*)param)->set_and_save(value_int8); break; case AP_PARAM_INT16: value_int16 = (int16_t)(argv[2].i); if(argv[2].i!=value_int16) { cliSerial->printf_P(PSTR("Value out of range for type INT16\n")); return 0; } ((AP_Int16*)param)->set_and_save(value_int16); break; //int32 and float don't need bounds checking, just use the value provoded by Menu::arg case AP_PARAM_INT32: ((AP_Int32*)param)->set_and_save(argv[2].i); break; case AP_PARAM_FLOAT: ((AP_Float*)param)->set_and_save(argv[2].f); break; default: cliSerial->printf_P(PSTR("Cannot set parameter of type %d.\n"), p_type); break; } return 0; } #endif // Initialise the EEPROM to 'factory' settings (mostly defined in APM_Config.h or via defaults). // Called by the setup menu 'factoryreset' command. static int8_t setup_factory(uint8_t argc, const Menu::arg *argv) { int c; cliSerial->printf_P(PSTR("\nType 'Y' and hit Enter to perform factory reset, any other key to abort: ")); do { c = cliSerial->read(); } while (-1 == c); if (('y' != c) && ('Y' != c)) return(-1); AP_Param::erase_all(); cliSerial->printf_P(PSTR("\nFACTORY RESET complete - please reset board to continue")); //default_flight_modes(); // This will not work here. Replacement code located in init_ardupilot() for (;; ) { } // note, cannot actually return here return(0); } // Perform radio setup. // Called by the setup menu 'radio' command. static int8_t setup_radio(uint8_t argc, const Menu::arg *argv) { cliSerial->printf_P(PSTR("\n\nRadio Setup:\n")); uint8_t i; for(i = 0; i < 100; i++) { delay(20); read_radio(); } if(channel_roll->radio_in < 500) { while(1) { cliSerial->printf_P(PSTR("\nNo radio; Check connectors.")); delay(1000); // stop here } } channel_roll->radio_min = channel_roll->radio_in; channel_pitch->radio_min = channel_pitch->radio_in; channel_throttle->radio_min = channel_throttle->radio_in; channel_rudder->radio_min = channel_rudder->radio_in; g.rc_5.radio_min = g.rc_5.radio_in; g.rc_6.radio_min = g.rc_6.radio_in; g.rc_7.radio_min = g.rc_7.radio_in; g.rc_8.radio_min = g.rc_8.radio_in; channel_roll->radio_max = channel_roll->radio_in; channel_pitch->radio_max = channel_pitch->radio_in; channel_throttle->radio_max = channel_throttle->radio_in; channel_rudder->radio_max = channel_rudder->radio_in; g.rc_5.radio_max = g.rc_5.radio_in; g.rc_6.radio_max = g.rc_6.radio_in; g.rc_7.radio_max = g.rc_7.radio_in; g.rc_8.radio_max = g.rc_8.radio_in; channel_roll->radio_trim = channel_roll->radio_in; channel_pitch->radio_trim = channel_pitch->radio_in; channel_rudder->radio_trim = channel_rudder->radio_in; g.rc_5.radio_trim = 1500; g.rc_6.radio_trim = 1500; g.rc_7.radio_trim = 1500; g.rc_8.radio_trim = 1500; cliSerial->printf_P(PSTR("\nMove all controls to each extreme. Hit Enter to save: \n")); while(1) { delay(20); // Filters radio input - adjust filters in the radio.pde file // ---------------------------------------------------------- read_radio(); channel_roll->update_min_max(); channel_pitch->update_min_max(); channel_throttle->update_min_max(); channel_rudder->update_min_max(); g.rc_5.update_min_max(); g.rc_6.update_min_max(); g.rc_7.update_min_max(); g.rc_8.update_min_max(); if(cliSerial->available() > 0) { while (cliSerial->available() > 0) { cliSerial->read(); } channel_roll->save_eeprom(); channel_pitch->save_eeprom(); channel_throttle->save_eeprom(); channel_rudder->save_eeprom(); g.rc_5.save_eeprom(); g.rc_6.save_eeprom(); g.rc_7.save_eeprom(); g.rc_8.save_eeprom(); print_done(); break; } } trim_radio(); report_radio(); return(0); } static int8_t setup_erase(uint8_t argc, const Menu::arg *argv) { int c; cliSerial->printf_P(PSTR("\nType 'Y' and hit Enter to erase all waypoint and parameter data, any other key to abort: ")); do { c = cliSerial->read(); } while (-1 == c); if (('y' != c) && ('Y' != c)) return(-1); zero_eeprom(); return 0; } static int8_t setup_level(uint8_t argc, const Menu::arg *argv) { startup_INS_ground(true); return 0; } #if !defined( __AVR_ATmega1280__ ) /* handle full accelerometer calibration via user dialog */ static int8_t setup_accel_scale(uint8_t argc, const Menu::arg *argv) { float trim_roll, trim_pitch; cliSerial->println_P(PSTR("Initialising gyros")); ahrs.init(); ahrs.set_fly_forward(true); ahrs.set_wind_estimation(true); ins.init(AP_InertialSensor::COLD_START, ins_sample_rate); AP_InertialSensor_UserInteractStream interact(hal.console); bool success = ins.calibrate_accel(&interact, trim_roll, trim_pitch); if (success) { // reset ahrs's trim to suggested values from calibration routine ahrs.set_trim(Vector3f(trim_roll, trim_pitch, 0)); } report_ins(); return(0); } #endif static int8_t setup_compass(uint8_t argc, const Menu::arg *argv) { if (!strcmp_P(argv[1].str, PSTR("on"))) { if (!compass.init()) { cliSerial->println_P(PSTR("Compass initialisation failed!")); g.compass_enabled = false; } else { g.compass_enabled = true; } } else if (!strcmp_P(argv[1].str, PSTR("off"))) { g.compass_enabled = false; } else if (!strcmp_P(argv[1].str, PSTR("reset"))) { compass.set_and_save_offsets(0,0,0,0); } else { cliSerial->printf_P(PSTR("\nOptions:[on,off,reset]\n")); report_compass(); return 0; } g.compass_enabled.save(); report_compass(); return 0; } /***************************************************************************/ // CLI reports /***************************************************************************/ static void report_radio() { //print_blanks(2); cliSerial->printf_P(PSTR("Radio\n")); print_divider(); // radio print_radio_values(); print_blanks(2); } static void report_ins() { //print_blanks(2); cliSerial->printf_P(PSTR("INS\n")); print_divider(); print_gyro_offsets(); print_accel_offsets_and_scaling(); print_blanks(2); } static void report_compass() { //print_blanks(2); cliSerial->printf_P(PSTR("Compass: ")); switch (compass.product_id) { case AP_COMPASS_TYPE_HMC5883L: cliSerial->println_P(PSTR("HMC5883L")); break; case AP_COMPASS_TYPE_HMC5843: cliSerial->println_P(PSTR("HMC5843")); break; case AP_COMPASS_TYPE_HIL: cliSerial->println_P(PSTR("HIL")); break; case AP_COMPASS_TYPE_PX4: cliSerial->println_P(PSTR("PX4")); break; case AP_COMPASS_TYPE_VRBRAIN: cliSerial->println_P(PSTR("VRBRAIN")); break; default: cliSerial->println_P(PSTR("(unknown)")); break; } print_divider(); print_enabled(g.compass_enabled); Vector3f offsets = compass.get_offsets(); // mag offsets cliSerial->printf_P(PSTR("Mag offsets: %4.4f, %4.4f, %4.4f\n"), offsets.x, offsets.y, offsets.z); print_blanks(2); } /***************************************************************************/ // CLI utilities /***************************************************************************/ static void print_radio_values() { cliSerial->printf_P(PSTR("CH1: %d | %d | %d\n"), (int)channel_roll->radio_min, (int)channel_roll->radio_trim, (int)channel_roll->radio_max); cliSerial->printf_P(PSTR("CH2: %d | %d | %d\n"), (int)channel_pitch->radio_min, (int)channel_pitch->radio_trim, (int)channel_pitch->radio_max); cliSerial->printf_P(PSTR("CH3: %d | %d | %d\n"), (int)channel_throttle->radio_min, (int)channel_throttle->radio_trim, (int)channel_throttle->radio_max); cliSerial->printf_P(PSTR("CH4: %d | %d | %d\n"), (int)channel_rudder->radio_min, (int)channel_rudder->radio_trim, (int)channel_rudder->radio_max); cliSerial->printf_P(PSTR("CH5: %d | %d | %d\n"), (int)g.rc_5.radio_min, (int)g.rc_5.radio_trim, (int)g.rc_5.radio_max); cliSerial->printf_P(PSTR("CH6: %d | %d | %d\n"), (int)g.rc_6.radio_min, (int)g.rc_6.radio_trim, (int)g.rc_6.radio_max); cliSerial->printf_P(PSTR("CH7: %d | %d | %d\n"), (int)g.rc_7.radio_min, (int)g.rc_7.radio_trim, (int)g.rc_7.radio_max); cliSerial->printf_P(PSTR("CH8: %d | %d | %d\n"), (int)g.rc_8.radio_min, (int)g.rc_8.radio_trim, (int)g.rc_8.radio_max); } static void print_done() { cliSerial->printf_P(PSTR("\nSaved Settings\n\n")); } static void print_blanks(int16_t num) { while(num > 0) { num--; cliSerial->println(); } } static void print_divider(void) { for (int16_t i = 0; i < 40; i++) { cliSerial->printf_P(PSTR("-")); } cliSerial->println(); } static void zero_eeprom(void) { uint8_t b = 0; cliSerial->printf_P(PSTR("\nErasing EEPROM\n")); for (uint16_t i = 0; i < HAL_STORAGE_SIZE_AVAILABLE; i++) { hal.storage->write_byte(i, b); } cliSerial->printf_P(PSTR("done\n")); } static void print_enabled(bool b) { if(b) cliSerial->printf_P(PSTR("en")); else cliSerial->printf_P(PSTR("dis")); cliSerial->printf_P(PSTR("abled\n")); } static void print_accel_offsets_and_scaling(void) { Vector3f accel_offsets = ins.get_accel_offsets(); Vector3f accel_scale = ins.get_accel_scale(); cliSerial->printf_P(PSTR("Accel offsets: %4.2f, %4.2f, %4.2f\tscale: %4.2f, %4.2f, %4.2f\n"), (float)accel_offsets.x, // Pitch (float)accel_offsets.y, // Roll (float)accel_offsets.z, // YAW (float)accel_scale.x, // Pitch (float)accel_scale.y, // Roll (float)accel_scale.z); // YAW } static void print_gyro_offsets(void) { Vector3f gyro_offsets = ins.get_gyro_offsets(); cliSerial->printf_P(PSTR("Gyro offsets: %4.2f, %4.2f, %4.2f\n"), (float)gyro_offsets.x, (float)gyro_offsets.y, (float)gyro_offsets.z); } #endif // CLI_ENABLED