2016-07-01 17:35:27 +00:00
/*
2016-09-11 20:25:22 +00:00
Bullet Continuous Collision Detection and Physics Library
Copyright ( c ) 2015 Google Inc . http : //bulletphysics.org
2016-07-01 17:35:27 +00:00
This software is provided ' as - is ' , without any express or implied warranty .
In no event will the authors be held liable for any damages arising from the use of this software .
Permission is granted to anyone to use this software for any purpose ,
including commercial applications , and to alter it and redistribute it freely ,
subject to the following restrictions :
1. The origin of this software must not be misrepresented ; you must not claim that you wrote the original software . If you use this software in a product , an acknowledgment in the product documentation would be appreciated but is not required .
2. Altered source versions must be plainly marked as such , and must not be misrepresented as being the original software .
3. This notice may not be removed or altered from any source distribution .
*/
# include "NN3DWalkers.h"
2016-07-04 17:17:10 +00:00
2016-07-01 17:35:27 +00:00
# include "btBulletDynamicsCommon.h"
# include "LinearMath/btIDebugDraw.h"
# include "LinearMath/btAlignedObjectArray.h"
2016-10-19 19:56:09 +00:00
# include "LinearMath/btHashMap.h"
2016-07-01 17:35:27 +00:00
class btBroadphaseInterface ;
class btCollisionShape ;
class btOverlappingPairCache ;
class btCollisionDispatcher ;
class btConstraintSolver ;
struct btCollisionAlgorithmCreateFunc ;
class btDefaultCollisionConfiguration ;
2016-09-11 20:25:22 +00:00
class NNWalker ;
2016-07-01 17:35:27 +00:00
2016-10-22 20:50:08 +00:00
# include "NN3DWalkersTimeWarpBase.h"
2016-07-04 17:17:10 +00:00
# include "../CommonInterfaces/CommonParameterInterface.h"
2016-07-01 17:35:27 +00:00
2016-09-11 20:25:22 +00:00
# include "../Utils/b3ReferenceFrameHelper.hpp"
2016-09-12 14:36:37 +00:00
# include "../RenderingExamples/TimeSeriesCanvas.h"
2016-07-03 17:54:47 +00:00
2017-06-05 21:15:51 +00:00
static btScalar gRootBodyRadius = 0.25f ;
static btScalar gRootBodyHeight = 0.1f ;
static btScalar gLegRadius = 0.1f ;
static btScalar gLegLength = 0.45f ;
static btScalar gForeLegLength = 0.75f ;
static btScalar gForeLegRadius = 0.08f ;
static btScalar gParallelEvaluations = 10.0f ;
# ifndef SIMD_PI_4
# define SIMD_PI_4 0.5 * SIMD_HALF_PI
2016-09-11 20:25:22 +00:00
# endif
2017-06-05 21:15:51 +00:00
# ifndef SIMD_PI_8
# define SIMD_PI_8 0.25 * SIMD_HALF_PI
2016-09-11 20:25:22 +00:00
# endif
2017-06-05 21:15:51 +00:00
# ifndef RANDOM_MOVEMENT
# define RANDOM_MOVEMENT false
2016-09-11 20:25:22 +00:00
# endif
2017-06-05 21:15:51 +00:00
# ifndef RANDOMIZE_DIMENSIONS
# define RANDOMIZE_DIMENSIONS false
2016-09-11 20:25:22 +00:00
# endif
2017-06-05 21:15:51 +00:00
# ifndef NUM_WALKERS
# define NUM_WALKERS 50
2016-09-11 20:25:22 +00:00
# endif
2017-06-05 21:15:51 +00:00
# ifndef EVALUATION_TIME
# define EVALUATION_TIME 10 // s
# endif
2016-09-11 20:25:22 +00:00
# ifndef REAP_QTY
2017-06-05 21:15:51 +00:00
# define REAP_QTY 0.3f // number of walkers reaped based on their bad performance
2016-09-11 20:25:22 +00:00
# endif
# ifndef SOW_CROSSOVER_QTY
2017-06-05 21:15:51 +00:00
# define SOW_CROSSOVER_QTY 0.2f // this means REAP_QTY-SOW_CROSSOVER_QTY = NEW_RANDOM_BREED_QTY
2016-09-11 20:25:22 +00:00
# endif
# ifndef SOW_ELITE_QTY
2017-06-05 21:15:51 +00:00
# define SOW_ELITE_QTY 0.2f // number of walkers kept using an elitist strategy
2016-09-11 20:25:22 +00:00
# endif
# ifndef SOW_MUTATION_QTY
2017-06-05 21:15:51 +00:00
# define SOW_MUTATION_QTY 0.5f // SOW_ELITE_QTY + SOW_MUTATION_QTY + REAP_QTY = 1
2016-09-11 20:25:22 +00:00
# endif
# ifndef MUTATION_RATE
2017-06-05 21:15:51 +00:00
# define MUTATION_RATE 0.5f // the mutation rate of for the walker with the worst performance
2016-09-11 20:25:22 +00:00
# endif
# ifndef SOW_ELITE_PARTNER
2017-06-05 21:15:51 +00:00
# define SOW_ELITE_PARTNER 0.8f
2016-09-11 20:25:22 +00:00
# endif
2017-06-05 21:15:51 +00:00
# define NUM_LEGS 6
# define BODYPART_COUNT (2 * NUM_LEGS + 1)
2016-09-11 20:25:22 +00:00
# define JOINT_COUNT (BODYPART_COUNT - 1)
2017-06-05 21:15:51 +00:00
# define DRAW_INTERPENETRATIONS false
2016-09-11 20:25:22 +00:00
2016-09-06 20:30:46 +00:00
void * GROUND_ID = ( void * ) 1 ;
2016-10-22 20:50:08 +00:00
class NN3DWalkersExample : public NN3DWalkersTimeWarpBase
2016-07-01 17:35:27 +00:00
{
2017-06-05 21:15:51 +00:00
btScalar m_Time ;
btScalar m_SpeedupTimestamp ;
btScalar m_targetAccumulator ;
btScalar m_targetFrequency ;
btScalar m_motorStrength ;
int m_evaluationsQty ;
int m_nextReaped ;
2016-07-01 17:35:27 +00:00
2017-06-05 21:15:51 +00:00
btAlignedObjectArray < class NNWalker * > m_walkersInPopulation ;
2016-07-01 17:35:27 +00:00
2017-06-05 21:15:51 +00:00
TimeSeriesCanvas * m_timeSeriesCanvas ;
2016-09-12 14:36:37 +00:00
2016-07-01 17:35:27 +00:00
public :
2016-09-06 20:30:46 +00:00
NN3DWalkersExample ( struct GUIHelperInterface * helper )
2016-10-22 20:50:08 +00:00
: NN3DWalkersTimeWarpBase ( helper ) ,
2017-06-05 21:15:51 +00:00
m_Time ( 0 ) ,
m_SpeedupTimestamp ( 0 ) ,
m_targetAccumulator ( 0 ) ,
m_targetFrequency ( 3 ) ,
m_motorStrength ( 0.5f ) ,
m_evaluationsQty ( 0 ) ,
m_nextReaped ( 0 ) ,
m_timeSeriesCanvas ( 0 )
2016-07-01 17:35:27 +00:00
{
2016-07-03 17:54:47 +00:00
}
2016-07-01 17:35:27 +00:00
2016-09-06 20:30:46 +00:00
virtual ~ NN3DWalkersExample ( )
2016-07-01 17:35:27 +00:00
{
2016-10-21 15:52:11 +00:00
delete m_timeSeriesCanvas ;
2016-07-01 17:35:27 +00:00
}
2016-09-11 20:25:22 +00:00
void initPhysics ( ) ;
virtual void exitPhysics ( ) ;
2016-07-01 17:35:27 +00:00
2017-06-05 21:15:51 +00:00
void spawnWalker ( int index , const btVector3 & startOffset , bool bFixed ) ;
virtual bool keyboardCallback ( int key , int state ) ;
2016-09-11 20:25:22 +00:00
bool detectCollisions ( ) ;
2016-09-06 20:30:46 +00:00
2016-07-01 17:35:27 +00:00
void resetCamera ( )
{
float dist = 11 ;
2017-06-01 22:30:37 +00:00
float pitch = - 35 ;
float yaw = 52 ;
2016-07-01 17:35:27 +00:00
float targetPos [ 3 ] = { 0 , 0.46 , 0 } ;
2017-06-01 22:30:37 +00:00
m_guiHelper - > resetCamera ( dist , yaw , pitch , targetPos [ 0 ] , targetPos [ 1 ] , targetPos [ 2 ] ) ;
2016-07-01 17:35:27 +00:00
}
2016-09-11 20:25:22 +00:00
// Evaluation
2016-07-01 17:35:27 +00:00
2016-09-12 21:59:42 +00:00
void update ( const btScalar timeSinceLastTick ) ;
2016-07-01 17:35:27 +00:00
2016-09-12 21:59:42 +00:00
void updateEvaluations ( const btScalar timeSinceLastTick ) ;
2016-07-01 17:35:27 +00:00
2016-09-11 20:25:22 +00:00
void scheduleEvaluations ( ) ;
2016-07-01 17:35:27 +00:00
2016-09-12 14:36:37 +00:00
void drawMarkings ( ) ;
2016-09-11 20:25:22 +00:00
// Reaper
2016-07-05 16:57:51 +00:00
2016-09-11 20:25:22 +00:00
void rateEvaluations ( ) ;
void reap ( ) ;
void sow ( ) ;
void crossover ( NNWalker * mother , NNWalker * father , NNWalker * offspring ) ;
2016-09-12 21:59:42 +00:00
void mutate ( NNWalker * mutant , btScalar mutationRate ) ;
2016-09-11 20:25:22 +00:00
NNWalker * getRandomElite ( ) ;
NNWalker * getRandomNonElite ( ) ;
NNWalker * getNextReaped ( ) ;
2016-09-12 14:36:37 +00:00
void printWalkerConfigs ( ) ;
2017-06-05 21:15:51 +00:00
2016-09-11 20:25:22 +00:00
} ;
static NN3DWalkersExample * nn3DWalkers = NULL ;
2016-07-01 17:35:27 +00:00
class NNWalker
{
2017-06-05 21:15:51 +00:00
btDynamicsWorld * m_ownerWorld ;
btCollisionShape * m_shapes [ BODYPART_COUNT ] ;
btRigidBody * m_bodies [ BODYPART_COUNT ] ;
btTransform m_bodyRelativeTransforms [ BODYPART_COUNT ] ;
btTypedConstraint * m_joints [ JOINT_COUNT ] ;
2016-10-19 19:56:09 +00:00
btHashMap < btHashPtr , int > m_bodyTouchSensorIndexMap ;
2017-06-05 21:15:51 +00:00
bool m_touchSensors [ BODYPART_COUNT ] ;
btScalar m_sensoryMotorWeights [ BODYPART_COUNT * JOINT_COUNT ] ;
2016-09-11 20:25:22 +00:00
2017-06-05 21:15:51 +00:00
bool m_inEvaluation ;
btScalar m_evaluationTime ;
bool m_reaped ;
btVector3 m_startPosition ;
int m_index ;
2016-07-01 17:35:27 +00:00
btRigidBody * localCreateRigidBody ( btScalar mass , const btTransform & startTransform , btCollisionShape * shape )
{
bool isDynamic = ( mass ! = 0.f ) ;
btVector3 localInertia ( 0 , 0 , 0 ) ;
if ( isDynamic )
shape - > calculateLocalInertia ( mass , localInertia ) ;
2016-07-03 17:54:47 +00:00
btDefaultMotionState * motionState = new btDefaultMotionState ( startTransform ) ;
btRigidBody : : btRigidBodyConstructionInfo rbInfo ( mass , motionState , shape , localInertia ) ;
2016-07-01 17:35:27 +00:00
btRigidBody * body = new btRigidBody ( rbInfo ) ;
return body ;
}
public :
2017-06-05 21:15:51 +00:00
void randomizeSensoryMotorWeights ( ) {
//initialize random weights
for ( int i = 0 ; i < BODYPART_COUNT ; i + + ) {
for ( int j = 0 ; j < JOINT_COUNT ; j + + ) {
m_sensoryMotorWeights [ i + j * BODYPART_COUNT ] = ( ( double ) rand ( ) / ( RAND_MAX ) ) * 2.0f - 1.0f ;
}
}
}
NNWalker ( int index , btDynamicsWorld * ownerWorld , const btVector3 & positionOffset , bool bFixed )
: m_ownerWorld ( ownerWorld ) ,
m_inEvaluation ( false ) ,
m_evaluationTime ( 0 ) ,
m_reaped ( false )
2016-09-11 20:25:22 +00:00
{
2017-06-05 21:15:51 +00:00
m_index = index ;
btVector3 vUp ( 0 , 1 , 0 ) ; // up in local reference frame
2016-09-11 20:25:22 +00:00
NN3DWalkersExample * nnWalkersDemo = ( NN3DWalkersExample * ) m_ownerWorld - > getWorldUserInfo ( ) ;
2017-06-05 21:15:51 +00:00
randomizeSensoryMotorWeights ( ) ;
//
2016-07-01 17:35:27 +00:00
// Setup geometry
2017-06-05 21:15:51 +00:00
m_shapes [ 0 ] = new btCapsuleShape ( gRootBodyRadius , gRootBodyHeight ) ; // root body capsule
2016-07-01 17:35:27 +00:00
int i ;
2017-06-05 21:15:51 +00:00
for ( i = 0 ; i < NUM_LEGS ; i + + )
2016-07-01 17:35:27 +00:00
{
2017-06-05 21:15:51 +00:00
m_shapes [ 1 + 2 * i ] = new btCapsuleShape ( gLegRadius , gLegLength ) ; // leg capsule
m_shapes [ 2 + 2 * i ] = new btCapsuleShape ( gForeLegRadius , gForeLegLength ) ; // fore leg capsule
2016-07-01 17:35:27 +00:00
}
2017-06-05 21:15:51 +00:00
//
2016-07-01 17:35:27 +00:00
// Setup rigid bodies
2017-06-05 21:15:51 +00:00
btScalar rootAboveGroundHeight = gForeLegLength ;
2016-07-01 17:35:27 +00:00
btTransform bodyOffset ; bodyOffset . setIdentity ( ) ;
2017-06-05 21:15:51 +00:00
bodyOffset . setOrigin ( positionOffset ) ;
2016-07-01 17:35:27 +00:00
2017-06-05 21:15:51 +00:00
// root body
btVector3 localRootBodyPosition = btVector3 ( btScalar ( 0. ) , rootAboveGroundHeight , btScalar ( 0. ) ) ; // root body position in local reference frame
2016-07-01 17:35:27 +00:00
btTransform transform ;
transform . setIdentity ( ) ;
2017-06-05 21:15:51 +00:00
transform . setOrigin ( localRootBodyPosition ) ;
2016-09-06 20:30:46 +00:00
2016-09-11 20:25:22 +00:00
btTransform originTransform = transform ;
2017-06-05 21:15:51 +00:00
m_bodies [ 0 ] = localCreateRigidBody ( btScalar ( bFixed ? 0. : 1. ) , bodyOffset * transform , m_shapes [ 0 ] ) ;
2016-09-06 20:30:46 +00:00
m_ownerWorld - > addRigidBody ( m_bodies [ 0 ] ) ;
2016-10-19 19:20:57 +00:00
m_bodyRelativeTransforms [ 0 ] = btTransform : : getIdentity ( ) ;
2016-09-06 20:30:46 +00:00
m_bodies [ 0 ] - > setUserPointer ( this ) ;
2016-10-19 19:56:09 +00:00
m_bodyTouchSensorIndexMap . insert ( btHashPtr ( m_bodies [ 0 ] ) , 0 ) ;
2016-07-01 17:35:27 +00:00
btHingeConstraint * hingeC ;
2017-06-05 21:15:51 +00:00
//btConeTwistConstraint* coneC;
2016-07-01 17:35:27 +00:00
btTransform localA , localB , localC ;
// legs
2017-06-05 21:15:51 +00:00
for ( i = 0 ; i < NUM_LEGS ; i + + )
2016-07-01 17:35:27 +00:00
{
2017-06-05 21:15:51 +00:00
float footAngle = 2 * SIMD_PI * i / NUM_LEGS ; // legs are uniformly distributed around the root body
float footYUnitPosition = sin ( footAngle ) ; // y position of the leg on the unit circle
float footXUnitPosition = cos ( footAngle ) ; // x position of the leg on the unit circle
2016-07-01 17:35:27 +00:00
transform . setIdentity ( ) ;
2017-06-05 21:15:51 +00:00
btVector3 legCOM = btVector3 ( btScalar ( footXUnitPosition * ( gRootBodyRadius + 0.5 * gLegLength ) ) , btScalar ( rootAboveGroundHeight ) , btScalar ( footYUnitPosition * ( gRootBodyRadius + 0.5 * gLegLength ) ) ) ;
2016-07-01 17:35:27 +00:00
transform . setOrigin ( legCOM ) ;
// thigh
2017-06-05 21:15:51 +00:00
btVector3 legDirection = ( legCOM - localRootBodyPosition ) . normalize ( ) ;
2016-07-01 17:35:27 +00:00
btVector3 kneeAxis = legDirection . cross ( vUp ) ;
2016-07-03 17:54:47 +00:00
transform . setRotation ( btQuaternion ( kneeAxis , SIMD_HALF_PI ) ) ;
2016-07-01 17:35:27 +00:00
m_bodies [ 1 + 2 * i ] = localCreateRigidBody ( btScalar ( 1. ) , bodyOffset * transform , m_shapes [ 1 + 2 * i ] ) ;
2016-09-11 20:25:22 +00:00
m_bodyRelativeTransforms [ 1 + 2 * i ] = transform ;
2016-09-06 20:30:46 +00:00
m_bodies [ 1 + 2 * i ] - > setUserPointer ( this ) ;
2016-10-19 19:56:09 +00:00
m_bodyTouchSensorIndexMap . insert ( btHashPtr ( m_bodies [ 1 + 2 * i ] ) , 1 + 2 * i ) ;
2016-07-01 17:35:27 +00:00
// shin
transform . setIdentity ( ) ;
2017-06-05 21:15:51 +00:00
transform . setOrigin ( btVector3 ( btScalar ( footXUnitPosition * ( gRootBodyRadius + gLegLength ) ) , btScalar ( rootAboveGroundHeight - 0.5 * gForeLegLength ) , btScalar ( footYUnitPosition * ( gRootBodyRadius + gLegLength ) ) ) ) ;
2016-07-01 17:35:27 +00:00
m_bodies [ 2 + 2 * i ] = localCreateRigidBody ( btScalar ( 1. ) , bodyOffset * transform , m_shapes [ 2 + 2 * i ] ) ;
2016-09-11 20:25:22 +00:00
m_bodyRelativeTransforms [ 2 + 2 * i ] = transform ;
2016-09-06 20:30:46 +00:00
m_bodies [ 2 + 2 * i ] - > setUserPointer ( this ) ;
2016-10-19 19:56:09 +00:00
m_bodyTouchSensorIndexMap . insert ( btHashPtr ( m_bodies [ 2 + 2 * i ] ) , 2 + 2 * i ) ;
2016-07-01 17:35:27 +00:00
// hip joints
localA . setIdentity ( ) ; localB . setIdentity ( ) ;
2017-06-05 21:15:51 +00:00
localA . getBasis ( ) . setEulerZYX ( 0 , - footAngle , 0 ) ; localA . setOrigin ( btVector3 ( btScalar ( footXUnitPosition * gRootBodyRadius ) , btScalar ( 0. ) , btScalar ( footYUnitPosition * gRootBodyRadius ) ) ) ;
2016-09-11 20:25:22 +00:00
localB = b3ReferenceFrameHelper : : getTransformWorldToLocal ( m_bodies [ 1 + 2 * i ] - > getWorldTransform ( ) , b3ReferenceFrameHelper : : getTransformLocalToWorld ( m_bodies [ 0 ] - > getWorldTransform ( ) , localA ) ) ;
2016-07-01 17:35:27 +00:00
hingeC = new btHingeConstraint ( * m_bodies [ 0 ] , * m_bodies [ 1 + 2 * i ] , localA , localB ) ;
2016-07-03 17:54:47 +00:00
hingeC - > setLimit ( btScalar ( - 0.75 * SIMD_PI_4 ) , btScalar ( SIMD_PI_8 ) ) ;
2017-06-05 21:15:51 +00:00
//hingeC->setLimit(btScalar(-0.1), btScalar(0.1));
2016-07-01 17:35:27 +00:00
m_joints [ 2 * i ] = hingeC ;
// knee joints
localA . setIdentity ( ) ; localB . setIdentity ( ) ; localC . setIdentity ( ) ;
2017-06-05 21:15:51 +00:00
localA . getBasis ( ) . setEulerZYX ( 0 , - footAngle , 0 ) ; localA . setOrigin ( btVector3 ( btScalar ( footXUnitPosition * ( gRootBodyRadius + gLegLength ) ) , btScalar ( 0. ) , btScalar ( footYUnitPosition * ( gRootBodyRadius + gLegLength ) ) ) ) ;
2016-09-11 20:25:22 +00:00
localB = b3ReferenceFrameHelper : : getTransformWorldToLocal ( m_bodies [ 1 + 2 * i ] - > getWorldTransform ( ) , b3ReferenceFrameHelper : : getTransformLocalToWorld ( m_bodies [ 0 ] - > getWorldTransform ( ) , localA ) ) ;
localC = b3ReferenceFrameHelper : : getTransformWorldToLocal ( m_bodies [ 2 + 2 * i ] - > getWorldTransform ( ) , b3ReferenceFrameHelper : : getTransformLocalToWorld ( m_bodies [ 0 ] - > getWorldTransform ( ) , localA ) ) ;
2016-07-01 17:35:27 +00:00
hingeC = new btHingeConstraint ( * m_bodies [ 1 + 2 * i ] , * m_bodies [ 2 + 2 * i ] , localB , localC ) ;
2017-06-05 21:15:51 +00:00
//hingeC->setLimit(btScalar(-0.01), btScalar(0.01));
2016-07-03 17:54:47 +00:00
hingeC - > setLimit ( btScalar ( - SIMD_PI_8 ) , btScalar ( 0.2 ) ) ;
2016-07-01 17:35:27 +00:00
m_joints [ 1 + 2 * i ] = hingeC ;
2016-09-06 20:30:46 +00:00
2017-06-05 21:15:51 +00:00
m_ownerWorld - > addRigidBody ( m_bodies [ 1 + 2 * i ] ) ; // add thigh bone
2016-09-06 20:30:46 +00:00
2017-06-05 21:15:51 +00:00
m_ownerWorld - > addConstraint ( m_joints [ 2 * i ] , true ) ; // connect thigh bone with root
2016-09-06 20:30:46 +00:00
2017-06-05 21:15:51 +00:00
if ( nnWalkersDemo - > detectCollisions ( ) ) { // if thigh bone causes collision, remove it again
2016-09-06 20:30:46 +00:00
m_ownerWorld - > removeRigidBody ( m_bodies [ 1 + 2 * i ] ) ;
2017-06-05 21:15:51 +00:00
m_ownerWorld - > removeConstraint ( m_joints [ 2 * i ] ) ; // disconnect thigh bone from root
2016-09-06 20:30:46 +00:00
}
else {
2017-06-05 21:15:51 +00:00
m_ownerWorld - > addRigidBody ( m_bodies [ 2 + 2 * i ] ) ; // add shin bone
m_ownerWorld - > addConstraint ( m_joints [ 1 + 2 * i ] , true ) ; // connect shin bone with thigh
2016-09-06 20:30:46 +00:00
2017-06-05 21:15:51 +00:00
if ( nnWalkersDemo - > detectCollisions ( ) ) { // if shin bone causes collision, remove it again
2016-09-06 20:30:46 +00:00
m_ownerWorld - > removeRigidBody ( m_bodies [ 2 + 2 * i ] ) ;
2017-06-05 21:15:51 +00:00
m_ownerWorld - > removeConstraint ( m_joints [ 1 + 2 * i ] ) ; // disconnect shin bone from thigh
2016-09-06 20:30:46 +00:00
}
}
2016-07-01 17:35:27 +00:00
}
// Setup some damping on the m_bodies
for ( i = 0 ; i < BODYPART_COUNT ; + + i )
{
2017-06-05 21:15:51 +00:00
m_bodies [ i ] - > setDamping ( 0.05 , 0.85 ) ;
m_bodies [ i ] - > setDeactivationTime ( 0.8 ) ;
//m_bodies[i]->setSleepingThresholds(1.6, 2.5);
m_bodies [ i ] - > setSleepingThresholds ( 0.5f , 0.5f ) ;
2016-07-01 17:35:27 +00:00
}
2016-09-11 20:25:22 +00:00
2017-06-05 21:15:51 +00:00
removeFromWorld ( ) ; // it should not yet be in the world
2016-07-01 17:35:27 +00:00
}
virtual ~ NNWalker ( )
{
int i ;
// Remove all constraints
for ( i = 0 ; i < JOINT_COUNT ; + + i )
{
2017-06-05 21:15:51 +00:00
m_ownerWorld - > removeConstraint ( m_joints [ i ] ) ;
delete m_joints [ i ] ; m_joints [ i ] = 0 ;
2016-07-01 17:35:27 +00:00
}
// Remove all bodies and shapes
for ( i = 0 ; i < BODYPART_COUNT ; + + i )
{
2017-06-05 21:15:51 +00:00
m_ownerWorld - > removeRigidBody ( m_bodies [ i ] ) ;
delete m_bodies [ i ] - > getMotionState ( ) ;
2016-07-01 17:35:27 +00:00
2017-06-05 21:15:51 +00:00
delete m_bodies [ i ] ; m_bodies [ i ] = 0 ;
delete m_shapes [ i ] ; m_shapes [ i ] = 0 ;
2016-07-01 17:35:27 +00:00
}
}
2016-09-11 20:25:22 +00:00
btTypedConstraint * * getJoints ( ) {
return & m_joints [ 0 ] ;
}
2016-07-04 17:17:10 +00:00
void setTouchSensor ( void * bodyPointer ) {
2016-10-19 19:56:09 +00:00
m_touchSensors [ * m_bodyTouchSensorIndexMap . find ( btHashPtr ( bodyPointer ) ) ] = true ;
2016-07-04 17:17:10 +00:00
}
void clearTouchSensors ( ) {
for ( int i = 0 ; i < BODYPART_COUNT ; i + + ) {
2016-07-04 17:43:26 +00:00
m_touchSensors [ i ] = false ;
2016-07-04 17:17:10 +00:00
}
}
2017-06-05 21:15:51 +00:00
bool getTouchSensor ( int i ) {
return m_touchSensors [ i ] ;
2016-09-11 20:25:22 +00:00
}
2016-07-01 17:35:27 +00:00
2017-06-05 21:15:51 +00:00
btScalar * getSensoryMotorWeights ( ) {
return m_sensoryMotorWeights ;
2016-07-04 17:17:10 +00:00
}
2016-07-01 17:35:27 +00:00
2016-09-11 20:25:22 +00:00
void addToWorld ( ) {
2017-06-05 21:15:51 +00:00
int i ;
// add all bodies and shapes
for ( i = 0 ; i < BODYPART_COUNT ; + + i )
{
m_ownerWorld - > addRigidBody ( m_bodies [ i ] ) ;
}
2016-12-30 00:46:36 +00:00
2017-06-05 21:15:51 +00:00
// add all constraints
for ( i = 0 ; i < JOINT_COUNT ; + + i )
{
m_ownerWorld - > addConstraint ( m_joints [ i ] , true ) ; // important! If you add constraints back, you must set bullet physics to disable collision between constrained bodies
2016-09-11 20:25:22 +00:00
}
2017-06-05 21:15:51 +00:00
m_startPosition = getPosition ( ) ;
2016-09-11 20:25:22 +00:00
}
2016-07-01 17:35:27 +00:00
2017-06-05 21:15:51 +00:00
void removeFromWorld ( ) {
int i ;
2016-07-01 17:35:27 +00:00
2017-06-05 21:15:51 +00:00
// Remove all constraints
for ( i = 0 ; i < JOINT_COUNT ; + + i )
{
m_ownerWorld - > removeConstraint ( m_joints [ i ] ) ;
}
2016-09-11 20:25:22 +00:00
2017-06-05 21:15:51 +00:00
// Remove all bodies
for ( i = 0 ; i < BODYPART_COUNT ; + + i )
{
m_ownerWorld - > removeRigidBody ( m_bodies [ i ] ) ;
2016-09-11 20:25:22 +00:00
}
}
btVector3 getPosition ( ) const {
btVector3 finalPosition ( 0 , 0 , 0 ) ;
for ( int i = 0 ; i < BODYPART_COUNT ; i + + )
{
2017-06-05 21:15:51 +00:00
finalPosition + = m_bodies [ i ] - > getCenterOfMassPosition ( ) ;
2016-09-11 20:25:22 +00:00
}
2017-06-05 21:15:51 +00:00
finalPosition / = BODYPART_COUNT ;
2016-09-11 20:25:22 +00:00
return finalPosition ;
}
btScalar getDistanceFitness ( ) const
{
btScalar distance = 0 ;
distance = ( getPosition ( ) - m_startPosition ) . length2 ( ) ;
return distance ;
}
btScalar getFitness ( ) const
{
return getDistanceFitness ( ) ; // for now it is only distance
}
2016-10-19 16:35:01 +00:00
void resetAt ( const btVector3 & position ) {
2016-09-12 21:59:42 +00:00
btTransform resetPosition ( btQuaternion : : getIdentity ( ) , position ) ;
2016-09-11 20:25:22 +00:00
for ( int i = 0 ; i < BODYPART_COUNT ; + + i )
{
2016-11-01 14:34:22 +00:00
m_bodies [ i ] - > setWorldTransform ( resetPosition * m_bodyRelativeTransforms [ i ] ) ;
if ( m_bodies [ i ] - > getMotionState ( ) ) {
m_bodies [ i ] - > getMotionState ( ) - > setWorldTransform ( resetPosition * m_bodyRelativeTransforms [ i ] ) ;
}
2017-06-05 21:15:51 +00:00
m_bodies [ i ] - > clearForces ( ) ;
m_bodies [ i ] - > setAngularVelocity ( btVector3 ( 0 , 0 , 0 ) ) ;
m_bodies [ i ] - > setLinearVelocity ( btVector3 ( 0 , 0 , 0 ) ) ;
2016-11-01 14:34:22 +00:00
2017-06-05 21:15:51 +00:00
}
2016-11-01 14:34:22 +00:00
2016-09-11 20:25:22 +00:00
clearTouchSensors ( ) ;
}
2016-09-12 21:59:42 +00:00
btScalar getEvaluationTime ( ) const {
2016-09-11 20:25:22 +00:00
return m_evaluationTime ;
}
2016-09-12 21:59:42 +00:00
void setEvaluationTime ( btScalar evaluationTime ) {
2016-09-11 20:25:22 +00:00
m_evaluationTime = evaluationTime ;
}
bool isInEvaluation ( ) const {
return m_inEvaluation ;
}
void setInEvaluation ( bool inEvaluation ) {
m_inEvaluation = inEvaluation ;
}
bool isReaped ( ) const {
return m_reaped ;
}
void setReaped ( bool reaped ) {
m_reaped = reaped ;
}
2016-09-12 17:42:07 +00:00
int getIndex ( ) const {
return m_index ;
}
2016-09-11 20:25:22 +00:00
} ;
void evaluationUpdatePreTickCallback ( btDynamicsWorld * world , btScalar timeStep ) ;
bool legContactProcessedCallback ( btManifoldPoint & cp , void * body0 , void * body1 )
2016-07-03 17:54:47 +00:00
{
btCollisionObject * o1 = static_cast < btCollisionObject * > ( body0 ) ;
btCollisionObject * o2 = static_cast < btCollisionObject * > ( body1 ) ;
2016-07-04 17:17:10 +00:00
void * ID1 = o1 - > getUserPointer ( ) ;
void * ID2 = o2 - > getUserPointer ( ) ;
2016-07-03 17:54:47 +00:00
2017-06-05 21:15:51 +00:00
if ( ID1 ! = GROUND_ID | | ID2 ! = GROUND_ID ) {
2016-07-03 17:54:47 +00:00
// Make a circle with a 0.9 radius at (0,0,0)
// with RGB color (1,0,0).
2017-06-05 21:15:51 +00:00
if ( nn3DWalkers - > m_dynamicsWorld - > getDebugDrawer ( ) ! = NULL ) {
if ( ! nn3DWalkers - > mIsHeadless ) {
nn3DWalkers - > m_dynamicsWorld - > getDebugDrawer ( ) - > drawSphere ( cp . getPositionWorldOnA ( ) , 0.1 , btVector3 ( 1. , 0. , 0. ) ) ;
}
2016-09-06 20:30:46 +00:00
}
2016-07-04 17:17:10 +00:00
2017-06-05 21:15:51 +00:00
if ( ID1 ! = GROUND_ID & & ID1 ) {
2016-07-04 17:17:10 +00:00
( ( NNWalker * ) ID1 ) - > setTouchSensor ( o1 ) ;
}
2017-06-05 21:15:51 +00:00
if ( ID2 ! = GROUND_ID & & ID2 ) {
2016-07-04 17:17:10 +00:00
( ( NNWalker * ) ID2 ) - > setTouchSensor ( o2 ) ;
}
2016-07-03 17:54:47 +00:00
}
return false ;
}
2017-06-05 21:15:51 +00:00
struct WalkerFilterCallback : public btOverlapFilterCallback
2016-09-12 17:42:07 +00:00
{
// return true when pairs need collision
virtual bool needBroadphaseCollision ( btBroadphaseProxy * proxy0 , btBroadphaseProxy * proxy1 ) const
{
btCollisionObject * obj0 = static_cast < btCollisionObject * > ( proxy0 - > m_clientObject ) ;
btCollisionObject * obj1 = static_cast < btCollisionObject * > ( proxy1 - > m_clientObject ) ;
if ( obj0 - > getUserPointer ( ) = = GROUND_ID | | obj1 - > getUserPointer ( ) = = GROUND_ID ) { // everything collides with ground
return true ;
}
2017-06-05 21:15:51 +00:00
else {
2016-09-12 17:42:07 +00:00
return ( ( NNWalker * ) obj0 - > getUserPointer ( ) ) - > getIndex ( ) = = ( ( NNWalker * ) obj1 - > getUserPointer ( ) ) - > getIndex ( ) ;
}
}
} ;
2016-09-06 20:30:46 +00:00
void NN3DWalkersExample : : initPhysics ( )
2016-07-01 17:35:27 +00:00
{
2016-09-16 07:49:18 +00:00
2016-09-17 22:42:34 +00:00
setupBasicParamInterface ( ) ; // parameter interface to use timewarp
2016-09-16 07:49:18 +00:00
2016-07-03 17:54:47 +00:00
gContactProcessedCallback = legContactProcessedCallback ;
2016-07-01 17:35:27 +00:00
m_guiHelper - > setUpAxis ( 1 ) ;
// Setup the basic world
2017-06-05 21:15:51 +00:00
m_Time = 0 ;
2016-07-01 17:35:27 +00:00
2017-06-05 21:15:51 +00:00
createEmptyDynamicsWorld ( ) ;
2016-07-04 17:17:10 +00:00
2017-06-05 21:15:51 +00:00
m_dynamicsWorld - > setInternalTickCallback ( evaluationUpdatePreTickCallback , this , true ) ;
m_guiHelper - > createPhysicsDebugDrawer ( m_dynamicsWorld ) ;
2017-05-29 19:56:18 +00:00
2017-06-05 21:15:51 +00:00
m_targetFrequency = 3 ;
2016-07-04 17:17:10 +00:00
// new SIMD solver for joints clips accumulated impulse, so the new limits for the motor
// should be (numberOfsolverIterations * oldLimits)
2017-06-05 21:15:51 +00:00
m_motorStrength = 0.05f * m_dynamicsWorld - > getSolverInfo ( ) . m_numIterations ;
2016-07-04 17:17:10 +00:00
{ // create a slider to change the motor update frequency
2017-06-05 21:15:51 +00:00
SliderParams slider ( " Motor update frequency " , & m_targetFrequency ) ;
2016-07-04 17:17:10 +00:00
slider . m_minVal = 0 ;
slider . m_maxVal = 10 ;
slider . m_clampToNotches = false ;
2017-06-05 21:15:51 +00:00
m_guiHelper - > getParameterInterface ( ) - > registerSliderFloatParameter (
slider ) ;
2016-07-04 17:17:10 +00:00
}
{ // create a slider to change the motor torque
2017-06-05 21:15:51 +00:00
SliderParams slider ( " Motor force " , & m_motorStrength ) ;
2016-07-04 17:17:10 +00:00
slider . m_minVal = 1 ;
slider . m_maxVal = 50 ;
slider . m_clampToNotches = false ;
2017-06-05 21:15:51 +00:00
m_guiHelper - > getParameterInterface ( ) - > registerSliderFloatParameter (
slider ) ;
2016-07-04 17:17:10 +00:00
}
2016-07-01 17:35:27 +00:00
2016-07-04 17:43:26 +00:00
{ // create a slider to change the root body radius
SliderParams slider ( " Root body radius " , & gRootBodyRadius ) ;
slider . m_minVal = 0.01f ;
slider . m_maxVal = 10 ;
slider . m_clampToNotches = false ;
2017-06-05 21:15:51 +00:00
m_guiHelper - > getParameterInterface ( ) - > registerSliderFloatParameter (
slider ) ;
2016-07-04 17:43:26 +00:00
}
{ // create a slider to change the root body height
SliderParams slider ( " Root body height " , & gRootBodyHeight ) ;
slider . m_minVal = 0.01f ;
slider . m_maxVal = 10 ;
slider . m_clampToNotches = false ;
2017-06-05 21:15:51 +00:00
m_guiHelper - > getParameterInterface ( ) - > registerSliderFloatParameter (
slider ) ;
2016-07-04 17:43:26 +00:00
}
{ // create a slider to change the leg radius
SliderParams slider ( " Leg radius " , & gLegRadius ) ;
slider . m_minVal = 0.01f ;
slider . m_maxVal = 10 ;
slider . m_clampToNotches = false ;
2017-06-05 21:15:51 +00:00
m_guiHelper - > getParameterInterface ( ) - > registerSliderFloatParameter (
slider ) ;
2016-07-04 17:43:26 +00:00
}
{ // create a slider to change the leg length
SliderParams slider ( " Leg length " , & gLegLength ) ;
slider . m_minVal = 0.01f ;
slider . m_maxVal = 10 ;
slider . m_clampToNotches = false ;
2017-06-05 21:15:51 +00:00
m_guiHelper - > getParameterInterface ( ) - > registerSliderFloatParameter (
slider ) ;
2016-07-04 17:43:26 +00:00
}
{ // create a slider to change the fore leg radius
2016-09-06 20:30:46 +00:00
SliderParams slider ( " Fore Leg radius " , & gForeLegRadius ) ;
slider . m_minVal = 0.01f ;
slider . m_maxVal = 10 ;
slider . m_clampToNotches = false ;
2017-06-05 21:15:51 +00:00
m_guiHelper - > getParameterInterface ( ) - > registerSliderFloatParameter (
slider ) ;
2016-09-06 20:30:46 +00:00
}
2016-07-04 17:43:26 +00:00
2016-09-06 20:30:46 +00:00
{ // create a slider to change the fore leg length
SliderParams slider ( " Fore Leg length " , & gForeLegLength ) ;
slider . m_minVal = 0.01f ;
slider . m_maxVal = 10 ;
slider . m_clampToNotches = false ;
2017-06-05 21:15:51 +00:00
m_guiHelper - > getParameterInterface ( ) - > registerSliderFloatParameter (
slider ) ;
2016-09-06 20:30:46 +00:00
}
2016-07-04 17:43:26 +00:00
2016-09-12 17:42:07 +00:00
{ // create a slider to change the number of parallel evaluations
SliderParams slider ( " Parallel evaluations " , & gParallelEvaluations ) ;
slider . m_minVal = 1 ;
2017-06-05 21:15:51 +00:00
slider . m_maxVal = NUM_WALKERS ;
slider . m_clampToIntegers = true ;
2016-09-12 17:42:07 +00:00
m_guiHelper - > getParameterInterface ( ) - > registerSliderFloatParameter (
slider ) ;
}
2017-05-29 19:56:18 +00:00
2017-06-05 21:15:51 +00:00
// Setup a big ground box
{
btCollisionShape * groundShape = new btBoxShape ( btVector3 ( btScalar ( 200. ) , btScalar ( 10. ) , btScalar ( 200. ) ) ) ;
m_collisionShapes . push_back ( groundShape ) ;
btTransform groundTransform ;
groundTransform . setIdentity ( ) ;
groundTransform . setOrigin ( btVector3 ( 0 , - 10 , 0 ) ) ;
btRigidBody * ground = createRigidBody ( btScalar ( 0. ) , groundTransform , groundShape ) ;
ground - > setFriction ( 5 ) ;
ground - > setUserPointer ( GROUND_ID ) ;
}
for ( int i = 0 ; i < NUM_WALKERS ; i + + ) {
if ( RANDOMIZE_DIMENSIONS ) {
float maxDimension = 0.2f ;
// randomize the dimensions
gRootBodyRadius = ( ( double ) rand ( ) / ( RAND_MAX ) ) * ( maxDimension - 0.01f ) + 0.01f ;
gRootBodyHeight = ( ( double ) rand ( ) / ( RAND_MAX ) ) * ( maxDimension - 0.01f ) + 0.01f ;
gLegRadius = ( ( double ) rand ( ) / ( RAND_MAX ) ) * ( maxDimension - 0.01f ) + 0.01f ;
gLegLength = ( ( double ) rand ( ) / ( RAND_MAX ) ) * ( maxDimension - 0.01f ) + 0.01f ;
gForeLegLength = ( ( double ) rand ( ) / ( RAND_MAX ) ) * ( maxDimension - 0.01f ) + 0.01f ;
gForeLegRadius = ( ( double ) rand ( ) / ( RAND_MAX ) ) * ( maxDimension - 0.01f ) + 0.01f ;
2016-07-04 17:17:10 +00:00
}
2016-07-01 17:35:27 +00:00
2017-06-05 21:15:51 +00:00
// Spawn one walker
btVector3 offset ( 0 , 0 , 0 ) ;
spawnWalker ( i , offset , false ) ;
2016-09-11 20:25:22 +00:00
}
2016-09-12 14:36:37 +00:00
2017-06-05 21:15:51 +00:00
btOverlapFilterCallback * filterCallback = new WalkerFilterCallback ( ) ;
m_dynamicsWorld - > getPairCache ( ) - > setOverlapFilterCallback ( filterCallback ) ;
2016-09-12 17:42:07 +00:00
2017-06-05 21:15:51 +00:00
m_timeSeriesCanvas = new TimeSeriesCanvas ( m_guiHelper - > getAppInterface ( ) - > m_2dCanvasInterface , 300 , 200 , " Fitness Performance " ) ;
m_timeSeriesCanvas - > setupTimeSeries ( 40 , NUM_WALKERS * EVALUATION_TIME , 0 ) ;
for ( int i = 0 ; i < NUM_WALKERS ; i + + ) {
m_timeSeriesCanvas - > addDataSource ( " " , 100 * i / NUM_WALKERS , 100 * ( NUM_WALKERS - i ) / NUM_WALKERS , 100 * ( i ) / NUM_WALKERS ) ;
2016-09-12 14:36:37 +00:00
}
2017-06-05 21:15:51 +00:00
}
2016-07-01 17:35:27 +00:00
2016-12-27 20:00:22 +00:00
2017-06-05 21:15:51 +00:00
void NN3DWalkersExample : : spawnWalker ( int index , const btVector3 & startOffset , bool bFixed )
{
NNWalker * walker = new NNWalker ( index , m_dynamicsWorld , startOffset , bFixed ) ;
m_walkersInPopulation . push_back ( walker ) ;
2016-07-01 17:35:27 +00:00
}
2016-09-11 20:25:22 +00:00
bool NN3DWalkersExample : : detectCollisions ( )
2016-07-01 17:35:27 +00:00
{
2016-09-11 20:25:22 +00:00
bool collisionDetected = false ;
2017-06-05 21:15:51 +00:00
if ( m_dynamicsWorld ) {
m_dynamicsWorld - > performDiscreteCollisionDetection ( ) ; // let the collisions be calculated
}
2016-07-01 17:35:27 +00:00
2017-06-05 21:15:51 +00:00
int numManifolds = m_dynamicsWorld - > getDispatcher ( ) - > getNumManifolds ( ) ;
for ( int i = 0 ; i < numManifolds ; i + + )
{
btPersistentManifold * contactManifold = m_dynamicsWorld - > getDispatcher ( ) - > getManifoldByIndexInternal ( i ) ;
const btCollisionObject * obA = contactManifold - > getBody0 ( ) ;
const btCollisionObject * obB = contactManifold - > getBody1 ( ) ;
2016-07-01 17:35:27 +00:00
2017-06-05 21:15:51 +00:00
if ( obA - > getUserPointer ( ) ! = GROUND_ID & & obB - > getUserPointer ( ) ! = GROUND_ID ) {
2016-07-03 17:54:47 +00:00
2017-06-05 21:15:51 +00:00
int numContacts = contactManifold - > getNumContacts ( ) ;
for ( int j = 0 ; j < numContacts ; j + + )
{
collisionDetected = true ;
btManifoldPoint & pt = contactManifold - > getContactPoint ( j ) ;
if ( pt . getDistance ( ) < 0.f )
2016-09-11 20:25:22 +00:00
{
2017-06-05 21:15:51 +00:00
//const btVector3& ptA = pt.getPositionWorldOnA();
//const btVector3& ptB = pt.getPositionWorldOnB();
//const btVector3& normalOnB = pt.m_normalWorldOnB;
2016-07-04 17:17:10 +00:00
2017-06-05 21:15:51 +00:00
if ( ! DRAW_INTERPENETRATIONS ) {
return collisionDetected ;
}
if ( m_dynamicsWorld - > getDebugDrawer ( ) ) {
m_dynamicsWorld - > getDebugDrawer ( ) - > drawSphere ( pt . getPositionWorldOnA ( ) , 0.1 , btVector3 ( 0. , 0. , 1. ) ) ;
m_dynamicsWorld - > getDebugDrawer ( ) - > drawSphere ( pt . getPositionWorldOnB ( ) , 0.1 , btVector3 ( 0. , 0. , 1. ) ) ;
2016-09-11 20:25:22 +00:00
}
2016-07-04 17:17:10 +00:00
}
2016-07-03 17:54:47 +00:00
}
2016-07-01 17:35:27 +00:00
}
}
2016-09-11 20:25:22 +00:00
return collisionDetected ;
2016-07-01 17:35:27 +00:00
}
2016-09-06 20:30:46 +00:00
bool NN3DWalkersExample : : keyboardCallback ( int key , int state )
2016-07-01 17:35:27 +00:00
{
switch ( key )
{
case ' [ ' :
2017-06-05 21:15:51 +00:00
m_motorStrength / = 1.1f ;
2016-07-01 17:35:27 +00:00
return true ;
case ' ] ' :
2017-06-05 21:15:51 +00:00
m_motorStrength * = 1.1f ;
2016-07-01 17:35:27 +00:00
return true ;
2016-09-12 14:36:37 +00:00
case ' l ' :
2017-06-05 21:15:51 +00:00
printWalkerConfigs ( ) ;
2016-09-17 22:42:34 +00:00
return true ;
2016-07-01 17:35:27 +00:00
default :
break ;
}
2016-10-22 20:50:08 +00:00
return NN3DWalkersTimeWarpBase : : keyboardCallback ( key , state ) ;
2016-07-01 17:35:27 +00:00
}
2016-09-06 20:30:46 +00:00
void NN3DWalkersExample : : exitPhysics ( )
2016-07-01 17:35:27 +00:00
{
2016-08-02 18:38:36 +00:00
gContactProcessedCallback = NULL ; // clear contact processed callback on exiting
2017-06-05 21:15:51 +00:00
int i ;
for ( i = 0 ; i < NUM_WALKERS ; i + + )
2016-07-01 17:35:27 +00:00
{
2016-09-11 20:25:22 +00:00
NNWalker * walker = m_walkersInPopulation [ i ] ;
2016-07-01 17:35:27 +00:00
delete walker ;
}
CommonRigidBodyBase : : exitPhysics ( ) ;
}
2016-09-11 20:25:22 +00:00
class CommonExampleInterface * ET_NN3DWalkersCreateFunc ( struct CommonExampleOptions & options )
2016-07-01 17:35:27 +00:00
{
2016-09-06 20:30:46 +00:00
nn3DWalkers = new NN3DWalkersExample ( options . m_guiHelper ) ;
2016-07-03 17:54:47 +00:00
return nn3DWalkers ;
2016-07-01 17:35:27 +00:00
}
2016-09-11 20:25:22 +00:00
bool fitnessComparator ( const NNWalker * a , const NNWalker * b )
{
2016-09-12 14:36:37 +00:00
return a - > getFitness ( ) > b - > getFitness ( ) ; // sort walkers descending
2016-09-11 20:25:22 +00:00
}
void NN3DWalkersExample : : rateEvaluations ( ) {
m_walkersInPopulation . quickSort ( fitnessComparator ) ; // Sort walkers by fitness
2016-07-01 17:35:27 +00:00
2016-09-11 20:25:22 +00:00
b3Printf ( " Best performing walker: %f meters " , btSqrt ( m_walkersInPopulation [ 0 ] - > getDistanceFitness ( ) ) ) ;
2017-06-05 21:15:51 +00:00
for ( int i = 0 ; i < NUM_WALKERS ; i + + ) {
m_timeSeriesCanvas - > insertDataAtCurrentTime ( btSqrt ( m_walkersInPopulation [ i ] - > getDistanceFitness ( ) ) , 0 , true ) ;
2016-09-12 14:36:37 +00:00
}
2017-06-05 21:15:51 +00:00
m_timeSeriesCanvas - > nextTick ( ) ;
2016-09-12 14:36:37 +00:00
2017-06-05 21:15:51 +00:00
for ( int i = 0 ; i < NUM_WALKERS ; i + + ) {
2016-09-11 20:25:22 +00:00
m_walkersInPopulation [ i ] - > setEvaluationTime ( 0 ) ;
}
2017-06-05 21:15:51 +00:00
m_nextReaped = 0 ;
2016-07-01 17:35:27 +00:00
}
2016-09-11 20:25:22 +00:00
void NN3DWalkersExample : : reap ( ) {
int reaped = 0 ;
2017-06-05 21:15:51 +00:00
for ( int i = NUM_WALKERS - 1 ; i > = ( NUM_WALKERS - 1 ) * ( 1 - REAP_QTY ) ; i - - ) { // reap a certain percentage
2016-09-11 20:25:22 +00:00
m_walkersInPopulation [ i ] - > setReaped ( true ) ;
reaped + + ;
2017-06-05 21:15:51 +00:00
b3Printf ( " %i Walker(s) reaped. " , reaped ) ;
2016-09-11 20:25:22 +00:00
}
2016-07-01 17:35:27 +00:00
}
2016-09-11 20:25:22 +00:00
NNWalker * NN3DWalkersExample : : getRandomElite ( ) {
2017-06-05 21:15:51 +00:00
return m_walkersInPopulation [ ( ( NUM_WALKERS - 1 ) * SOW_ELITE_QTY ) * ( rand ( ) / RAND_MAX ) ] ;
2016-07-01 17:35:27 +00:00
}
2016-09-11 20:25:22 +00:00
NNWalker * NN3DWalkersExample : : getRandomNonElite ( ) {
2017-06-05 21:15:51 +00:00
return m_walkersInPopulation [ ( NUM_WALKERS - 1 ) * SOW_ELITE_QTY + ( NUM_WALKERS - 1 ) * ( 1.0f - SOW_ELITE_QTY ) * ( rand ( ) / RAND_MAX ) ] ;
2016-07-01 17:35:27 +00:00
}
2016-07-03 17:54:47 +00:00
2016-09-11 20:25:22 +00:00
NNWalker * NN3DWalkersExample : : getNextReaped ( ) {
2017-06-05 21:15:51 +00:00
if ( ( NUM_WALKERS - 1 ) - m_nextReaped > = ( NUM_WALKERS - 1 ) * ( 1 - REAP_QTY ) ) {
m_nextReaped + + ;
2016-09-11 20:25:22 +00:00
}
2017-06-05 21:15:51 +00:00
if ( m_walkersInPopulation [ ( NUM_WALKERS - 1 ) - m_nextReaped + 1 ] - > isReaped ( ) ) {
return m_walkersInPopulation [ ( NUM_WALKERS - 1 ) - m_nextReaped + 1 ] ;
2016-09-12 14:36:37 +00:00
}
else {
return NULL ; // we asked for too many
}
2016-09-11 20:25:22 +00:00
}
void NN3DWalkersExample : : sow ( ) {
int sow = 0 ;
2017-06-05 21:15:51 +00:00
for ( int i = 0 ; i < NUM_WALKERS * ( SOW_CROSSOVER_QTY ) ; i + + ) { // create number of new crossover creatures
2016-09-11 20:25:22 +00:00
sow + + ;
2017-06-05 21:15:51 +00:00
b3Printf ( " %i Walker(s) sown. " , sow ) ;
2016-09-12 21:59:42 +00:00
NNWalker * mother = getRandomElite ( ) ; // Get elite partner (mother)
NNWalker * father = ( SOW_ELITE_PARTNER < rand ( ) / RAND_MAX ) ? getRandomElite ( ) : getRandomNonElite ( ) ; //Get elite or random partner (father)
2016-09-11 20:25:22 +00:00
NNWalker * offspring = getNextReaped ( ) ;
crossover ( mother , father , offspring ) ;
}
2017-06-05 21:15:51 +00:00
for ( int i = NUM_WALKERS * SOW_ELITE_QTY ; i < NUM_WALKERS * ( SOW_ELITE_QTY + SOW_MUTATION_QTY ) ; i + + ) { // create mutants
mutate ( m_walkersInPopulation [ i ] , btScalar ( MUTATION_RATE / ( NUM_WALKERS * SOW_MUTATION_QTY ) * ( i - NUM_WALKERS * SOW_ELITE_QTY ) ) ) ;
2016-09-11 20:25:22 +00:00
}
2017-06-05 21:15:51 +00:00
for ( int i = 0 ; i < ( NUM_WALKERS - 1 ) * ( REAP_QTY - SOW_CROSSOVER_QTY ) ; i + + ) {
2016-09-11 20:25:22 +00:00
sow + + ;
2017-06-05 21:15:51 +00:00
b3Printf ( " %i Walker(s) sown. " , sow ) ;
2016-09-11 20:25:22 +00:00
NNWalker * reaped = getNextReaped ( ) ;
reaped - > setReaped ( false ) ;
reaped - > randomizeSensoryMotorWeights ( ) ;
}
}
void NN3DWalkersExample : : crossover ( NNWalker * mother , NNWalker * father , NNWalker * child ) {
for ( int i = 0 ; i < BODYPART_COUNT * JOINT_COUNT ; i + + ) {
btScalar random = ( ( double ) rand ( ) / ( RAND_MAX ) ) ;
if ( random > = 0.5f ) {
child - > getSensoryMotorWeights ( ) [ i ] = mother - > getSensoryMotorWeights ( ) [ i ] ;
}
else
{
child - > getSensoryMotorWeights ( ) [ i ] = father - > getSensoryMotorWeights ( ) [ i ] ;
}
}
2016-07-03 17:54:47 +00:00
}
2016-09-12 21:59:42 +00:00
void NN3DWalkersExample : : mutate ( NNWalker * mutant , btScalar mutationRate ) {
2016-09-11 20:25:22 +00:00
for ( int i = 0 ; i < BODYPART_COUNT * JOINT_COUNT ; i + + ) {
btScalar random = ( ( double ) rand ( ) / ( RAND_MAX ) ) ;
if ( random > = mutationRate ) {
mutant - > getSensoryMotorWeights ( ) [ i ] = ( ( double ) rand ( ) / ( RAND_MAX ) ) * 2.0f - 1.0f ;
}
}
}
void evaluationUpdatePreTickCallback ( btDynamicsWorld * world , btScalar timeStep ) {
NN3DWalkersExample * nnWalkersDemo = ( NN3DWalkersExample * ) world - > getWorldUserInfo ( ) ;
2017-06-05 21:15:51 +00:00
2016-09-11 20:25:22 +00:00
nnWalkersDemo - > update ( timeStep ) ;
}
2016-09-12 21:59:42 +00:00
void NN3DWalkersExample : : update ( const btScalar timeSinceLastTick ) {
2016-09-11 20:25:22 +00:00
updateEvaluations ( timeSinceLastTick ) ; /**!< We update all evaluations that are in the loop */
scheduleEvaluations ( ) ; /**!< Start new evaluations and finish the old ones. */
2016-09-12 14:36:37 +00:00
2016-09-12 21:59:42 +00:00
drawMarkings ( ) ; /**!< Draw markings on the ground */
2016-09-17 22:42:34 +00:00
2017-06-05 21:15:51 +00:00
if ( m_Time > m_SpeedupTimestamp + 2.0f ) { // print effective speedup
2016-09-17 22:52:08 +00:00
b3Printf ( " Avg Effective speedup: %f real time " , calculatePerformedSpeedup ( ) ) ;
2017-06-05 21:15:51 +00:00
m_SpeedupTimestamp = m_Time ;
2016-09-17 22:52:08 +00:00
}
2016-09-11 20:25:22 +00:00
}
void NN3DWalkersExample : : updateEvaluations ( const btScalar timeSinceLastTick ) {
btScalar delta = timeSinceLastTick ;
2016-09-12 21:59:42 +00:00
btScalar minFPS = 1.f / 60.f ;
2016-09-11 20:25:22 +00:00
if ( delta > minFPS ) {
delta = minFPS ;
}
2017-06-05 21:15:51 +00:00
m_Time + = delta ;
2016-09-11 20:25:22 +00:00
2017-06-05 21:15:51 +00:00
m_targetAccumulator + = delta ;
for ( int i = 0 ; i < NUM_WALKERS ; i + + ) // evaluation time passes
2016-09-11 20:25:22 +00:00
{
2017-06-05 21:15:51 +00:00
if ( m_walkersInPopulation [ i ] - > isInEvaluation ( ) ) {
m_walkersInPopulation [ i ] - > setEvaluationTime ( m_walkersInPopulation [ i ] - > getEvaluationTime ( ) + delta ) ; // increase evaluation time
}
}
2016-09-11 20:25:22 +00:00
2017-06-05 21:15:51 +00:00
if ( m_targetAccumulator > = 1.0f / ( ( double ) m_targetFrequency ) )
{
m_targetAccumulator = 0 ;
2016-09-11 20:25:22 +00:00
2017-06-05 21:15:51 +00:00
for ( int r = 0 ; r < NUM_WALKERS ; r + + )
{
if ( m_walkersInPopulation [ r ] - > isInEvaluation ( ) )
2016-09-11 20:25:22 +00:00
{
2017-06-05 21:15:51 +00:00
for ( int i = 0 ; i < 2 * NUM_LEGS ; i + + )
2016-09-11 20:25:22 +00:00
{
2017-06-05 21:15:51 +00:00
btScalar targetAngle = 0 ;
2016-09-11 20:25:22 +00:00
btHingeConstraint * hingeC = static_cast < btHingeConstraint * > ( m_walkersInPopulation [ r ] - > getJoints ( ) [ i ] ) ;
2017-06-05 21:15:51 +00:00
if ( RANDOM_MOVEMENT ) {
targetAngle = ( ( double ) rand ( ) / ( RAND_MAX ) ) ;
}
else { // neural network movement
2016-09-11 20:25:22 +00:00
2017-06-05 21:15:51 +00:00
// accumulate sensor inputs with weights
for ( int j = 0 ; j < JOINT_COUNT ; j + + ) {
2016-09-11 20:25:22 +00:00
targetAngle + = m_walkersInPopulation [ r ] - > getSensoryMotorWeights ( ) [ i + j * BODYPART_COUNT ] * m_walkersInPopulation [ r ] - > getTouchSensor ( i ) ;
}
2017-06-05 21:15:51 +00:00
// apply the activation function
targetAngle = ( tanh ( targetAngle ) + 1.0f ) * 0.5f ;
}
btScalar targetLimitAngle = hingeC - > getLowerLimit ( ) + targetAngle * ( hingeC - > getUpperLimit ( ) - hingeC - > getLowerLimit ( ) ) ;
btScalar currentAngle = hingeC - > getHingeAngle ( ) ;
btScalar angleError = targetLimitAngle - currentAngle ;
btScalar desiredAngularVel = 0 ;
if ( delta ) {
desiredAngularVel = angleError / delta ;
2016-09-12 21:59:42 +00:00
}
2017-06-05 21:15:51 +00:00
else {
desiredAngularVel = angleError / 0.0001f ;
}
hingeC - > enableAngularMotor ( true , desiredAngularVel , m_motorStrength ) ;
2016-09-11 20:25:22 +00:00
}
2016-11-01 14:34:22 +00:00
2017-06-05 21:15:51 +00:00
// clear sensor signals after usage
m_walkersInPopulation [ r ] - > clearTouchSensors ( ) ;
}
2016-09-11 20:25:22 +00:00
}
}
}
void NN3DWalkersExample : : scheduleEvaluations ( ) {
2017-06-05 21:15:51 +00:00
for ( int i = 0 ; i < NUM_WALKERS ; i + + ) {
2016-09-11 20:25:22 +00:00
2017-06-05 21:15:51 +00:00
if ( m_walkersInPopulation [ i ] - > isInEvaluation ( ) & & m_walkersInPopulation [ i ] - > getEvaluationTime ( ) > = EVALUATION_TIME ) { /**!< tear down evaluations */
b3Printf ( " An evaluation finished at %f s. Distance: %f m " , m_Time , btSqrt ( m_walkersInPopulation [ i ] - > getDistanceFitness ( ) ) ) ;
m_walkersInPopulation [ i ] - > setInEvaluation ( false ) ;
2016-09-11 20:25:22 +00:00
m_walkersInPopulation [ i ] - > removeFromWorld ( ) ;
2017-06-05 21:15:51 +00:00
m_evaluationsQty - - ;
2016-09-11 20:25:22 +00:00
}
2017-06-05 21:15:51 +00:00
if ( m_evaluationsQty < gParallelEvaluations & & ! m_walkersInPopulation [ i ] - > isInEvaluation ( ) & & m_walkersInPopulation [ i ] - > getEvaluationTime ( ) = = 0 ) { /**!< Setup the new evaluations */
b3Printf ( " An evaluation started at %f s. " , m_Time ) ;
m_evaluationsQty + + ;
m_walkersInPopulation [ i ] - > setInEvaluation ( true ) ;
2016-09-12 17:42:07 +00:00
2017-06-05 21:15:51 +00:00
if ( m_walkersInPopulation [ i ] - > getEvaluationTime ( ) = = 0 ) { // reset to origin if the evaluation did not yet run
m_walkersInPopulation [ i ] - > resetAt ( btVector3 ( 0 , 0 , 0 ) ) ;
2016-11-01 14:34:22 +00:00
}
2017-06-05 21:15:51 +00:00
2016-09-11 20:25:22 +00:00
m_walkersInPopulation [ i ] - > addToWorld ( ) ;
2017-06-05 21:15:51 +00:00
m_guiHelper - > autogenerateGraphicsObjects ( m_dynamicsWorld ) ;
2016-09-11 20:25:22 +00:00
}
}
2017-06-05 21:15:51 +00:00
if ( m_evaluationsQty = = 0 ) { // if there are no more evaluations possible
rateEvaluations ( ) ; // rate evaluations by sorting them based on their fitness
2016-09-11 20:25:22 +00:00
2017-06-05 21:15:51 +00:00
reap ( ) ; // reap worst performing walkers
2016-09-11 20:25:22 +00:00
2017-06-05 21:15:51 +00:00
sow ( ) ; // crossover & mutate and sow new walkers
2016-09-11 20:25:22 +00:00
b3Printf ( " ### A new generation started. ### " ) ;
}
2016-07-03 17:54:47 +00:00
}
2016-09-12 14:36:37 +00:00
void NN3DWalkersExample : : drawMarkings ( ) {
2017-06-05 21:15:51 +00:00
if ( ! mIsHeadless ) {
for ( int i = 0 ; i < NUM_WALKERS ; i + + ) // draw current distance plates of moving walkers
2016-09-17 22:42:34 +00:00
{
if ( m_walkersInPopulation [ i ] - > isInEvaluation ( ) ) {
btVector3 walkerPosition = m_walkersInPopulation [ i ] - > getPosition ( ) ;
2016-10-19 19:20:57 +00:00
char performance [ 20 ] ;
2016-09-17 22:42:34 +00:00
sprintf ( performance , " %.2f m " , btSqrt ( m_walkersInPopulation [ i ] - > getDistanceFitness ( ) ) ) ;
2017-06-05 21:15:51 +00:00
m_guiHelper - > drawText3D ( performance , walkerPosition . x ( ) , walkerPosition . y ( ) + 1 , walkerPosition . z ( ) , 1 ) ;
2016-09-17 22:42:34 +00:00
}
2016-09-12 14:36:37 +00:00
}
2017-06-05 21:15:51 +00:00
for ( int i = 2 ; i < 50 ; i + = 2 ) { // draw distance circles
if ( m_dynamicsWorld - > getDebugDrawer ( ) ) {
m_dynamicsWorld - > getDebugDrawer ( ) - > drawArc ( btVector3 ( 0 , 0 , 0 ) , btVector3 ( 0 , 1 , 0 ) , btVector3 ( 1 , 0 , 0 ) , btScalar ( i ) , btScalar ( i ) , btScalar ( 0 ) , btScalar ( SIMD_2_PI ) , btVector3 ( 10 * i , 0 , 0 ) , false ) ;
2016-09-17 22:42:34 +00:00
}
2016-09-12 14:36:37 +00:00
}
}
}
2017-06-05 21:15:51 +00:00
void NN3DWalkersExample : : printWalkerConfigs ( ) {
#if 0
char configString [ 25 + NUM_WALKERS * BODYPART_COUNT * JOINT_COUNT * ( 3 + 15 + 1 ) + NUM_WALKERS * 4 + 1 ] ; // 15 precision + [],\n
char * runner = configString ;
sprintf ( runner , " Population configuration: " ) ;
runner + = 25 ;
for ( int i = 0 ; i < NUM_WALKERS ; i + + ) {
runner [ 0 ] = ' \n ' ;
runner + + ;
runner [ 0 ] = ' [ ' ;
runner + + ;
for ( int j = 0 ; j < BODYPART_COUNT * JOINT_COUNT ; j + + ) {
sprintf ( runner , " %.15f " , m_walkersInPopulation [ i ] - > getSensoryMotorWeights ( ) [ j ] ) ;
runner + = 15 ;
if ( j + 1 < BODYPART_COUNT * JOINT_COUNT ) {
runner [ 0 ] = ' , ' ;
}
else {
runner [ 0 ] = ' ] ' ;
}
runner + + ;
}
}
runner [ 0 ] = ' \0 ' ;
b3Printf ( configString ) ;
# endif
}