Skip to main content

EvoLisa - Comparing two images: performance tuning

It looks like the program is rather slow. Using DotTrace, I took a look at where the bottleneck is.

The method that calculates the 'fitnesse', how closely the created image resembles the original image, is where most of the time goes to. Can we improve that?

So what happens in this function? We compare each pixel of both images to compare their ARGB values. The better they match, the better the result! How can be compare this?

First, I tried using the naive approach: compare each pixel using a bitmaps GetPixel(x, y). Do that however, and you'll quickly see how excruciatingly slow that is. Especially considering that even with an image of 333x333 you're already comparing a million pixels each time you compare two images!

So, instead I tried using a 'FastBitmap'. It's a project I found which wraps the Bitmap and provides much faster Get and SetPixel performance. Using this instead, the performance went up by an order of magnitude!

But we can still do better. Taking a look at the code, it's using the 'LockBits' to get the entire Bitmap in memory. It's then using pointer magic and some basic multiplications to get each pixel. But these multiplications happen for each of the million pixels.

In this case however, we don't have to reinvent the wheel. Someone else already solved this problem for us! Dan Bystrom provided some code to make the fitnesse function even faster. Before, using the FastBitmap each GetPixel would do a multiplication like this:
private void LockBitmap()
{
    GraphicsUnit unit = GraphicsUnit.Pixel;
    RectangleF boundsF = Subject.GetBounds(ref unit);
    Rectangle bounds = new Rectangle(
        (int)boundsF.X,
        (int)boundsF.Y,
        (int)boundsF.Width,
            (int)boundsF.Height);
    SubjectWidth = (int)boundsF.Width * sizeof(PixelData);
    if (SubjectWidth % 4 != 0)
    {
        SubjectWidth = 4 * (SubjectWidth / 4 + 1);
    }
    bitmapData = Subject.LockBits(bounds, ImageLockMode.ReadWrite, PixelFormat.Format24bppRgb);
    pBase = (Byte*)bitmapData.Scan0.ToPointer();
}

private PixelData* PixelAt(int x, int y)
{
    return (PixelData*)(pBase + y * SubjectWidth + x * sizeof(PixelData));
}

Now, we no longer have to use multiplication on each line! The code now looks like this:
public long CalculateFitness(Population pop)
{
    long fitnesse = 0;

    var picture = pop.GetPicture();
    BitmapData bd = picture.LockBits(
        new Rectangle(0, 0, Settings.ScreenWidth, Settings.ScreenHeight),
        ImageLockMode.ReadOnly,
        PixelFormat.Format32bppArgb);
    BitmapData obd = _bitmap.LockBits(
        new Rectangle(0, 0, Settings.ScreenWidth, Settings.ScreenHeight),
        ImageLockMode.ReadOnly,
        PixelFormat.Format32bppArgb);

    unchecked
    {
        unsafe
        {
            //Shamelessly copied from https://danbystrom.se/2008/12/14/improving-performance/
            Pixel* p1 = (Pixel*)bd.Scan0.ToPointer();
            Pixel* p2 = (Pixel*)obd.Scan0.ToPointer();
            var size = Settings.ScreenWidth * Settings.ScreenHeight;
            for (int i = size; i > 0; i--, p1++, p2++)
            {
                int r = p1->R - p2->R;
                int g = p1->G - p2->G;
                int b = p1->B - p2->B;
                fitnesse += r * r + g * g + b * b;
            }                    
        }
    }

    _bitmap.UnlockBits(obd);
    picture.UnlockBits(bd);

    return fitnesse;
}

Usually you don't need this kind of micro optimisation but in this case, because this code is done so many times, the performance gain is enormous! And to finish, here's a 200 polygon image of Mila Kunis:

Comments

Popular posts from this blog

Welcome back

Seeing my World of Warcraft account getting hacked wasn't much fun but it does have one positive side. Since I had to pay €15 to get my deleted items back I decided to log in a few times this month and see what changed. I haven't played WoW since April this year so there's quite a bit of new content to go through. And of course I wanted to say hello again to all the friends I haven't spoken to in months. As I logged in I was quickly greeted by some of my guild members. Time is never standing still so I wasn't surprised to see that my guild looks nothing like it did half a year ago. Roughly half of the people I enjoyed chatting and playing with had all decided to join a guild that was more to their liking. Only the officer team seems to be relatively intact. The player gap has been filled in with lots of new players but it doesn't feel like the same guild. I had a chat with the old officers and my friends from ancient times. In this short time I even got a few...

Circumventing the Steam Regional lockin for Europeans

Thiefsie at rps.com found a nice way to get the steam games in pounds. I tested it and it works! 1) Put ?cc=uk after a title. You'll now see the prices in pound. 2) Change your country to United Kingdom. That's it, you can now buy the game in pounds. Possible to save quite some money this way. Left 4 dead 2 costs €37.49 compared to the us £22.5(~€24.5). And the THQ complete pack costs €49.99 as compared to £26.49 (~€29.2). As always with these things: use at your own risk. It's probably not going to last very long.

EvoLisa - My own version

It's an old idea by now, can we recreate the Mona Lisa with fifty polygons by using random changes? The idea and original implementation by Roger Johansson can be found here . So, how does it work? You start with an source image. Then you create an empty image. We keep on doing small changes to this image. In my case this is one of these changes: Recolor our polygon Change the position of one of the polygon points Add or remove a new point to our polygon Add or remove a new polygon Switch two polygons After each action, we take a look and check if the newly changed image looks more like the original image by comparing each pixel. If it is, we continue using this one. If it's not, we discard the changes. Looks like a fun project! So I built my own version from scratch eight years ago, reusing some of the same ideas. Here's my result using the Mona Lisa: Mona Lisa with 50 polygons: You can see that it's the Mona Lisa but the details around the eyes and mou...