warp-lang 1.0.1__py3-none-win_amd64.whl → 1.1.0__py3-none-win_amd64.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Potentially problematic release.
This version of warp-lang might be problematic. Click here for more details.
- warp/__init__.py +108 -97
- warp/__init__.pyi +1 -1
- warp/bin/warp-clang.dll +0 -0
- warp/bin/warp.dll +0 -0
- warp/build.py +115 -113
- warp/build_dll.py +383 -375
- warp/builtins.py +3425 -3354
- warp/codegen.py +2878 -2792
- warp/config.py +40 -36
- warp/constants.py +45 -45
- warp/context.py +5194 -5102
- warp/dlpack.py +442 -442
- warp/examples/__init__.py +16 -16
- warp/examples/assets/bear.usd +0 -0
- warp/examples/assets/bunny.usd +0 -0
- warp/examples/assets/cartpole.urdf +110 -110
- warp/examples/assets/crazyflie.usd +0 -0
- warp/examples/assets/cube.usd +0 -0
- warp/examples/assets/nv_ant.xml +92 -92
- warp/examples/assets/nv_humanoid.xml +183 -183
- warp/examples/assets/quadruped.urdf +267 -267
- warp/examples/assets/rocks.nvdb +0 -0
- warp/examples/assets/rocks.usd +0 -0
- warp/examples/assets/sphere.usd +0 -0
- warp/examples/benchmarks/benchmark_api.py +383 -383
- warp/examples/benchmarks/benchmark_cloth.py +278 -279
- warp/examples/benchmarks/benchmark_cloth_cupy.py +88 -88
- warp/examples/benchmarks/benchmark_cloth_jax.py +97 -100
- warp/examples/benchmarks/benchmark_cloth_numba.py +146 -142
- warp/examples/benchmarks/benchmark_cloth_numpy.py +77 -77
- warp/examples/benchmarks/benchmark_cloth_pytorch.py +86 -86
- warp/examples/benchmarks/benchmark_cloth_taichi.py +112 -112
- warp/examples/benchmarks/benchmark_cloth_warp.py +146 -146
- warp/examples/benchmarks/benchmark_launches.py +295 -295
- warp/examples/browse.py +29 -28
- warp/examples/core/example_dem.py +234 -221
- warp/examples/core/example_fluid.py +293 -267
- warp/examples/core/example_graph_capture.py +144 -129
- warp/examples/core/example_marching_cubes.py +188 -176
- warp/examples/core/example_mesh.py +174 -154
- warp/examples/core/example_mesh_intersect.py +205 -193
- warp/examples/core/example_nvdb.py +176 -169
- warp/examples/core/example_raycast.py +105 -89
- warp/examples/core/example_raymarch.py +199 -178
- warp/examples/core/example_render_opengl.py +185 -141
- warp/examples/core/example_sph.py +405 -389
- warp/examples/core/example_torch.py +222 -181
- warp/examples/core/example_wave.py +263 -249
- warp/examples/fem/bsr_utils.py +378 -380
- warp/examples/fem/example_apic_fluid.py +407 -391
- warp/examples/fem/example_convection_diffusion.py +182 -168
- warp/examples/fem/example_convection_diffusion_dg.py +219 -209
- warp/examples/fem/example_convection_diffusion_dg0.py +204 -194
- warp/examples/fem/example_deformed_geometry.py +177 -159
- warp/examples/fem/example_diffusion.py +201 -173
- warp/examples/fem/example_diffusion_3d.py +177 -152
- warp/examples/fem/example_diffusion_mgpu.py +221 -214
- warp/examples/fem/example_mixed_elasticity.py +244 -222
- warp/examples/fem/example_navier_stokes.py +259 -243
- warp/examples/fem/example_stokes.py +220 -192
- warp/examples/fem/example_stokes_transfer.py +265 -249
- warp/examples/fem/mesh_utils.py +133 -109
- warp/examples/fem/plot_utils.py +292 -287
- warp/examples/optim/example_bounce.py +260 -248
- warp/examples/optim/example_cloth_throw.py +222 -210
- warp/examples/optim/example_diffray.py +566 -535
- warp/examples/optim/example_drone.py +864 -835
- warp/examples/optim/example_inverse_kinematics.py +176 -169
- warp/examples/optim/example_inverse_kinematics_torch.py +185 -170
- warp/examples/optim/example_spring_cage.py +239 -234
- warp/examples/optim/example_trajectory.py +223 -201
- warp/examples/optim/example_walker.py +306 -292
- warp/examples/sim/example_cartpole.py +139 -128
- warp/examples/sim/example_cloth.py +196 -184
- warp/examples/sim/example_granular.py +124 -113
- warp/examples/sim/example_granular_collision_sdf.py +197 -185
- warp/examples/sim/example_jacobian_ik.py +236 -213
- warp/examples/sim/example_particle_chain.py +118 -106
- warp/examples/sim/example_quadruped.py +193 -179
- warp/examples/sim/example_rigid_chain.py +197 -189
- warp/examples/sim/example_rigid_contact.py +189 -176
- warp/examples/sim/example_rigid_force.py +127 -126
- warp/examples/sim/example_rigid_gyroscopic.py +109 -97
- warp/examples/sim/example_rigid_soft_contact.py +134 -124
- warp/examples/sim/example_soft_body.py +190 -178
- warp/fabric.py +337 -335
- warp/fem/__init__.py +60 -27
- warp/fem/cache.py +401 -388
- warp/fem/dirichlet.py +178 -179
- warp/fem/domain.py +262 -263
- warp/fem/field/__init__.py +100 -101
- warp/fem/field/field.py +148 -149
- warp/fem/field/nodal_field.py +298 -299
- warp/fem/field/restriction.py +22 -21
- warp/fem/field/test.py +180 -181
- warp/fem/field/trial.py +183 -183
- warp/fem/geometry/__init__.py +15 -19
- warp/fem/geometry/closest_point.py +69 -70
- warp/fem/geometry/deformed_geometry.py +270 -271
- warp/fem/geometry/element.py +744 -744
- warp/fem/geometry/geometry.py +184 -186
- warp/fem/geometry/grid_2d.py +380 -373
- warp/fem/geometry/grid_3d.py +441 -435
- warp/fem/geometry/hexmesh.py +953 -953
- warp/fem/geometry/partition.py +374 -376
- warp/fem/geometry/quadmesh_2d.py +532 -532
- warp/fem/geometry/tetmesh.py +840 -840
- warp/fem/geometry/trimesh_2d.py +577 -577
- warp/fem/integrate.py +1630 -1615
- warp/fem/operator.py +190 -191
- warp/fem/polynomial.py +214 -213
- warp/fem/quadrature/__init__.py +2 -2
- warp/fem/quadrature/pic_quadrature.py +243 -245
- warp/fem/quadrature/quadrature.py +295 -294
- warp/fem/space/__init__.py +294 -292
- warp/fem/space/basis_space.py +488 -489
- warp/fem/space/collocated_function_space.py +100 -105
- warp/fem/space/dof_mapper.py +236 -236
- warp/fem/space/function_space.py +148 -145
- warp/fem/space/grid_2d_function_space.py +267 -267
- warp/fem/space/grid_3d_function_space.py +305 -306
- warp/fem/space/hexmesh_function_space.py +350 -352
- warp/fem/space/partition.py +350 -350
- warp/fem/space/quadmesh_2d_function_space.py +368 -369
- warp/fem/space/restriction.py +158 -160
- warp/fem/space/shape/__init__.py +13 -15
- warp/fem/space/shape/cube_shape_function.py +738 -738
- warp/fem/space/shape/shape_function.py +102 -103
- warp/fem/space/shape/square_shape_function.py +611 -611
- warp/fem/space/shape/tet_shape_function.py +565 -567
- warp/fem/space/shape/triangle_shape_function.py +429 -429
- warp/fem/space/tetmesh_function_space.py +294 -292
- warp/fem/space/topology.py +297 -295
- warp/fem/space/trimesh_2d_function_space.py +223 -221
- warp/fem/types.py +77 -77
- warp/fem/utils.py +495 -495
- warp/jax.py +166 -141
- warp/jax_experimental.py +341 -339
- warp/native/array.h +1072 -1025
- warp/native/builtin.h +1560 -1560
- warp/native/bvh.cpp +398 -398
- warp/native/bvh.cu +525 -525
- warp/native/bvh.h +429 -429
- warp/native/clang/clang.cpp +495 -464
- warp/native/crt.cpp +31 -31
- warp/native/crt.h +334 -334
- warp/native/cuda_crt.h +1049 -1049
- warp/native/cuda_util.cpp +549 -540
- warp/native/cuda_util.h +288 -203
- warp/native/cutlass_gemm.cpp +34 -34
- warp/native/cutlass_gemm.cu +372 -372
- warp/native/error.cpp +66 -66
- warp/native/error.h +27 -27
- warp/native/fabric.h +228 -228
- warp/native/hashgrid.cpp +301 -278
- warp/native/hashgrid.cu +78 -77
- warp/native/hashgrid.h +227 -227
- warp/native/initializer_array.h +32 -32
- warp/native/intersect.h +1204 -1204
- warp/native/intersect_adj.h +365 -365
- warp/native/intersect_tri.h +322 -322
- warp/native/marching.cpp +2 -2
- warp/native/marching.cu +497 -497
- warp/native/marching.h +2 -2
- warp/native/mat.h +1498 -1498
- warp/native/matnn.h +333 -333
- warp/native/mesh.cpp +203 -203
- warp/native/mesh.cu +293 -293
- warp/native/mesh.h +1887 -1887
- warp/native/nanovdb/NanoVDB.h +4782 -4782
- warp/native/nanovdb/PNanoVDB.h +2553 -2553
- warp/native/nanovdb/PNanoVDBWrite.h +294 -294
- warp/native/noise.h +850 -850
- warp/native/quat.h +1084 -1084
- warp/native/rand.h +299 -299
- warp/native/range.h +108 -108
- warp/native/reduce.cpp +156 -156
- warp/native/reduce.cu +348 -348
- warp/native/runlength_encode.cpp +61 -61
- warp/native/runlength_encode.cu +46 -46
- warp/native/scan.cpp +30 -30
- warp/native/scan.cu +36 -36
- warp/native/scan.h +7 -7
- warp/native/solid_angle.h +442 -442
- warp/native/sort.cpp +94 -94
- warp/native/sort.cu +97 -97
- warp/native/sort.h +14 -14
- warp/native/sparse.cpp +337 -337
- warp/native/sparse.cu +544 -544
- warp/native/spatial.h +630 -630
- warp/native/svd.h +562 -562
- warp/native/temp_buffer.h +30 -30
- warp/native/vec.h +1132 -1132
- warp/native/volume.cpp +297 -297
- warp/native/volume.cu +32 -32
- warp/native/volume.h +538 -538
- warp/native/volume_builder.cu +425 -425
- warp/native/volume_builder.h +19 -19
- warp/native/warp.cpp +1057 -1052
- warp/native/warp.cu +2943 -2828
- warp/native/warp.h +313 -305
- warp/optim/__init__.py +9 -9
- warp/optim/adam.py +120 -120
- warp/optim/linear.py +1104 -939
- warp/optim/sgd.py +104 -92
- warp/render/__init__.py +10 -10
- warp/render/render_opengl.py +3217 -3204
- warp/render/render_usd.py +768 -749
- warp/render/utils.py +152 -150
- warp/sim/__init__.py +52 -59
- warp/sim/articulation.py +685 -685
- warp/sim/collide.py +1594 -1590
- warp/sim/import_mjcf.py +489 -481
- warp/sim/import_snu.py +220 -221
- warp/sim/import_urdf.py +536 -516
- warp/sim/import_usd.py +887 -881
- warp/sim/inertia.py +316 -317
- warp/sim/integrator.py +234 -233
- warp/sim/integrator_euler.py +1956 -1956
- warp/sim/integrator_featherstone.py +1910 -1991
- warp/sim/integrator_xpbd.py +3294 -3312
- warp/sim/model.py +4473 -4314
- warp/sim/particles.py +113 -112
- warp/sim/render.py +417 -403
- warp/sim/utils.py +413 -410
- warp/sparse.py +1227 -1227
- warp/stubs.py +2109 -2469
- warp/tape.py +1162 -225
- warp/tests/__init__.py +1 -1
- warp/tests/__main__.py +4 -4
- warp/tests/assets/torus.usda +105 -105
- warp/tests/aux_test_class_kernel.py +26 -26
- warp/tests/aux_test_compile_consts_dummy.py +10 -10
- warp/tests/aux_test_conditional_unequal_types_kernels.py +21 -21
- warp/tests/aux_test_dependent.py +22 -22
- warp/tests/aux_test_grad_customs.py +23 -23
- warp/tests/aux_test_reference.py +11 -11
- warp/tests/aux_test_reference_reference.py +10 -10
- warp/tests/aux_test_square.py +17 -17
- warp/tests/aux_test_unresolved_func.py +14 -14
- warp/tests/aux_test_unresolved_symbol.py +14 -14
- warp/tests/disabled_kinematics.py +239 -239
- warp/tests/run_coverage_serial.py +31 -31
- warp/tests/test_adam.py +157 -157
- warp/tests/test_arithmetic.py +1124 -1124
- warp/tests/test_array.py +2417 -2326
- warp/tests/test_array_reduce.py +150 -150
- warp/tests/test_async.py +668 -656
- warp/tests/test_atomic.py +141 -141
- warp/tests/test_bool.py +204 -149
- warp/tests/test_builtins_resolution.py +1292 -1292
- warp/tests/test_bvh.py +164 -171
- warp/tests/test_closest_point_edge_edge.py +228 -228
- warp/tests/test_codegen.py +566 -553
- warp/tests/test_compile_consts.py +97 -101
- warp/tests/test_conditional.py +246 -246
- warp/tests/test_copy.py +232 -215
- warp/tests/test_ctypes.py +632 -632
- warp/tests/test_dense.py +67 -67
- warp/tests/test_devices.py +91 -98
- warp/tests/test_dlpack.py +530 -529
- warp/tests/test_examples.py +400 -378
- warp/tests/test_fabricarray.py +955 -955
- warp/tests/test_fast_math.py +62 -54
- warp/tests/test_fem.py +1277 -1278
- warp/tests/test_fp16.py +130 -130
- warp/tests/test_func.py +338 -337
- warp/tests/test_generics.py +571 -571
- warp/tests/test_grad.py +746 -640
- warp/tests/test_grad_customs.py +333 -336
- warp/tests/test_hash_grid.py +210 -164
- warp/tests/test_import.py +39 -39
- warp/tests/test_indexedarray.py +1134 -1134
- warp/tests/test_intersect.py +67 -67
- warp/tests/test_jax.py +307 -307
- warp/tests/test_large.py +167 -164
- warp/tests/test_launch.py +354 -354
- warp/tests/test_lerp.py +261 -261
- warp/tests/test_linear_solvers.py +191 -171
- warp/tests/test_lvalue.py +421 -493
- warp/tests/test_marching_cubes.py +65 -65
- warp/tests/test_mat.py +1801 -1827
- warp/tests/test_mat_lite.py +115 -115
- warp/tests/test_mat_scalar_ops.py +2907 -2889
- warp/tests/test_math.py +126 -193
- warp/tests/test_matmul.py +500 -499
- warp/tests/test_matmul_lite.py +410 -410
- warp/tests/test_mempool.py +188 -190
- warp/tests/test_mesh.py +284 -324
- warp/tests/test_mesh_query_aabb.py +228 -241
- warp/tests/test_mesh_query_point.py +692 -702
- warp/tests/test_mesh_query_ray.py +292 -303
- warp/tests/test_mlp.py +276 -276
- warp/tests/test_model.py +110 -110
- warp/tests/test_modules_lite.py +39 -39
- warp/tests/test_multigpu.py +163 -163
- warp/tests/test_noise.py +248 -248
- warp/tests/test_operators.py +250 -250
- warp/tests/test_options.py +123 -125
- warp/tests/test_peer.py +133 -137
- warp/tests/test_pinned.py +78 -78
- warp/tests/test_print.py +54 -54
- warp/tests/test_quat.py +2086 -2086
- warp/tests/test_rand.py +288 -288
- warp/tests/test_reload.py +217 -217
- warp/tests/test_rounding.py +179 -179
- warp/tests/test_runlength_encode.py +190 -190
- warp/tests/test_sim_grad.py +243 -0
- warp/tests/test_sim_kinematics.py +91 -97
- warp/tests/test_smoothstep.py +168 -168
- warp/tests/test_snippet.py +305 -266
- warp/tests/test_sparse.py +468 -460
- warp/tests/test_spatial.py +2148 -2148
- warp/tests/test_streams.py +486 -473
- warp/tests/test_struct.py +710 -675
- warp/tests/test_tape.py +173 -148
- warp/tests/test_torch.py +743 -743
- warp/tests/test_transient_module.py +87 -87
- warp/tests/test_types.py +556 -659
- warp/tests/test_utils.py +490 -499
- warp/tests/test_vec.py +1264 -1268
- warp/tests/test_vec_lite.py +73 -73
- warp/tests/test_vec_scalar_ops.py +2099 -2099
- warp/tests/test_verify_fp.py +94 -94
- warp/tests/test_volume.py +737 -736
- warp/tests/test_volume_write.py +255 -265
- warp/tests/unittest_serial.py +37 -37
- warp/tests/unittest_suites.py +363 -359
- warp/tests/unittest_utils.py +603 -578
- warp/tests/unused_test_misc.py +71 -71
- warp/tests/walkthrough_debug.py +85 -85
- warp/thirdparty/appdirs.py +598 -598
- warp/thirdparty/dlpack.py +143 -143
- warp/thirdparty/unittest_parallel.py +566 -561
- warp/torch.py +321 -295
- warp/types.py +4504 -4450
- warp/utils.py +1008 -821
- {warp_lang-1.0.1.dist-info → warp_lang-1.1.0.dist-info}/LICENSE.md +126 -126
- {warp_lang-1.0.1.dist-info → warp_lang-1.1.0.dist-info}/METADATA +338 -400
- warp_lang-1.1.0.dist-info/RECORD +352 -0
- warp/examples/assets/cube.usda +0 -42
- warp/examples/assets/sphere.usda +0 -56
- warp/examples/assets/torus.usda +0 -105
- warp_lang-1.0.1.dist-info/RECORD +0 -352
- {warp_lang-1.0.1.dist-info → warp_lang-1.1.0.dist-info}/WHEEL +0 -0
- {warp_lang-1.0.1.dist-info → warp_lang-1.1.0.dist-info}/top_level.txt +0 -0
warp/sim/import_mjcf.py
CHANGED
|
@@ -1,481 +1,489 @@
|
|
|
1
|
-
# Copyright (c) 2022 NVIDIA CORPORATION. All rights reserved.
|
|
2
|
-
# NVIDIA CORPORATION and its licensors retain all intellectual property
|
|
3
|
-
# and proprietary rights in and to this software, related documentation
|
|
4
|
-
# and any modifications thereto. Any use, reproduction, disclosure or
|
|
5
|
-
# distribution of this software and related documentation without an express
|
|
6
|
-
# license agreement from NVIDIA CORPORATION is strictly prohibited.
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
import math
|
|
10
|
-
import os
|
|
11
|
-
import re
|
|
12
|
-
import xml.etree.ElementTree as ET
|
|
13
|
-
|
|
14
|
-
import numpy as np
|
|
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
|
-
if
|
|
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
|
-
return wp.
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
return wp.
|
|
193
|
-
if "
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
return wp.
|
|
200
|
-
if "
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
yaxis = wp.normalize(wp.cross(zaxis, xaxis))
|
|
205
|
-
rot_matrix = np.array([xaxis, yaxis, zaxis]).T
|
|
206
|
-
return wp.quat_from_matrix(rot_matrix)
|
|
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
|
-
if
|
|
236
|
-
|
|
237
|
-
else:
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
if
|
|
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
|
-
if
|
|
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
|
-
if
|
|
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
|
-
|
|
1
|
+
# Copyright (c) 2022 NVIDIA CORPORATION. All rights reserved.
|
|
2
|
+
# NVIDIA CORPORATION and its licensors retain all intellectual property
|
|
3
|
+
# and proprietary rights in and to this software, related documentation
|
|
4
|
+
# and any modifications thereto. Any use, reproduction, disclosure or
|
|
5
|
+
# distribution of this software and related documentation without an express
|
|
6
|
+
# license agreement from NVIDIA CORPORATION is strictly prohibited.
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
import math
|
|
10
|
+
import os
|
|
11
|
+
import re
|
|
12
|
+
import xml.etree.ElementTree as ET
|
|
13
|
+
|
|
14
|
+
import numpy as np
|
|
15
|
+
|
|
16
|
+
import warp as wp
|
|
17
|
+
|
|
18
|
+
|
|
19
|
+
def parse_mjcf(
|
|
20
|
+
mjcf_filename,
|
|
21
|
+
builder,
|
|
22
|
+
xform=None,
|
|
23
|
+
density=1000.0,
|
|
24
|
+
stiffness=0.0,
|
|
25
|
+
damping=0.0,
|
|
26
|
+
contact_ke=1000.0,
|
|
27
|
+
contact_kd=100.0,
|
|
28
|
+
contact_kf=100.0,
|
|
29
|
+
contact_ka=0.0,
|
|
30
|
+
contact_mu=0.5,
|
|
31
|
+
contact_restitution=0.5,
|
|
32
|
+
contact_thickness=0.0,
|
|
33
|
+
limit_ke=100.0,
|
|
34
|
+
limit_kd=10.0,
|
|
35
|
+
scale=1.0,
|
|
36
|
+
armature=0.0,
|
|
37
|
+
armature_scale=1.0,
|
|
38
|
+
parse_meshes=True,
|
|
39
|
+
enable_self_collisions=False,
|
|
40
|
+
up_axis="Z",
|
|
41
|
+
ignore_classes=None,
|
|
42
|
+
collapse_fixed_joints=False,
|
|
43
|
+
):
|
|
44
|
+
"""
|
|
45
|
+
Parses MuJoCo XML (MJCF) file and adds the bodies and joints to the given ModelBuilder.
|
|
46
|
+
|
|
47
|
+
Args:
|
|
48
|
+
mjcf_filename (str): The filename of the MuJoCo file to parse.
|
|
49
|
+
builder (ModelBuilder): The :class:`ModelBuilder` to add the bodies and joints to.
|
|
50
|
+
xform (:ref:`transform <transform>`): The transform to apply to the imported mechanism.
|
|
51
|
+
density (float): The density of the shapes in kg/m^3 which will be used to calculate the body mass and inertia.
|
|
52
|
+
stiffness (float): The stiffness of the joints.
|
|
53
|
+
damping (float): The damping of the joints.
|
|
54
|
+
contact_ke (float): The stiffness of the shape contacts.
|
|
55
|
+
contact_kd (float): The damping of the shape contacts.
|
|
56
|
+
contact_kf (float): The friction stiffness of the shape contacts.
|
|
57
|
+
contact_ka (float): The adhesion distance of the shape contacts.
|
|
58
|
+
contact_mu (float): The friction coefficient of the shape contacts.
|
|
59
|
+
contact_restitution (float): The restitution coefficient of the shape contacts.
|
|
60
|
+
contact_thickness (float): The thickness to add to the shape geometry.
|
|
61
|
+
limit_ke (float): The stiffness of the joint limits.
|
|
62
|
+
limit_kd (float): The damping of the joint limits.
|
|
63
|
+
scale (float): The scaling factor to apply to the imported mechanism.
|
|
64
|
+
armature (float): Default joint armature to use if `armature` has not been defined for a joint in the MJCF.
|
|
65
|
+
armature_scale (float): Scaling factor to apply to the MJCF-defined joint armature values.
|
|
66
|
+
parse_meshes (bool): Whether geometries of type `"mesh"` should be parsed. If False, geometries of type `"mesh"` are ignored.
|
|
67
|
+
enable_self_collisions (bool): If True, self-collisions are enabled.
|
|
68
|
+
up_axis (str): The up axis of the mechanism. Can be either `"X"`, `"Y"` or `"Z"`. The default is `"Z"`.
|
|
69
|
+
ignore_classes (List[str]): A list of regular expressions. Bodies and joints with a class matching one of the regular expressions will be ignored.
|
|
70
|
+
collapse_fixed_joints (bool): If True, fixed joints are removed and the respective bodies are merged.
|
|
71
|
+
|
|
72
|
+
Note:
|
|
73
|
+
The inertia and masses of the bodies are calculated from the shape geometry and the given density. The values defined in the MJCF are not respected at the moment.
|
|
74
|
+
|
|
75
|
+
The handling of advanced features, such as MJCF classes, is still experimental.
|
|
76
|
+
"""
|
|
77
|
+
if xform is None:
|
|
78
|
+
xform = wp.transform()
|
|
79
|
+
|
|
80
|
+
if ignore_classes is None:
|
|
81
|
+
ignore_classes = []
|
|
82
|
+
|
|
83
|
+
mjcf_dirname = os.path.dirname(mjcf_filename)
|
|
84
|
+
file = ET.parse(mjcf_filename)
|
|
85
|
+
root = file.getroot()
|
|
86
|
+
|
|
87
|
+
contact_vars = {
|
|
88
|
+
"ke": contact_ke,
|
|
89
|
+
"kd": contact_kd,
|
|
90
|
+
"kf": contact_kf,
|
|
91
|
+
"ka": contact_ka,
|
|
92
|
+
"mu": contact_mu,
|
|
93
|
+
"restitution": contact_restitution,
|
|
94
|
+
"thickness": contact_thickness,
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
use_degrees = True # angles are in degrees by default
|
|
98
|
+
euler_seq = [1, 2, 3] # XYZ by default
|
|
99
|
+
|
|
100
|
+
compiler = root.find("compiler")
|
|
101
|
+
if compiler is not None:
|
|
102
|
+
use_degrees = compiler.attrib.get("angle", "degree").lower() == "degree"
|
|
103
|
+
euler_seq = ["xyz".index(c) + 1 for c in compiler.attrib.get("eulerseq", "xyz").lower()]
|
|
104
|
+
mesh_dir = compiler.attrib.get("meshdir", ".")
|
|
105
|
+
|
|
106
|
+
mesh_assets = {}
|
|
107
|
+
for asset in root.findall("asset"):
|
|
108
|
+
for mesh in asset.findall("mesh"):
|
|
109
|
+
if "file" in mesh.attrib:
|
|
110
|
+
fname = os.path.join(mesh_dir, mesh.attrib["file"])
|
|
111
|
+
# handle stl relative paths
|
|
112
|
+
if not os.path.isabs(fname):
|
|
113
|
+
fname = os.path.abspath(os.path.join(mjcf_dirname, fname))
|
|
114
|
+
if "name" in mesh.attrib:
|
|
115
|
+
mesh_assets[mesh.attrib["name"]] = fname
|
|
116
|
+
else:
|
|
117
|
+
name = ".".join(os.path.basename(fname).split(".")[:-1])
|
|
118
|
+
mesh_assets[name] = fname
|
|
119
|
+
|
|
120
|
+
class_parent = {}
|
|
121
|
+
class_children = {}
|
|
122
|
+
class_defaults = {"__all__": {}}
|
|
123
|
+
|
|
124
|
+
def get_class(element):
|
|
125
|
+
return element.get("class", "__all__")
|
|
126
|
+
|
|
127
|
+
def parse_default(node, parent):
|
|
128
|
+
nonlocal class_parent
|
|
129
|
+
nonlocal class_children
|
|
130
|
+
nonlocal class_defaults
|
|
131
|
+
class_name = "__all__"
|
|
132
|
+
if "class" in node.attrib:
|
|
133
|
+
class_name = node.attrib["class"]
|
|
134
|
+
class_parent[class_name] = parent
|
|
135
|
+
parent = parent or "__all__"
|
|
136
|
+
if parent not in class_children:
|
|
137
|
+
class_children[parent] = []
|
|
138
|
+
class_children[parent].append(class_name)
|
|
139
|
+
|
|
140
|
+
if class_name not in class_defaults:
|
|
141
|
+
class_defaults[class_name] = {}
|
|
142
|
+
for child in node:
|
|
143
|
+
if child.tag == "default":
|
|
144
|
+
parse_default(child, node.get("class"))
|
|
145
|
+
else:
|
|
146
|
+
class_defaults[class_name][child.tag] = child.attrib
|
|
147
|
+
|
|
148
|
+
for default in root.findall("default"):
|
|
149
|
+
parse_default(default, None)
|
|
150
|
+
|
|
151
|
+
def merge_attrib(default_attrib: dict, incoming_attrib: dict):
|
|
152
|
+
attrib = default_attrib.copy()
|
|
153
|
+
attrib.update(incoming_attrib)
|
|
154
|
+
return attrib
|
|
155
|
+
|
|
156
|
+
if isinstance(up_axis, str):
|
|
157
|
+
up_axis = "XYZ".index(up_axis.upper())
|
|
158
|
+
sqh = np.sqrt(0.5)
|
|
159
|
+
if up_axis == 0:
|
|
160
|
+
xform = wp.transform(xform.p, wp.quat(0.0, 0.0, -sqh, sqh) * xform.q)
|
|
161
|
+
elif up_axis == 2:
|
|
162
|
+
xform = wp.transform(xform.p, wp.quat(sqh, 0.0, 0.0, -sqh) * xform.q)
|
|
163
|
+
# do not apply scaling to the root transform
|
|
164
|
+
xform = wp.transform(np.array(xform.p) / scale, xform.q)
|
|
165
|
+
|
|
166
|
+
def parse_float(attrib, key, default):
|
|
167
|
+
if key in attrib:
|
|
168
|
+
return float(attrib[key])
|
|
169
|
+
else:
|
|
170
|
+
return default
|
|
171
|
+
|
|
172
|
+
def parse_vec(attrib, key, default):
|
|
173
|
+
if key in attrib:
|
|
174
|
+
out = np.fromstring(attrib[key], sep=" ", dtype=np.float32)
|
|
175
|
+
else:
|
|
176
|
+
out = np.array(default, dtype=np.float32)
|
|
177
|
+
|
|
178
|
+
length = len(out)
|
|
179
|
+
if length == 1:
|
|
180
|
+
return wp.vec(len(default), wp.float32)(out[0], out[0], out[0])
|
|
181
|
+
|
|
182
|
+
return wp.vec(length, wp.float32)(out)
|
|
183
|
+
|
|
184
|
+
def parse_orientation(attrib):
|
|
185
|
+
if "quat" in attrib:
|
|
186
|
+
wxyz = np.fromstring(attrib["quat"], sep=" ")
|
|
187
|
+
return wp.normalize(wp.quat(*wxyz[1:], wxyz[0]))
|
|
188
|
+
if "euler" in attrib:
|
|
189
|
+
euler = np.fromstring(attrib["euler"], sep=" ")
|
|
190
|
+
if use_degrees:
|
|
191
|
+
euler *= np.pi / 180
|
|
192
|
+
return wp.quat_from_euler(euler, *euler_seq)
|
|
193
|
+
if "axisangle" in attrib:
|
|
194
|
+
axisangle = np.fromstring(attrib["axisangle"], sep=" ")
|
|
195
|
+
angle = axisangle[3]
|
|
196
|
+
if use_degrees:
|
|
197
|
+
angle *= np.pi / 180
|
|
198
|
+
axis = wp.normalize(wp.vec3(*axisangle[:3]))
|
|
199
|
+
return wp.quat_from_axis_angle(axis, angle)
|
|
200
|
+
if "xyaxes" in attrib:
|
|
201
|
+
xyaxes = np.fromstring(attrib["xyaxes"], sep=" ")
|
|
202
|
+
xaxis = wp.normalize(wp.vec3(*xyaxes[:3]))
|
|
203
|
+
zaxis = wp.normalize(wp.vec3(*xyaxes[3:]))
|
|
204
|
+
yaxis = wp.normalize(wp.cross(zaxis, xaxis))
|
|
205
|
+
rot_matrix = np.array([xaxis, yaxis, zaxis]).T
|
|
206
|
+
return wp.quat_from_matrix(rot_matrix)
|
|
207
|
+
if "zaxis" in attrib:
|
|
208
|
+
zaxis = np.fromstring(attrib["zaxis"], sep=" ")
|
|
209
|
+
zaxis = wp.normalize(wp.vec3(*zaxis))
|
|
210
|
+
xaxis = wp.normalize(wp.cross(wp.vec3(0, 0, 1), zaxis))
|
|
211
|
+
yaxis = wp.normalize(wp.cross(zaxis, xaxis))
|
|
212
|
+
rot_matrix = np.array([xaxis, yaxis, zaxis]).T
|
|
213
|
+
return wp.quat_from_matrix(rot_matrix)
|
|
214
|
+
return wp.quat_identity()
|
|
215
|
+
|
|
216
|
+
def parse_mesh(geom):
|
|
217
|
+
import trimesh
|
|
218
|
+
|
|
219
|
+
faces = []
|
|
220
|
+
vertices = []
|
|
221
|
+
stl_file = mesh_assets[geom["mesh"]]
|
|
222
|
+
m = trimesh.load(stl_file)
|
|
223
|
+
|
|
224
|
+
for v in m.vertices:
|
|
225
|
+
vertices.append(np.array(v) * scale)
|
|
226
|
+
|
|
227
|
+
for f in m.faces:
|
|
228
|
+
faces.append(int(f[0]))
|
|
229
|
+
faces.append(int(f[1]))
|
|
230
|
+
faces.append(int(f[2]))
|
|
231
|
+
return wp.sim.Mesh(vertices, faces), m.scale
|
|
232
|
+
|
|
233
|
+
def parse_body(body, parent, incoming_defaults: dict):
|
|
234
|
+
body_class = body.get("childclass")
|
|
235
|
+
if body_class is None:
|
|
236
|
+
defaults = incoming_defaults
|
|
237
|
+
else:
|
|
238
|
+
for pattern in ignore_classes:
|
|
239
|
+
if re.match(pattern, body_class):
|
|
240
|
+
return
|
|
241
|
+
defaults = merge_attrib(incoming_defaults, class_defaults[body_class])
|
|
242
|
+
if "body" in defaults:
|
|
243
|
+
body_attrib = merge_attrib(defaults["body"], body.attrib)
|
|
244
|
+
else:
|
|
245
|
+
body_attrib = body.attrib
|
|
246
|
+
body_name = body_attrib["name"]
|
|
247
|
+
body_pos = parse_vec(body_attrib, "pos", (0.0, 0.0, 0.0))
|
|
248
|
+
body_ori = parse_orientation(body_attrib)
|
|
249
|
+
if parent == -1:
|
|
250
|
+
body_pos = wp.transform_point(xform, body_pos)
|
|
251
|
+
body_ori = xform.q * body_ori
|
|
252
|
+
body_pos *= scale
|
|
253
|
+
|
|
254
|
+
joint_armature = []
|
|
255
|
+
joint_name = []
|
|
256
|
+
joint_pos = []
|
|
257
|
+
|
|
258
|
+
linear_axes = []
|
|
259
|
+
angular_axes = []
|
|
260
|
+
joint_type = None
|
|
261
|
+
|
|
262
|
+
freejoint_tags = body.findall("freejoint")
|
|
263
|
+
if len(freejoint_tags) > 0:
|
|
264
|
+
joint_type = wp.sim.JOINT_FREE
|
|
265
|
+
joint_name.append(freejoint_tags[0].attrib.get("name", f"{body_name}_freejoint"))
|
|
266
|
+
else:
|
|
267
|
+
joints = body.findall("joint")
|
|
268
|
+
for _i, joint in enumerate(joints):
|
|
269
|
+
if "joint" in defaults:
|
|
270
|
+
joint_attrib = merge_attrib(defaults["joint"], joint.attrib)
|
|
271
|
+
else:
|
|
272
|
+
joint_attrib = joint.attrib
|
|
273
|
+
|
|
274
|
+
# default to hinge if not specified
|
|
275
|
+
joint_type_str = joint_attrib.get("type", "hinge")
|
|
276
|
+
|
|
277
|
+
joint_name.append(joint_attrib["name"])
|
|
278
|
+
joint_pos.append(parse_vec(joint_attrib, "pos", (0.0, 0.0, 0.0)) * scale)
|
|
279
|
+
joint_range = parse_vec(joint_attrib, "range", (-3.0, 3.0))
|
|
280
|
+
joint_armature.append(parse_float(joint_attrib, "armature", armature) * armature_scale)
|
|
281
|
+
|
|
282
|
+
if joint_type_str == "free":
|
|
283
|
+
joint_type = wp.sim.JOINT_FREE
|
|
284
|
+
break
|
|
285
|
+
if joint_type_str == "fixed":
|
|
286
|
+
joint_type = wp.sim.JOINT_FIXED
|
|
287
|
+
break
|
|
288
|
+
is_angular = joint_type_str == "hinge"
|
|
289
|
+
mode = wp.sim.JOINT_MODE_FORCE
|
|
290
|
+
if stiffness > 0.0 or "stiffness" in joint_attrib:
|
|
291
|
+
mode = wp.sim.JOINT_MODE_TARGET_POSITION
|
|
292
|
+
axis_vec = parse_vec(joint_attrib, "axis", (0.0, 0.0, 0.0))
|
|
293
|
+
ax = wp.sim.model.JointAxis(
|
|
294
|
+
axis=axis_vec,
|
|
295
|
+
limit_lower=(np.deg2rad(joint_range[0]) if is_angular and use_degrees else joint_range[0]),
|
|
296
|
+
limit_upper=(np.deg2rad(joint_range[1]) if is_angular and use_degrees else joint_range[1]),
|
|
297
|
+
target_ke=parse_float(joint_attrib, "stiffness", stiffness),
|
|
298
|
+
target_kd=parse_float(joint_attrib, "damping", damping),
|
|
299
|
+
limit_ke=limit_ke,
|
|
300
|
+
limit_kd=limit_kd,
|
|
301
|
+
mode=mode,
|
|
302
|
+
)
|
|
303
|
+
if is_angular:
|
|
304
|
+
angular_axes.append(ax)
|
|
305
|
+
else:
|
|
306
|
+
linear_axes.append(ax)
|
|
307
|
+
|
|
308
|
+
link = builder.add_body(
|
|
309
|
+
origin=wp.transform(body_pos, body_ori), # will be evaluated in fk()
|
|
310
|
+
armature=joint_armature[0] if len(joint_armature) > 0 else armature,
|
|
311
|
+
name=body_name,
|
|
312
|
+
)
|
|
313
|
+
|
|
314
|
+
if joint_type is None:
|
|
315
|
+
if len(linear_axes) == 0:
|
|
316
|
+
if len(angular_axes) == 0:
|
|
317
|
+
joint_type = wp.sim.JOINT_FIXED
|
|
318
|
+
elif len(angular_axes) == 1:
|
|
319
|
+
joint_type = wp.sim.JOINT_REVOLUTE
|
|
320
|
+
elif len(angular_axes) == 2:
|
|
321
|
+
joint_type = wp.sim.JOINT_UNIVERSAL
|
|
322
|
+
elif len(angular_axes) == 3:
|
|
323
|
+
joint_type = wp.sim.JOINT_COMPOUND
|
|
324
|
+
elif len(linear_axes) == 1 and len(angular_axes) == 0:
|
|
325
|
+
joint_type = wp.sim.JOINT_PRISMATIC
|
|
326
|
+
else:
|
|
327
|
+
joint_type = wp.sim.JOINT_D6
|
|
328
|
+
|
|
329
|
+
joint_pos = joint_pos[0] if len(joint_pos) > 0 else (0.0, 0.0, 0.0)
|
|
330
|
+
builder.add_joint(
|
|
331
|
+
joint_type,
|
|
332
|
+
parent,
|
|
333
|
+
link,
|
|
334
|
+
linear_axes,
|
|
335
|
+
angular_axes,
|
|
336
|
+
name="_".join(joint_name),
|
|
337
|
+
parent_xform=wp.transform(body_pos + joint_pos, body_ori),
|
|
338
|
+
child_xform=wp.transform(joint_pos, wp.quat_identity()),
|
|
339
|
+
armature=joint_armature[0] if len(joint_armature) > 0 else armature,
|
|
340
|
+
)
|
|
341
|
+
|
|
342
|
+
# -----------------
|
|
343
|
+
# add shapes
|
|
344
|
+
|
|
345
|
+
for geo_count, geom in enumerate(body.findall("geom")):
|
|
346
|
+
geom_defaults = defaults
|
|
347
|
+
if "class" in geom.attrib:
|
|
348
|
+
geom_class = geom.attrib["class"]
|
|
349
|
+
ignore_geom = False
|
|
350
|
+
for pattern in ignore_classes:
|
|
351
|
+
if re.match(pattern, geom_class):
|
|
352
|
+
ignore_geom = True
|
|
353
|
+
break
|
|
354
|
+
if ignore_geom:
|
|
355
|
+
continue
|
|
356
|
+
if geom_class in class_defaults:
|
|
357
|
+
geom_defaults = merge_attrib(defaults, class_defaults[geom_class])
|
|
358
|
+
if "geom" in geom_defaults:
|
|
359
|
+
geom_attrib = merge_attrib(geom_defaults["geom"], geom.attrib)
|
|
360
|
+
else:
|
|
361
|
+
geom_attrib = geom.attrib
|
|
362
|
+
|
|
363
|
+
geom_name = geom_attrib.get("name", f"{body_name}_geom_{geo_count}")
|
|
364
|
+
geom_type = geom_attrib.get("type", "sphere")
|
|
365
|
+
if "mesh" in geom_attrib:
|
|
366
|
+
geom_type = "mesh"
|
|
367
|
+
|
|
368
|
+
geom_size = parse_vec(geom_attrib, "size", [1.0, 1.0, 1.0]) * scale
|
|
369
|
+
geom_pos = parse_vec(geom_attrib, "pos", (0.0, 0.0, 0.0)) * scale
|
|
370
|
+
geom_rot = parse_orientation(geom_attrib)
|
|
371
|
+
geom_density = parse_float(geom_attrib, "density", density)
|
|
372
|
+
|
|
373
|
+
if geom_type == "sphere":
|
|
374
|
+
builder.add_shape_sphere(
|
|
375
|
+
link,
|
|
376
|
+
pos=geom_pos,
|
|
377
|
+
rot=geom_rot,
|
|
378
|
+
radius=geom_size[0],
|
|
379
|
+
density=geom_density,
|
|
380
|
+
**contact_vars,
|
|
381
|
+
)
|
|
382
|
+
|
|
383
|
+
elif geom_type == "box":
|
|
384
|
+
builder.add_shape_box(
|
|
385
|
+
link,
|
|
386
|
+
pos=geom_pos,
|
|
387
|
+
rot=geom_rot,
|
|
388
|
+
hx=geom_size[0],
|
|
389
|
+
hy=geom_size[1],
|
|
390
|
+
hz=geom_size[2],
|
|
391
|
+
density=geom_density,
|
|
392
|
+
**contact_vars,
|
|
393
|
+
)
|
|
394
|
+
|
|
395
|
+
elif geom_type == "mesh" and parse_meshes:
|
|
396
|
+
mesh, _ = parse_mesh(geom_attrib)
|
|
397
|
+
if "mesh" in defaults:
|
|
398
|
+
mesh_scale = parse_vec(defaults["mesh"], "scale", [1.0, 1.0, 1.0])
|
|
399
|
+
else:
|
|
400
|
+
mesh_scale = [1.0, 1.0, 1.0]
|
|
401
|
+
# as per the Mujoco XML reference, ignore geom size attribute
|
|
402
|
+
assert len(geom_size) == 3, "need to specify size for mesh geom"
|
|
403
|
+
builder.add_shape_mesh(
|
|
404
|
+
body=link,
|
|
405
|
+
pos=geom_pos,
|
|
406
|
+
rot=geom_rot,
|
|
407
|
+
mesh=mesh,
|
|
408
|
+
scale=mesh_scale,
|
|
409
|
+
density=density,
|
|
410
|
+
**contact_vars,
|
|
411
|
+
)
|
|
412
|
+
|
|
413
|
+
elif geom_type in {"capsule", "cylinder"}:
|
|
414
|
+
if "fromto" in geom_attrib:
|
|
415
|
+
geom_fromto = parse_vec(geom_attrib, "fromto", (0.0, 0.0, 0.0, 1.0, 0.0, 0.0))
|
|
416
|
+
|
|
417
|
+
start = wp.vec3(geom_fromto[0:3]) * scale
|
|
418
|
+
end = wp.vec3(geom_fromto[3:6]) * scale
|
|
419
|
+
|
|
420
|
+
# compute rotation to align the Warp capsule (along x-axis), with mjcf fromto direction
|
|
421
|
+
axis = wp.normalize(end - start)
|
|
422
|
+
angle = math.acos(wp.dot(axis, wp.vec3(0.0, 1.0, 0.0)))
|
|
423
|
+
axis = wp.normalize(wp.cross(axis, wp.vec3(0.0, 1.0, 0.0)))
|
|
424
|
+
|
|
425
|
+
geom_pos = (start + end) * 0.5
|
|
426
|
+
geom_rot = wp.quat_from_axis_angle(axis, -angle)
|
|
427
|
+
|
|
428
|
+
geom_radius = geom_size[0]
|
|
429
|
+
geom_height = wp.length(end - start) * 0.5
|
|
430
|
+
geom_up_axis = 1
|
|
431
|
+
|
|
432
|
+
else:
|
|
433
|
+
geom_radius = geom_size[0]
|
|
434
|
+
geom_height = geom_size[1]
|
|
435
|
+
geom_up_axis = up_axis
|
|
436
|
+
|
|
437
|
+
if geom_type == "cylinder":
|
|
438
|
+
builder.add_shape_cylinder(
|
|
439
|
+
link,
|
|
440
|
+
pos=geom_pos,
|
|
441
|
+
rot=geom_rot,
|
|
442
|
+
radius=geom_radius,
|
|
443
|
+
half_height=geom_height,
|
|
444
|
+
density=density,
|
|
445
|
+
up_axis=geom_up_axis,
|
|
446
|
+
**contact_vars,
|
|
447
|
+
)
|
|
448
|
+
else:
|
|
449
|
+
builder.add_shape_capsule(
|
|
450
|
+
link,
|
|
451
|
+
pos=geom_pos,
|
|
452
|
+
rot=geom_rot,
|
|
453
|
+
radius=geom_radius,
|
|
454
|
+
half_height=geom_height,
|
|
455
|
+
density=density,
|
|
456
|
+
up_axis=geom_up_axis,
|
|
457
|
+
**contact_vars,
|
|
458
|
+
)
|
|
459
|
+
|
|
460
|
+
else:
|
|
461
|
+
print(f"MJCF parsing shape {geom_name} issue: geom type {geom_type} is unsupported")
|
|
462
|
+
|
|
463
|
+
# -----------------
|
|
464
|
+
# recurse
|
|
465
|
+
|
|
466
|
+
for child in body.findall("body"):
|
|
467
|
+
parse_body(child, link, defaults)
|
|
468
|
+
|
|
469
|
+
# -----------------
|
|
470
|
+
# start articulation
|
|
471
|
+
|
|
472
|
+
start_shape_count = len(builder.shape_geo_type)
|
|
473
|
+
builder.add_articulation()
|
|
474
|
+
|
|
475
|
+
world = root.find("worldbody")
|
|
476
|
+
world_class = get_class(world)
|
|
477
|
+
world_defaults = merge_attrib(class_defaults["__all__"], class_defaults.get(world_class, {}))
|
|
478
|
+
for body in world.findall("body"):
|
|
479
|
+
parse_body(body, -1, world_defaults)
|
|
480
|
+
|
|
481
|
+
end_shape_count = len(builder.shape_geo_type)
|
|
482
|
+
|
|
483
|
+
if not enable_self_collisions:
|
|
484
|
+
for i in range(start_shape_count, end_shape_count):
|
|
485
|
+
for j in range(i + 1, end_shape_count):
|
|
486
|
+
builder.shape_collision_filter_pairs.add((i, j))
|
|
487
|
+
|
|
488
|
+
if collapse_fixed_joints:
|
|
489
|
+
builder.collapse_fixed_joints()
|