next up previous
Next: About this document ...

PHYS 410/555: Computational Physics      Fall 2003       Homework 5
DUE: Thursday, November 27, 10:00 AM       Report bugs to [email protected]

The following assignment involves writing and testing two Fortran 77 programs which use lsoda to solve ordinary differential equations. As usual, all files required by the assignment must reside in the correct places on your lnx account for the homework to be considered complete. Contact me immediately if you are having undue difficulties with any part of the homework.



Problem 1: Consider the following ODE, known as the Van der Pol equation:

\begin{displaymath}
\frac{d^2u(t)}{dt^2} - \epsilon \left( 1 - u(t)^2\right)
\frac{du(t)}{dt} + u(t) = 0 \quad\quad t \ge 0
\end{displaymath} (1)

where $\epsilon > 0$ is a specified positive constant. Physically, this ODE describes the voltage behaviour of a tunnel diode oscillator, although for the purposes of this homework we will simply be viewing it as providing an interesting example of non-linear oscillator dynamics.

In directory $\sim$/hw5/a1, write a Fortran 77 program vdp.f, with corresponding executable vdp which solves the above ODE subject to the initial conditions:
$\displaystyle u(0)$ $\textstyle =$ $\displaystyle {\tt u0}$ (2)
$\displaystyle \frac{du}{dt}(0)$ $\textstyle =$ $\displaystyle {\tt du0}$ (3)

vdp must have the following usage:
   usage: vdp <tmax> <u0> <du0> <epsilon> <tol> <olevel>
where
  1. tmax is the real*8 final integration time ( ${\tt tmax} > 0$).
  2. u0 is the real*8 initial oscillator displacement.
  3. du0 is the real*8 initial oscillator velocity.
  4. epsilon is the real*8 value of $\epsilon$ ( ${\tt epsilon} > 0$).
  5. tol is the real*8 lsoda tolerance. Use itol = 1, and pure absolute tolerance.
    ( $10^{-12} \le {\tt tol} \le 10^{-2}$)
  6. olevel is the integer output level. ( ${\tt olevel} > 0$).

Your program must be commented, and must perform rudimentary error-checking of command-line arguments: arguments should be of the proper type, and should be in the appropriate ranges as given above (as usual, you can use i4arg and r8arg to detect whether an argument is of the correct type by using default values which a user is unlikely to enter).

At requested output times, vdp must write on standard output the time, and the computed displacement and velocity of the oscillator at that time, using an output statement such as
    write(*,*) t, y(1), y(2)

The requested output times, $t_j$, are defined by
\begin{displaymath}
t_j = 0, \, h, \, 2h, \, \cdots \, {\tt tmax}
\end{displaymath} (4)

where $h$ is given by
\begin{displaymath}
h \equiv \frac{{\tt tmax}}{2^{\tt olevel}}
\end{displaymath} (5)

To test your implementation, code an independent residual evaluation program, chk-vdp.f--with corresponding executable chk-vdp--which applies an $O(h^2)$ finite-difference approximation of (1) to the output of vdp. chk-vdp.f should be identical in construction, as well as what it inputs and outputs, to the chk-tlsoda.f example covered in class, except that chk-vdp.f is to accept a single, real*8 command-line argument, epsilon. Use the usual $O(h^2)$ difference approximations for the first and second time-derivatives in (1); namely:
$\displaystyle \frac{du}{dt}(t_j)$ $\textstyle =$ $\displaystyle \frac{u(t_{j+1}) - u(t_{j-1})}{2h} + O(h^2)$ (6)
$\displaystyle \frac{d^2u}{dt^2}(t_j)$ $\textstyle =$ $\displaystyle \frac{u(t_{j+1}) - 2u(t_j) + u(t_{j-1})}{h^2} + O(h^2)$ (7)

Create a script Do-chk which contains the following
#!/bin/sh 

epsilon=1.0
vdp 10.0 0.0 1.0 $epsilon 1.0d-8 10 | nth 1 2 > vdp-out

for inc in 8 4 2 1; do
   nth 1 2 < vdp-out | lines 1 . $inc | chk-vdp $epsilon
done
Once you are satisfied that both vdp and chk-vdp are working properly, execute the script and save the output as chk-vdp-out.

In order to get a feel for the behaviour of the oscillator, use vdp to investigate the solution of (1) for a variety of values of $\epsilon$ ( $0 \le \epsilon \le 5$ suggested), u0 ( $10^{-3} \le {\tt u0} \le 10$ suggested) and du0, ( $-5 \le {\tt du0} \le 5$ suggested). When performing your studies, ensure that you specify tmax large enough to determine the long-time behaviour of the oscillator. You can use gnuplot to plot the results of your computations.

An interesting way to examine the dynamics of an oscillator is to make a phase-space plot, i.e. a parametric plot (in $t$) of the oscillator's velocity versus its displacement. Use gnuplot to make a single phase-space Postscript plot called vdp-phase showing the $du/dt$ vs $u$ trajectories from the following three computations:
   vdp 50.0   0.01  0.0  1.0 1.0e-8 11
   vdp 50.0    1.0 -1.0  1.0 1.0e-8 11
   vdp 50.0    3.0  1.0  1.0 1.0e-8 11

Using gnuplot, make a Postscript plot called vdp.ps showing the oscillator displacement versus $t$ for the following computation
   vdp 100.0 0.01 0.0 10.0 1.0e-8 12

