Program Listing for File model.hxx

Return to documentation for file (include/pinocchio/parsers/urdf/model.hxx)

//
// Copyright (c) 2015-2021 CNRS INRIA
// Copyright (c) 2015 Wandercraft, 86 rue de Paris 91400 Orsay, France.
//

#ifndef __pinocchio_multibody_parsers_urdf_model_hxx__
#define __pinocchio_multibody_parsers_urdf_model_hxx__

#include "pinocchio/math/matrix.hpp"
#include "pinocchio/parsers/config.hpp"
#include "pinocchio/parsers/urdf.hpp"
#include "pinocchio/multibody/model.hpp"

#include <sstream>
#include <boost/foreach.hpp>
#include <limits>
#include <iostream>
#include <boost/optional.hpp>
#include <boost/none.hpp>

namespace pinocchio
{
  namespace urdf
  {
    namespace details
    {
      typedef double urdf_scalar_type;

      template<typename _Scalar, int Options>
      struct MimicInfo;

      template<typename _Scalar, int Options>
      class UrdfVisitorBaseTpl
      {
      public:
        enum JointType
        {
          REVOLUTE,
          CONTINUOUS,
          PRISMATIC,
          FLOATING,
          PLANAR,
          SPHERICAL,
          MIMIC
        };

        typedef enum ::pinocchio::FrameType FrameType;
        typedef _Scalar Scalar;
        typedef SE3Tpl<Scalar, Options> SE3;
        typedef InertiaTpl<Scalar, Options> Inertia;

        typedef Eigen::Matrix<Scalar, 3, 1> Vector3;
        typedef Eigen::Matrix<Scalar, Eigen::Dynamic, 1> Vector;
        typedef Eigen::Ref<Vector> VectorRef;
        typedef Eigen::Ref<const Vector> VectorConstRef;
        typedef MimicInfo<Scalar, Options> MimicInfo_;

        virtual void setName(const std::string & name) = 0;

        virtual void addRootJoint(const Inertia & Y, const std::string & body_name) = 0;

        virtual void addJointAndBody(
          JointType type,
          const Vector3 & axis,
          const FrameIndex & parentFrameId,
          const SE3 & placement,
          const std::string & joint_name,
          const Inertia & Y,
          const std::string & body_name,
          const VectorConstRef & max_effort,
          const VectorConstRef & max_velocity,
          const VectorConstRef & min_config,
          const VectorConstRef & max_config,
          const VectorConstRef & friction,
          const VectorConstRef & damping,
          const boost::optional<MimicInfo_> & mimic_info = boost::none) = 0;

        virtual void addJointAndBody(
          JointType type,
          const Vector3 & axis,
          const FrameIndex & parentFrameId,
          const SE3 & placement,
          const std::string & joint_name,
          const Inertia & Y,
          const SE3 & frame_placement,
          const std::string & body_name,
          const VectorConstRef & max_effort,
          const VectorConstRef & max_velocity,
          const VectorConstRef & min_config,
          const VectorConstRef & max_config,
          const VectorConstRef & friction,
          const VectorConstRef & damping,
          const boost::optional<MimicInfo_> & mimic_info = boost::none) = 0;

        virtual void addFixedJointAndBody(
          const FrameIndex & parentFrameId,
          const SE3 & joint_placement,
          const std::string & joint_name,
          const Inertia & Y,
          const std::string & body_name) = 0;

        virtual void appendBodyToJoint(
          const FrameIndex fid,
          const Inertia & Y,
          const SE3 & placement,
          const std::string & body_name) = 0;

        virtual FrameIndex getBodyId(const std::string & frame_name) const = 0;

        virtual JointIndex getJointId(const std::string & joint_name) const = 0;

        virtual const std::string & getJointName(const JointIndex jointId) const = 0;

        virtual Frame getBodyFrame(const std::string & frame_name) const = 0;

        virtual JointIndex getParentId(const std::string & frame_name) const = 0;

        virtual bool existFrame(const std::string & frame_name, const FrameType type) const = 0;

        UrdfVisitorBaseTpl()
        : log(NULL)
        {
        }

