Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Added function to apply external forces (simple payload) #4519

Open
wants to merge 1 commit into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions AirLib/include/api/RpcLibClientBase.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -161,6 +161,8 @@ namespace airlib
bool isRecording();

void simSetWind(const Vector3r& wind) const;
void simSetExtForce(const Vector3r& ext_force) const;

vector<string> listVehicles();

std::string getSettingsString() const;
Expand Down
1 change: 1 addition & 0 deletions AirLib/include/api/WorldSimApiBase.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -86,6 +86,7 @@ namespace airlib
virtual bool isRecording() const = 0;

virtual void setWind(const Vector3r& wind) const = 0;
virtual void setExtForce(const Vector3r& ext_force) const = 0;
virtual vector<string> listVehicles() const = 0;

virtual std::string getSettingsString() const = 0;
Expand Down
11 changes: 10 additions & 1 deletion AirLib/include/common/AirSimSettings.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -414,6 +414,7 @@ namespace airlib
std::string speed_unit_label = "m\\s";
std::map<std::string, std::shared_ptr<SensorSetting>> sensor_defaults;
Vector3r wind = Vector3r::Zero();
Vector3r ext_force = Vector3r::Zero();
CameraSettingMap external_cameras;

std::string settings_text_ = "";
Expand Down Expand Up @@ -1203,6 +1204,15 @@ namespace airlib
wind = createVectorSetting(child_json, wind);
}
}
{
// External Force Settings
Settings child_json;
if (settings_json.getChild("ExternalForce", child_json)) {
ext_force = createVectorSetting(child_json, ext_force);
}
}


}

static void loadDefaultCameraSetting(const Settings& settings_json, CameraSetting& camera_defaults)
Expand All @@ -1212,7 +1222,6 @@ namespace airlib
camera_defaults = createCameraSetting(child_json, camera_defaults);
}
}

static void loadCameraDirectorSetting(const Settings& settings_json,
CameraDirectorSetting& camera_director, const std::string& simmode_name)
{
Expand Down
30 changes: 22 additions & 8 deletions AirLib/include/physics/FastPhysicsEngine.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -22,8 +22,8 @@ namespace airlib
class FastPhysicsEngine : public PhysicsEngineBase
{
public:
FastPhysicsEngine(bool enable_ground_lock = true, Vector3r wind = Vector3r::Zero())
: enable_ground_lock_(enable_ground_lock), wind_(wind)
FastPhysicsEngine(bool enable_ground_lock = true, Vector3r wind = Vector3r::Zero(), Vector3r ext_force = Vector3r::Zero())
: enable_ground_lock_(enable_ground_lock), wind_(wind), ext_force_(ext_force)
{
setName("FastPhysicsEngine");
}
Expand Down Expand Up @@ -69,6 +69,12 @@ namespace airlib
{
wind_ = wind;
}
// Set External Force
void setExtForce(const Vector3r& ext_force) override
{
ext_force_ = ext_force;
}


private:
void initPhysicsBody(PhysicsBody* body_ptr)
Expand All @@ -88,7 +94,7 @@ namespace airlib

//first compute the response as if there was no collision
//this is necessary to take in to account forces and torques generated by body
getNextKinematicsNoCollision(dt, body, current, next, next_wrench, wind_);
getNextKinematicsNoCollision(dt, body, current, next, next_wrench, wind_, ext_force_);

//if there is collision, see if we need collision response
const CollisionInfo collision_info = body.getCollisionInfo();
Expand Down Expand Up @@ -261,8 +267,11 @@ namespace airlib
}
}

static Wrench getDragWrench(const PhysicsBody& body, const Quaternionr& orientation,
const Vector3r& linear_vel, const Vector3r& angular_vel_body, const Vector3r& wind_world)
static Wrench getDragWrench(const PhysicsBody& body,
const Quaternionr& orientation,
const Vector3r& linear_vel,
const Vector3r& angular_vel_body,
const Vector3r& wind_world)
{
//add linear drag due to velocity we had since last dt seconds + wind
//drag vector magnitude is proportional to v^2, direction opposite of velocity
Expand Down Expand Up @@ -323,7 +332,7 @@ namespace airlib
}

