-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathmanyrotations.py
84 lines (66 loc) · 2.36 KB
/
manyrotations.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
"""
Solve a very simple toy problem involving a rotation
min_w || R p - p* ||**2 with R:= exp(w)
where p and p* are two known 3d vectors.
Display the result in Meshcat.
"""
import casadi
import pinocchio as pin
from casadi_so3 import exp3,log3
from pinocchio.utils import rotate
import numpy as np
import time
from utils.meshcat_viewer_wrapper import MeshcatVisualizer
p = np.array([0.1, 0.2, 0.3])
omega = 2*np.pi*.5
pdes = [ rotate('x',t)@rotate('y',2*t-5)@ ((1+.2*np.sin(t*omega))*p)
for t in np.arange(0,5,.02) ]
T = len(pdes)
# ## PROBLEM
# Create the casadi optimization problem
opti = casadi.Opti()
# The optimization variable is the angle-vector w and the associated rotation R=exp(w)
ws = [ opti.variable(3) for t in range(T) ]
Rs = [ exp3(w) for w in ws ]
def new_func(p, pdes, T, ws, Rs):
totalcost = 0
# Beware: casadi matrix product is @ like numpy array product
for t in range(T):
totalcost += 0.5 * casadi.sumsqr(Rs[t] @ p - pdes[t])
if t>0:
# totalcost += 0.5 * casadi.sumsqr( log3(Rs[t-1].T @ Rs[t]) )
totalcost += 0.5 * casadi.sumsqr( ws[t] - ws[t-1])
return totalcost
totalcost = new_func(p, pdes, T, ws, Rs)
# ## SOLVE
opti.minimize(totalcost)
opti.solver("ipopt")
# # Initial guess with optimal solution for debug
# R0s=[pin.Quaternion.FromTwoVectors(p,pd).matrix() for pd in pdes]
# w0s=[pin.log3(R) for R in R0s]
# # Uncomment if you want the warm start
# opti.set_initial(ws,w0s)
# Warm start
for t in range(T):
opti.set_initial(ws[t],np.array([.1,.1,.1]))
sol = opti.solve()
# The value of the decision variable at the optimum are stored in 2 arrays.
ws_sol = [ opti.value(w) for w in ws ]
Rs_sol = [ opti.value(R) for R in Rs ]
# ## Sanity check
for R_sol in Rs_sol:
assert np.allclose(R_sol @ R_sol.T, np.eye(3))
# ## Display
viz = MeshcatVisualizer()
pointID = "world/point"
viz.addSphere(pointID, 0.1, [1, 0, 0, 1])
pointdesID = "world/pointdes"
viz.addSphere(pointdesID, 0.1, [0, 1, 0, 1])
boxID = "world/box"
viz.addBox(boxID, (p * 2).tolist(), [1, 1, 0, .1])
def viewtraj():
for t,[R_sol,pt] in enumerate(zip(Rs_sol,pdes)):
viz.applyConfiguration(pointdesID, pt.tolist() + [0, 0, 0, 1])
viz.applyConfiguration(pointID, (R_sol @ p).tolist() + [0, 0, 0, 1])
viz.applyConfiguration(boxID, [0, 0, 0] + pin.Quaternion(R_sol).coeffs().tolist())
time.sleep(1e-0)