What can you say about the long-time behaviour of the oscillator as a function of $u(0)$ and $du/dt(0)$ for fixed $\epsilon$? Answer in $\sim$/hw5/a1/README.



Problem 2: Consider the following partial differential equation, (frequently called the diffusion equation, or heat equation):

\begin{displaymath}
\frac{\partial u(x,t)}{\partial t} =
\frac{\partial^2 u(x...
...ad\quad {\hbox{\rm on}} \quad\quad 0 \le x \le 1 \quad t \ge 0
\end{displaymath} (8)

subject to (i) the initial conditions:
\begin{displaymath}
u(x,0) = u_0(x)
\end{displaymath} (9)

(where $u_0(x)$ is a specified function), and (ii) the boundary conditions:
\begin{displaymath}
u(0,t) = u(1,t) = 0
\end{displaymath} (10)

In directory $\sim$/hw5/a2, write a Fortran 77 program, diffusion.f, with corresponding executable diffusion, which solves the above PDE using the method of lines, and lsoda. Specifically, convert (8) into a system of $N$ coupled ODEs by (i) introducing a regular mesh $x_j$ and discrete unknowns $u_j$:

\begin{displaymath}
x_j \equiv (j - 1) h \quad j = 1,2,\,\cdots\, N \quad\quad
h \equiv (N - 1)^{-1} \quad\quad u_j(t) \equiv u(x_j,t)
\end{displaymath} (11)

(ii) replacing $\partial^2 u/\partial x^2$ with the usual $O(h^2)$ finite difference approximation, yielding
\begin{displaymath}
\frac{d\,u_j(t)}{dt} = \frac{u_{j+1} - 2 u_j + u_{j-1}}{h^2} \quad\quad
j = 2,\,\cdots\, N - 1
\end{displaymath} (12)

and (iii) incorporating the Dirichlet boundary conditions (10) as follows:
$\displaystyle \frac{du_1(t)}{dt}$ $\textstyle =$ $\displaystyle 0$ (13)
$\displaystyle \frac{du_N(t)}{dt}$ $\textstyle =$ $\displaystyle 0$ (14)

diffusion must have the following usage:
   usage: diffusion <xlevel> <olevel> <tfinal> <dtout> <tol>
where
  1. xlevel is the integer level of spatial discretization: $N \equiv {\tt nx} = 2^{\tt xlevel} + 1$. ( $1 \le {\tt xlevel} \le 9$).
  2. olevel is the integer level of spatial discretization for output to be subsequently plotted as a gnuplot parametric surface plot. See the discussion of the stride argument of the instructor-supplied gnuout routine below. ( $1 \le {\tt olevel} \le {\tt xlevel}$).
  3. tfinal is the real*8 final integration time. ( ${\tt tfinal > 0}$).
  4. dtout is the real*8 output time interval: i.e. diffusion produces output at $t =
{\tt dtout}, 2 \,{\tt dtout}, ...,
N_{\tt out} \, {\tt dtout}$ where $N_{\tt out}$ is the largest integer satisfying $N_{\tt out} \, {\tt dtout} \le {\tt tfinal}$. ( ${\tt dtout} > 0$).
  5. tol is the real*8 lsoda tolerance. Use itol = 1, and atol = rtol = tol. ( $10^{-12} \le {\tt tol} \le 10^{-2}$)

For the initial conditions (9), use the following gaussian profile:
$\displaystyle u(x,0)$ $\textstyle =$ $\displaystyle u_0(x) = \exp(-((x-0.5)/0.1)^2) \quad\quad 0 < x < 1$ (15)
$\displaystyle u(0,0)$ $\textstyle =$ $\displaystyle u(1,0) = 0$ (16)

diffusion must output the approximate solution to standard output using the gnuout routine, which can be copied from $\sim$phys410/hw5/diffusion/gnuout.f:
c===========================================================
c     Output to standard out for subsequent plotting via 
c     gnuplot.
c===========================================================
      subroutine gnuout(u,x,nx,t,stride)
         implicit       none

         integer        nx,           stride
         real*8         u(nx),        x(nx),        t

         integer        j
         
         do j = 1 , nx , stride
            write(*,*) t, x(j), u(j)
         end do
         write(*,*)

         return

      end
Note that the stride argument is used to ``down-sample'' the data being written, since at high spatial resolutions, direct surface plotting via gnuplot will produce very large, mostly illegible plots. In the current case stride is defined as follows:
\begin{displaymath}
{\tt stride} \equiv 2^{\tt xlevel - olevel}
\end{displaymath} (17)

Your program must be commented and must perform error-checking of command-line arguments as in Problem 1.

In developing your program, you may find it useful to use the xvs interface, the xvs visualization server, and the sdftoxvs utility program which sends .sdf files to xvs. See the Course Software page for information on these topics, as well as the Supplementary Material for Homework 5 section of the ODE Notes page.

After your program has been de-bugged to your satisfaction, use it to generate a file out8 which contains the standard output of the invocation
   diffusion 8 6 0.20 0.005 1.0e-6
Using gnuplot, make a parametric surface plot of this data and save it as the Postscript file out8.ps.

Finally, in $\sim$hw5/a2/README, briefly discuss how you might go about verifying that your program is correct (i.e. that as ${\tt tol} \to 0$ and $h \to 0$ we have $u_j(t) \to u(t)$).


next up previous
Next: About this document ...
Matthew W. Choptuik 2003-11-05