static void getNextKinematicsNoCollision(TTimeDelta dt, PhysicsBody& body, const Kinematics::State& current,
Kinematics::State& next, Wrench& next_wrench, const Vector3r& wind)
Kinematics::State& next, Wrench& next_wrench, const Vector3r& wind, const Vector3r& ext_force)
{
const real_T dt_real = static_cast<real_T>(dt);

Expand Down Expand Up @@ -354,9 +363,13 @@ namespace airlib
//To find the drag force, we find the magnitude in the body frame and unit vector direction in world frame
avg_linear = current.twist.linear + current.accelerations.linear * (0.5f * dt_real);
avg_angular = current.twist.angular + current.accelerations.angular * (0.5f * dt_real);
const Wrench drag_wrench = getDragWrench(body, current.pose.orientation, avg_linear, avg_angular, wind);
const Wrench drag_wrench = getDragWrench(body, current.pose.orientation, avg_linear, avg_angular, wind);//, ext_force);

// ext_force is defined in world space
Wrench ext_force_wrench = Wrench::zero();
ext_force_wrench.force = ext_force;

next_wrench = body_wrench + drag_wrench;
next_wrench = body_wrench + drag_wrench + ext_force_wrench;

//Utils::log(Utils::stringf("B-WRN %s: ", VectorMath::toString(body_wrench.force).c_str()));
//Utils::log(Utils::stringf("D-WRN %s: ", VectorMath::toString(drag_wrench.force).c_str()));
Expand Down Expand Up @@ -459,6 +472,7 @@ namespace airlib
bool enable_ground_lock_;
TTimePoint last_message_time;
Vector3r wind_;
Vector3r ext_force_;
};
}
} //namespace
Expand Down
1 change: 1 addition & 0 deletions AirLib/include/physics/PhysicsEngineBase.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@ namespace airlib
}

virtual void setWind(const Vector3r& wind) { unused(wind); };
virtual void setExtForce(const Vector3r& ext_force) { unused(ext_force); };
};
}
} //namespace
Expand Down
6 changes: 5 additions & 1 deletion AirLib/src/api/RpcLibClientBase.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -620,7 +620,11 @@ __pragma(warning(disable : 4239))
RpcLibAdaptorsBase::Vector3r conv_wind(wind);
pimpl_->client.call("simSetWind", conv_wind);
}

void RpcLibClientBase::simSetExtForce(const Vector3r& ext_force) const
{
RpcLibAdaptorsBase::Vector3r conv_ext_force(ext_force);
pimpl_->client.call("simSetExtForce", conv_ext_force);
}
vector<string> RpcLibClientBase::listVehicles()
{
return pimpl_->client.call("listVehicles").as<vector<string>>();
Expand Down
5 changes: 4 additions & 1 deletion AirLib/src/api/RpcLibServerBase.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -278,7 +278,6 @@ namespace airlib
const auto& response = getWorldSimApi()->getDetections(type, CameraDetails(camera_name, vehicle_name, external));
return RpcLibAdaptorsBase::DetectionInfo::from(response);
});

pimpl_->server.bind("reset", [&]() -> void {
//Exit if already resetting.
static bool resetInProgress;
Expand Down Expand Up @@ -501,6 +500,10 @@ namespace airlib
getWorldSimApi()->setWind(wind.to());
});

pimpl_->server.bind("simSetExtForce", [&](const RpcLibAdaptorsBase::Vector3r& ext_force) -> void {
getWorldSimApi()->setExtForce(ext_force.to());
});

pimpl_->server.bind("listVehicles", [&]() -> vector<string> {
return getWorldSimApi()->listVehicles();
});
Expand Down
14 changes: 12 additions & 2 deletions PythonClient/airsim/client.py
Original file line number Diff line number Diff line change
Expand Up @@ -1113,7 +1113,17 @@ def getSettingsString(self):
"""
return self.client.call('getSettingsString')

#----------------------------------- Multirotor APIs ---------------------------------------------
def simSetExtForce(self, ext_force):
"""
Set arbitrary external forces, in World frame, NED direction. Can be used
for implementing simple payloads.

