1 module gfm.freeimage.freeimage;
2 
3 import std.conv,
4        std..string;
5 
6 import derelict.freeimage.freeimage,
7        derelict.util.exception;
8 
9 /// The one exception type thrown in this wrapper.
10 /// A failing FreeImage function should <b>always</b> throw an FreeImageException.
11 class FreeImageException : Exception
12 {
13     public
14     {
15         @safe pure nothrow this(string message, string file =__FILE__, size_t line = __LINE__, Throwable next = null)
16         {
17             super(message, file, line, next);
18         }
19     }
20 }
21 
22 /// FreeImage library wrapper.
23 final class FreeImage
24 {
25     public
26     {
27         /// Loads the FreeImage library and logs some information.
28         /// Throws: FreeImageException on error.
29         this(bool useExternalPlugins = false)
30         {
31             try
32             {
33                 DerelictFI.load();
34             }
35             catch(DerelictException e)
36             {
37                 throw new FreeImageException(e.msg);
38             }
39 
40             //FreeImage_Initialise(useExternalPlugins ? TRUE : FALSE); // documentation says it's useless
41             _libInitialized = true;
42         }
43 
44         ~this()
45         {
46             if (_libInitialized)
47             {
48                 debug ensureNotInGC("FreeImage");
49                 //FreeImage_DeInitialise(); // documentation says it's useless
50                 _libInitialized = false;
51             }
52         }
53         deprecated("Use .destroy instead") void close(){}
54 
55         const(char)[] getVersion()
56         {
57             const(char)* versionZ = FreeImage_GetVersion();
58             return fromStringz(versionZ);
59         }
60 
61         const(char)[] getCopyrightMessage()
62         {
63             const(char)* copyrightZ = FreeImage_GetCopyrightMessage();
64             return fromStringz(copyrightZ);
65         }
66     }
67 
68     private
69     {
70         bool _libInitialized;
71     }
72 }
73 
74 /// Crash if the GC is running.
75 /// Useful in destructors to avoid reliance GC resource release.
76 package void ensureNotInGC(string resourceName) nothrow
77 {
78     import core.exception;
79     try
80     {
81         import core.memory;
82         cast(void) GC.malloc(1); // not ideal since it allocates
83         return;
84     }
85     catch(InvalidMemoryOperationError e)
86     {
87 
88         import core.stdc.stdio;
89         fprintf(stderr, "Error: clean-up of %s incorrectly depends on destructors called by the GC.\n", resourceName.ptr);
90         assert(false);
91     }
92 }