-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathdebug_contact_physics.py
More file actions
96 lines (75 loc) · 3.55 KB
/
debug_contact_physics.py
File metadata and controls
96 lines (75 loc) · 3.55 KB
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
#!/usr/bin/env python3
"""
Debug contact physics to understand why L2 keeps rotating backward.
"""
import mujoco
import numpy as np
import os
# Load model
model_path = os.path.join(os.path.dirname(__file__), 'ezgripper.xml')
model = mujoco.MjModel.from_xml_path(model_path)
data = mujoco.MjData(model)
print("="*80)
print("CONTACT PHYSICS DEBUG")
print("="*80)
# Get IDs
f1_palm_id = model.joint('F1_palm_knuckle').id
f1_tip_id = model.joint('F1_knuckle_tip').id
f2_palm_id = model.joint('F2_palm_knuckle').id
f2_tip_id = model.joint('F2_knuckle_tip').id
f1_actuator_id = model.actuator('gripper_actuator_f1').id
f2_actuator_id = model.actuator('gripper_actuator_f2').id
print(f"\nTesting contact physics step by step:")
# Test 1: Start from open position
mujoco.mj_resetData(model, data)
data.ctrl[f1_actuator_id] = 0.200 # Open
data.ctrl[f2_actuator_id] = 0.200
for i in range(100):
mujoco.mj_step(model, data)
print(f"\nAfter opening (100 steps):")
print(f"F1: Palm={np.degrees(data.qpos[f1_palm_id]):6.1f}°, Tip={np.degrees(data.qpos[f1_tip_id]):6.1f}°")
print(f"F2: Palm={np.degrees(data.qpos[f2_palm_id]):6.1f}°, Tip={np.degrees(data.qpos[f2_tip_id]):6.1f}°")
print(f"Contacts: {data.ncon}")
# Test 2: Close slowly until contact
target_tendon = 0.140
data.ctrl[f1_actuator_id] = target_tendon
data.ctrl[f2_actuator_id] = target_tendon
print(f"\nClosing to tendon position {target_tendon}:")
for step in range(500):
mujoco.mj_step(model, data)
if step % 50 == 0:
f1_palm = np.degrees(data.qpos[f1_palm_id])
f1_tip = np.degrees(data.qpos[f1_tip_id])
f2_palm = np.degrees(data.qpos[f2_palm_id])
f2_tip = np.degrees(data.qpos[f2_tip_id])
print(f" Step {step:3d}: F1({f1_palm:5.1f}°,{f1_tip:5.1f}°) F2({f2_palm:5.1f}°,{f2_tip:5.1f}°) Contacts={data.ncon}")
if data.ncon > 0:
print(f" CONTACT DETECTED!")
for i in range(min(data.ncon, 3)):
contact = data.contact[i]
geom1_name = mujoco.mj_id2name(model, mujoco.mjtObj.mjOBJ_GEOM, contact.geom1)
geom2_name = mujoco.mj_id2name(model, mujoco.mjtObj.mjOBJ_GEOM, contact.geom2)
force = np.linalg.norm(data.efc_force[contact.efc_address:contact.efc_address+6])
print(f" {i}: {geom1_name} <-> {geom2_name}, force={force:.1f}N")
break
# Test 3: Continue applying force after contact
print(f"\nContinuing to apply force after contact:")
for step in range(200):
mujoco.mj_step(model, data)
if step % 25 == 0:
f1_palm = np.degrees(data.qpos[f1_palm_id])
f1_tip = np.degrees(data.qpos[f1_tip_id])
f2_palm = np.degrees(data.qpos[f2_palm_id])
f2_tip = np.degrees(data.qpos[f2_tip_id])
print(f" Step {step:3d}: F1({f1_palm:5.1f}°,{f1_tip:5.1f}°) F2({f2_palm:5.1f}°,{f2_tip:5.1f}°) Contacts={data.ncon}")
if data.ncon > 0:
for i in range(min(data.ncon, 2)):
contact = data.contact[i]
geom1_name = mujoco.mj_id2name(model, mujoco.mjtObj.mjOBJ_GEOM, contact.geom1)
geom2_name = mujoco.mj_id2name(model, mujoco.mjtObj.mjOBJ_GEOM, contact.geom2)
force = np.linalg.norm(data.efc_force[contact.efc_address:contact.efc_address+6])
pos = data.efc_pos[contact.efc_address:contact.efc_address+3]
print(f" {i}: {geom1_name} <-> {geom2_name}, force={force:.1f}N, pos=({pos[0]:.3f},{pos[1]:.3f},{pos[2]:.3f})")
print("\n" + "="*80)
print("CONTACT PHYSICS DEBUG COMPLETE")
print("="*80)