        template<typename T>
        UrdfVisitorBaseTpl & operator<<(const T & t)
        {
          if (log != NULL)
            *log << t;
          return *this;
        }

        std::ostream * log;
      };

      template<typename _Scalar, int Options>
      struct MimicInfo
      {
        typedef _Scalar Scalar;
        typedef Eigen::Matrix<Scalar, 3, 1> Vector3;

        std::string mimicked_name;
        Scalar multiplier;
        Scalar offset;
        Vector3 axis;

        // Use the JointType from UrdfVisitorBaseTpl
        typedef typename UrdfVisitorBaseTpl<_Scalar, Options>::JointType JointType;
        JointType jointType;

        MimicInfo() = default;

        MimicInfo(
          std::string _mimicked_name,
          Scalar _multiplier,
          Scalar _offset,
          Vector3 _axis,
          JointType _jointType)
        : mimicked_name(_mimicked_name)
        , multiplier(_multiplier)
        , offset(_offset)
        , axis(_axis)
        , jointType(_jointType)
        {
        }
      };

      template<typename Scalar, int Options, template<typename, int> class JointCollectionTpl>
      class UrdfVisitor : public UrdfVisitorBaseTpl<Scalar, Options>
      {
      public:
        typedef UrdfVisitorBaseTpl<Scalar, Options> Base;
        typedef typename Base::JointType JointType;
        typedef typename Base::Vector3 Vector3;
        typedef typename Base::VectorConstRef VectorConstRef;
        typedef typename Base::SE3 SE3;
        typedef typename Base::Inertia Inertia;

        typedef ModelTpl<Scalar, Options, JointCollectionTpl> Model;
        typedef typename Model::JointCollection JointCollection;
        typedef typename Model::JointModel JointModel;
        typedef typename Model::Frame Frame;

        typedef MimicInfo<Scalar, Options> MimicInfo_;

        Model & model;
        std::string root_joint_name;

        UrdfVisitor(Model & model)
        : model(model)
        {
        }

        UrdfVisitor(Model & model, const std::string & rjn)
        : model(model)
        , root_joint_name(rjn)
        {
        }

        void setName(const std::string & name)
        {
          model.name = name;
        }

        virtual void addRootJoint(const Inertia & Y, const std::string & body_name)
        {
          const Frame & parent_frame = model.frames[0];

          model.addFrame(
            Frame(body_name, parent_frame.parentJoint, 0, parent_frame.placement, BODY, Y));
        }

        void addJointAndBody(
          JointType type,
          const Vector3 & axis,
          const FrameIndex & parentFrameId,
          const SE3 & placement,
          const std::string & joint_name,
          const Inertia & Y,
          const std::string & body_name,
          const VectorConstRef & max_effort,
          const VectorConstRef & max_velocity,
          const VectorConstRef & min_config,
          const VectorConstRef & max_config,
          const VectorConstRef & friction,
          const VectorConstRef & damping,
          const boost::optional<MimicInfo_> & mimic_info = boost::none)
        {
          addJointAndBody(
            type, axis, parentFrameId, placement, joint_name, Y, SE3::Identity(), body_name,
            max_effort, max_velocity, min_config, max_config, friction, damping, mimic_info);
        }

