#ifndef MILLARD2016TORQUEMUSCLE_H_ #define MILLARD2016TORQUEMUSCLE_H_ /* * RBDL - Rigid Body Dynamics Library: Addon : forceElements * Copyright (c) 2016 Matthew Millard * * Licensed under the zlib license. See LICENSE for more details. */ #include #include #include "../geometry/SmoothSegmentedFunction.h" namespace RigidBodyDynamics { namespace Addons { namespace Muscle{ /** This struct contains an enumerated list of the data sets which constain torque muscles. Thus far this list includes -Anderson2007: Table 3 from Anderson et al. -Gymnast: A set of torque muscles for the whole body (in progress) For details of these data sets please refer to the main description of this class. */ const static struct DataSet{ enum item{ Anderson2007 = 0, Gymnast, Last }; const static char* names[]; DataSet(){} } DataSet; /** This struct contains an enumerated list of the genders for which data torque data has been reported. */ const static struct GenderSet{ enum item { Male = 0, Female, Last }; const static char* names[]; GenderSet(){} } GenderSet; /** This struct contains an enumerated list of the age groups for which data torque data has been reported. */ const static struct AgeGroupSet{ enum item { Young18To25 = 0, Middle55To65, SeniorOver65, Last }; const static char* names[]; AgeGroupSet(){} } AgeGroupSet; /** This struct contains an enumerated list of the joint-torque-directions for which data torque data has been reported. */ const static struct JointTorqueSet{ enum item{ HipExtension = 0, HipFlexion = 1, KneeExtension = 2, KneeFlexion = 3, AnkleExtension = 4, AnkleFlexion = 5, ElbowExtension = 6, ElbowFlexion = 7, ShoulderExtension = 8, ShoulderFlexion = 9, WristExtension = 10, WristFlexion = 11, ShoulderHorizontalAdduction = 12, ShoulderHorizontalAbduction = 13, ShoulderInternalRotation = 14, ShoulderExternalRotation = 15, WristUlnarDeviation = 16, WristRadialDeviation = 17, WristPronation = 18, WristSupination = 19, LumbarExtension = 20, LumbarFlexion = 21, Last = 22 }; const static char* names[]; JointTorqueSet(){} } JointTorqueSet; /** This struct contains 3 enumerated lists (Gender, AgeGroup, JointTorque) that lists the genders, age groups, and joint-torque-directions for which the Anderson2007 data set contains data. Please refer to the class description for more details about this data set and how to use it. */ const static struct Anderson2007{ enum Gender { Male = GenderSet::Male, Female = GenderSet::Female, LastGender }; enum AgeGroup { Young18To25 = AgeGroupSet::Young18To25, Middle55To65 = AgeGroupSet::Middle55To65, SeniorOver65 = AgeGroupSet::SeniorOver65, LastAgeGroup }; enum JointTorque{ HipExtension = JointTorqueSet::HipExtension , HipFlexion = JointTorqueSet::HipFlexion , KneeExtension = JointTorqueSet::KneeExtension , KneeFlexion = JointTorqueSet::KneeFlexion , AnkleExtension = JointTorqueSet::AnkleExtension, AnkleFlexion = JointTorqueSet::AnkleFlexion, LastJointTorque }; const static char* GenderNames[]; const static char* AgeGroupNames[]; const static char* JointTorqueNames[]; Anderson2007(){} } Anderson2007; /** This struct contains 3 enumerated lists (Gender, AgeGroup, JointTorque) that lists the genders, age groups, and joint-torque-directions for which the Gymnast data set contains data. At the present time Gymnast data set only contains data appropriate for a young (18-25) elite male gymnast. Please refer to the main class description for details on this data set and how to use it. */ const static struct Gymnast{ enum Gender { Male = GenderSet::Male, LastGender }; enum AgeGroup { Young18To25 = AgeGroupSet::Young18To25, LastAgeGroup }; enum TableIndex { TauMax = 0, OmegaMax, ActiveAngleAtOneNormTorque, ActiveAngularStandardDeviation, TvAtMaxEccentricVelocity, TvAtHalfMaxConcentricVelocity, PassiveAngleAtZeroTorque, PassiveAngleAtOneNormTorque, LastTableIndex }; enum JointTorque { HipExtension = JointTorqueSet::HipExtension , HipFlexion = JointTorqueSet::HipFlexion , KneeExtension = JointTorqueSet::KneeExtension , KneeFlexion = JointTorqueSet::KneeFlexion , AnkleExtension = JointTorqueSet::AnkleExtension , AnkleFlexion = JointTorqueSet::AnkleFlexion , ElbowExtension = JointTorqueSet::ElbowExtension , ElbowFlexion = JointTorqueSet::ElbowFlexion , ShoulderExtension = JointTorqueSet::ShoulderExtension , ShoulderFlexion = JointTorqueSet::ShoulderFlexion , WristExtension = JointTorqueSet::WristExtension , WristFlexion = JointTorqueSet::WristFlexion , ShoulderHorizontalAdduction = JointTorqueSet::ShoulderHorizontalAdduction, ShoulderHorizontalAbduction = JointTorqueSet::ShoulderHorizontalAbduction, ShoulderInternalRotation = JointTorqueSet::ShoulderInternalRotation , ShoulderExternalRotation = JointTorqueSet::ShoulderExternalRotation , WristUlnarDeviation = JointTorqueSet::WristUlnarDeviation , WristRadialDeviation = JointTorqueSet::WristRadialDeviation, WristPronation = JointTorqueSet::WristPronation , WristSupination = JointTorqueSet::WristSupination , LumbarExtension = JointTorqueSet::LumbarExtension, LumbarFlexion = JointTorqueSet::LumbarFlexion, LastJointTorque }; const static char* GenderNames[]; const static char* AgeGroupNames[]; const static char* JointTorqueNames[]; Gymnast(){} } Gymnast; /** This is a struct that contains subject-specific information that does not change for a given subject. @param gender Male/Female. Selecting a gender that is not present in a data set will result in the program aborting and an error message being printed to the terminal. @param ageGroup: Presently the age group options include Young (18-25), middle aged (55-65), and senior (>65). Selecting an age group that is not present in a data set will result in the program aborting and an error message being printed to the terminal. @param subjectHeightInMeters This parameter is used to scale from the normalized curves reported by Anderson et al. See the class description for details. @param subjectMassInKg This parameter is used to scale from the normalized curves reported by Anderson et al. See the class description for details. */ struct SubjectInformation{ GenderSet::item gender; AgeGroupSet::item ageGroup; double heightInMeters; double massInKg; }; struct TorqueMuscleInfo{ /**The angle of the joint (radians)*/ double jointAngle; /**The angular velocity of the joint, where the sign convention is chosen by the user at the time the torque muscle is created (radians/sec)*/ double jointAngularVelocity; /**The angle that the muscle fiber spans (radians)*/ double fiberAngle; //The angle that the tendon spans (radians) //double tendonAngle; /**The rate-of-angular-lengthening of the fiber. A positive sign is for a concentric contraction, that is where the fibers are shortening. (radians/sec)*/ double fiberAngularVelocity; /*The rate-of-angular-lengthening of the tendon. A positive sign is for a concentric contraction, that is where the tendon is shortening. (radians/sec)*/ //double tendonAngularVelocity; /** The normalized value of the passive-torque-angle curve. Here a value of 1 means 1 maximum-isometric-torque. (Nm/Nm)*/ double fiberPassiveTorqueAngleMultiplier; /**The normalized value of the active-torque-angle curve. Here a value of 1 means 1 maximum-isometric-torque. (Nm/Nm)*/ double fiberActiveTorqueAngleMultiplier; /**The normalized value of the torque-angular-velocity curve. Here a value of 1 means 1 maximum-isometric-torque. (Nm/Nm)*/ double fiberActiveTorqueAngularVelocityMultiplier; /*The normalized value of the tendon-torque-angle curve. Here a value of 1 means 1 maximum-isometric-torque. (Nm/Nm)*/ //double tendonTorqueAngleMultiplier; /**The activation of the muscle*/ double activation; /**The torque generated by the active element of the muscle fiber (Nm)*/ double fiberActiveTorque; /**The torque generated by the passive element of the muscle fiber (Nm)*/ double fiberPassiveTorque; /**The torque generated by the damping element (Nm)*/ double fiberDampingTorque; /**The torque generated by the damping element (Nm)*/ double fiberNormDampingTorque; /**The torque generated by the entire fiber (Nm)*/ double fiberTorque; /*The torque transmitted through the tendon across the joint (Nm)*/ //double tendonTorque; /**The joint torque developed by the muscle. This is signed so that it is consistent with the sign convention of the joint chosen by the user. (Nm)*/ double jointTorque; /**The stiffness of the fiber (Nm/rad)*/ double fiberStiffness; /*The stiffness of the tendon (Nm/rad)*/ //double tendonStiffness; /** The stiffness of the joint. This is signed so that it is consistent with the sign convention of the joint chosen by the user. (Nm/rad)*/ double jointStiffness; /**The power output of the active fiber element. A positive power means that the fiber is contracting concentrically. (Watts - Nm/s)*/ double fiberActivePower; /**The power output of the passive fiber element. A positive power means that the passive element is recoiling concentrically (Watts - Nm/s)*/ double fiberPassivePower; /**The total power output of the fiber element.(Watts - Nm/s)*/ double fiberPower; /**The partial derivative of joint torque w.r.t the joint angle*/ double DjointTorqueDjointAngle; /**The partial derivative of joint torque w.r.t the joint angular velocity*/ double DjointTorqueDjointAngularVelocity; /**The partial derivative of joint torque w.r.t activation*/ double DjointTorqueDactivation; /*The power output of the tendon. A positive power means that the tendon is physically shortening. (Watts - Nm/s)*/ //double tendonPower; /** The power output by this muscle at the joint. This is signed so that it is consistent with the sign convention of the joint chosen by the user. (Nm/rad)*/ double jointPower; TorqueMuscleInfo(): jointAngle(nan("1")), jointAngularVelocity(nan("1")), fiberAngle(nan("1")), fiberAngularVelocity(nan("1")), fiberPassiveTorqueAngleMultiplier(nan("1")), fiberActiveTorqueAngleMultiplier(nan("1")), fiberActiveTorqueAngularVelocityMultiplier(nan("1")), activation(nan("1")), fiberActiveTorque(nan("1")), fiberPassiveTorque(nan("1")), fiberTorque(nan("1")), jointTorque(nan("1")), fiberStiffness(nan("1")), jointStiffness(nan("1")), fiberActivePower(nan("1")), fiberPassivePower(nan("1")), fiberPower(nan("1")), jointPower(nan("1")){} }; /** This class implements a rigid-tendon torque muscle for a growing list of joints and torque-directions. This rigid-tendon torque muscle model provides modeling support for 3 phenomena -torque-angle curve (\f$\mathbf{t}_A(\theta)\f$): the variation of active isometric torque in one direction as a function of joint angle -torque-velocity curve (\f$\mathbf{t}_V(\dot{\theta})\f$): the variation of torque as a function of angular velocity -passive-torque-angle curve (\f$s_P\mathbf{t}_P(\theta-\theta_S)\f$): the variation of passive torque as a function of joint angle. Here \f$s_P\f$ and \f$\theta_S\f$ are user-defined scaling and shift parameters. each of which are represented as smooth normalized curves that vary between 0 and 1. These three phenomena are used to compute the torque developed \f$\tau\f$ given the angle of the joint \f$\theta\f$, the angular-velocity of the joint \f$\dot{\theta}\f$, and the activation of the muscle \f$\mathbf{a}\f$ (a 0-1 quantity that defines how much the muscle is turned-on, or activated), and the maximum-isometric torque \f$\tau_{ISO}\f$ \f[ \tau (\mathbf{a}, \theta,\dot{\theta}) = \begin{cases} \tau_{ISO} ( \mathbf{a} \, \mathbf{t}_A(\theta) \mathbf{t}_V(\dot{\theta}/\dot{\theta}_{MAX}) - \beta \dot{\theta}/\dot{\theta}_{MAX} + s_P \mathbf{t}_P(\theta-\theta_S) \, ) & \text{ if } \,\, \tau (\mathbf{a}, \theta,\dot{\theta}) > 0 \\ 0 & \text{otherwise} \end{cases} \f] The This model does not yet provide support for the following phenomena but will in the future. -activation dynamics: this is to be decided by the modeler. -tendon-elasticity -muscle short-range-stiffness All of these characteristic curves are represented using \f$C_2\f$ continuous \f$5^{th}\f$ order Bezier curves that have been fitted to the data from data in the literature. In many cases these curves have been carefully edited so that they fit the curves of the original papers, but have more desireable numerical properties for optimal control work. The characterisic curves provided by this class have been fitted to a growing list of data sets: -Anderson Data Set: from Anderson et al. 2007 -Whole-body Gymnast Data Set: from Jackson, Kentel et al., Anderson et al., Dolan et al. and Raschke et al. Data Set: Anderson2007 This data set uses the mean value of the coefficients published in Anderson et al. The standard deviation table has also been entered. However, since it is unclear how to use the standard deviation in a consistent way across all joints/parameters this table is not yet accessible through the constructor. This data set includes coefficients for the following -Number of subjects: 34 -Gender: male and female -Age: young (18-25, 14 subjects), middle-aged (55-65, 14 subjects), senior (> 65, 6 subjects) -Joint: hip/knee/ankle -Direction: extension/flexion Notes -# Angles are plotted using units of degrees for readability. The actual curves are described in units of radians -# See Anderson et al. for further details. \image html fig_MuscleAddon_Anderson2007AllPositiveSigns.png "Characteristic from Anderson et al. 2007 [1]" Data Set: Gymnast This data set is an attempt at making enough torque muscles for a whole body. Since no single source in the literature comes close to measuring the characteristics of all of the joints, data from Jackson et al., Kentel et al., Anderson et al., Dolan et al, and Raschke et al. have been combined. Since the subjects used in these various studies are wildly different (Jackson et al. measured an elite male gymnast; Kentel et al. measured an elite tennis player; Anderson et al. measured, in the category of young male, a selection of active undergraduate students, Dolan et al from 126 women and 23 men, and Raschke from 5 male subjects) scaling has been used to make the strength of the subject consistent. Scaling coefficients for the lower body, shoulders and elbow, and forearm/wrist using measurements that overlapped between datasets. Presently this data set includes curves for 22 joint and direction specific torque muscles. - Number of subjects: 1 elite gymnast (69.6 kg, 1.732 m) - Gender: male - Age: 21 years old - Joint and Directions available -# Ankle: flexion/extension (scaled from Anderson) -# Knee: flexion/extension (from Jackson) -# Hip: flexion/extension (from Jackson) -# Lumbar: active extension curves (\f$\mathbf{t}_A\f$ and \f$\mathbf{t}_P\f$) from Raschke et al. passive extension from Dolan et al. -# Lumbar: active flexion \f$\tau_{ISO}\f$ from Beimborn et al. -# Shoulder: flexion/extension (from Jackson) -# Shoulder: horizontal adduction/abduction (from Kentel, scaled to Jackson's subject) -# Shoulder: internal rotation/external rotation (from Kentel, scaled to Jackson's subject) -# Elbow: flexion/extension (from Kentel, scaled to Jackson's subject) -# Wrist: pronation/supination (from Kentel, scaled to Jackson's subject) -# Wrist: extension/flextion (from Jackson) -# Wrist: ulnar/radial deviation (from Kentel, scaled to Jackson's subject) - Missing Joint and directions -# Ankle inversion/eversion -# Hip adduction/abduction -# Hip internal rotation/external rotation -# Lumbar extension/flexion -# Lumbar bending -# Lumbar twisting -# Shoulder Adduction -# Shoulder Abduction -# Scapular elevation/depression -# Scapular adduction/abduction -# Scapular upward/downward rotation In all cases the curves have been fitted to Bezier curves that are constructed using functions in TorqueMuscleFunctionFactory. Notes -# Angles are plotted using units of degrees for readability. The actual curves are described in units of radians -# Hip and Knee characteristics taken from Jackson. Ankle extension is from Anderson et al., scaled using Jackson-to-Anderson hip/knee strength ratios from Jackson ratios -# Shoulder horizontal adduction/abduction and internal/external rotation is a scaled version of the Kentel. Strength was scaled using the Jackson-to-Kentel shoulder flex/ext ratios. -# Elbow extension/flexion and forearm pronation/supination. Elbow strength scaled from Kentel using the ratio of maximum isometric shoulder ext/flextion between Kentel and Jackson. Forearm pronation/supination scaled using the maximum torque strength ratio of wrist extension/flextion between Kentel and Jackson -# Wrist ext/flexion directly from Jackson, while the curves for ulnar and radial deviation have been scaled (using the maximum isometric torque ratios of wrist extension and flexion from both models) from Kentel et al. -# Lumbar-extension active-torque-angle-curve, and torque-velocity-curve in extension comes from Raschke et al. -# Lumbar-flexion \f$\tau_{ISO}\f$ comes from Beimborn et al.'s observation that the strength ratio of back extensors to flextors at a flextion angle of zero is most often repored as 1.3:1. The torque velocity curve is a guess (slower than back extesors due to the larger moment arm). -# Any passive curve that is not accompanied by a curve from the literature (see the plots for details) is an educated guess. \image html fig_MuscleAddon_Gymnast_HipKneeAnkle.png " Hip/Knee/Ankle: from Jackson and Anderson et al. " \image html fig_MuscleAddon_Gymnast_Lumbar.png " Lumbar Extension/Flexion: from Dolan et al.,Raschke et al., and Beimborn et al." \image html fig_MuscleAddon_Gymnast_Shoulder3Dof.png " Shoulder 3 DoF torques: from Jackson and Kentel et al. " \image html fig_MuscleAddon_Gymnast_ElbowForearm.png " Elbow flexion/extension: from Kentel et al." \image html fig_MuscleAddon_Gymnast_Wrist3Dof.png " Wrist 3 DoF torques: from Jackson and Kentel et al." Parameterized Curves used here vs. Literature The curves used in this implementation are 2nd order 2-dimensional Bezier curves. The curves described in Anderson et al., Jackson, Kentel were not directly used because they are not continuous to the second derivative (a requirement for most gradient based optimization routines). There are some other detailed differences that might be of interest: -# Anderson et al.'s torque-velocity curve tends to large negative values for fast eccentric contractions. This is in contrast to the literature which says that at large eccentric contractions the torque-velocity curve (or the force-velocity-curve) tends to a value between 1.0 and 1.4. -# Anderson et al.'s torque-velcity curve for ankle extension did not cross the x-axis on the concentric side of the curve. This would endow the plantar flexors with super-human abilities. This error has been corrected by fitting a Bezier curve to a Hill-type curve that passes through the point where \f$\dot{\theta}= \frac{1}{2} \dot{\theta}_{MAX}\f$ -# Anderson et al.'s, Jackson, and Kentel et al. had discintinuities in the first derivative of the force velocity curve at \f$\dot{\theta}=0\f$. While this follows Huxley's famous observations that the slope does discontinuously change at at \f$\dot{\theta}=0\f$. This is not a phenomena that is not compatible with most optimal control formulations and thus this discontinuity is not present in the force velocity curves used in this model. -# Anderson et al. and Kentel et al.'s active-torque-angle curves can achieve negative values - this is obviously undesirable as it will allow a muscle to push. -# Kentel et al.'s activation inhibiition function does not always cross 1.0 for \f$\dot{\theta}=0\f$, which means that \f$\tau_{ISO}\f$ is not reached. This makes for a confusing model to use. Coordinate Mapping Every author chose a particular convention for measuring the angles of the hip, knee, ankle joint, shoulder, elbow, wrist and lumbar --- see the figure for details. These conventions have all been mapped to the one used in the illustrations. You will need to use the figure, your model, and the constructors appropriately so that -# the joint angle of your model is correctly mapped to the fiber angle of the Millard2016TorqueMuscle; -# the sign of the muscle's output torque matches the sign associated with your model. To map from your model's joint coordinates to the joint coordines used in this model (see the figure in the description) the followinq equation is used at the torque level \f$ jointTorque = signOfJointTorque*fiberTorque \f$ where fiberTorque is the torque produced by Anderson et al.'s curves, which is always positive. At the position level, the angles from your models joint angle to Anderson et al.'s joint angle (called fiberAngle) are mapped using \f$ fiberAngle = signOfJointAngleRelativeToAnderson2007*(jointAngle-jointAngleOffsetRelativeToAnderson2007). \f$ Internally the sign of the fiber velocity follows signOfJointTorque so that the signs of joint power and muscle power are consistent. Strength Scaling The leg strength (here we mean \f$\tau_{ISO}\f$) predicted by Anderson et al.'s curves should be taken as a good first approximation. While Anderson et al.'s data set is the most comprehensive in the literature, they only measured torques from active people: they did not include people at the extremes (both very weak, and very strong), nor did they include children. Finally, the torques produced by each subject were normalized by subjectMassInKg*subjectHeightInM*accelerationDueToGravity. Strength is a strange phenomena which is not nicely normalized by just these quantites, and so the strength predicted by Anderson et al.'s curves might not fit your subject even if they are represented in Anderson et al.'s data set. The strength used in the Gymnast data set is fitted to an elite male gymnast. It goes without saying that an elite gymnast has strength proportions, and an absolute strength that are not typical. In the future it would be nice to have a function that could provide an educated guess about how to map Gymnast's strengths to that of another subject. For the moment I have no idea how to do this, nor am I aware of any works in the literature that can provide insight of how to do this. For now the whole-body Gymnast model should be viewed as being a representation of what is possible for a human, but not a typical human. At the present time the default strength settings of the Gymnast are not scaled by subject height, nor weight. If you happen to know the maximum-isometric-active-torque (note this does not include the passive component) that your subject can produce,you can update the strength of the torque-muscle using the functions getMaximumActiveIsometricTorque(), and setMaximumActiveIsometricTorque(). Limitations This rigid-tendon torque muscle has some limitations that you should be aware of: -# There are no elastic tendons. That means that the mapping between the mechanical work done by this torque actuator will greatly differ from the positive mechanical work done by a torque actuator that includes an elastic tendon. This difference is greatest for those muscles with long tendons - namely the Achilles tendon. If you are interested in fiber kinematics, fiber work, or metabolic energy consumption you cannot use this model especially for muscles that have long tendons. -# This model formulation predicts torque well, but does a poor job of predicting joint stiffness. In this model stiffness is given by the partial derivative of torque w.r.t. joint angle. Since the active-torque-angle curve fits a cosine function, it is possible to construct a torque muscle that has a region of physically impossible negative stiffness. Real muscle, in constrast, always has a positive stiffness even on the descending limb of the active-torque-angle curve (see Rassier et al. for details). -# Muscles that cross 2 joints (e.g. the hamstrings) produce coupled torques at both of those joints. In this model there is no coupling between joints. Furthermore, because of the lack of coupling the curves used here are only valid for the posture that Anderson et al., Jackson, and Kentel et al. used when they made their data collection. If you are interested in simulating postures that are very different from those described in by these authors then the results produced by this model should be treated as very rough. -# Because this is a joint-torque muscle, none of the joint contact forces predicted will come close to matching what is produced by line-type muscles. If you are interested in joint-contact forces you cannot use this model. This simple model is a fast approximate means to constrain the joint torque developed in the body to something that is physiologically possible. That is it. Units Although the figure in this description has angles in units of degrees, this is only to help intuitition: when using the model, use radians. This model uses MKS: -Distance: m -Angles: radians -Angular velocity: radians/s -Mass: kg -Torque: Nm -Time: second -Power: Nm/second References -# Anderson, D. E., Madigan, M. L., & Nussbaum, M. A. (2007). Maximum voluntary joint torque as a function of joint angle and angular velocity: model development and application to the lower limb. Journal of biomechanics, 40(14), 3105-3113. -# Beimborn, D. S., & Morrissey, M. C. (1988). A review of the literature related to trunk muscle performance. Spine, 13(6), 655-660. -# Dolan, P., A. F. Mannion, and M. A. Adams. Passive tissues help the back muscles to generate extensor moments during lifting. Journal of Biomechanics 27, no. 8 (1994): 1077-1085. -# Jackson, M.I. (2010). The mechanics of the Table Contact Phase of Gymnastics Vaulting. Doctoral Thesis, Loughborough University. -# Kentel, B.B., King, M.A., & Mitchell, S.R. (2011). Evaluation of a subject-specific torque-driven computer simulation model of one-handed tennis backhand ground strokes. Journal of Applied Biomechanics, 27(4),345-354. -# Millard, M., Uchida, T., Seth, A., & Delp, S. L. (2013). Flexing computational muscle: modeling and simulation of musculotendon dynamics. Journal of biomechanical engineering, 135(2), 021005. -# Raschke, U., & Chaffin, D. B. (1996). Support for a linear length-tension relation of the torso extensor muscles: an investigation of the length and velocity EMG-force relationships. Journal of biomechanics, 29(12), 1597-1604. -# Rassier, D. E., Herzog, W., Wakeling, J., & Syme, D. A. (2003). Stretch-induced, steady-state force enhancement in single skeletal muscle fibers exceeds the isometric force at optimum fiber length. Journal of biomechanics, 36(9), 1309-1316. */ class Millard2016TorqueMuscle { public: /** Default constructor, which for the moment does nothing. Calling any of the models functions after the default construction will result in a runtime error. */ Millard2016TorqueMuscle(); /** This constructor allows you to easily access the large table of built-in torque muscle coefficients to create a torque muscle that best represents the joint of interest. Note: directions This constructs a single joint-torque muscle: it can only generate torque in one direction. If you want to generate a torque in two directions, you need 2 torque muscles. Note: signs and offsets All of the angles in these models are defined anatomically. You will need to set a series of variables to correctly map from your model's joint coordinates and sign conventions to that of the models: jointAngleOffsetRelativeToDoxygenFigures, signOfJointAngleRelativeToDoxygenFigures, signOfJointTorqueToDoxygenFigures. Also note that due to the anatomical angle definitions some left and right handed joints will require different signs. This will be true for internal/external rotation at the shoulder, horizontal adduction/abduction at the shoulder, ulnar/radial deviation at the wrist, pronation/supination of the wrist, and others as the list of directions grows. @param dataSet The desired source of joint torque coefficients. Use the DataSet structure to choose the desired data set (e.g. DataSet::Anderson2007, or DataSet::Gymnast) @param subjectInfo A struct that contains metadata about the subject which is used to scale the maximum torque of the torque muscle. @param jointTorque Select the joint and torque direction of interest. Use the struct for each data set to choose a joint-torque-direction that is in the set (e.g. Anderson2007::HipExtension, or Gymnast::ShoulderHorizontalAdduction) @param jointAngleOffsetRelativeToDoxygenFigures Offset angle between your model's joints and the reference figures in class description. @param signOfJointAngleRelativeToDoxygenFigures The sign convention that converts your model's joint angles to the angles used in the reference figures. @param signOfJointTorqueToDoxygenFigures The sign that maps fiberTorque from Anderson's model (which is always positive) to the correctly signed joint torque for your model. @param name The name of the muscle. This is needed to do useful things like provide error messages that are human readable. @throws abort() when -# The combination of dataSet, gender, joint, and jointDirection does not correspond to a valid entry -# subjectHeightInMeters <= 0 -# subjectMassInKg <= 0 -# abs(signOfJointTorque)-1 > epsilon */ Millard2016TorqueMuscle( DataSet::item dataSet, const SubjectInformation &subjectInfo, int jointTorque, double jointAngleOffsetRelativeToDoxygenFigures, double signOfJointAngleRelativeToDoxygenFigures, double signOfJointTorqueToDoxygenFigures, const std::string& name ); /** Calculates the joint torque developed by the muscle. @param jointAngle (radians) @param jointAngularVelocity (radians/sec) @param activation: the percentage of the muscle that is turned on [0-1]. This function allows activations to be outside [0,1], because this is useful during the intermediate solutions of an optimization run. However, you must ensure after the fact that your activations fall within a bound of [0,1]. @returns torque developed by the musce in (Nm). */ double calcJointTorque( double jointAngle, double jointAngularVelocity, double activation) const; /** Calculates a large number of internal quantities of the torque muscle ranging from the values of the muscle's components, the stiffness of the muscle, and its power output. @param jointAngle (radians) @param jointAngularVelocity (radians/sec) @param activation: the percentage of the muscle that is turned on [0-1]. This function allows activations to be outside [0,1], because this is useful during the intermediate solutions of an optimization run. However, you must ensure after the fact that your activations fall within a bound of [0,1]. @param torqueMuscleInfoStruct: A torque muscle struct */ void calcTorqueMuscleInfo( double jointAngle, double jointAngularVelocity, double activation, TorqueMuscleInfo& torqueMuscleInfoStruct) const; /** @return the sign of the joint torque (+/- 1) */ double getJointTorqueSign() const; /** @return the sign of the angle sign relative to the figures in the class description (+/- 1) */ double getJointAngleSign() const; /** @return the offset angle between the model's joint and the figures in the class description (rad) */ double getJointAngleOffset() const; /** @return the maximum-active-isometric torque that this muscle can produce in Nm. */ double getMaximumActiveIsometricTorque() const; /** @return the joint angle at which the normalized active-torque-angle curve peaks at its maximum value of 1.0. Angle is in radians */ double getJointAngleAtMaximumActiveIsometricTorque() const; /** @return the joint angle at which the normalized passive-torque-angle curve reaches a value of 1.0. If this curve never reaches a value of 1.0 (because it is flat, or the passiveTorqueScale has been set to 0) a value of std::numeric_limits::signaling_NaN() is returned. Use the std function isfinite to test if a signaling_NaN has been returned. Angle is in radians */ double getJointAngleAtOneNormalizedPassiveIsometricTorque() const; /** @return the maximum concentric joint angular velocity in radians. */ double getMaximumJointAngularVelocity() const; /** @return the passive-torque-scale \f$s_P\f$ that is applied to the passive-torque-curve. */ double getPassiveTorqueScale() const; /** @return the angle \f$\theta_S\f$ that the passive curve has been shifted (radians). */ double getPassiveCurveAngleOffset() const; /** @return The normalized damping term \f$\beta\f$ term in the torque muscle model. See class description for details. */ double getNormalizedDampingCoefficient() const; /** @param beta -t normalized damping term \f$\beta\f$ term in the torque muscle model. See class description for details. @throw abort() if beta < 0 */ void setNormalizedDampingCoefficient(double beta); /** Sets the scaling of the passive-joint-torques. By default this scale is one. @param passiveTorqueScale The scale \f$s_P\f$ applied to the passive-joint-torque curve (unitless) */ void setPassiveTorqueScale(double passiveTorqueScale); /** @param passiveCurveAngleOffsetVal the angle \f$\theta_S\f$ that the passive curve should be shifted. Angles in radians */ void setPassiveCurveAngleOffset( double passiveCurveAngleOffsetVal); /** This function iteratively solves for the passiveTorqueScale so that at the specified jointAngle the passive curve develops the specified passiveTorque @param jointAngle the target joint angle in radians @param passiveTorque the target passive joint torque in Nm. @throws abort if jointAngle is not in the domain of the curve @throws abort if passiveTorque < sqrt(eps) */ void fitPassiveTorqueScale(double jointAngle, double passiveTorque); /** This function solves for the passive curve angle offset so that at the specified jointAngle the passive curve develops the specified passiveTorque @param jointAngle the target joint angle in radians @param passiveTorque the target passive joint torque in Nm. @throws abort if passiveTorque < sqrt(eps) */ void fitPassiveCurveAngleOffset(double jointAngle, double passiveTorque); /** Sets the strength of the muscle to match a desired value. @param maxIsometricTorque The desired maximum-active-isometric torque of the muscle (Nm) */ void setMaximumActiveIsometricTorque( double maxIsometricTorque); /** @return the SmoothSegmentedFunction the has been fitted to Anderson et al.'s passive torque angle curve. */ const RigidBodyDynamics::Addons::Geometry:: SmoothSegmentedFunction& getActiveTorqueAngleCurve() const; /** @return the SmoothSegmentedFunction the has been fitted to Anderson et al.'s active torque angle curve. */ const RigidBodyDynamics::Addons::Geometry:: SmoothSegmentedFunction& getPassiveTorqueAngleCurve() const; /** @return the SmoothSegmentedFunction the has been fitted to Anderson et al.'s torque velocity curve. */ const RigidBodyDynamics::Addons::Geometry:: SmoothSegmentedFunction& getTorqueAngularVelocityCurve() const; /** Prints 2 csv files: -# 'fileName' + '_variableLengthfixedVelocity': All of the fields in TorqueMuscleInfo are recorded to file as the jointAngle varies but the jointAngularVelocity is zero. -#'fileName' + '_fixedLengthVariableVelocity': All of the fields in TorqueMuscleInfo are recorded to file as the jointAngle is fixed but the jointAngularVelocity varies. Each column has a header, so that you can tell what each piece of data means. @param path: the path to the destination folder. Don't put an '\' on the end. @param fileNameWithoutExtension: the name of the file, but without an extension. @param numberOfSamplePoints: the number of sample points to use in the files. */ void printJointTorqueProfileToFile( const std::string& path, const std::string& fileNameWithoutExtension, int numberOfSamplePoints); std::string getName(); void setName(std::string& name); private: /** @return the parameters c1,...,c6 that desribe the active-torque-angle and torque-velocity curves of this torque muscle model. See the Anderson et al. paper metioned in the class description for detail. */ //const RigidBodyDynamics::Math::VectorNd& //getParametersC1C2C3C4C5C6(); /** @return the parameters b1,k1,b2,k2 that desribe the passive-torque-angle curves of this model. See the Anderson et al. paper metioned in the class description for detail. */ //const RigidBodyDynamics::Math::VectorNd& //getParametersB1K1B2K2(); bool muscleCurvesAreDirty; void updateTorqueMuscleCurves(); TorqueMuscleInfo tmInfo; RigidBodyDynamics::Addons::Geometry:: SmoothSegmentedFunction taCurve; RigidBodyDynamics::Addons::Geometry:: SmoothSegmentedFunction tpCurve; RigidBodyDynamics::Addons::Geometry:: SmoothSegmentedFunction tvCurve; RigidBodyDynamics::Math::VectorNd c1c2c3c4c5c6Anderson2007; RigidBodyDynamics::Math::VectorNd b1k1b2k2Anderson2007; RigidBodyDynamics::Math::VectorNd gymnastParams; DataSet::item dataSet; double maxActiveIsometricTorque; double angleAtOneNormActiveTorque; double omegaMax; double angleAtOneNormPassiveTorque; double passiveTorqueScale; double passiveCurveAngleOffset; double beta; //passive damping coefficient double subjectHeightInMeters; double subjectMassInKg; double scaleFactorAnderson2007; double signOfJointAngle; double signOfConcentricAnglularVelocity; double signOfJointTorque; double angleOffset; std::string muscleName; double calcJointAngle(double fiberAngle) const; double calcFiberAngle(double jointAngle) const; double calcFiberAngularVelocity( double jointAngularVelocity) const; //const static RigidBodyDynamics::Math::MatrixNd& //getAnderson2007ParameterMatrix(); static double const Anderson2007Table3Mean[36][14]; static double const Anderson2007Table3Std[36][14]; static double const GymnastWholeBody[22][12]; }; } } } #endif