From 613e5cf938f9f87c2078fe69761e0572492e7f28 Mon Sep 17 00:00:00 2001 From: Guglielmo Date: Tue, 6 Feb 2024 15:01:41 +0100 Subject: [PATCH] Fixes to base estimator squashed in one single commit. --- .../BaseEstimatorFromFootIMU.h | 125 ++++-- .../src/BaseEstimatorFromFootIMU.cpp | 425 ++++++++++++++---- 2 files changed, 418 insertions(+), 132 deletions(-) diff --git a/src/Estimators/include/BipedalLocomotion/FloatingBaseEstimators/BaseEstimatorFromFootIMU.h b/src/Estimators/include/BipedalLocomotion/FloatingBaseEstimators/BaseEstimatorFromFootIMU.h index 22abc4998d..73dd49d988 100644 --- a/src/Estimators/include/BipedalLocomotion/FloatingBaseEstimators/BaseEstimatorFromFootIMU.h +++ b/src/Estimators/include/BipedalLocomotion/FloatingBaseEstimators/BaseEstimatorFromFootIMU.h @@ -16,6 +16,11 @@ #include #include +// YARP +#include +#include +#include + #include namespace BipedalLocomotion @@ -30,6 +35,12 @@ struct BaseEstimatorFromFootIMUState { manif::SE3d basePose; /**< final output of the estimator - pose of the robot root link. */ + manif::SE3d footPose_L; /**< pose of the left foot */ + manif::SE3d footPose_R; /**< pose of the right foot */ + Eigen::Vector3d centerOfMassPosition; + std::vector stanceFootShadowCorners; + std::vector stanceFootCorners; + int supportCornerIndex; }; /** @@ -37,12 +48,16 @@ struct BaseEstimatorFromFootIMUState */ struct BaseEstimatorFromFootIMUInput { + bool isLeftStance; /**< true if the left foot is in contact with the ground */ + bool isRightStance; /**< true if the right foot is in contact with the ground */ Eigen::VectorXd jointPositions; /**< vector of the robot joint positions */ Eigen::VectorXd jointVelocities; /**< vector of the robot joint velocities */ - manif::SE3d desiredFootPose; /**< desired orientation and position of the foot - as per footstep planner output */ - manif::SO3d measuredRotation; /**< actual orientation of the foot measured by - on-board IMU */ + manif::SE3d offsetStanceFootPose; /**< Optional offset orientation and position of the foot. + E.g. as per footstep planner output */ + manif::SO3d measuredRotation_L; /**< actual orientation of the left foot measured by + on-board IMU */ + manif::SO3d measuredRotation_R; /**< actual orientation of the right foot measured by + on-board IMU */ }; /** @@ -50,13 +65,17 @@ struct BaseEstimatorFromFootIMUInput * kinematic chain given by the leg joints positions. * * This class assumes that the foot has a rectangular shape as shown in the following schematics - * + - * p2 __|__ p1 __ - * | | | | - * ___|__|__|___+ | FOOT - * | | | | LENGTH - * |__|__| __| - * p3 | p4 + * + * FRONT + * +X + * p1 __|__ p0 __ + * | | | | + * +Y___|__|__|___ | FOOT + * | | | | LENGTH + * |__|__| __| + * p2 | p3 + * + * HIND * * |_____| * FOOT @@ -77,7 +96,7 @@ class BaseEstimatorFromFootIMU // clang-format off /** * Initialize the BaseEstimatorFromFootIMU block. - * The setModel and setStanceFootRF methods must be called before initialization. + * The setModel method must be called before initialization. * @param handler pointer to the parameter handler. * @note The following parameters are required * | Parameter Name | Type | Description | Mandatory | @@ -132,6 +151,9 @@ class BaseEstimatorFromFootIMU BaseEstimatorFromFootIMUInput m_input; /**< Last input stored in the estimator */ BaseEstimatorFromFootIMUState m_state; /**< Current state stored in the estimator */ + yarp::os::BufferedPort m_port; /**< Port used to send the output of the + estimator to the WalkingModule */ + // Geometric quantities of the foot double m_footWidth; /**< Lateral dimension of the robot foot */ double m_footLength; /**< Frontal dimension of the robot foot */ @@ -139,50 +161,71 @@ class BaseEstimatorFromFootIMU /** * Define the 4 foot vertices in World reference frame: * - * + - * m_p2 __|__ m_p1 __ - * | | | | - * ___|__|__|___+ | FOOT - * | | | | LENGTH - * |__|__| __| - * m_p3 | m_p4 - * - * |_____| - * FOOT - * WIDTH + * +x FRONT + * m_p1 __|__ m_p0 __ + * | | | | + * +y___|__|__|___ | FOOT + * | | | | LENGTH + * |__|__| __| + * m_p2 | m_p3 + * HIND + * |_____| + * FOOT + * WIDTH * */ - std::vector m_cornersInLocalFrame; /**< this implementation is considering + std::vector m_cornersInInertialFrame; /**< this implementation is considering rectangular feet (4 corners) */ - std::vector m_transformedFootCorners; /**< vector of the foot corners + std::vector m_tiltedFootCorners; /**< vector of the foot corners transformed into the inertial frame */ - Eigen::Vector3d m_desiredTranslation; /**< the position vector extracted from the - `desiredFootPose` */ - Eigen::Matrix3d m_desiredRotation; /**< the rotation matrix extracted from the `desiredFootPose` - */ - Eigen::Vector3d m_desiredRPY; /**< the rotation matrix extracted from the `desiredFootPose` + Eigen::Vector3d m_offsetTranslation; /**< the position vector extracted from the + `offsetFootPose` */ + Eigen::Matrix3d m_offsetRotation; /**< the rotation matrix extracted from the `offsetFootPose` + */ + Eigen::Vector3d m_offsetRPY; /**< the rotation matrix extracted from the `offsetFootPose` converted into Euler angles */ Eigen::Matrix3d m_measuredRotation; /**< the measured foot orientation casted manif::SO3d --> Eigen::Matrix3d */ Eigen::Vector3d m_measuredRPY; /**< the measured foot orientation converted into Euler angles */ manif::SO3d m_measuredRotationCorrected; /**< rotation matrix that employs: measured Roll, - measured Pitch, desired Yaw */ + measured Pitch, offset Yaw */ + manif::SO3d m_offsetRotationCasted; + manif::SO3d m_measuredTilt; manif::SE3d m_measuredFootPose; /**< the final foot pose matrix obtained through measured and - desired quantities */ - manif::SE3d m_frame_H_link; /**< coordinate change matrix from foot link frame to foot sole frame - */ + offset quantities */ + manif::SE3d m_T_walk; + manif::SE3d m_T_yawDrift; + double m_yawOld; + manif::SE3d m_footFrame_H_link_L; /**< coordinate change matrix from left foot link frame to + * left foot sole frame + */ + manif::SE3d m_footFrame_H_link_R; /**< coordinate change matrix from right foot link frame to + * right foot sole frame + */ Eigen::Vector3d m_gravity; + const Eigen::Vector3d m_noTras{0.0, 0.0, 0.0}; iDynTree::KinDynComputations m_kinDyn; iDynTree::Model m_model; - iDynTree::LinkIndex m_linkIndex; - int m_frameIndex; - std::string m_frameName; - int m_baseFrame; /**< Index of the frame whose pose needs to be estimated */ - std::string m_footFrameName; /**< reference frames of the possible stance feet (whose - orientations are measured)*/ - bool m_isOutputValid{false}; + iDynTree::LinkIndex m_stanceLinkIndex; + + std::string m_baseFrameName; /**< Base link of the robot (whose pose must be estimated) */ + int m_baseFrameIndex; /**< Index of the frame whose pose needs to be estimated */ + + std::string m_footFrameName_L; /**< reference frame of the left stance foot (whose + orientation is measured)*/ + int m_footFrameIndex_L; /**< Index of the left foot frame */ + + std::string m_footFrameName_R; /**< reference frame of the right stance foot (whose + orientation is measured)*/ + int m_footFrameIndex_R; /**< Index of the right foot frame */ + + bool m_isLastStanceFoot_L{false}; /**< true if the last stance foot was the left one */ + bool m_isLastStanceFoot_R{false}; /**< true if the last stance foot was the right one */ + bool m_isModelSet{false}; bool m_isInitialized{false}; + bool m_isInputSet{false}; + bool m_isOutputValid{false}; }; } // namespace Estimators diff --git a/src/Estimators/src/BaseEstimatorFromFootIMU.cpp b/src/Estimators/src/BaseEstimatorFromFootIMU.cpp index 9bb484f846..5b54dd57d5 100644 --- a/src/Estimators/src/BaseEstimatorFromFootIMU.cpp +++ b/src/Estimators/src/BaseEstimatorFromFootIMU.cpp @@ -11,12 +11,12 @@ using namespace BipedalLocomotion; -Eigen::Vector3d toZYX(const Eigen::Matrix3d& r) +Eigen::Vector3d toXYZ(const Eigen::Matrix3d& r) { Eigen::Vector3d output; - double& thetaZ = output[0]; // Roll + double& thetaX = output[0]; // Roll double& thetaY = output[1]; // Pitch - double& thetaX = output[2]; // Yaw + double& thetaZ = output[2]; // Yaw if (r(2, 0) < +1) { @@ -51,14 +51,14 @@ bool BaseEstimatorFromFootIMU::setModel(const iDynTree::Model& model) m_isModelSet = false; - m_model = model; - if (!m_kinDyn.loadRobotModel(model)) { log()->error("{} Unable to load the model.", logPrefix); return false; } + m_model = model; + m_isModelSet = true; return true; @@ -111,47 +111,76 @@ bool BaseEstimatorFromFootIMU::initialize( return true; }; - // Base link of the robot (whose pose must be estimated) - std::string baseFrameName; - bool ok = populateParameter(ptr->getGroup("MODEL_INFO"), "base_frame", baseFrameName); - - // Frame associated to the foot of the robot (whose orientation is measured) - ok = populateParameter(ptr->getGroup("MODEL_INFO"), "foot_frame", m_footFrameName); - - ok = ok && populateParameter(ptr, "foot_width_in_m", m_footWidth); - ok = ok && populateParameter(ptr, "foot_length_in_m", m_footLength); + auto baseEstimatorPtr = ptr->getGroup("BASE_ESTIMATOR").lock(); + if (baseEstimatorPtr == nullptr) + { + log()->error("{} Invalid parameter handler for the group 'BASE_ESTIMATOR'", logPrefix); + return false; + } - // Set the 4 foot vertices in World reference frame [dimensions in meters] - m_cornersInLocalFrame.emplace_back(+m_footWidth / 2, +m_footLength / 2, 0); - m_cornersInLocalFrame.emplace_back(-m_footWidth / 2, +m_footLength / 2, 0); - m_cornersInLocalFrame.emplace_back(-m_footWidth / 2, -m_footLength / 2, 0); - m_cornersInLocalFrame.emplace_back(+m_footWidth / 2, -m_footLength / 2, 0); + // Frame associated to the base of the robot (whose pose is estimated) + bool ok = populateParameter(baseEstimatorPtr->getGroup("MODEL_INFO"), + "base_frame", + m_baseFrameName); + + // Frame associated to the left foot of the robot (whose orientation is measured) + ok = ok + && populateParameter(baseEstimatorPtr->getGroup("MODEL_INFO"), + "foot_frame_L", + m_footFrameName_L); + // Frame associated to the right foot of the robot (whose orientation is measured) + ok = ok + && populateParameter(baseEstimatorPtr->getGroup("MODEL_INFO"), + "foot_frame_R", + m_footFrameName_R); + // Foot dimensions. Same for both feet. + ok = ok && populateParameter(baseEstimatorPtr, "foot_width_in_m", m_footWidth); + ok = ok && populateParameter(baseEstimatorPtr, "foot_length_in_m", m_footLength); m_gravity << 0, 0, -BipedalLocomotion::Math::StandardAccelerationOfGravitation; - m_frameIndex = m_kinDyn.getFrameIndex(m_footFrameName); - if (m_frameIndex == iDynTree::FRAME_INVALID_INDEX) + + m_baseFrameIndex = m_kinDyn.getFrameIndex(m_baseFrameName); + if (m_baseFrameIndex == iDynTree::FRAME_INVALID_INDEX) { - log()->error("{} Invalid frame named: {}", logPrefix, m_footFrameName); + log()->error("{} Invalid frame named: {}", logPrefix, m_baseFrameName); return false; } - m_baseFrame = m_kinDyn.getFrameIndex(baseFrameName); - if (m_baseFrame == iDynTree::FRAME_INVALID_INDEX) + + m_footFrameIndex_L = m_kinDyn.getFrameIndex(m_footFrameName_L); + if (m_footFrameIndex_L == iDynTree::FRAME_INVALID_INDEX) { - log()->error("{} Invalid frame named: {}", logPrefix, baseFrameName); + log()->error("{} Invalid frame named: {}", logPrefix, m_footFrameName_L); return false; } + iDynTree::Transform frame_H_link_L = m_model.getFrameTransform(m_footFrameIndex_L).inverse(); + m_footFrame_H_link_L = Conversions::toManifPose(frame_H_link_L); - m_frameName = m_kinDyn.getFrameName(m_frameIndex); - m_linkIndex = m_model.getFrameLink(m_frameIndex); - - if (!m_kinDyn.setFloatingBase(m_model.getLinkName(m_linkIndex))) + m_footFrameIndex_R = m_kinDyn.getFrameIndex(m_footFrameName_R); + if (m_footFrameIndex_R == iDynTree::FRAME_INVALID_INDEX) { - log()->error("{} Unable to set the floating base.", logPrefix); + log()->error("{} Invalid frame named: {}", logPrefix, m_footFrameName_R); return false; } + iDynTree::Transform frame_H_link_R = m_model.getFrameTransform(m_footFrameIndex_R).inverse(); + m_footFrame_H_link_R = Conversions::toManifPose(frame_H_link_R); - iDynTree::Transform frame_H_link = m_model.getFrameTransform(m_frameIndex).inverse(); - m_frame_H_link = Conversions::toManifPose(frame_H_link); + // resetting the vector of foot corners to be sure it is correctly initialized. + m_cornersInInertialFrame.clear(); + + // Set the 4 foot vertices in World reference frame [dimensions in meters] + m_cornersInInertialFrame.emplace_back(+m_footLength / 2, -m_footWidth / 2, 0); + m_cornersInInertialFrame.emplace_back(+m_footLength / 2, +m_footWidth / 2, 0); + m_cornersInInertialFrame.emplace_back(-m_footLength / 2, +m_footWidth / 2, 0); + m_cornersInInertialFrame.emplace_back(-m_footLength / 2, -m_footWidth / 2, 0); + + m_yawOld = 0.0; + m_T_yawDrift.setIdentity(); + m_measuredFootPose.setIdentity(); + m_T_walk.setIdentity(); + m_state.stanceFootShadowCorners.resize(m_cornersInInertialFrame.size()); + m_state.stanceFootCorners.resize(m_cornersInInertialFrame.size()); + + m_port.open("/baseEstimatorFromFootIMU/state:o"); m_isInitialized = true; @@ -160,7 +189,17 @@ bool BaseEstimatorFromFootIMU::initialize( bool BaseEstimatorFromFootIMU::setInput(const BaseEstimatorFromFootIMUInput& input) { + constexpr auto logPrefix = "[BaseEstimatorFromFootIMU::setInput]"; + m_isInputSet = false; + + if (!m_isInitialized) + { + log()->error("{} The estimator must be initialized before setting the input.", logPrefix); + return false; + } + m_input = input; + m_isInputSet = true; return true; } @@ -170,97 +209,242 @@ bool BaseEstimatorFromFootIMU::advance() m_isOutputValid = false; - if (!m_isInitialized) + if (!m_isInputSet) + { + log()->error("{} The estimator input has not been set.", logPrefix); + return false; + } + + // checking the stance foot + if (m_input.isLeftStance && m_input.isRightStance) + { + log()->error("{} Both feet are stance feet. The estimator accept only one stance foot.", + logPrefix); + return false; + } + if (!m_input.isLeftStance && !m_input.isRightStance) + { + log()->error("{} No stance foot set. The estimator needs one stance foot in input.", + logPrefix); + return false; + } + manif::SE3d stanceFootFrame_H_link = manif::SE3d::Identity(); + int stanceFootFrameIndex = iDynTree::FRAME_INVALID_INDEX; + + if (m_input.isLeftStance) { - log()->error("{} The estimator is not initialized.", logPrefix); + stanceFootFrame_H_link = m_footFrame_H_link_L; + stanceFootFrameIndex = m_footFrameIndex_L; + // casting the measured foot orientation manif::SO3d --> Eigen::Matrix3d. + m_measuredRotation = m_input.measuredRotation_L.rotation(); + // has the stance foot just changed? + if (m_isLastStanceFoot_R) + { + // updating the yawOld value. + Eigen::Vector3d lastRPY_R = toXYZ(m_state.footPose_R.rotation()); + // Eigen::Vector3d lastRPY_L = toXYZ(m_input.measuredRotation_L.rotation()); + m_yawOld = lastRPY_R(2); + + // updating the m_T_walk matrix. + Eigen::Vector3d stepTras = (m_state.footPose_L.translation() - m_state.footPose_R.translation()); + stepTras(2) = 0.0; + manif::SE3d T_walkTras(stepTras, manif::SO3d::Identity()); + + // Eigen::Matrix3d stepRot = m_state.footPose_R.rotation().inverse() * m_state.footPose_L.rotation(); + // Eigen::Vector3d stepRotRPY = toXYZ(stepRot); + // manif::SO3d stepRotManif = manif::SO3d(stepRotRPY(0), stepRotRPY(1), stepRotRPY(2)); + // manif::SE3d T_walkRot(m_noTras, stepRotManif); + // manif::SE3d temp = T_walkTras * T_walkRot * m_T_walk * m_T_yawDrift; + + manif::SE3d temp = T_walkTras * m_T_walk; + m_T_walk = temp; + + // m_T_yawDrift.setIdentity(); + } + } + + if (m_input.isRightStance) + { + stanceFootFrame_H_link = m_footFrame_H_link_R; + stanceFootFrameIndex = m_footFrameIndex_R; + // casting the measured foot orientation manif::SO3d --> Eigen::Matrix3d. + m_measuredRotation = m_input.measuredRotation_R.rotation(); + // has the stance foot just changed? + if (m_isLastStanceFoot_L) + { + // updating the yawOld value. + Eigen::Vector3d lastRPY_L = toXYZ(m_state.footPose_L.rotation()); + // Eigen::Vector3d lastRPY_R = toXYZ(m_input.measuredRotation_R.rotation()); + m_yawOld = lastRPY_L(2); + + // updating the m_T_walk matrix. + Eigen::Vector3d stepTras = (m_state.footPose_R.translation() - m_state.footPose_L.translation()); + stepTras(2) = 0.0; + manif::SE3d T_walkTras(stepTras, manif::SO3d::Identity()); + + // Eigen::Matrix3d stepRot = m_state.footPose_L.rotation().inverse() * m_state.footPose_R.rotation(); + // Eigen::Vector3d stepRotRPY = toXYZ(stepRot); + // manif::SO3d stepRotManif = manif::SO3d(stepRotRPY(0), stepRotRPY(1), stepRotRPY(2)); + // manif::SE3d T_walkRot(m_noTras, stepRotManif); + // manif::SE3d temp = T_walkTras * T_walkRot * m_T_walk * m_T_yawDrift; + + manif::SE3d temp = T_walkTras * m_T_walk; + m_T_walk = temp; + + // m_T_yawDrift.setIdentity(); + } + } + + m_stanceLinkIndex = m_model.getFrameLink(stanceFootFrameIndex); + if (!m_kinDyn.setFloatingBase(m_model.getLinkName(m_stanceLinkIndex))) + { + log()->error("{} Unable to set the stance foot as floating base.", logPrefix); return false; } - // `desiredFootPose` is intended as the output of the footstep planner. + // `offsetStanceFootPose` is intended as the output of the footstep planner. - // extracting the position part of the `desiredFootPose`. - m_desiredTranslation = m_input.desiredFootPose.translation(); + // extracting the position part of the `offsetStanceFootPose`. + m_offsetTranslation = m_input.offsetStanceFootPose.translation(); - // extracting the orientation part of the `desiredFootPose` and expressing it + // extracting the orientation part of the `offsetStanceFootPose` and expressing it // through RPY Euler angles. - m_desiredRotation = m_input.desiredFootPose.rotation(); - m_desiredRPY = toZYX(m_desiredRotation); + m_offsetRotation = m_input.offsetStanceFootPose.rotation(); + m_offsetRPY = toXYZ(m_offsetRotation); + m_offsetRotationCasted = manif::SO3d(m_offsetRPY(0), m_offsetRPY(1), m_offsetRPY(2)); + + // expressing measured orientation through RPY Euler angles. + m_measuredRPY = toXYZ(m_measuredRotation); - // casting the measured foot orientation manif::SO3d --> Eigen::Matrix3d. - m_measuredRotation = m_input.measuredRotation.rotation(); - // expressing this orientation through RPY Euler angles. - m_measuredRPY = toZYX(m_measuredRotation); + // offset Yaw is used instead of measured Yaw. + // m_measuredRPY(2) = m_offsetRPY(2); - // desired Yaw is used instead of measured Yaw. - m_measuredRPY(2) = m_desiredRPY(2); + // MANUAL CORRECTION: measured Roll, Pitch and Yaw. + // double temp_roll = -m_measuredRPY(1); + // double temp_pitch = -m_measuredRPY(0); + // double temp_yaw = -m_measuredRPY(2); + // m_measuredRPY(0) = temp_roll; + // m_measuredRPY(1) = temp_pitch; + // m_measuredRPY(2) = temp_yaw; - // manif::SO3d rotation matrix that employs: measured Roll, measured Pitch, - // desired Yaw. + // manif::SO3d rotation matrix that employs: measured Roll, measured Pitch, offset Yaw. m_measuredRotationCorrected = manif::SO3d(m_measuredRPY(0), m_measuredRPY(1), m_measuredRPY(2)); + m_measuredTilt = manif::SO3d(m_measuredRPY(0), m_measuredRPY(1), 0.0); + double measuredYaw = m_measuredRPY(2); - // manif::SE3d pose matrix that employs: desired Position, measured Roll, - // measured Pitch, desired Yaw. - manif::SE3d T_foot_raw(m_desiredTranslation, m_measuredRotationCorrected); + // manif::SE3d pose matrix that employs: offset Position, measured Roll, measured Pitch, offset + // Yaw. + manif::SE3d T_foot_imu(m_noTras, m_measuredRotationCorrected); + manif::SE3d T_foot_tilt(m_noTras, m_measuredTilt); + manif::SE3d T_foot_offset(m_offsetTranslation, m_offsetRotationCasted); - // finding the positions of the foot corners in world frame given `T_foot_raw` + // finding the positions of the foot corners in world frame given `T_foot_imu` // pose matrix. + // resetting the vector of transformed foot corners from previous iteration. + m_tiltedFootCorners.clear(); + // for each corner we compute the position in the inertial frame - for (const auto& corner : m_cornersInLocalFrame) + for (const auto& corner : m_cornersInInertialFrame) { - m_transformedFootCorners.emplace_back(T_foot_raw.act(corner)); + m_tiltedFootCorners.emplace_back(T_foot_tilt.act(corner)); } // The center of the foot sole is at ground level --> some corners may be - // under ground level. The foot may need to be lifted by an offset value. + // under ground level. The foot may need to be translated by a 3D offset value. // extracting the vertical quotes of the foot corners and finding the lowest // corner. - auto lowestCorner = std::min_element(m_transformedFootCorners.begin(), - m_transformedFootCorners.end(), - [](const Eigen::Vector3d& a, const Eigen::Vector3d& b) { - return a[2] < b[2]; - }); + Eigen::VectorXd cornersZ; + cornersZ.resize(m_cornersInInertialFrame.size()); + cornersZ.setZero(); + int index = 0; + for (const auto& corner : m_tiltedFootCorners) + { + cornersZ[index] = corner[2]; + index++; + } // finding the index of the lowest corner. - int supportCornerIndex = std::distance(m_transformedFootCorners.begin(), lowestCorner); - - Eigen::Vector3d p_desired(0, 0, 0); - Eigen::Vector3d vertexOffset(0, 0, 0); + double minZ = cornersZ[0]; + int indexMinZ = 0; + for (int i = 1; i < cornersZ.size(); i++) + { + if (cornersZ[i] < minZ) + { + minZ = cornersZ[i]; + indexMinZ = i; + } + } + m_state.supportCornerIndex = indexMinZ; - // finding the offset vector needed to bring the lowest corner back to its - // desired position. - switch (supportCornerIndex) + // checking that the index of the lowest corner is within the range [0, 3]. + if (!(0 <= m_state.supportCornerIndex <= 3)) { - case 0: - p_desired = m_input.desiredFootPose.act(m_cornersInLocalFrame[supportCornerIndex]); - vertexOffset = p_desired - m_transformedFootCorners[supportCornerIndex]; - break; - case 1: - p_desired = m_input.desiredFootPose.act(m_cornersInLocalFrame[supportCornerIndex]); - vertexOffset = p_desired - m_transformedFootCorners[supportCornerIndex]; - break; - case 2: - p_desired = m_input.desiredFootPose.act(m_cornersInLocalFrame[supportCornerIndex]); - vertexOffset = p_desired - m_transformedFootCorners[supportCornerIndex]; - break; - case 3: - p_desired = m_input.desiredFootPose.act(m_cornersInLocalFrame[supportCornerIndex]); - vertexOffset = p_desired - m_transformedFootCorners[supportCornerIndex]; - break; - default: log()->error("{} Foot vertex index out of bounds (0, 3): {}.", logPrefix, - supportCornerIndex); + m_state.supportCornerIndex); return false; } + // finding the index of the highest corner. + // double maxZ = cornersZ[0]; + // int indexMaxZ = 0; + // for (int i = 1; i < cornersZ.size(); i++) + // { + // if (cornersZ[i] > maxZ) + // { + // maxZ = cornersZ[i]; + // indexMaxZ = i; + // } + // } + + // double deltaZ = cornersZ[indexMaxZ] - cornersZ[m_state.supportCornerIndex]; + // std::cerr << "Foot deltaZ: " << deltaZ << std::endl; + + // finding the translation vector needed to bring the lowest corner back to its + // untilted position. + Eigen::Vector3d p_untilted(0, 0, 0); + Eigen::Vector3d supportCornerTranslation(0, 0, 0); + p_untilted = (m_cornersInInertialFrame[m_state.supportCornerIndex]); + supportCornerTranslation + = p_untilted - (m_tiltedFootCorners[m_state.supportCornerIndex]); // TODO: change if + // flat ground + // assumption is + // removed + // transforming the offset vector into a translation matrix. - manif::SE3d T_vertexOffset(vertexOffset, manif::SO3d::Identity()); + manif::SE3d T_supportCornerTranslation(supportCornerTranslation, manif::SO3d::Identity()); + + // calculating the yaw drift - VALID FOR BOTH FEET ONLY IF FRAMES ARE ORIENTED IN THE SAME WAY. + double deltaYaw = 0.0; + deltaYaw = measuredYaw - m_yawOld; + m_yawOld = measuredYaw; + // manif::SO3d rotation matrix that employs: zero Roll, zero Pitch, measured Yaw variation. + auto R_deltaYaw = manif::SO3d(0.0, 0.0, deltaYaw); + manif::SE3d T_deltaYaw(m_noTras, R_deltaYaw); + + std::vector tempCorners; + tempCorners.resize(m_cornersInInertialFrame.size()); + int j = 0; + for (const auto& corner : m_cornersInInertialFrame) + { + tempCorners[j] = T_deltaYaw.act(corner); + j++; + } + Eigen::Vector3d yawDrift(0, 0, 0); + yawDrift = m_cornersInInertialFrame[m_state.supportCornerIndex] + - tempCorners[m_state.supportCornerIndex]; + // manif::SE3d T_yawDrift(yawDrift, R_deltaYaw); + manif::SE3d T_yawDrift(m_noTras, R_deltaYaw); + + manif::SE3d temp = T_yawDrift * m_T_yawDrift; + m_T_yawDrift = temp; - // obtaining the final foot pose using both measured and desired quantities. + // obtaining the final foot pose using both measured and offset quantities. // cordinate change is performed from foot sole frame to foot link frame. - m_measuredFootPose = T_vertexOffset * T_foot_raw * m_frame_H_link; + m_measuredFootPose = T_foot_offset * m_T_walk * m_T_yawDrift * T_supportCornerTranslation + * T_foot_tilt * stanceFootFrame_H_link; Eigen::VectorXd baseVelocity(6); baseVelocity.setZero(); @@ -272,12 +456,71 @@ bool BaseEstimatorFromFootIMU::advance() m_input.jointVelocities, m_gravity)) { - log()->error("{} Unable to set the robot state.", logPrefix); + log()->error("{} Unable to set the robot state from the stance foot pose.", logPrefix); return false; } - // calculating the pose of the root link given the robot state. - m_state.basePose = Conversions::toManifPose(m_kinDyn.getWorldTransform(m_baseFrame)); + // calculating the output of the estimator given the robot state. + m_state.basePose = Conversions::toManifPose(m_kinDyn.getWorldTransform(m_baseFrameIndex)); + m_state.footPose_L = Conversions::toManifPose(m_kinDyn.getWorldTransform(m_footFrameIndex_L)); + m_state.footPose_R = Conversions::toManifPose(m_kinDyn.getWorldTransform(m_footFrameIndex_R)); + m_kinDyn.getCenterOfMassPosition(m_state.centerOfMassPosition); + + for (int i = 0; i < m_cornersInInertialFrame.size(); i++) + { + m_state.stanceFootShadowCorners[i] + = m_T_walk.act(m_T_yawDrift.act(m_cornersInInertialFrame[i])); + m_state.stanceFootCorners[i] = m_T_walk.act( + m_T_yawDrift.act(T_supportCornerTranslation.act(m_tiltedFootCorners[i]))); + } + + // yarp write the basePose in the port + yarp::sig::Vector output; + output.clear(); + // output.resize(12); + output.push_back(m_state.basePose.translation().x()); + output.push_back(m_state.basePose.translation().y()); + output.push_back(m_state.basePose.translation().z()); + output.push_back(toXYZ(m_state.basePose.rotation())[0]); + output.push_back(toXYZ(m_state.basePose.rotation())[1]); + output.push_back(toXYZ(m_state.basePose.rotation())[2]); + output.push_back(0.0); + output.push_back(0.0); + output.push_back(0.0); + output.push_back(0.0); + output.push_back(0.0); + output.push_back(0.0); + m_port.prepare() = output; + m_port.write(); + + double orientationError_L = (toXYZ(m_state.footPose_L.rotation()) - toXYZ(m_input.measuredRotation_L.rotation())).norm(); + double orientationError_R = (toXYZ(m_state.footPose_R.rotation()) - toXYZ(m_input.measuredRotation_R.rotation())).norm(); + double orientationErrorThreshold = 0.01; // [rad]. TODO: get parameter from config file. 0,01 rad = 0,5729578 deg. + + if ( (orientationError_L > orientationErrorThreshold) || (orientationError_R > orientationErrorThreshold) ) + { + log()->warn("{} Foot orientation error above {} threshold: {} Left, {} Right.", + logPrefix, + orientationErrorThreshold, + orientationError_L, + orientationError_R); + } + // std::cerr << "L FOOT ROTATION IN: " << toXYZ(m_input.measuredRotation_L.rotation()).transpose() << std::endl; + // std::cerr << "L FOOT ROTATION OUT: " << toXYZ(m_state.footPose_L.rotation()).transpose() << std::endl; + // std::cerr << "R FOOT ROTATION IN: " << toXYZ(m_input.measuredRotation_R.rotation()).transpose() << std::endl; + // std::cerr << "R FOOT ROTATION OUT: " << toXYZ(m_state.footPose_R.rotation()).transpose() << std::endl; + + // updating the stance foot flags. + if (m_input.isLeftStance) + { + m_isLastStanceFoot_L = true; + m_isLastStanceFoot_R = false; + } + if (m_input.isRightStance) + { + m_isLastStanceFoot_L = false; + m_isLastStanceFoot_R = true; + } m_isOutputValid = true;