Args:
ext_force (Vector3r): Force, in World frame, NED direction, in N
"""
self.client.call('simSetExtForce', ext_force)

# ----------------------------------- Multirotor APIs ---------------------------------------------
class MultirotorClient(VehicleClient, object):
def __init__(self, ip = "", port = 41451, timeout_value = 3600):
super(MultirotorClient, self).__init__(ip, port, timeout_value)
Expand Down Expand Up @@ -1618,4 +1628,4 @@ def getCarControls(self, vehicle_name=''):
CarControls:
"""
controls_raw = self.client.call('getCarControls', vehicle_name)
return CarControls.from_msgpack(controls_raw)
return CarControls.from_msgpack(controls_raw)
7 changes: 7 additions & 0 deletions Unreal/Plugins/AirSim/Source/SimMode/SimModeBase.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -312,6 +312,13 @@ void ASimModeBase::setWind(const msr::airlib::Vector3r& wind) const
throw std::domain_error("setWind not implemented by SimMode");
}

void ASimModeBase::setExtForce(const msr::airlib::Vector3r& ext_force) const
{
// should be overridden by derived class
unused(ext_force);
throw std::domain_error("setExtForce not implemented by SimMode");
}

std::unique_ptr<msr::airlib::ApiServerBase> ASimModeBase::createApiServer() const
{
//this will be the case when compilation with RPCLIB is disabled or simmode doesn't support APIs
Expand Down
1 change: 1 addition & 0 deletions Unreal/Plugins/AirSim/Source/SimMode/SimModeBase.h
Original file line number Diff line number Diff line change
Expand Up @@ -63,6 +63,7 @@ class AIRSIM_API ASimModeBase : public AActor
virtual void continueForFrames(uint32_t frames);

virtual void setWind(const msr::airlib::Vector3r& wind) const;
virtual void setExtForce(const msr::airlib::Vector3r& ext_force) const;

virtual void setTimeOfDay(bool is_enabled, const std::string& start_datetime, bool is_start_datetime_dst,
float celestial_clock_speed, float update_interval_secs, bool move_sun);
Expand Down
6 changes: 6 additions & 0 deletions Unreal/Plugins/AirSim/Source/SimMode/SimModeWorldBase.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -74,6 +74,7 @@ std::unique_ptr<ASimModeWorldBase::PhysicsEngineBase> ASimModeWorldBase::createP
}

physics_engine->setWind(getSettings().wind);
physics_engine->setExtForce(getSettings().ext_force);
}
else if (physics_engine_name == "ExternalPhysicsEngine") {
physics_engine.reset(new msr::airlib::ExternalPhysicsEngine());
Expand Down Expand Up @@ -136,6 +137,11 @@ void ASimModeWorldBase::setWind(const msr::airlib::Vector3r& wind) const
physics_engine_->setWind(wind);
}

void ASimModeWorldBase::setExtForce(const msr::airlib::Vector3r& ext_force) const
{
physics_engine_->setExtForce(ext_force);
}

void ASimModeWorldBase::updateDebugReport(msr::airlib::StateReporterWrapper& debug_reporter)
{
unused(debug_reporter);
Expand Down
1 change: 1 addition & 0 deletions Unreal/Plugins/AirSim/Source/SimMode/SimModeWorldBase.h
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@ class AIRSIM_API ASimModeWorldBase : public ASimModeBase
virtual void continueForFrames(uint32_t frames) override;

virtual void setWind(const msr::airlib::Vector3r& wind) const override;
virtual void setExtForce(const msr::airlib::Vector3r& ext_force) const override;

protected:
void startAsyncUpdator();
Expand Down
5 changes: 5 additions & 0 deletions Unreal/Plugins/AirSim/Source/WorldSimApi.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -727,6 +727,11 @@ void WorldSimApi::setWind(const Vector3r& wind) const
simmode_->setWind(wind);
}

void WorldSimApi::setExtForce(const Vector3r& ext_force) const
{
simmode_->setExtForce(ext_force);
}

std::vector<std::string> WorldSimApi::listVehicles() const
{
std::vector<std::string> vehicle_names;
Expand Down
1 change: 1 addition & 0 deletions Unreal/Plugins/AirSim/Source/WorldSimApi.h
Original file line number Diff line number Diff line change
Expand Up @@ -76,6 +76,7 @@ class WorldSimApi : public msr::airlib::WorldSimApiBase
virtual bool isRecording() const override;

virtual void setWind(const Vector3r& wind) const override;
virtual void setExtForce(const Vector3r& ext_force) const override;
virtual bool createVoxelGrid(const Vector3r& position, const int& x_size, const int& y_size, const int& z_size, const float& res, const std::string& output_file) override;
virtual std::vector<std::string> listVehicles() const override;

Expand Down