From e88a46f24ee1c5dce2adb6c02ef77380d0acdd8f Mon Sep 17 00:00:00 2001 From: Frederik Markus <80588263+frede791@users.noreply.github.com> Date: Thu, 9 Nov 2023 19:14:06 +0900 Subject: [PATCH 1/2] wind addition to advanced_lift_drag plugin (#2226) * wind addition to advanced_lift_drag plugin Signed-off-by: frederik * added note on wind Signed-off-by: frederik --------- Signed-off-by: frederik Co-authored-by: frederik --- src/systems/advanced_lift_drag/AdvancedLiftDrag.cc | 14 +++++++++++++- src/systems/advanced_lift_drag/AdvancedLiftDrag.hh | 2 ++ 2 files changed, 15 insertions(+), 1 deletion(-) diff --git a/src/systems/advanced_lift_drag/AdvancedLiftDrag.cc b/src/systems/advanced_lift_drag/AdvancedLiftDrag.cc index a4da3147e4..a7c60109fd 100644 --- a/src/systems/advanced_lift_drag/AdvancedLiftDrag.cc +++ b/src/systems/advanced_lift_drag/AdvancedLiftDrag.cc @@ -59,6 +59,7 @@ #include "gz/sim/components/Name.hh" #include "gz/sim/components/ExternalWorldWrenchCmd.hh" #include "gz/sim/components/Pose.hh" +#include "gz/sim/components/Wind.hh" using namespace gz; using namespace sim; @@ -481,6 +482,13 @@ void AdvancedLiftDragPrivate::Update(EntityComponentManager &_ecm) const auto worldPose = _ecm.Component(this->linkEntity); + // get wind as a component from the _ecm + components::WorldLinearVelocity *windLinearVel = nullptr; + if(_ecm.EntityByComponents(components::Wind()) != kNullEntity){ + Entity windEntity = _ecm.EntityByComponents(components::Wind()); + windLinearVel = + _ecm.Component(windEntity); + } std::vector controlJointPosition_vec( this->num_ctrl_surfaces); @@ -499,8 +507,12 @@ void AdvancedLiftDragPrivate::Update(EntityComponentManager &_ecm) const auto &pose = worldPose->Data(); const auto cpWorld = pose.Rot().RotateVector(this->cp); - const auto air_velocity = worldLinVel->Data() + worldAngVel->Data().Cross( + auto air_velocity = worldLinVel->Data() + worldAngVel->Data().Cross( cpWorld); + if (windLinearVel != nullptr){ + air_velocity = worldLinVel->Data() + worldAngVel->Data().Cross( + cpWorld) - windLinearVel->Data(); + } // Define body frame: X forward, Z downward, Y out the right wing gz::math::Vector3d body_x_axis = pose.Rot().RotateVector(this->forward); diff --git a/src/systems/advanced_lift_drag/AdvancedLiftDrag.hh b/src/systems/advanced_lift_drag/AdvancedLiftDrag.hh index 0a5be89e8f..31b420d745 100644 --- a/src/systems/advanced_lift_drag/AdvancedLiftDrag.hh +++ b/src/systems/advanced_lift_drag/AdvancedLiftDrag.hh @@ -44,6 +44,8 @@ namespace systems /// https://github.com/PX4/PX4-Autopilot/blob/main/Tools/simulation/ /// gz/tools/avl_automation/ /// + /// Note: Wind calculations can be enabled by setting the wind parameter + /// in the world file. /// /// ## System Parameters /// From 409efa7df74271426f68faee92e4250afa66f1c6 Mon Sep 17 00:00:00 2001 From: Steve Peters Date: Mon, 13 Nov 2023 08:06:38 -0800 Subject: [PATCH 2/2] Use sdf FindElement API to avoid const_cast (#2231) (#2236) Several systems use const_cast in order to call sdf::Element::GetElement with const ElementPtrs, but the FindElement API can be used instead. Signed-off-by: Steve Peters --- src/systems/diff_drive/DiffDrive.cc | 8 ++------ .../JointStatePublisher.cc | 3 +-- .../JointTrajectoryController.cc | 3 +-- src/systems/log/LogRecord.cc | 3 +-- .../log_video_recorder/LogVideoRecorder.cc | 14 +++++--------- src/systems/mecanum_drive/MecanumDrive.cc | 12 ++++-------- src/systems/shader_param/ShaderParam.cc | 17 +++++++---------- src/systems/tracked_vehicle/TrackedVehicle.cc | 12 ++++-------- src/systems/velocity_control/VelocityControl.cc | 8 ++------ 9 files changed, 27 insertions(+), 53 deletions(-) diff --git a/src/systems/diff_drive/DiffDrive.cc b/src/systems/diff_drive/DiffDrive.cc index cd72571dc6..b05b210925 100644 --- a/src/systems/diff_drive/DiffDrive.cc +++ b/src/systems/diff_drive/DiffDrive.cc @@ -182,18 +182,14 @@ void DiffDrive::Configure(const Entity &_entity, return; } - // Ugly, but needed because the sdf::Element::GetElement is not a const - // function and _sdf is a const shared pointer to a const sdf::Element. - auto ptr = const_cast(_sdf.get()); - // Get params from SDF - sdf::ElementPtr sdfElem = ptr->GetElement("left_joint"); + auto sdfElem = _sdf->FindElement("left_joint"); while (sdfElem) { this->dataPtr->leftJointNames.push_back(sdfElem->Get()); sdfElem = sdfElem->GetNextElement("left_joint"); } - sdfElem = ptr->GetElement("right_joint"); + sdfElem = _sdf->FindElement("right_joint"); while (sdfElem) { this->dataPtr->rightJointNames.push_back(sdfElem->Get()); diff --git a/src/systems/joint_state_publisher/JointStatePublisher.cc b/src/systems/joint_state_publisher/JointStatePublisher.cc index de11de8c2c..03ab241c0d 100644 --- a/src/systems/joint_state_publisher/JointStatePublisher.cc +++ b/src/systems/joint_state_publisher/JointStatePublisher.cc @@ -64,8 +64,7 @@ void JointStatePublisher::Configure( // specified joints. Otherwise, publish all the joints. if (_sdf->HasElement("joint_name")) { - sdf::Element *ptr = const_cast(_sdf.get()); - sdf::ElementPtr elem = ptr->GetElement("joint_name"); + auto elem = _sdf->FindElement("joint_name"); while (elem) { std::string jointName = elem->Get(); diff --git a/src/systems/joint_traj_control/JointTrajectoryController.cc b/src/systems/joint_traj_control/JointTrajectoryController.cc index 5c0b332b8f..6b081e0b5c 100644 --- a/src/systems/joint_traj_control/JointTrajectoryController.cc +++ b/src/systems/joint_traj_control/JointTrajectoryController.cc @@ -811,8 +811,7 @@ std::vector JointParameters::Parse( if (_sdf->HasElement(_parameterName)) { - sdf::ElementPtr param = const_cast( - _sdf.get())->GetElement(_parameterName); + auto param = _sdf->FindElement(_parameterName); while (param) { output.push_back(param->Get()); diff --git a/src/systems/log/LogRecord.cc b/src/systems/log/LogRecord.cc index 940d5bc2f2..fefa5f4ac8 100644 --- a/src/systems/log/LogRecord.cc +++ b/src/systems/log/LogRecord.cc @@ -338,8 +338,7 @@ bool LogRecordPrivate::Start(const std::string &_logPath, // Get the topics to record, if any. if (this->sdf->HasElement("record_topic")) { - auto ptr = const_cast(this->sdf.get()); - sdf::ElementPtr recordTopicElem = ptr->GetElement("record_topic"); + auto recordTopicElem = this->sdf->FindElement("record_topic"); // This is used to determine if a topic is a regular expression. std::regex regexMatch(".*[\\*\\?\\[\\]\\(\\)\\.]+.*"); diff --git a/src/systems/log_video_recorder/LogVideoRecorder.cc b/src/systems/log_video_recorder/LogVideoRecorder.cc index 71c68c7999..53d23b08da 100644 --- a/src/systems/log_video_recorder/LogVideoRecorder.cc +++ b/src/systems/log_video_recorder/LogVideoRecorder.cc @@ -168,13 +168,9 @@ void LogVideoRecorder::Configure( this->dataPtr->node.Advertise( "/log_video_recorder/status"); - // Ugly, but needed because the sdf::Element::GetElement is not a const - // function and _sdf is a const shared pointer to a const sdf::Element. - auto ptr = const_cast(_sdf.get()); - if (_sdf->HasElement("entity")) { - auto entityElem = ptr->GetElement("entity"); + auto entityElem = _sdf->FindElement("entity"); while (entityElem) { this->dataPtr->modelsToRecord.insert(entityElem->Get()); @@ -184,7 +180,7 @@ void LogVideoRecorder::Configure( if (_sdf->HasElement("region")) { - sdf::ElementPtr regionElem = ptr->GetElement("region"); + auto regionElem = _sdf->FindElement("region"); while (regionElem) { auto min = regionElem->Get("min"); @@ -198,14 +194,14 @@ void LogVideoRecorder::Configure( if (_sdf->HasElement("start_time")) { - auto t = ptr->Get("start_time"); + auto t = _sdf->Get("start_time"); this->dataPtr->startTime = std::chrono::milliseconds(static_cast(t * 1000.0)); } if (_sdf->HasElement("end_time")) { - auto t = ptr->Get("end_time"); + auto t = _sdf->Get("end_time"); std::chrono::milliseconds ms(static_cast(t * 1000.0)); if (this->dataPtr->startTime > ms) { @@ -219,7 +215,7 @@ void LogVideoRecorder::Configure( if (_sdf->HasElement("exit_on_finish")) { - this->dataPtr->exitOnFinish = ptr->Get("exit_on_finish"); + this->dataPtr->exitOnFinish = _sdf->Get("exit_on_finish"); } this->dataPtr->loadTime = std::chrono::system_clock::now(); diff --git a/src/systems/mecanum_drive/MecanumDrive.cc b/src/systems/mecanum_drive/MecanumDrive.cc index bb4d3f4202..01cacea629 100644 --- a/src/systems/mecanum_drive/MecanumDrive.cc +++ b/src/systems/mecanum_drive/MecanumDrive.cc @@ -192,31 +192,27 @@ void MecanumDrive::Configure(const Entity &_entity, return; } - // Ugly, but needed because the sdf::Element::GetElement is not a const - // function and _sdf is a const shared pointer to a const sdf::Element. - auto ptr = const_cast(_sdf.get()); - // Get params from SDF - sdf::ElementPtr sdfElem = ptr->GetElement("front_left_joint"); + auto sdfElem = _sdf->FindElement("front_left_joint"); while (sdfElem) { this->dataPtr->frontLeftJointNames.push_back(sdfElem->Get()); sdfElem = sdfElem->GetNextElement("front_left_joint"); } - sdfElem = ptr->GetElement("front_right_joint"); + sdfElem = _sdf->FindElement("front_right_joint"); while (sdfElem) { this->dataPtr->frontRightJointNames.push_back(sdfElem->Get()); sdfElem = sdfElem->GetNextElement("front_right_joint"); } - sdfElem = ptr->GetElement("back_left_joint"); + sdfElem = _sdf->FindElement("back_left_joint"); while (sdfElem) { this->dataPtr->backLeftJointNames.push_back(sdfElem->Get()); sdfElem = sdfElem->GetNextElement("back_left_joint"); } - sdfElem = ptr->GetElement("back_right_joint"); + sdfElem = _sdf->FindElement("back_right_joint"); while (sdfElem) { this->dataPtr->backRightJointNames.push_back(sdfElem->Get()); diff --git a/src/systems/shader_param/ShaderParam.cc b/src/systems/shader_param/ShaderParam.cc index 74d547800a..212cdb46b6 100644 --- a/src/systems/shader_param/ShaderParam.cc +++ b/src/systems/shader_param/ShaderParam.cc @@ -138,14 +138,11 @@ void ShaderParam::Configure(const Entity &_entity, EventManager &_eventMgr) { GZ_PROFILE("ShaderParam::Configure"); - // Ugly, but needed because the sdf::Element::GetElement is not a const - // function and _sdf is a const shared pointer to a const sdf::Element. - auto sdf = const_cast(_sdf.get()); - if (sdf->HasElement("param")) + if (_sdf->HasElement("param")) { // loop and parse all shader params - sdf::ElementPtr paramElem = sdf->GetElement("param"); + auto paramElem = _sdf->FindElement("param"); while (paramElem) { if (!paramElem->HasElement("shader") || @@ -170,7 +167,7 @@ void ShaderParam::Configure(const Entity &_entity, if (paramElem->HasElement("arg")) { - sdf::ElementPtr argElem = paramElem->GetElement("arg"); + auto argElem = paramElem->FindElement("arg"); while (argElem) { spv.args.push_back(argElem->Get()); @@ -191,14 +188,14 @@ void ShaderParam::Configure(const Entity &_entity, } // parse path to shaders - if (!sdf->HasElement("shader")) + if (!_sdf->HasElement("shader")) { gzerr << "Unable to load shader param system. " << "Missing SDF element." << std::endl; return; } // allow mulitple shader SDF element for different shader languages - sdf::ElementPtr shaderElem = sdf->GetElement("shader"); + auto shaderElem = _sdf->FindElement("shader"); while (shaderElem) { if (!shaderElem->HasElement("vertex") || @@ -217,11 +214,11 @@ void ShaderParam::Configure(const Entity &_entity, ShaderParamPrivate::ShaderUri shader; shader.language = api; - sdf::ElementPtr vertexElem = shaderElem->GetElement("vertex"); + auto vertexElem = shaderElem->FindElement("vertex"); shader.vertexShaderUri = common::findFile( asFullPath(vertexElem->Get(), this->dataPtr->modelPath)); - sdf::ElementPtr fragmentElem = shaderElem->GetElement("fragment"); + auto fragmentElem = shaderElem->FindElement("fragment"); shader.fragmentShaderUri = common::findFile( asFullPath(fragmentElem->Get(), this->dataPtr->modelPath)); diff --git a/src/systems/tracked_vehicle/TrackedVehicle.cc b/src/systems/tracked_vehicle/TrackedVehicle.cc index 094b7ebab1..b7273cfb47 100644 --- a/src/systems/tracked_vehicle/TrackedVehicle.cc +++ b/src/systems/tracked_vehicle/TrackedVehicle.cc @@ -227,17 +227,13 @@ void TrackedVehicle::Configure(const Entity &_entity, if (!links.empty()) this->dataPtr->canonicalLink = Link(links[0]); - // Ugly, but needed because the sdf::Element::GetElement is not a const - // function and _sdf is a const shared pointer to a const sdf::Element. - auto ptr = const_cast(_sdf.get()); - - std::unordered_map tracks; + std::unordered_map tracks; if (_sdf->HasElement("body_link")) this->dataPtr->bodyLinkName = _sdf->Get("body_link"); // Get params from SDF - sdf::ElementPtr sdfElem = ptr->GetElement("left_track"); + auto sdfElem = _sdf->FindElement("left_track"); while (sdfElem) { const auto& linkName = sdfElem->Get("link"); @@ -245,7 +241,7 @@ void TrackedVehicle::Configure(const Entity &_entity, tracks[linkName] = sdfElem; sdfElem = sdfElem->GetNextElement("left_track"); } - sdfElem = ptr->GetElement("right_track"); + sdfElem = _sdf->FindElement("right_track"); while (sdfElem) { const auto& linkName = sdfElem->Get("link"); @@ -288,7 +284,7 @@ void TrackedVehicle::Configure(const Entity &_entity, if (!_sdf->HasElement(tag)) continue; - auto sdf = ptr->GetElement(tag); + auto sdf = _sdf->FindElement(tag); // Parse speed limiter parameters. bool hasVelocityLimits = false; diff --git a/src/systems/velocity_control/VelocityControl.cc b/src/systems/velocity_control/VelocityControl.cc index c50f0a57ce..39dad86f36 100644 --- a/src/systems/velocity_control/VelocityControl.cc +++ b/src/systems/velocity_control/VelocityControl.cc @@ -154,14 +154,10 @@ void VelocityControl::Configure(const Entity &_entity, << modelTopic << "]" << std::endl; - // Ugly, but needed because the sdf::Element::GetElement is not a const - // function and _sdf is a const shared pointer to a const sdf::Element. - auto ptr = const_cast(_sdf.get()); - - if (!ptr->HasElement("link_name")) + if (!_sdf->HasElement("link_name")) return; - sdf::ElementPtr sdfElem = ptr->GetElement("link_name"); + auto sdfElem = _sdf->FindElement("link_name"); while (sdfElem) { this->dataPtr->linkNames.push_back(sdfElem->Get());