Image Processing Example - PNWScala
The following example is based on the miniboxing presentation at PNWScala. A huge thanks to Thomas Lockney and the PNWScala organizers for the wonderful conference and the confreaks guys who recorded the presentation!
This page will guide you through the steps necessary to run the example in the presentation. The pureimage library is located at github.com/stephenjudkins/pureimage. The mock-up used for the presentation is located at github.com/VladUreche/image-example.
To start, clone the pureimage mock-up:
$ git clone https://github.com/VladUreche/image-example.git
Cloning into 'image-example'...
...
$ cd image-example
Let’s generate the Eclipse project files to load the project in the IDE:
$ sbt eclipse
[info] Loading global plugins from /home/sun/.sbt/0.13/plugins
...
[info] Done updating.
[info] Successfully created Eclipse project files for project(s):
[info] image-example
Cool. Now, all you need is the latest Scala IDE for Eclipse to load the project. Please use a version for Scala 2.11, in order to stay compatible with the rest of the project. Now, to load the project:
File
>Import ...
>General
>Existing Projects into Workspace
- point Eclipse to the
image-example
directory, selectimage-example
and hitFinish
You should see the project in Eclipse:
data:image/s3,"s3://crabby-images/34eef/34eefcc9392d91008fceeba08678b1ccbf25bd96" alt=""
Open Test.scala
from package image.example
, located in src/main/scala
, and run it:
data:image/s3,"s3://crabby-images/b913b/b913b1e639a2ec6db7a31b5f6fbf640d3ca10de1" alt=""
On my laptop, I got the following times:
Operation took 4192 milliseconds.
Operation took 4957 milliseconds.
Operation took 4755 milliseconds.
Operation took 3969 milliseconds.
Operation took 4073 milliseconds.
The miniboxing plugin generated a couple of warnings during compilation:
data:image/s3,"s3://crabby-images/35c09/35c09311805a80bfd77f8c5993a9c8e5d6c0d416" alt=""
One such warning is:
The class image.example.Pixel would benefit from miniboxing type parameter Repr, since it is
instantiated by a primitive type.
So let’s go in and fix that. Open Pixel.scala
and mark the type parameter Repr
of class Pixel
by @miniboxed
:
abstract class Pixel[@miniboxed Repr: Manifest] {
def r(t: Repr): Double // red
def g(t: Repr): Double // green
def b(t: Repr): Double // blue
def a(t: Repr): Double // alpha
def pack(r: Double, g: Double, b: Double, a: Double): Repr
// to allow Image to build an array:
implicit def manifest = implicitly[Manifest[Repr]]
}
data:image/s3,"s3://crabby-images/14cab/14cab55dc90852b573d030226e314dd73604df08" alt=""
We can fix the other 8 warnings that appear and recompile. Surprisingly, after compiling again, we get an error:
The method map in class ImageImpl overrides method map in class Image therefore needs to
have the follwing type parameters marked with @miniboxed: type Repr2.
data:image/s3,"s3://crabby-images/d3bf4/d3bf4261168768f7449a97af0ab6a8e2a72959fd" alt=""
This is because we added the @miniboxed
annotation to method map
in Image
:
abstract class Image[Repr: Pixel] {
def width: Int
def height: Int
def apply(x: Int, y: Int): Repr
def map[@miniboxed Repr2: Pixel](f: Image[Repr] => Generator[Repr2]): Image[Repr2]
}
but did not add the annotation to method map
in ImageImpl
, which extends Image
:
private class ImageImpl[Repr: Pixel](val width: Int, val height: Int) extends Image[Repr] {
...
def map[Repr2: Pixel](f: Image[Repr] => Generator[Repr2]): Image[Repr2] = {
...
}
}
Let’s fix that:
private class ImageImpl[Repr: Pixel](val width: Int, val height: Int) extends Image[Repr] {
...
def map[@miniboxed Repr2: Pixel](f: Image[Repr] => Generator[Repr2]): Image[Repr2] = {
...
}
}
If we recompile, the error is gone, but we get another batch of 26 warnings
:
data:image/s3,"s3://crabby-images/acf54/acf5471fca97bc6f74b02f247c1c4d6f4df21937" alt=""
Let’s try to run the program now, despite the warnings:
data:image/s3,"s3://crabby-images/8f151/8f151e4b15bdc819c248f362d987852ab360581c" alt=""
So by adding a couple of annotations, we just shaved off one quarter of the running time:
Operation took 3082 milliseconds.
Operation took 2998 milliseconds.
Operation took 3017 milliseconds.
Operation took 2535 milliseconds.
Operation took 2615 milliseconds.
Maybe this miniboxing thing is actually worth it…
Let’s now address all the warnings. But let’s be smart about it:
- go to
Project
>Properties
(if that’s disabled, make sure theimage-example
project is selected) - in the window go to
Scala Compiler
and add-P:minibox:mark-all
to theAdditional command line parameters
field.
As explained in the command line options, the -P:minibox:mark-all
flag will mark all type parameters with @miniboxed
automatically.
data:image/s3,"s3://crabby-images/50116/50116f1f3237dd601c5a2171e9677ec78ced9e80" alt=""
data:image/s3,"s3://crabby-images/558fe/558fed0e8d7208ea00362f2764d024f545543b88" alt=""
Recompile and you should not see any more warnings. Now let’s run again:
data:image/s3,"s3://crabby-images/0b079/0b079144b68c59c6b3024b70637777ae8693eed6" alt=""
Operation took 1346 milliseconds.
Operation took 1187 milliseconds.
Operation took 1178 milliseconds.
Operation took 1094 milliseconds.
Operation took 1163 milliseconds.
Wow, that’s four times faster than our initial running time. That’s what miniboxing can do for numeric-intensive applications!
Comments
Comments are always welcome! But to make the best use of them, please consider this:
- If you have questions or feedback regarding the content of this page, please leave us a comment!
- If you have general questions about the miniboxing plugin, please ask on the mailing List.
- If you found a bug, please let us know on the github issue tracker.
Thanks! Looking forward to your messages!
comments powered by Disqus