        void addJointAndBody(
          JointType type,
          const Vector3 & axis,
          const FrameIndex & parentFrameId,
          const SE3 & placement,
          const std::string & joint_name,
          const Inertia & Y,
          const SE3 & frame_placement,
          const std::string & body_name,
          const VectorConstRef & max_effort,
          const VectorConstRef & max_velocity,
          const VectorConstRef & min_config,
          const VectorConstRef & max_config,
          const VectorConstRef & friction,
          const VectorConstRef & damping,
          const boost::optional<MimicInfo_> & mimic_info = boost::none)
        {
          JointIndex joint_id;
          const Frame & frame = model.frames[parentFrameId];
          switch (type)
          {
          case Base::FLOATING:
            joint_id = model.addJoint(
              frame.parentJoint, typename JointCollection::JointModelFreeFlyer(),
              frame.placement * placement, joint_name, max_effort, max_velocity, min_config,
              max_config, friction, damping);
            break;
          case Base::REVOLUTE:
            joint_id = addJoint<
              typename JointCollection::JointModelRX, typename JointCollection::JointModelRY,
              typename JointCollection::JointModelRZ,
              typename JointCollection::JointModelRevoluteUnaligned>(
              axis, frame, placement, joint_name, max_effort, max_velocity, min_config, max_config,
              friction, damping);
            break;
          case Base::CONTINUOUS:
            joint_id = addJoint<
              typename JointCollection::JointModelRUBX, typename JointCollection::JointModelRUBY,
              typename JointCollection::JointModelRUBZ,
              typename JointCollection::JointModelRevoluteUnboundedUnaligned>(
              axis, frame, placement, joint_name, max_effort, max_velocity, min_config, max_config,
              friction, damping);
            break;
          case Base::PRISMATIC:
            joint_id = addJoint<
              typename JointCollection::JointModelPX, typename JointCollection::JointModelPY,
              typename JointCollection::JointModelPZ,
              typename JointCollection::JointModelPrismaticUnaligned>(
              axis, frame, placement, joint_name, max_effort, max_velocity, min_config, max_config,
              friction, damping);
            break;
          case Base::PLANAR:
            joint_id = model.addJoint(
              frame.parentJoint, typename JointCollection::JointModelPlanar(),
              frame.placement * placement, joint_name, max_effort, max_velocity, min_config,
              max_config, friction, damping);
            break;
          case Base::SPHERICAL:
            joint_id = model.addJoint(
              frame.parentJoint, typename JointCollection::JointModelSpherical(),
              frame.placement * placement, joint_name, max_effort, max_velocity, min_config,
              max_config, friction, damping);
            break;
          case Base::MIMIC:
            if (mimic_info)
              switch (mimic_info->jointType)
              {
              case Base::REVOLUTE:
                joint_id = addMimicJoint<
                  typename JointCollection::JointModelRX, typename JointCollection::JointModelRY,
                  typename JointCollection::JointModelRZ,
                  typename JointCollection::JointModelRevoluteUnaligned>(
                  frame, placement, joint_name, max_effort, max_velocity, min_config, max_config,
                  friction, damping, *mimic_info);
                break;
              case Base::PRISMATIC:
                joint_id = addMimicJoint<
                  typename JointCollection::JointModelPX, typename JointCollection::JointModelPY,
                  typename JointCollection::JointModelPZ,
                  typename JointCollection::JointModelPrismaticUnaligned>(
                  frame, placement, joint_name, max_effort, max_velocity, min_config, max_config,
                  friction, damping, *mimic_info);
                break;
              case Base::CONTINUOUS:
                joint_id = addMimicJoint<
                  typename JointCollection::JointModelRUBX,
                  typename JointCollection::JointModelRUBY,
                  typename JointCollection::JointModelRUBZ,
                  typename JointCollection::JointModelRevoluteUnboundedUnaligned>(
                  frame, placement, joint_name, max_effort, max_velocity, min_config, max_config,
                  friction, damping, *mimic_info);
                break;
              default:
                PINOCCHIO_CHECK_INPUT_ARGUMENT(
                  false, "Cannot mimic this type. Only revolute, prismatic and helicoidal can be "
                         "mimicked");
                break;
              }
            else
              PINOCCHIO_CHECK_INPUT_ARGUMENT(false, "Cannot create mimic joint. Missing info.")
            break;
          default:
            PINOCCHIO_CHECK_INPUT_ARGUMENT(false, "The joint type is not correct.");
          };

          FrameIndex jointFrameId = model.addJointFrame(joint_id, (int)parentFrameId);
          appendBodyToJoint(jointFrameId, Y, frame_placement, body_name);
        }

