2016-06-23 15:12:57 +03:00
Grenade
=======
```
First shalt thou take out the Holy Pin, then shalt thou count to three, no more, no less.
Three shall be the number thou shalt count, and the number of the counting shall be three.
Four shalt thou not count, neither count thou two, excepting that thou then proceed to three.
Five is right out.
```
💣 Machine learning which might blow up in your face 💣
2016-06-27 07:29:41 +03:00
Grenade is a dependently typed, practical, and pretty quick neural network library for concise and precise
2016-06-23 15:12:57 +03:00
specifications of complex networks in Haskell.
2016-12-02 16:04:01 +03:00
As an example, a network which can achieve less than 1.5% error on MNIST can be specified and
2016-12-06 13:07:38 +03:00
initialised with random weights in a few lines of code with
2016-06-23 15:12:57 +03:00
```haskell
2016-12-06 13:07:38 +03:00
randomMnist :: MonadRandom m
=> m (Network '[ Convolution 1 10 5 5 1 1, Pooling 2 2 2 2, Relu, Convolution 10 16 5 5 1 1, Pooling 2 2 2 2, FlattenLayer, Relu, FullyConnected 256 80, Logit, FullyConnected 80 10, Logit]
'[ 'D2 28 28, 'D3 24 24 10, 'D3 12 12 10, 'D3 12 12 10, 'D3 8 8 16, 'D3 4 4 16, 'D1 256, 'D1 256, 'D1 80, 'D1 80, 'D1 10, 'D1 10])
randomMnist = randomNetwork
2016-06-23 15:12:57 +03:00
```
The network can be thought of as a heterogeneous list of layers, and its type signature includes a type
level list of the shapes of the data passed between the layers of the network.
In the above example, the input layer can be seen to be a two dimensional (`D2`) image with 28 by 28 pixels.
The last item in the list is one dimensional (`D1`) with 10 values, representing the categories of the mnist data.
Layers in Grenade are represented as Haskell classes, so creating one's own is easy in downstream code. If the shapes
of a network are not specified correctly and a layer can not sensibly perform the operation between two shapes, then
it will result in a compile time error.
2016-06-29 13:34:53 +03:00
Build Instructions
------------------
2016-06-29 13:55:02 +03:00
Grenade currently only builds with the [mafia ](https://github.com/ambiata/mafia ) scipt that is located in the
2016-06-29 13:34:53 +03:00
repository. You will also need the `lapack` and `blas` libraries and development tools. Once you have all
that, Grenade can be build using:
```
./mafia build
```
and the tests run using:
```
./mafia test
```
2016-06-30 11:56:16 +03:00
Grenade is currently known to build with ghc 7.10 and 8.0.
2016-06-29 13:34:53 +03:00
2016-06-23 15:12:57 +03:00
Thanks
------
2016-10-17 13:10:13 +03:00
Writing a library like this has been on my mind for a while now, but a big shout out must go to [Justin Le ](https://github.com/mstksg ), whose
[dependently typed fully connected network ](https://blog.jle.im/entry/practical-dependent-types-in-haskell-1.html ) inspired me to get cracking, gave many ideas for the type level tools I
2016-06-23 15:12:57 +03:00
needed, and was a great starting point for writing this library.
Performance
-----------
Grenade is backed by hmatrix and blas, and uses a pretty clever convolution trick popularised by Caffe, which
is surprisingly effective and fast. So for many small scale problems it should be sufficient.
That said, it's currently stuck on a single core and doesn't hit up the GPU, so there's a fair bit of performance
sitting there begging.
Training 15 generations over Kaggle's mnist training data took a few hours.
Contributing
------------
Contributions are welcome.