OpenCLNet/examples/OpenCLTest/Mandelbrot.cs

147 lines
5.4 KiB
C#
Raw Normal View History

2015-11-15 23:13:50 +01:00
/*
* Copyright (c) 2009 Olav Kalgraf(olav.kalgraf@gmail.com)
*
* Permission is hereby granted, free of charge, to any person
* obtaining a copy of this software and associated documentation
* files (the "Software"), to deal in the Software without
* restriction, including without limitation the rights to use,
* copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the
* Software is furnished to do so, subject to the following
* conditions:
*
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
* OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
* HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
* WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
* OTHER DEALINGS IN THE SOFTWARE.
*/
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
using System.Diagnostics;
using System.IO;
using System.Drawing.Imaging;
using OpenCLNet;
namespace OpenCLTest
{
public class Mandelbrot
{
public float Left { get; set; }
public float Top { get; set; }
public float Right { get; set; }
public float Bottom { get; set; }
public int BitmapWidth { get; set; }
public int BitmapHeight { get; set; }
public Bitmap Bitmap { get; protected set; }
Platform openCLPlatform;
Device[] openCLDevices;
Context openCLContext;
CommandQueue openCLCQ;
Program mandelBrotProgram;
Kernel mandelbrotKernel;
Mem mandelbrotMemBuffer;
public Mandelbrot( Platform platform, int width, int height )
{
openCLPlatform = platform;
openCLDevices = openCLPlatform.QueryDevices(DeviceType.ALL);
openCLContext = openCLPlatform.CreateDefaultContext();
openCLCQ = openCLContext.CreateCommandQueue(openCLDevices[0], CommandQueueProperties.PROFILING_ENABLE);
mandelBrotProgram = openCLContext.CreateProgramWithSource(File.ReadAllText("Mandelbrot.cl"));
try
{
mandelBrotProgram.Build();
}
catch (OpenCLException ocle)
{
string buildLog = mandelBrotProgram.GetBuildLog(openCLDevices[0]);
MessageBox.Show(buildLog,"Build error(64 bit debug sessions in vs2008 always fail like this - debug in 32 bit)");
Application.Exit();
}
mandelbrotKernel = mandelBrotProgram.CreateKernel("Mandelbrot");
Left = -2.0f;
Top = 2.0f;
Right = 2.0f;
Bottom = -2.0f;
BitmapWidth = width;
BitmapHeight = height;
mandelbrotMemBuffer = openCLContext.CreateBuffer((MemFlags)((long)MemFlags.WRITE_ONLY), width*height*4, IntPtr.Zero);
}
public void AllocBuffers()
{
Bitmap = new Bitmap( BitmapWidth, BitmapHeight, System.Drawing.Imaging.PixelFormat.Format32bppArgb );
}
public void Calculate()
{
BitmapData bd = Bitmap.LockBits(new Rectangle(0, 0, Bitmap.Width, Bitmap.Height), ImageLockMode.WriteOnly, PixelFormat.Format32bppArgb);
int bitmapSize = bd.Width * bd.Height;
mandelbrotKernel.SetArg( 0, Left );
mandelbrotKernel.SetArg( 1, Top );
mandelbrotKernel.SetArg( 2, Right );
mandelbrotKernel.SetArg( 3, Bottom );
mandelbrotKernel.SetArg( 4, BitmapWidth );
mandelbrotKernel.SetArg( 5, mandelbrotMemBuffer );
Event calculationStart;
Event calculationEnd;
openCLCQ.EnqueueMarker(out calculationStart);
IntPtr[] globalWorkSize = new IntPtr[2];
globalWorkSize[0] = (IntPtr)BitmapWidth;
globalWorkSize[1] = (IntPtr)BitmapHeight;
openCLCQ.EnqueueNDRangeKernel(mandelbrotKernel, 2, null, globalWorkSize, null);
for (int y = 0; y < BitmapHeight; y++)
openCLCQ.EnqueueReadBuffer(mandelbrotMemBuffer, true, (IntPtr)(BitmapWidth*4*y), (IntPtr)(BitmapWidth*4), (IntPtr)(bd.Scan0.ToInt64()+y*bd.Stride));
openCLCQ.Finish();
openCLCQ.EnqueueMarker(out calculationEnd);
openCLCQ.Finish();
ulong start = 0;
ulong end = 0;
try
{
calculationStart.GetEventProfilingInfo(ProfilingInfo.QUEUED, out start);
calculationEnd.GetEventProfilingInfo(ProfilingInfo.END, out end);
}
catch (OpenCLException ocle)
{
}
finally
{
CalculationTimeMS = (end - start) / 1000000;
calculationStart.Dispose();
calculationEnd.Dispose();
}
Bitmap.UnlockBits(bd);
}
public ulong CalculationTimeMS;
}
}