        void addFixedJointAndBody(
          const FrameIndex & parent_frame_id,
          const SE3 & joint_placement,
          const std::string & joint_name,
          const Inertia & Y,
          const std::string & body_name)
        {
          const Frame & parent_frame = model.frames[parent_frame_id];
          const JointIndex parent_frame_parent = parent_frame.parentJoint;

          const SE3 placement = parent_frame.placement * joint_placement;
          FrameIndex fid = model.addFrame(Frame(
            joint_name, parent_frame.parentJoint, parent_frame_id, placement, FIXED_JOINT, Y));

          model.addBodyFrame(body_name, parent_frame_parent, placement, (int)fid);
        }

        void appendBodyToJoint(
          const FrameIndex fid,
          const Inertia & Y,
          const SE3 & placement,
          const std::string & body_name)
        {
          const Frame & frame = model.frames[fid];
          const SE3 & p = frame.placement * placement;
          assert(frame.parentJoint >= 0);
          if (!Y.isZero(Scalar(0)))
          {
            model.appendBodyToJoint(frame.parentJoint, Y, p);
          }

          model.addBodyFrame(body_name, frame.parentJoint, p, (int)fid);
          // Reference to model.frames[fid] can has changed because the vector
          // may have been reallocated.
          assert(model.frames[fid].parentJoint >= 0);
          {
            assert(
              !hasNaN(model.inertias[model.frames[fid].parentJoint].lever())
              && !hasNaN(model.inertias[model.frames[fid].parentJoint].inertia().data()));
          }
        }

        FrameIndex getBodyId(const std::string & frame_name) const
        {

          if (model.existFrame(frame_name, BODY))
          {
            FrameIndex fid = model.getFrameId(frame_name, BODY);
            assert(model.frames[fid].type == BODY);
            return fid;
          }
          else
            throw std::invalid_argument("Model does not have any body named " + frame_name);
        }

        FrameIndex getJointId(const std::string & joint_name) const
        {

          if (model.existJointName(joint_name))
          {
            JointIndex jid = model.getJointId(joint_name);
            return jid;
          }
          else
            throw std::invalid_argument("Model does not have any joint named " + joint_name);
        }

        const std::string & getJointName(const JointIndex jointId) const
        {
          return model.names[jointId];
        }

        Frame getBodyFrame(const std::string & frame_name) const
        {

          if (model.existFrame(frame_name, BODY))
          {
            FrameIndex fid = model.getFrameId(frame_name, BODY);
            assert(model.frames[fid].type == BODY);
            return model.frames[fid];
          }
          else
            throw std::invalid_argument("Model does not have any body named " + frame_name);
        }

        JointIndex getParentId(const std::string & frame_name) const
        {

          if (model.existFrame(frame_name, BODY))
          {
            FrameIndex fid = model.getFrameId(frame_name, BODY);
            assert(model.frames[fid].type == BODY);
            return model.frames[fid].parentJoint;
          }
          else
            throw std::invalid_argument("Model does not have any body named " + frame_name);
        }

        bool existFrame(const std::string & frame_name, const FrameType type) const
        {
          return model.existFrame(frame_name, type);
        }

        template<typename TypeX, typename TypeY, typename TypeZ, typename TypeUnaligned>
        JointIndex addJoint(
          const Vector3 & axis,
          const Frame & frame,
          const SE3 & placement,
          const std::string & joint_name,
          const VectorConstRef & max_effort,
          const VectorConstRef & max_velocity,
          const VectorConstRef & min_config,
          const VectorConstRef & max_config,
          const VectorConstRef & friction,
          const VectorConstRef & damping)
        {
          CartesianAxis axisType = extractCartesianAxis(axis);
          switch (axisType)
          {
          case AXIS_X:
            return model.addJoint(
              frame.parentJoint, TypeX(), frame.placement * placement, joint_name, max_effort,
              max_velocity, min_config, max_config, friction, damping);
            break;

          case AXIS_Y:
            return model.addJoint(
              frame.parentJoint, TypeY(), frame.placement * placement, joint_name, max_effort,
              max_velocity, min_config, max_config, friction, damping);
            break;

          case AXIS_Z:
            return model.addJoint(
              frame.parentJoint, TypeZ(), frame.placement * placement, joint_name, max_effort,
              max_velocity, min_config, max_config, friction, damping);
            break;

          case AXIS_UNALIGNED:
            return model.addJoint(
              frame.parentJoint, TypeUnaligned(axis.normalized()), frame.placement * placement,
              joint_name, max_effort, max_velocity, min_config, max_config, friction, damping);
            break;
          default:
            PINOCCHIO_CHECK_INPUT_ARGUMENT(false, "The axis type of the joint is of wrong type.");
            break;
          }
        }

