-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathPyBulletSimulator.py
791 lines (629 loc) · 36.5 KB
/
PyBulletSimulator.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
import numpy as np
import pybullet as pyb # Pybullet server
import pybullet_data
import time as time
import sys
import pinocchio as pin
class pybullet_simulator:
"""Wrapper for the PyBullet simulator to initialize the simulation, interact with it
and use various utility functions
Args:
q_init (array): the default position of the robot
envID (int): identifier of the current environment to be able to handle different scenarios
use_flat_plane (bool): to use either a flat ground or a rough ground
enable_pyb_GUI (bool): to display PyBullet GUI or not
dt (float): time step of the inverse dynamics
"""
def __init__(self, q_init, envID, use_flat_plane, enable_pyb_GUI, dt=0.001):
self.applied_force = np.zeros(3)
# Start the client for PyBullet
if enable_pyb_GUI:
pyb.connect(pyb.GUI)
pyb.configureDebugVisualizer(pyb.COV_ENABLE_GUI, 0)
pyb.configureDebugVisualizer(pyb.COV_ENABLE_SHADOWS, 0)
else:
pyb.connect(pyb.DIRECT)
# p.GUI for graphical version
# p.DIRECT for non-graphical version
# Load horizontal plane
pyb.setAdditionalSearchPath(pybullet_data.getDataPath())
# Roll and Pitch angle of the ground
p_roll = 0.0 / 57.3 # Roll angle of ground
p_pitch = 0.0 / 57.3 # Pitch angle of ground
# Either use a flat ground or a rough terrain
if use_flat_plane:
self.planeId = pyb.loadURDF("plane.urdf") # Flat plane
self.planeIdbis = pyb.loadURDF("plane.urdf") # Flat plane
# Tune position and orientation of plane
pyb.resetBasePositionAndOrientation(self.planeId, [0, 0, 0.0], pin.Quaternion(pin.rpy.rpyToMatrix(p_roll, p_pitch, 0.0)).coeffs())
pyb.resetBasePositionAndOrientation(self.planeIdbis, [200.0, 0, -100.0 * np.sin(p_pitch)], pin.Quaternion(pin.rpy.rpyToMatrix(p_roll, p_pitch, 0.0)).coeffs())
else:
import random
random.seed(41)
# p.configureDebugVisualizer(p.COV_ENABLE_RENDERING,0)
heightPerturbationRange = 0.05
numHeightfieldRows = 256*2
numHeightfieldColumns = 256*2
heightfieldData = [0]*numHeightfieldRows*numHeightfieldColumns
height_prev = 0.0
for j in range(int(numHeightfieldColumns/2)):
for i in range(int(numHeightfieldRows/2)):
# uniform distribution
height = random.uniform(0, heightPerturbationRange)
# height = 0.25*np.sin(2*np.pi*(i-128)/46) # sinus pattern
heightfieldData[2*i+2*j * numHeightfieldRows] = (height + height_prev) * 0.5
heightfieldData[2*i+1+2*j*numHeightfieldRows] = height
heightfieldData[2*i+(2*j+1) * numHeightfieldRows] = (height + height_prev) * 0.5
heightfieldData[2*i+1+(2*j+1)*numHeightfieldRows] = height
height_prev = height
# Create the collision shape based on the height field
terrainShape = pyb.createCollisionShape(shapeType=pyb.GEOM_HEIGHTFIELD, meshScale=[.05, .05, 1],
heightfieldTextureScaling=(numHeightfieldRows-1)/2,
heightfieldData=heightfieldData,
numHeightfieldRows=numHeightfieldRows,
numHeightfieldColumns=numHeightfieldColumns)
self.planeId = pyb.createMultiBody(0, terrainShape)
pyb.resetBasePositionAndOrientation(self.planeId, [0, 0, 0], [0, 0, 0, 1])
pyb.changeVisualShape(self.planeId, -1, rgbaColor=[1, 1, 1, 1])
if envID == 1:
# Add stairs with platform and bridge
self.stairsId = pyb.loadURDF("bauzil_stairs.urdf") # ,
"""basePosition=[-1.25, 3.5, -0.1],
baseOrientation=pyb.getQuaternionFromEuler([0.0, 0.0, 3.1415]))"""
pyb.changeDynamics(self.stairsId, -1, lateralFriction=1.0)
# Create the red steps to act as small perturbations
mesh_scale = [1.0, 0.1, 0.02]
visualShapeId = pyb.createVisualShape(shapeType=pyb.GEOM_MESH,
fileName="cube.obj",
halfExtents=[0.5, 0.5, 0.1],
rgbaColor=[1.0, 0.0, 0.0, 1.0],
specularColor=[0.4, .4, 0],
visualFramePosition=[0.0, 0.0, 0.0],
meshScale=mesh_scale)
collisionShapeId = pyb.createCollisionShape(shapeType=pyb.GEOM_MESH,
fileName="cube.obj",
collisionFramePosition=[0.0, 0.0, 0.0],
meshScale=mesh_scale)
for i in range(4):
tmpId = pyb.createMultiBody(baseMass=0.0,
baseInertialFramePosition=[0, 0, 0],
baseCollisionShapeIndex=collisionShapeId,
baseVisualShapeIndex=visualShapeId,
basePosition=[0.0, 0.5+0.2*i, 0.01],
useMaximalCoordinates=True)
pyb.changeDynamics(tmpId, -1, lateralFriction=1.0)
tmpId = pyb.createMultiBody(baseMass=0.0,
baseInertialFramePosition=[0, 0, 0],
baseCollisionShapeIndex=collisionShapeId,
baseVisualShapeIndex=visualShapeId,
basePosition=[0.5, 0.5+0.2*4, 0.01],
useMaximalCoordinates=True)
pyb.changeDynamics(tmpId, -1, lateralFriction=1.0)
tmpId = pyb.createMultiBody(baseMass=0.0,
baseInertialFramePosition=[0, 0, 0],
baseCollisionShapeIndex=collisionShapeId,
baseVisualShapeIndex=visualShapeId,
basePosition=[0.5, 0.5+0.2*5, 0.01],
useMaximalCoordinates=True)
pyb.changeDynamics(tmpId, -1, lateralFriction=1.0)
# Create the green steps to act as bigger perturbations
mesh_scale = [0.2, 0.1, 0.01]
visualShapeId = pyb.createVisualShape(shapeType=pyb.GEOM_MESH,
fileName="cube.obj",
halfExtents=[0.5, 0.5, 0.1],
rgbaColor=[0.0, 1.0, 0.0, 1.0],
specularColor=[0.4, .4, 0],
visualFramePosition=[0.0, 0.0, 0.0],
meshScale=mesh_scale)
collisionShapeId = pyb.createCollisionShape(shapeType=pyb.GEOM_MESH,
fileName="cube.obj",
collisionFramePosition=[0.0, 0.0, 0.0],
meshScale=mesh_scale)
for i in range(3):
tmpId = pyb.createMultiBody(baseMass=0.0,
baseInertialFramePosition=[0, 0, 0],
baseCollisionShapeIndex=collisionShapeId,
baseVisualShapeIndex=visualShapeId,
basePosition=[0.15 * (-1)**i, 0.9+0.2*i, 0.025],
useMaximalCoordinates=True)
pyb.changeDynamics(tmpId, -1, lateralFriction=1.0)
# Create sphere obstacles that will be thrown toward the quadruped
mesh_scale = [0.1, 0.1, 0.1]
visualShapeId = pyb.createVisualShape(shapeType=pyb.GEOM_MESH,
fileName="sphere_smooth.obj",
halfExtents=[0.5, 0.5, 0.1],
rgbaColor=[1.0, 0.0, 0.0, 1.0],
specularColor=[0.4, .4, 0],
visualFramePosition=[0.0, 0.0, 0.0],
meshScale=mesh_scale)
collisionShapeId = pyb.createCollisionShape(shapeType=pyb.GEOM_MESH,
fileName="sphere_smooth.obj",
collisionFramePosition=[0.0, 0.0, 0.0],
meshScale=mesh_scale)
self.sphereId1 = pyb.createMultiBody(baseMass=0.4,
baseInertialFramePosition=[0, 0, 0],
baseCollisionShapeIndex=collisionShapeId,
baseVisualShapeIndex=visualShapeId,
basePosition=[-0.6, 0.9, 0.1],
useMaximalCoordinates=True)
self.sphereId2 = pyb.createMultiBody(baseMass=0.4,
baseInertialFramePosition=[0, 0, 0],
baseCollisionShapeIndex=collisionShapeId,
baseVisualShapeIndex=visualShapeId,
basePosition=[0.6, 1.1, 0.1],
useMaximalCoordinates=True)
# Flag to launch the two spheres in the environment toward the robot
self.flag_sphere1 = True
self.flag_sphere2 = True
# Create blue spheres without collision box for debug purpose
mesh_scale = [0.015, 0.015, 0.015]
visualShapeId = pyb.createVisualShape(shapeType=pyb.GEOM_MESH,
fileName="sphere_smooth.obj",
halfExtents=[0.5, 0.5, 0.1],
rgbaColor=[0.0, 0.0, 1.0, 1.0],
specularColor=[0.4, .4, 0],
visualFramePosition=[0.0, 0.0, 0.0],
meshScale=mesh_scale)
self.ftps_Ids = np.zeros((4, 5), dtype=np.int)
for i in range(4):
for j in range(5):
self.ftps_Ids[i, j] = pyb.createMultiBody(baseMass=0.0,
baseInertialFramePosition=[0, 0, 0],
baseVisualShapeIndex=visualShapeId,
basePosition=[0.0, 0.0, -0.1],
useMaximalCoordinates=True)
# Create green spheres without collision box for debug purpose
visualShapeId = pyb.createVisualShape(shapeType=pyb.GEOM_MESH,
fileName="sphere_smooth.obj",
halfExtents=[0.5, 0.5, 0.1],
rgbaColor=[0.0, 1.0, 0.0, 1.0],
specularColor=[0.4, .4, 0],
visualFramePosition=[0.0, 0.0, 0.0],
meshScale=mesh_scale)
self.ftps_Ids_deb = [0] * 4
for i in range(4):
self.ftps_Ids_deb[i] = pyb.createMultiBody(baseMass=0.0,
baseInertialFramePosition=[0, 0, 0],
baseVisualShapeIndex=visualShapeId,
basePosition=[0.0, 0.0, -0.1],
useMaximalCoordinates=True)
# Create a red line for debug purpose
self.lineId_red = []
# Create a blue line for debug purpose
self.lineId_blue = []
"""cubeStartPos = [0.0, 0.45, 0.0]
cubeStartOrientation = pyb.getQuaternionFromEuler([0, 0, 0])
self.cubeId = pyb.loadURDF("cube_small.urdf",
cubeStartPos, cubeStartOrientation)
pyb.changeDynamics(self.cubeId, -1, mass=0.5)
print("Mass of cube:", pyb.getDynamicsInfo(self.cubeId, -1)[0])"""
# Set the gravity
pyb.setGravity(0, 0, -9.81)
# Load Quadruped robot
robotStartPos = [0, 0, 0.0]
robotStartOrientation = pyb.getQuaternionFromEuler([0.0, 0.0, 0.0]) # -np.pi/2
pyb.setAdditionalSearchPath("/opt/openrobots/share/example-robot-data/robots/solo_description/robots")
self.robotId = pyb.loadURDF("solo12.urdf", robotStartPos, robotStartOrientation)
# Disable default motor control for revolute joints
self.revoluteJointIndices = [0, 1, 2, 4, 5, 6, 8, 9, 10, 12, 13, 14]
pyb.setJointMotorControlArray(self.robotId, jointIndices=self.revoluteJointIndices,
controlMode=pyb.VELOCITY_CONTROL,
targetVelocities=[0.0 for m in self.revoluteJointIndices],
forces=[0.0 for m in self.revoluteJointIndices])
# Initialize the robot in a specific configuration
self.q_init = np.array([q_init]).transpose()
pyb.resetJointStatesMultiDof(self.robotId, self.revoluteJointIndices, self.q_init) # q0[7:])
# Enable torque control for revolute joints
jointTorques = [0.0 for m in self.revoluteJointIndices]
pyb.setJointMotorControlArray(self.robotId, self.revoluteJointIndices,
controlMode=pyb.TORQUE_CONTROL, forces=jointTorques)
# Get position of feet in world frame with base at (0, 0, 0)
feetLinksID = [3, 7, 11, 15]
linkStates = pyb.getLinkStates(self.robotId, feetLinksID)
# Get minimum height of feet (they are in the ground since base is at 0, 0, 0)
z_min = linkStates[0][4][2]
i_min = 0
i = 1
for link in linkStates[1:]:
if link[4][2] < z_min:
z_min = link[4][2]
i_min = i
i += 1
# Set base at (0, 0, -z_min) so that the lowest foot is at z = 0
pyb.resetBasePositionAndOrientation(self.robotId, [0.0, 0.0, -z_min], pin.Quaternion(pin.rpy.rpyToMatrix(p_roll, p_pitch, 0.0)).coeffs())
# Progressively raise the base to achieve proper contact (take into account radius of the foot)
while (pyb.getClosestPoints(self.robotId, self.planeId, distance=0.005,
linkIndexA=feetLinksID[i_min]))[0][8] < -0.001:
z_min -= 0.001
pyb.resetBasePositionAndOrientation(self.robotId, [0.0, 0.0, -z_min], pin.Quaternion(pin.rpy.rpyToMatrix(p_roll, p_pitch, 0.0)).coeffs())
# Fix the base in the world frame
# pyb.createConstraint(self.robotId, -1, -1, -1, pyb.JOINT_FIXED, [0, 0, 0], [0, 0, 0], [0, 0, robotStartPos[2]])
# Set time step for the simulation
pyb.setTimeStep(dt)
# Change camera position
pyb.resetDebugVisualizerCamera(cameraDistance=0.6, cameraYaw=45, cameraPitch=-39.9,
cameraTargetPosition=[0.0, 0.0, robotStartPos[2]-0.2])
def check_pyb_env(self, k, envID, velID, qmes12):
"""Check the state of the robot to trigger events and update camera
Args:
k (int): Number of inv dynamics iterations since the start of the simulation
envID (int): Identifier of the current environment to be able to handle different scenarios
velID (int): Identifier of the current velocity profile to be able to handle different scenarios
qmes12 (19x1 array): the position/orientation of the trunk and angular position of actuators
"""
# If spheres are loaded
if envID == 1:
# Check if the robot is in front of the first sphere to trigger it
if self.flag_sphere1 and (qmes12[1, 0] >= 0.9):
pyb.resetBaseVelocity(self.sphereId1, linearVelocity=[2.5, 0.0, 2.0])
self.flag_sphere1 = False
# Check if the robot is in front of the second sphere to trigger it
if self.flag_sphere2 and (qmes12[1, 0] >= 1.1):
pyb.resetBaseVelocity(self.sphereId2, linearVelocity=[-2.5, 0.0, 2.0])
self.flag_sphere2 = False
# Create the red steps to act as small perturbations
"""if k == 10:
pyb.setAdditionalSearchPath(pybullet_data.getDataPath())
mesh_scale = [2.0, 2.0, 0.3]
visualShapeId = pyb.createVisualShape(shapeType=pyb.GEOM_MESH,
fileName="cube.obj",
halfExtents=[0.5, 0.5, 0.1],
rgbaColor=[0.0, 0.0, 1.0, 1.0],
specularColor=[0.4, .4, 0],
visualFramePosition=[0.0, 0.0, 0.0],
meshScale=mesh_scale)
collisionShapeId = pyb.createCollisionShape(shapeType=pyb.GEOM_MESH,
fileName="cube.obj",
collisionFramePosition=[0.0, 0.0, 0.0],
meshScale=mesh_scale)
tmpId = pyb.createMultiBody(baseMass=0.0,
baseInertialFramePosition=[0, 0, 0],
baseCollisionShapeIndex=collisionShapeId,
baseVisualShapeIndex=visualShapeId,
basePosition=[0.0, 0.0, 0.15],
useMaximalCoordinates=True)
pyb.changeDynamics(tmpId, -1, lateralFriction=1.0)
pyb.resetBasePositionAndOrientation(self.robotId, [0, 0, 0.5], [0, 0, 0, 1])"""
if k == 10:
pyb.setAdditionalSearchPath(pybullet_data.getDataPath())
mesh_scale = [0.1, 0.1, 0.04]
visualShapeId = pyb.createVisualShape(shapeType=pyb.GEOM_MESH,
fileName="cube.obj",
halfExtents=[0.5, 0.5, 0.1],
rgbaColor=[0.0, 0.0, 1.0, 1.0],
specularColor=[0.4, .4, 0],
visualFramePosition=[0.0, 0.0, 0.0],
meshScale=mesh_scale)
collisionShapeId = pyb.createCollisionShape(shapeType=pyb.GEOM_MESH,
fileName="cube.obj",
collisionFramePosition=[0.0, 0.0, 0.0],
meshScale=mesh_scale)
tmpId = pyb.createMultiBody(baseMass=0.0,
baseInertialFramePosition=[0, 0, 0],
baseCollisionShapeIndex=collisionShapeId,
baseVisualShapeIndex=visualShapeId,
basePosition=[0.19, 0.15005, 0.02],
useMaximalCoordinates=True)
pyb.changeDynamics(tmpId, -1, lateralFriction=1.0)
pyb.resetBasePositionAndOrientation(self.robotId, [0, 0, 0.25], [0, 0, 0, 1])
# Apply perturbations by directly applying external forces on the robot trunk
if velID == 4:
self.apply_external_force(k, 4250, 500, np.array([0.0, 0.0, -3.0]), np.zeros((3,)))
self.apply_external_force(k, 5250, 500, np.array([0.0, +3.0, 0.0]), np.zeros((3,)))
"""if velID == 0:
self.apply_external_force(k, 1000, 1000, np.array([0.0, 0.0, -6.0]), np.zeros((3,)))
self.apply_external_force(k, 2000, 1000, np.array([0.0, +12.0, 0.0]), np.zeros((3,)))"""
# Update the PyBullet camera on the robot position to do as if it was attached to the robot
"""pyb.resetDebugVisualizerCamera(cameraDistance=0.75, cameraYaw=+50, cameraPitch=-35,
cameraTargetPosition=[qmes12[0, 0], qmes12[1, 0] + 0.0, 0.0])"""
# Get the orientation of the robot to change the orientation of the camera with the rotation of the robot
oMb_tmp = pin.SE3(pin.Quaternion(qmes12[3:7]), np.array([0.0, 0.0, 0.0]))
RPY = pin.rpy.matrixToRpy(oMb_tmp.rotation)
# Update the PyBullet camera on the robot position to do as if it was attached to the robot
pyb.resetDebugVisualizerCamera(cameraDistance=0.6, cameraYaw=(0.0*RPY[2]*(180/3.1415)+45), cameraPitch=-39.9,
cameraTargetPosition=[qmes12[0,0], qmes12[1,0] + 0.0, 0.0])
return 0
def retrieve_pyb_data(self):
"""Retrieve the position and orientation of the base in world frame as well as its linear and angular velocities
"""
# Retrieve data from the simulation
self.jointStates = pyb.getJointStates(self.robotId, self.revoluteJointIndices) # State of all joints
self.baseState = pyb.getBasePositionAndOrientation(self.robotId) # Position and orientation of the trunk
self.baseVel = pyb.getBaseVelocity(self.robotId) # Velocity of the trunk
# Joints configuration and velocity vector for free-flyer + 12 actuators
self.qmes12 = np.vstack((np.array([self.baseState[0]]).T, np.array([self.baseState[1]]).T,
np.array([[state[0] for state in self.jointStates]]).T))
self.vmes12 = np.vstack((np.array([self.baseVel[0]]).T, np.array([self.baseVel[1]]).T,
np.array([[state[1] for state in self.jointStates]]).T))
"""robotVirtualOrientation = pyb.getQuaternionFromEuler([0, 0, np.pi / 4])
self.qmes12[3:7, 0] = robotVirtualOrientation"""
# Add uncertainty to feedback from PyBullet to mimic imperfect measurements
"""tmp = np.array([pyb.getQuaternionFromEuler(pin.rpy.matrixToRpy(
pin.Quaternion(self.qmes12[3:7, 0:1]).toRotationMatrix())
+ np.random.normal(0, 0.03, (3,)))])
self.qmes12[3:7, 0] = tmp[0, :]
self.vmes12[0:6, 0] += np.random.normal(0, 0.01, (6,))"""
return 0
def apply_external_force(self, k, start, duration, F, loc):
"""Apply an external force/momentum to the robot
4-th order polynomial: zero force and force velocity at start and end
(bell-like force trajectory)
Args:
k (int): numero of the current iteration of the simulation
start (int): numero of the iteration for which the force should start to be applied
duration (int): number of iterations the force should last
F (3x array): components of the force in PyBullet world frame
loc (3x array): position on the link where the force is applied
"""
if ((k < start) or (k > (start+duration))):
return 0.0
"""if k == start:
print("Applying [", F[0], ", ", F[1], ", ", F[2], "]")"""
ev = k - start
t1 = duration
A4 = 16 / (t1**4)
A3 = - 2 * t1 * A4
A2 = t1**2 * A4
alpha = A2*ev**2 + A3*ev**3 + A4*ev**4
self.applied_force[:] = alpha * F
pyb.applyExternalForce(self.robotId, -1, alpha * F, loc, pyb.LINK_FRAME)
return 0.0
def get_to_default_position(self, qtarget):
"""Controler that tries to get the robot back to a default angular positions
of its 12 actuators using polynomials to generate trajectories and a PD controller
to make the actuators follow them
Args:
qtarget (12x1 array): the target position for the 12 actuators
"""
qmes = np.zeros((12, 1))
vmes = np.zeros((12, 1))
# Retrieve angular position and velocity of actuators
jointStates = pyb.getJointStates(self.robotId, self.revoluteJointIndices)
qmes[:, 0] = [state[0] for state in jointStates]
vmes[:, 0] = [state[1] for state in jointStates]
# Create trajectory
dt_traj = 0.0020
t1 = 4.0 # seconds
cpt = 0
# PD settings
P = 1.0 * 3.0
D = 0.05 * np.array([[1.0, 0.3, 0.3, 1.0, 0.3, 0.3,
1.0, 0.3, 0.3, 1.0, 0.3, 0.3]]).transpose()
while True or np.max(np.abs(qtarget - qmes)) > 0.1:
time_loop = time.time()
# Retrieve angular position and velocity of actuators
jointStates = pyb.getJointStates(self.robotId, self.revoluteJointIndices)
qmes[:, 0] = [state[0] for state in jointStates]
vmes[:, 0] = [state[1] for state in jointStates]
# Torque PD controller
if (cpt * dt_traj < t1):
ev = dt_traj * cpt
A3 = 2 * (qmes - qtarget) / t1**3
A2 = (-3/2) * t1 * A3
qdes = qmes + A2*(ev**2) + A3*(ev**3)
vdes = 2*A2*ev + 3*A3*(ev**2)
jointTorques = P * (qdes - qmes) + D * (vdes - vmes)
# Saturation to limit the maximal torque
t_max = 2.5
jointTorques[jointTorques > t_max] = t_max
jointTorques[jointTorques < -t_max] = -t_max
# Set control torque for all joints
pyb.setJointMotorControlArray(self.robotId, self.revoluteJointIndices,
controlMode=pyb.TORQUE_CONTROL, forces=jointTorques)
# Compute one step of simulation
pyb.stepSimulation()
# Increment loop counter
cpt += 1
while (time.time() - time_loop) < dt_traj:
pass
class Hardware():
"""Dummy class that simulates the Hardware class used to communicate with the real masterboard"""
def __init__(self):
self.is_timeout = False
self.roll = 0.0
self.pitch = 0.0
self.yaw = 0.0
def IsTimeout(self):
return self.is_timeout
def Stop(self):
return 0
def imu_data_attitude(self, i):
if i == 0:
return self.roll
elif i == 1:
return self.pitch
elif i == 2:
return self.yaw
class IMU():
"""Dummy class that simulates the IMU class used to communicate with the real masterboard"""
def __init__(self):
self.linear_acceleration = np.zeros((3, ))
self.accelerometer = np.zeros((3, ))
self.gyroscope = np.zeros((3, ))
self.attitude_euler = np.zeros((3, ))
self.attitude_quaternion = np.array([[0.0, 0.0, 0.0, 1.0]])
class Powerboard():
"""Dummy class that simulates the Powerboard class used to communicate with the real masterboard"""
def __init__(self):
self.current = 0.0
self.voltage = 0.0
self.energy = 0.0
class Joints():
"""Dummy class that simulates the Joints class used to communicate with the real masterboard"""
def __init__(self, parent_class):
self.parent = parent_class
self.positions = np.zeros((12, ))
self.velocities = np.zeros((12, ))
self.measured_torques = np.zeros((12, ))
def set_torques(self, torques):
"""Set desired joint torques
Args:
torques (12 x 0): desired articular feedforward torques
"""
# Save desired torques in a storage array
self.parent.tau_ff = torques.copy()
return
def set_position_gains(self, P):
"""Set desired P gains for articular low level control
Args:
P (12 x 0 array): desired position gains
"""
self.parent.P = P
def set_velocity_gains(self, D):
"""Set desired D gains for articular low level control
Args:
D (12 x 0 array): desired velocity gains
"""
self.parent.D = D
def set_desired_positions(self, q_des):
"""Set desired joint positions
Args:
q_des (12 x 0 array): desired articular positions
"""
self.parent.q_des = q_des
def set_desired_velocities(self, v_des):
"""Set desired joint velocities
Args:
v_des (12 x 0 array): desired articular velocities
"""
self.parent.v_des = v_des
class RobotInterface():
"""Dummy class that simulates the robot_interface class used to communicate with the real masterboard"""
def __init__(self):
pass
def PrintStats(self):
pass
class PyBulletSimulator():
"""Class that wraps a PyBullet simulation environment to seamlessly switch between the real robot or
simulation by having the same interface in both cases (calling the same functions/variables)
"""
def __init__(self):
self.cpt = 0
self.nb_motors = 12
self.jointTorques = np.zeros(self.nb_motors)
self.revoluteJointIndices = [0, 1, 2, 4, 5, 6, 8, 9, 10, 12, 13, 14]
self.is_timeout = False
self.powerboard = Powerboard()
self.imu = IMU()
self.joints = Joints(self)
self.robot_interface = RobotInterface()
# Measured data
self.o_baseVel = np.zeros((3, 1))
self.o_imuVel = np.zeros((3, 1))
self.prev_o_imuVel = np.zeros((3, 1))
# PD+ quantities
self.P = 0.0
self.D = 0.0
self.q_des = np.zeros(12)
self.v_des = np.zeros(12)
self.tau_ff = np.zeros(12)
# LPF parameters
self._alpha = 0.0
self.filterTorques = np.zeros(self.nb_motors)
def Init(self, calibrateEncoders=False, q_init=None, envID=0, use_flat_plane=True, enable_pyb_GUI=False, dt=0.002, alpha=0.0):
"""Initialize the PyBullet simultor with a given environment and a given state of the robot
Args:
calibrateEncoders (bool): dummy variable, not used for simulation but used for real robot
q_init (12 x 0 array): initial angular positions of the joints of the robot
envID (int): which environment should be loaded in the simulation
use_flat_plane (bool): to use either a flat ground or a rough ground
enable_pyb_GUI (bool): to display PyBullet GUI or not
dt (float): time step of the simulation
"""
# Initialisation of the PyBullet simulator
self.pyb_sim = pybullet_simulator(q_init, envID, use_flat_plane, enable_pyb_GUI, dt)
self.q_init = q_init
self.joints.positions[:] = q_init
self.dt = dt
self.time_loop = time.time()
self._alpha = alpha
return
def cross3(self, left, right):
"""Numpy is inefficient for this
Args:
left (3x0 array): left term of the cross product
right (3x0 array): right term of the cross product
"""
return np.array([[left[1] * right[2] - left[2] * right[1]],
[left[2] * right[0] - left[0] * right[2]],
[left[0] * right[1] - left[1] * right[0]]])
def parse_sensor_data(self):
"""Retrieve data about the robot from the simulation to mimic what the masterboard does
"""
# Position and velocity of actuators
jointStates = pyb.getJointStates(self.pyb_sim.robotId, self.revoluteJointIndices) # State of all joints
self.joints.positions[:] = np.array([state[0] for state in jointStates])
self.joints.velocities[:] = np.array([state[1] for state in jointStates])
# Measured torques
self.joints.measured_torques[:] = self.jointTorques[:].ravel()
# Position and orientation of the trunk (PyBullet world frame)
self.baseState = pyb.getBasePositionAndOrientation(self.pyb_sim.robotId)
self.dummyHeight = np.array(self.baseState[0])
self.dummyHeight[2] = 0.20
self.dummyPos = np.array(self.baseState[0])
# Linear and angular velocity of the trunk (PyBullet world frame)
self.baseVel = pyb.getBaseVelocity(self.pyb_sim.robotId)
# print("baseVel: ", np.array([self.baseVel[0]]))
# Orientation of the base (quaternion)
self.imu.attitude_quaternion[0,:] = np.array(self.baseState[1])
self.rot_oMb = pin.Quaternion(self.imu.attitude_quaternion.T).toRotationMatrix()
self.imu.attitude_euler[:] = pin.rpy.matrixToRpy(self.rot_oMb)
self.oMb = pin.SE3(self.rot_oMb, np.array([self.dummyHeight]).transpose())
# Angular velocities of the base
self.imu.gyroscope[:] = (self.oMb.rotation.transpose() @ np.array([self.baseVel[1]]).transpose()).ravel()
# Linear Acceleration of the base
self.o_baseVel = np.array([self.baseVel[0]]).transpose()
self.b_baseVel = (self.oMb.rotation.transpose() @ self.o_baseVel).ravel()
self.o_imuVel = self.o_baseVel + self.oMb.rotation @ self.cross3(np.array([0.1163, 0.0, 0.02]), self.imu.gyroscope[:])
self.imu.linear_acceleration[:] = (self.oMb.rotation.transpose() @ (self.o_imuVel - self.prev_o_imuVel)).ravel() / self.dt
self.prev_o_imuVel[:, 0:1] = self.o_imuVel
self.imu.accelerometer[:] = self.imu.linear_acceleration + \
(self.oMb.rotation.transpose() @ np.array([[0.0], [0.0], [-9.81]])).ravel()
return
def send_command_and_wait_end_of_cycle(self, WaitEndOfCycle=True):
"""Send control commands to the robot
Args:
WaitEndOfCycle (bool): wait to have simulation time = real time
"""
# Position and velocity of actuators
jointStates = pyb.getJointStates(self.pyb_sim.robotId, self.revoluteJointIndices) # State of all joints
self.joints.positions[:] = np.array([state[0] for state in jointStates])
self.joints.velocities[:] = np.array([state[1] for state in jointStates])
# Compute PD torques
tau_pd = self.P * (self.q_des - self.joints.positions) + self.D * (self.v_des - self.joints.velocities)
# Save desired torques in a storage array
self.jointTorques = (tau_pd + self.tau_ff).clip(-3,3)
# Low pass filter the torques
self.filterTorques = self._alpha * self.filterTorques + (1 - self._alpha) * self.jointTorques
# Set control torque for all joints
pyb.setJointMotorControlArray(self.pyb_sim.robotId, self.pyb_sim.revoluteJointIndices,
controlMode=pyb.TORQUE_CONTROL, forces=self.filterTorques)
# Compute one step of simulation
pyb.stepSimulation()
pyb.resetDebugVisualizerCamera(cameraDistance=0.6,
cameraYaw=(0.0*self.imu.attitude_euler[2]*(180/3.1415)+45),
cameraPitch=-39.9,
cameraTargetPosition=[self.baseState[0][0], self.baseState[0][1] + 0.0, 0.0])
# Wait to have simulation time = real time
if WaitEndOfCycle:
while (time.time() - self.time_loop) < self.dt:
pass
self.cpt += 1
self.time_loop = time.time()
return
def Print(self):
"""Print simulation parameters in the console"""
np.set_printoptions(precision=2)
# print(chr(27) + "[2J")
print("#######")
print("q_mes = ", self.joints.positions)
print("v_mes = ", self.joints.velocities)
print("torques = ", self.joints.measured_torques)
print("orientation = ", self.imu.attitude_quaternion)
print("lin acc = ", self.imu.linear_acceleration)
print("ang vel = ", self.imu.gyroscope)
sys.stdout.flush()
def Stop(self):
"""Stop the simulation environment"""
# Disconnect the PyBullet server (also close the GUI)
pyb.disconnect()