/* DEMO3.I Chaotic pendulum demo $Id$ */ /* Copyright (c) 1995. The Regents of the University of California. All rights reserved. */ func demo3(time_limit, kick=, damp_time=) /* DOCUMENT demo3 Solve Lagrange's equations for a famous chaotic pendulum (as on exhibit at the San Francisco Exploratorium). Run a movie of the result. By default, the movie runs for 60 seconds, but if you supply an argument to the demo3 function, it will run for that many seconds. The kick= keyword may be used to adjsut the initial amplitude; kick= 1.2 is the default. You may also wish to supply a damp_time= keyword to see the effect of an ad hoc damping term. damp_time=100 is nice. */ { require, "rkutta.i"; a= 2.0; b= 3.0; c= 1.5; gravity= 1.0; widget_setup, a, b, c, 1.00, 0.67, gravity; dt= 0.05*sqrt(min(a,b,c)/gravity); if (is_void(damp_time)) damp_time= 1.e30; if (is_void(kick)) kick= 1.2; /* roll dice to get initial velocities */ thetadot= kick; gammadot= 0.1*random(); qdotq= [[thetadot,0.,0.,gammadot], [0.,0.,0.,0.]]; window, wait=1; s= 1.1*(b+c); limits, -s,s,-s,s; display, qdotq; /* without this, won't get axis numbers */ require, "movie.i"; movie, draw_frame, time_limit, 0.02; } func draw_frame (i) { display, qdotq; qdotq= rkdumb(lagrange, qdotq,0., dt, 1, nosave=1); return 1; /* just go until time limit expires */ } func widget_setup (a, b, c, m_tee, m_bar, gravity) { /* set up the values used by lagrange */ extern Tdiag, Ta, Tb, Tc, Ba, Bb, Bg, Dg; B= 0.5*m_bar*c^2; C= 0.5*B*c; A= m_tee*(2.*a^3+b^3)/3. + m_bar*c*(2.*a^2+b^2) + C; Ba= B*a; Bb= B*b; Dg= (0.5*m_tee*b^2 + m_bar*b*c)*gravity; Bg= B*gravity; Tdiag= [[A,0.,0.,0.], [0.,C,0.,0.], [0.,0.,C,0.], [0.,0.,0.,C]]; Ta= [[0.,Ba,0.,0.], [Ba,0.,0.,0.], [0.,0.,0.,0.], [0.,0.,0.,0.]]; Tb= [[0.,0.,Ba,0.], [0.,0.,0.,0.], [Ba,0.,0.,0.], [0.,0.,0.,0.]]; Tc= [[0.,0.,0.,Bb], [0.,0.,0.,0.], [0.,0.,0.,0.], [Bb,0.,0.,0.]]; } func lagrange (qdotq, time/*unused*/) { /* compute the time derivatives of [qdot,q] Lagrange's equations d(dL/dqdot)/dt - dL/dq = 0 turn out to be T*qdotdot+L=0, with T and L computed here. */ theta= qdotq(1,2); alpha= qdotq(2,2); beta= qdotq(3,2); gamma= qdotq(4,2); qdot= qdotq(,1); t2= qdot(1)^2; a2= qdot(2)^2; b2= qdot(3)^2; g2= qdot(4)^2; amt= alpha-theta; bpt= beta+theta; gmt= gamma-theta; T= Tdiag + Ta*sin(amt) + Tb*sin(bpt) - Tc*cos(gmt); camt= cos(amt); cbpt= cos(bpt); sgmt= sin(gmt); L= [Dg*sin(theta) + Ba*(a2*camt+b2*cbpt)+Bb*g2*sgmt, Bg*sin(alpha) - Ba*t2*camt, Bg*sin(beta) + Ba*t2*cbpt, Bg*sin(gamma) - Bb*t2*sgmt]; qdotdot= LUsolve(T, -L); qdotdot(,1)-= qdot/damp_time; return [qdotdot, qdot]; } func display(qdotq) { /* draw the widget given its current state */ theta= qdotq(1,2); alpha= qdotq(2,2); beta= qdotq(3,2); gamma= qdotq(4,2); st= sin(theta); ct= cos(theta); p1x= -a*ct; p1y= -a*st; p2x= -p1x; p2y= -p1y; p3x= b*st; p3y= -b*ct; q1x= p1x + c*sin(alpha); q1y= p1y - c*cos(alpha); q2x= p2x + c*sin(beta); q2y= p2y - c*cos(beta); q3x= p3x + c*sin(gamma); q3y= p3y - c*cos(gamma); plg, [p1y,p2y], [p1x,p2x], marks=0,type=1,width=36,color="red"; plg, [0.,p3y], [0.,p3x], marks=0,type=1,width=36,color="red"; plg, [p1y,q1y], [p1x,q1x], marks=0,type=1,width=24,color="green"; plg, [p2y,q2y], [p2x,q2x], marks=0,type=1,width=24,color="green"; plg, [p3y,q3y], [p3x,q3x], marks=0,type=1,width=24,color="green"; }