23#include <condition_variable>
31#include <rclcpp/rclcpp.hpp>
32#include <realtime_tools/realtime_publisher.hpp>
33#include <rosgraph_msgs/msg/clock.hpp>
35#include "glfw_adapter.h"
38#include <mujoco/mujoco.h>
40#include <mujoco_ros2_control_msgs/srv/reset_world.hpp>
41#include <mujoco_ros2_control_msgs/srv/set_pause.hpp>
42#include <mujoco_ros2_control_msgs/srv/step_simulation.hpp>
43#include <mujoco_ros2_control_plugins/mujoco_ros2_control_plugins_base.hpp>
114 bool initialize(rclcpp::Node::SharedPtr node,
const std::string& model_path,
const std::string& mujoco_model_topic,
115 double sim_speed_factor,
bool headless);
211 RCLCPP_WARN_EXPRESSION(logger_, sim_mutex_ ==
nullptr,
"Sim recursive mutex is still nullptr");
222 return step_count_.load();
231 void handle_xfrc_applied();
241 void publish_clock();
246 void update_sim_display();
249 void reset_world_callback(
const std::shared_ptr<mujoco_ros2_control_msgs::srv::ResetWorld::Request> request,
250 std::shared_ptr<mujoco_ros2_control_msgs::srv::ResetWorld::Response> response);
251 void set_pause_callback(
const std::shared_ptr<mujoco_ros2_control_msgs::srv::SetPause::Request> request,
252 std::shared_ptr<mujoco_ros2_control_msgs::srv::SetPause::Response> response);
253 void step_simulation_callback(
const std::shared_ptr<mujoco_ros2_control_msgs::srv::StepSimulation::Request> request,
254 std::shared_ptr<mujoco_ros2_control_msgs::srv::StepSimulation::Response> response);
256 rclcpp::Logger get_logger()
const
262 rclcpp::Logger logger_ = rclcpp::get_logger(
"MujocoSimulation");
265 rclcpp::Node::SharedPtr node_;
268 std::string model_path_;
269 std::string mujoco_model_topic_;
273 mjModel* mj_model_{
nullptr };
277 mjData* mj_data_{
nullptr };
281 std::vector<mjtNum> xfrc_plugin_desired_;
282 std::vector<mjtNum> xfrc_viewer_capture_;
283 std::vector<mjtNum> xfrc_last_written_;
292 double sim_speed_factor_{ -1.0 };
295 bool headless_{
false };
298 std::unique_ptr<mujoco::Simulate> sim_;
301 std::thread physics_thread_;
302 std::thread ui_thread_;
305 std::shared_ptr<rclcpp::Publisher<rosgraph_msgs::msg::Clock>> clock_publisher_;
313 std::recursive_mutex* sim_mutex_{
nullptr };
316 rclcpp::CallbackGroup::SharedPtr reset_world_cb_group_;
317 rclcpp::Service<mujoco_ros2_control_msgs::srv::ResetWorld>::SharedPtr reset_world_service_;
320 rclcpp::CallbackGroup::SharedPtr set_pause_cb_group_;
321 rclcpp::Service<mujoco_ros2_control_msgs::srv::SetPause>::SharedPtr set_pause_service_;
324 rclcpp::CallbackGroup::SharedPtr step_simulation_cb_group_;
325 rclcpp::Service<mujoco_ros2_control_msgs::srv::StepSimulation>::SharedPtr step_simulation_service_;
328 std::atomic<uint32_t> pending_steps_{ 0 };
329 std::atomic<bool> step_diverged_{
false };
330 std::atomic<bool> steps_interrupted_{
false };
331 std::atomic<bool> keyboard_step_requested_{
false };
332 std::atomic<uint64_t> step_count_{ 0 };
333 std::mutex steps_cv_mutex_;
334 std::condition_variable steps_cv_;
337 std::vector<mjtNum> initial_qpos_;
338 std::vector<mjtNum> initial_qvel_;
339 std::vector<mjtNum> initial_ctrl_;
ROS 2-based container for the mujoco Simulate application.
Definition mujoco_simulation.hpp:87
void capture_initial_state()
Can be called by consumers of this class to store the current state as the "initial" state.
Definition mujoco_simulation.cpp:681
void reset_world_state(bool fill_initial_state)
Reset simulation state (qpos/qvel/ctrl/sensors/forces) to the captured initial state.
Definition mujoco_simulation.cpp:750
std::recursive_mutex & mutex() const
Accessor for the mutex which locks access to the data and model.
Definition mujoco_simulation.hpp:209
void start_physics_thread()
Start the physics thread. Must be called after load_model().
Definition mujoco_simulation.cpp:694
std::function< void(bool fill_initial_state)> ResetCallback
Callback invoked when the simulation's state must be reset.
Definition mujoco_simulation.hpp:99
void apply_control_data(mjData *control_data)
Copies control fields from control_data into the sim data in a thread safe way.
Definition mujoco_simulation.cpp:950
void copy_physics_data(mjData *&destination)
Copies mj_data_ into the provided container in a thread safe way.
Definition mujoco_simulation.cpp:939
void shutdown()
Stop the physics and UI threads if they are running.
Definition mujoco_simulation.cpp:731
bool apply_keyframe(const std::string &keyframe_name)
Apply a keyframe to the simulation by name.
Definition mujoco_simulation.cpp:663
void copy_physics_model(mjModel *&destination)
Copies mj_model_ into the provided container in a thread safe way.
Definition mujoco_simulation.cpp:927
MujocoSimulation()=default
Construct a new Mujoco Simulation object. This is a no-op until initialization.
mjModel * model()
Accessor for the mujoco model.
Definition mujoco_simulation.hpp:151
mjData * data()
Accessor for the raw mujoco simulation data.
Definition mujoco_simulation.hpp:163
uint64_t step_count() const
Returns the number of steps takein by the physics simulation.
Definition mujoco_simulation.hpp:220
void overwrite_physics_data(mjData *source)
Copies the provided mjData into mj_data_ in a thread safe way.
Definition mujoco_simulation.cpp:933
void set_reset_callback(ResetCallback callback)
Register a callback function to be called on reset_world_state.
Definition mujoco_simulation.cpp:689
bool initialize(rclcpp::Node::SharedPtr node, const std::string &model_path, const std::string &mujoco_model_topic, double sim_speed_factor, bool headless)
Construct the Simulate application and start the UI thread (if not headless).
Definition mujoco_simulation.cpp:482