Plan 9 from Bell Labs’s /usr/web/sources/contrib/fernan/nhc98/tests/nofib/spectral/atom/Main.hs

Copyright © 2021 Plan 9 Foundation.
Distributed under the MIT License.
Download the Plan 9 distribution.


{-	A kernel fragment from a program written by 
		Ron Legere  -- http://www.its.caltech.edu/~legere
		Caltech Quantum Optics

	It has the interesting property that Classic Hugs 
	runs it 20x faster than GHC!
	Reason: runExperiment calls itself with identical parameters,
		and Hugs commons that up for some reason.

	(Even with that fixed, STG Hugs ran the program a lot
	slower than Classic Hugs.)

	So it seems like an interesting program.  It appears here
	in the form with the silly self-call, because that makes
	it run a nice long time.  It thrashes floating point
	multiplication and lists.
-}

module Main where

infixl 9 .*
infix 9 <*>

main = putStr (show (take 1000 test))

test :: StateStream 
test = runExperiment testforce 0.02 [1.0] (State [1.0] [0.0]) 


testforce :: ForceLaw [Float]
testforce k [] = [] 
testforce k ( (State pos vel):atoms) = (-1.0) .* k * pos: 
		                       testforce k atoms

{-
The test force: K is a list of spring 
constants. (But here I am only doing one dimension for the purposes
of demonstrating the approach)
-}
 


{-
******************
******************
Module: Numerical classical atom (atom.lhs)
******************
******************
-}


-- We will want types for the whole simulation (where we can configure
-- the dimensions, etc), for the results (a state stream), and the force laws.

data AtomState = State Position Velocity
type Position   = [Float]
type Velocity   = [Float]
type Force  = [Float]


type StateStream = [AtomState]



{-
I made AtomState a data type, just so I could play with them. I think
I would prefer to keep it just a synonym, because that would
be simpler!

Now we need a function to write out the results to a file in a nice format.
I think I would prefer a simple x y z /n x y z /n etc

NOTE that show AtomState only shows the position! 
-}

instance Show AtomState where
  show  (State pos vel) = concat [ (show component) ++ "\t"  | component <- pos ]    
  showList states = showString (concat [(show state) ++ "\n" | state <- states])


{-
Note that I used lists for the position and velocity to allow for
unknown number of dimensions. I suspect this will have to
be optimized into tuples at some point!



Ok, so how shall we define the ForceLaw type?
-}

type ForceLaw a = a -> StateStream -> [Force]



{-
The force law maps a stream of states to a stream of forces so that time
dependant forces can be used. The parametric type 'a' is to allow the force law
to depend on some parameter, for example (the common case!) a seed for a random number
generater, and/or the timestep, or the spring constant 
-}
 
runExperiment :: ForceLaw a -> Float -> a -> AtomState -> StateStream 

{-
	In this form this program takes 1 min when compiled under ghc-4.05,
	but takes 3 seconds under hugs....
-}
runExperiment law dt param init = init : zipWith (propagate dt)
              				     (law param stream)
              				     stream
                   where  stream =
                             runExperiment law dt param init

{-
-- In this form GHC is (as expected) much faster
runExperiment law dt param init = stream
	where
	  stream = init : zipWith (propagate dt)
              		  (law param stream)
               		  stream
-}

{-
runExperiment forces timestep param initialcondition :: [AtomState] 
is an infinite stream of atom states. We can then use this to 
generate necessary averages, temperatures, allen variences , or wtf 
you want. 
 
We could for example, start the random number generator with seed param, if
the type is int . 

It is an error to have the initial 
atom state not have the correct number of dimensions. 
-}

propagate :: Float -> Force -> AtomState -> AtomState

{-
Ok, I see one problem with this, not general enough! Some better propagators
exist that can use previous atom states. Actually, by using previous atom states,
we will not even need to seperately track the velocities either.  Oh well, for now
I will stick with that.
-}

propagate dt aforce (State pos vel)  = State newpos newvel
        where newpos = pos  + (dt .* vel)
              newvel = vel  + (dt .* aforce)

-- Note assumes mass =1



{-
********************************************************
********************************************************

Numerical Lists

********************************************************
********************************************************
-}

instance Num a => Num [a] where
	negate (f:fs) = (negate f):(negate fs)
	l + []        = l
	[] + l	  = l
	(f:fs) + (g:gs) = (f+g):fs+gs
	_ * []     = []
	[] * _	   = []
	(f:fs) * (g:gs) = (f*g):(gs*gs)
	fromInteger c = fromInteger c : [0]


(.*):: Num a => a-> [a] -> [a]
c .* []     = []


c .* (f:fs) = c*f : c .* fs

(<*>):: Num a => [a] -> [a] -> a
f <*> g = sum (f*g)



Bell Labs OSI certified Powered by Plan 9

(Return to Plan 9 Home Page)

Copyright © 2021 Plan 9 Foundation. All Rights Reserved.
Comments to webmaster@9p.io.