        template<typename TypeX, typename TypeY, typename TypeZ, typename TypeUnaligned>
        JointIndex addMimicJoint(
          const Frame & frame,
          const SE3 & placement,
          const std::string & joint_name,
          const VectorConstRef & max_effort,
          const VectorConstRef & max_velocity,
          const VectorConstRef & min_config,
          const VectorConstRef & max_config,
          const VectorConstRef & friction,
          const VectorConstRef & damping,
          const MimicInfo_ & mimic_info)
        {
          if (!model.existJointName(mimic_info.mimicked_name))
            PINOCCHIO_CHECK_INPUT_ARGUMENT(
              false, "The parent joint of the mimic joint is not in the kinematic tree");

          auto mimicked_joint = model.joints[getJointId(mimic_info.mimicked_name)];

          CartesianAxis axisType = extractCartesianAxis(mimic_info.axis);
          switch (axisType)
          {
          case AXIS_X:
            return model.addJoint(
              frame.parentJoint,
              typename JointCollection::JointModelMimic(
                TypeX(), mimicked_joint, mimic_info.multiplier, mimic_info.offset),
              frame.placement * placement, joint_name, max_effort, max_velocity, min_config,
              max_config, friction, damping);
            break;
          case AXIS_Y:
            return model.addJoint(
              frame.parentJoint,
              typename JointCollection::JointModelMimic(
                TypeY(), mimicked_joint, mimic_info.multiplier, mimic_info.offset),
              frame.placement * placement, joint_name, max_effort, max_velocity, min_config,
              max_config, friction, damping);
            break;

          case AXIS_Z:
            return model.addJoint(
              frame.parentJoint,
              typename JointCollection::JointModelMimic(
                TypeZ(), mimicked_joint, mimic_info.multiplier, mimic_info.offset),
              frame.placement * placement, joint_name, max_effort, max_velocity, min_config,
              max_config, friction, damping);
            break;

          case AXIS_UNALIGNED:
            return model.addJoint(
              frame.parentJoint,
              typename JointCollection::JointModelMimic(
                TypeUnaligned(mimic_info.axis.normalized()), mimicked_joint, mimic_info.multiplier,
                mimic_info.offset),
              frame.placement * placement, joint_name, max_effort, max_velocity, min_config,
              max_config, friction, damping);
            break;

          default:
            PINOCCHIO_CHECK_INPUT_ARGUMENT(false, "The axis type of the joint is of wrong type.");
            break;
          }
        }

      private:
        enum CartesianAxis
        {
          AXIS_X = 0,
          AXIS_Y = 1,
          AXIS_Z = 2,
          AXIS_UNALIGNED
        };

        static inline CartesianAxis extractCartesianAxis(const Vector3 & axis)
        {
          if (axis.isApprox(Vector3::UnitX()))
            return AXIS_X;
          else if (axis.isApprox(Vector3::UnitY()))
            return AXIS_Y;
          else if (axis.isApprox(Vector3::UnitZ()))
            return AXIS_Z;
          else
            return AXIS_UNALIGNED;
        }
      };

      template<typename Scalar, int Options, template<typename, int> class JointCollectionTpl>
      class UrdfVisitorWithRootJoint : public UrdfVisitor<Scalar, Options, JointCollectionTpl>
      {
      public:
        typedef UrdfVisitor<Scalar, Options, JointCollectionTpl> Base;
        typedef typename Base::Inertia Inertia;
        using Base::appendBodyToJoint;
        using Base::model;

        typedef ModelTpl<Scalar, Options, JointCollectionTpl> Model;
        typedef typename Model::JointCollection JointCollection;
        typedef typename Model::JointModel JointModel;

