/* * ControllerQuad.h * * Created on: Jun 30, 2011 * Author: jgoppert */ #ifndef CONTROLLERQUAD_H_ #define CONTROLLERQUAD_H_ #include "../APO/AP_Controller.h" #include "../APO/AP_BatteryMonitor.h" #include "../APO/AP_ArmingMechanism.h" namespace apo { class ControllerQuad: public AP_Controller { public: ControllerQuad(AP_Navigator * nav, AP_Guide * guide, AP_Board * board) : AP_Controller(nav, guide, board, new AP_ArmingMechanism(board,this,ch_thrust,ch_yaw,0.1,-0.9,0.9), ch_mode, k_cntrl), pidRoll(new AP_Var_group(k_pidRoll, PSTR("ROLL_")), 1, PID_ATT_P, PID_ATT_I, PID_ATT_D, PID_ATT_AWU, PID_ATT_LIM), pidPitch(new AP_Var_group(k_pidPitch, PSTR("PITCH_")), 1, PID_ATT_P, PID_ATT_I, PID_ATT_D, PID_ATT_AWU, PID_ATT_LIM), pidYaw(new AP_Var_group(k_pidYaw, PSTR("YAW_")), 1, PID_YAWPOS_P, PID_YAWPOS_I, PID_YAWPOS_D, PID_YAWPOS_AWU, PID_YAWPOS_LIM), pidYawRate(new AP_Var_group(k_pidYawRate, PSTR("YAWRT_")), 1, PID_YAWSPEED_P, PID_YAWSPEED_I, PID_YAWSPEED_D, PID_YAWSPEED_AWU, PID_YAWSPEED_LIM, PID_YAWSPEED_DFCUT), pidSpeed(new AP_Var_group(k_pidSpeed, PSTR("SPD_")), 1, PID_SPD_P, PID_SPD_I, PID_SPD_D, PID_SPD_AWU, PID_SPD_LIM, PID_SPD_DFCUT), pidTilt(new AP_Var_group(k_pidTilt, PSTR("TILT_")), 1, PID_TILT_P, PID_TILT_I, PID_TILT_D, PID_TILT_AWU, PID_TILT_LIM, PID_TILT_DFCUT), pidPD(new AP_Var_group(k_pidPD, PSTR("DOWN_")), 1, PID_POS_Z_P, PID_POS_Z_I, PID_POS_Z_D, PID_POS_Z_AWU, PID_POS_Z_LIM), _thrustMix(0), _pitchMix(0), _rollMix(0), _yawMix(0), _cmdRoll(0), _cmdPitch(0), _cmdYawRate(0) { _board->debug->println_P(PSTR("initializing quad controller")); /* * allocate radio channels * the order of the channels has to match the enumeration above */ _board->rc.push_back( new AP_RcChannel(k_chMode, PSTR("MODE_"), board->radio, 5, 1100, 1500, 1900, RC_MODE_IN, false)); _board->rc.push_back( new AP_RcChannel(k_chRight, PSTR("RIGHT_"), board->radio, 0, 1100, 1100, 1900, RC_MODE_OUT, false)); _board->rc.push_back( new AP_RcChannel(k_chLeft, PSTR("LEFT_"), board->radio, 1, 1100, 1100, 1900, RC_MODE_OUT, false)); _board->rc.push_back( new AP_RcChannel(k_chFront, PSTR("FRONT_"), board->radio, 2, 1100, 1100, 1900, RC_MODE_OUT, false)); _board->rc.push_back( new AP_RcChannel(k_chBack, PSTR("BACK_"), board->radio, 3, 1100, 1100, 1900, RC_MODE_OUT, false)); _board->rc.push_back( new AP_RcChannel(k_chRoll, PSTR("ROLL_"), board->radio, 0, 1100, 1500, 1900, RC_MODE_IN, false)); _board->rc.push_back( new AP_RcChannel(k_chPitch, PSTR("PITCH_"), board->radio, 1, 1100, 1500, 1900, RC_MODE_IN, false)); _board->rc.push_back( new AP_RcChannel(k_chThr, PSTR("THRUST_"), board->radio, 2, 1100, 1100, 1900, RC_MODE_IN, false)); _board->rc.push_back( new AP_RcChannel(k_chYaw, PSTR("YAW_"), board->radio, 3, 1100, 1500, 1900, RC_MODE_IN, false)); } private: // methods void manualLoop(const float dt) { setAllRadioChannelsManually(); _cmdRoll = -0.5 * _board->rc[ch_roll]->getPosition(); _cmdPitch = -0.5 * _board->rc[ch_pitch]->getPosition(); _cmdYawRate = -1 * _board->rc[ch_yaw]->getPosition(); _thrustMix = _board->rc[ch_thrust]->getPosition(); autoAttitudeLoop(dt); } void autoLoop(const float dt) { autoPositionLoop(dt); autoAttitudeLoop(dt); } void autoPositionLoop(float dt) { float cmdSpeed = pidSpeed.update(_guide->getGroundSpeedError(),dt); float cmdDown = pidPD.update(_guide->getPDError(),-_nav->getVD(),dt); // tilt based control float cmdTilt = pidTilt.update(_guide->getDistanceToNextWaypoint(),dt); _cmdPitch = -cmdTilt*cos(_guide->getHeadingError()); _cmdRoll = cmdTilt*sin(_guide->getHeadingError()); // add velocity based control _cmdPitch -= cmdSpeed*cos(_nav->getRelativeCourseOverGround()); _cmdRoll += cmdSpeed*sin(_nav->getRelativeCourseOverGround()); _cmdYawRate = pidYaw.update(_guide->getYawError(),-_nav->getYawRate(),dt); // always points to next waypoint _thrustMix = THRUST_HOVER_OFFSET - cmdDown; // "thrust-trim-adjust" if (fabs(_cmdRoll) > 0.5) _thrustMix *= 1.13949393; else _thrustMix /= cos(_cmdRoll); if (fabs(_cmdPitch) > 0.5) _thrustMix *= 1.13949393; else _thrustMix /= cos(_cmdPitch); // debug for position loop //_board->debug->printf_P(PSTR("cmd: tilt(%f), down(%f), pitch(%f), roll(%f)\n"),cmdTilt,cmdDown,_cmdPitch,_cmdRoll); } void autoAttitudeLoop(float dt) { _rollMix = pidRoll.update(_cmdRoll - _nav->getRoll(), -_nav->getRollRate(), dt); _pitchMix = pidPitch.update(_cmdPitch - _nav->getPitch(), -_nav->getPitchRate(), dt); _yawMix = pidYawRate.update(_cmdYawRate - _nav->getYawRate(), dt); } void setMotors() { // turn all motors off if below 0.1 throttle if (fabs(_board->rc[ch_thrust]->getRadioPosition()) < 0.1) { setAllRadioChannelsToNeutral(); } else { _board->rc[ch_right]->setPosition(_thrustMix - _rollMix + _yawMix); _board->rc[ch_left]->setPosition(_thrustMix + _rollMix + _yawMix); _board->rc[ch_front]->setPosition(_thrustMix + _pitchMix - _yawMix); _board->rc[ch_back]->setPosition(_thrustMix - _pitchMix - _yawMix); } } void handleFailsafe() { // turn off setMode(MAV_MODE_LOCKED); } // attributes /** * note that these are not the controller radio channel numbers, they are just * unique keys so they can be reaccessed from the board rc vector */ enum { ch_mode = 0, // note scicoslab channels set mode, left, right, front, back order ch_right, ch_left, ch_front, ch_back, ch_roll, ch_pitch, ch_thrust, ch_yaw }; // must match channel enum enum { k_chMode = k_radioChannelsStart, k_chRight, k_chLeft, k_chFront, k_chBack, k_chRoll, k_chPitch, k_chThr, k_chYaw }; enum { k_pidGroundSpeed2Throttle = k_controllersStart, k_pidStr, k_pidTilt, k_pidSpeed, k_pidPD, k_pidRoll, k_pidPitch, k_pidYawRate, k_pidYaw, }; BlockPIDDfb pidRoll, pidPitch, pidYaw; BlockPID pidYawRate; BlockPID pidTilt, pidSpeed; BlockPIDDfb pidPD; float _thrustMix, _pitchMix, _rollMix, _yawMix; float _cmdRoll, _cmdPitch, _cmdYawRate; }; } // namespace apo #endif /* CONTROLLERQUAD_H_ */ // vim:ts=4:sw=4:expandtab