        JointModel root_joint;

        UrdfVisitorWithRootJoint(
          Model & model,
          const JointModelBase<JointModel> & root_joint,
          const std::string & rjn = "root_joint")
        : Base(model, rjn)
        , root_joint(root_joint.derived())
        {
        }

        void addRootJoint(const Inertia & Y, const std::string & body_name)
        {
          const Frame & frame = model.frames[0];

          PINOCCHIO_THROW(
            !model.existJointName(this->root_joint_name), std::invalid_argument,
            "root_joint already exists as a joint in the kinematic tree.");

          JointIndex idx = model.addJoint(
            frame.parentJoint, root_joint, SE3::Identity(), this->root_joint_name
            // TODO ,max_effort,max_velocity,min_config,max_config
          );

          FrameIndex jointFrameId = model.addJointFrame(idx, 0);
          appendBodyToJoint(jointFrameId, Y, SE3::Identity(), body_name);
        }
      };

      typedef UrdfVisitorBaseTpl<double, 0> UrdfVisitorBase;
      typedef MimicInfo<double, 0> MimicInfo_;

      PINOCCHIO_PARSERS_DLLAPI void parseRootTree(
        const ::urdf::ModelInterface * urdfTree, UrdfVisitorBase & model, const bool mimic = false);

      PINOCCHIO_PARSERS_DLLAPI void parseRootTree(
        const std::string & filename, UrdfVisitorBase & model, const bool mimic = false);

      PINOCCHIO_PARSERS_DLLAPI void parseRootTreeFromXML(
        const std::string & xmlString, UrdfVisitorBase & model, const bool mimic = false);
    } // namespace details

    template<typename Scalar, int Options, template<typename, int> class JointCollectionTpl>
    ModelTpl<Scalar, Options, JointCollectionTpl> & buildModel(
      const std::string & filename,
      const typename ModelTpl<Scalar, Options, JointCollectionTpl>::JointModel & rootJoint,
      const std::string & rootJointName,
      ModelTpl<Scalar, Options, JointCollectionTpl> & model,
      const bool verbose,
      const bool mimic)
    {
      if (rootJointName.empty())
        throw std::invalid_argument(
          "rootJoint was given without a name. Please fill the argument root_joint_name");

      details::UrdfVisitorWithRootJoint<Scalar, Options, JointCollectionTpl> visitor(
        model, rootJoint, rootJointName);
      if (verbose)
        visitor.log = &std::cout;
      details::parseRootTree(filename, visitor, mimic);
      return model;
    }

    template<typename Scalar, int Options, template<typename, int> class JointCollectionTpl>
    ModelTpl<Scalar, Options, JointCollectionTpl> & buildModel(
      const std::string & filename,
      const typename ModelTpl<Scalar, Options, JointCollectionTpl>::JointModel & rootJoint,
      ModelTpl<Scalar, Options, JointCollectionTpl> & model,
      const bool verbose,
      const bool mimic)
    {
      return buildModel(filename, rootJoint, "root_joint", model, verbose, mimic);
    }

    template<typename Scalar, int Options, template<typename, int> class JointCollectionTpl>
    ModelTpl<Scalar, Options, JointCollectionTpl> & buildModel(
      const std::string & filename,
      ModelTpl<Scalar, Options, JointCollectionTpl> & model,
      const bool verbose,
      const bool mimic)
    {
      details::UrdfVisitor<Scalar, Options, JointCollectionTpl> visitor(model);
      if (verbose)
        visitor.log = &std::cout;
      details::parseRootTree(filename, visitor, mimic);
      return model;
    }

    template<typename Scalar, int Options, template<typename, int> class JointCollectionTpl>
    ModelTpl<Scalar, Options, JointCollectionTpl> & buildModelFromXML(
      const std::string & xmlStream,
      const typename ModelTpl<Scalar, Options, JointCollectionTpl>::JointModel & rootJoint,
      const std::string & rootJointName,
      ModelTpl<Scalar, Options, JointCollectionTpl> & model,
      const bool verbose,
      const bool mimic)
    {
      if (rootJointName.empty())
        throw std::invalid_argument(
          "rootJoint was given without a name. Please fill the argument rootJointName");

      details::UrdfVisitorWithRootJoint<Scalar, Options, JointCollectionTpl> visitor(
        model, rootJoint, rootJointName);
      if (verbose)
        visitor.log = &std::cout;
      details::parseRootTreeFromXML(xmlStream, visitor, mimic);
      return model;
    }

    template<typename Scalar, int Options, template<typename, int> class JointCollectionTpl>
    ModelTpl<Scalar, Options, JointCollectionTpl> & buildModelFromXML(
      const std::string & xmlStream,
      const typename ModelTpl<Scalar, Options, JointCollectionTpl>::JointModel & rootJoint,
      ModelTpl<Scalar, Options, JointCollectionTpl> & model,
      const bool verbose,
      const bool mimic)
    {
      return buildModelFromXML(xmlStream, rootJoint, "root_joint", model, verbose, mimic);
    }

    template<typename Scalar, int Options, template<typename, int> class JointCollectionTpl>
    ModelTpl<Scalar, Options, JointCollectionTpl> & buildModelFromXML(
      const std::string & xmlStream,
      ModelTpl<Scalar, Options, JointCollectionTpl> & model,
      const bool verbose,
      const bool mimic)
    {
      details::UrdfVisitor<Scalar, Options, JointCollectionTpl> visitor(model);
      if (verbose)
        visitor.log = &std::cout;
      details::parseRootTreeFromXML(xmlStream, visitor, mimic);
      return model;
    }

    template<typename Scalar, int Options, template<typename, int> class JointCollectionTpl>
    ModelTpl<Scalar, Options, JointCollectionTpl> & buildModel(
      const std::shared_ptr<::urdf::ModelInterface> urdfTree,
      const typename ModelTpl<Scalar, Options, JointCollectionTpl>::JointModel & rootJoint,
      const std::string & rootJointName,
      ModelTpl<Scalar, Options, JointCollectionTpl> & model,
      const bool verbose,
      const bool mimic)
    {
      if (rootJointName.empty())
        throw std::invalid_argument(
          "rootJoint was given without a name. Please fill the argument rootJointName");

      PINOCCHIO_CHECK_INPUT_ARGUMENT(urdfTree != NULL);
      details::UrdfVisitorWithRootJoint<Scalar, Options, JointCollectionTpl> visitor(
        model, rootJoint, rootJointName);
      if (verbose)
        visitor.log = &std::cout;
      details::parseRootTree(urdfTree.get(), visitor, mimic);
      return model;
    }

    template<typename Scalar, int Options, template<typename, int> class JointCollectionTpl>
    ModelTpl<Scalar, Options, JointCollectionTpl> & buildModel(
      const std::shared_ptr<::urdf::ModelInterface> urdfTree,
      const typename ModelTpl<Scalar, Options, JointCollectionTpl>::JointModel & rootJoint,
      ModelTpl<Scalar, Options, JointCollectionTpl> & model,
      const bool verbose,
      const bool mimic)
    {
      return buildModel(urdfTree, rootJoint, "root_joint", model, verbose, mimic);
    }

    template<typename Scalar, int Options, template<typename, int> class JointCollectionTpl>
    ModelTpl<Scalar, Options, JointCollectionTpl> & buildModel(
      const std::shared_ptr<::urdf::ModelInterface> urdfTree,
      ModelTpl<Scalar, Options, JointCollectionTpl> & model,
      const bool verbose,
      const bool mimic)
    {
      PINOCCHIO_CHECK_INPUT_ARGUMENT(urdfTree != NULL);
      details::UrdfVisitor<Scalar, Options, JointCollectionTpl> visitor(model);
      if (verbose)
        visitor.log = &std::cout;
      details::parseRootTree(urdfTree.get(), visitor, mimic);
      return model;
    }

  } // namespace urdf
} // namespace pinocchio

#endif // ifndef __pinocchio_multibody_parsers_urdf_model_hxx__