mirror of
				https://github.com/Ryujinx/Opentk.git
				synced 2025-10-26 08:18:59 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			712 lines
		
	
	
		
			38 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			712 lines
		
	
	
		
			38 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| #region --- License ---
 | |
| /* Licensed under the MIT/X11 license.
 | |
|  * Copyright (c) 2006-2008 the OpenTK Team.
 | |
|  * This notice may not be removed from any source distribution.
 | |
|  * See license.txt for licensing details.
 | |
|  */
 | |
| #endregion
 | |
| 
 | |
| // #define READALL
 | |
| // uncomment so ALL fields read from file are interpreted and filled. Necessary to implement uncompressed DDS
 | |
| 
 | |
| // TODO: Find app that can build compressed dds cubemaps and verify that the import code works.
 | |
| 
 | |
| using System;
 | |
| using System.IO;
 | |
| using System.Diagnostics;
 | |
| 
 | |
| using OpenTK;
 | |
| using OpenTK.Graphics;
 | |
| using OpenTK.Graphics.OpenGL;
 | |
| 
 | |
| namespace Examples.TextureLoaders
 | |
| {
 | |
|     /// <summary> 
 | |
|     /// Expects the presence of a valid OpenGL Context and Texture Compression Extensions (GL 1.5) and Cube Maps (GL 1.3).
 | |
|     /// You will get what you give. No automatic Mipmap generation or automatic compression is done. (both bad quality)
 | |
|     /// Textures are never rescaled or checked if Power of 2, but you should make the Width and Height a multiple of 4 because DXTn uses 4x4 blocks.
 | |
|     /// (Image displays correctly but runs extremely slow with non-power-of-two Textures on FX5600, Cache misses?)
 | |
|     /// CubeMap support is experimental and the file must specify all 6 faces to work at all.
 | |
|     /// </summary>
 | |
|     static class ImageDDS
 | |
|     {
 | |
|         #region Constants
 | |
|         private const byte HeaderSizeInBytes = 128; // all non-image data together is 128 Bytes
 | |
|         private const uint BitMask = 0x00000007; // bits = 00 00 01 11
 | |
|        
 | |
| 
 | |
|         private static NotImplementedException Unfinished = new NotImplementedException( "ERROR: Only 2 Dimensional DXT1/3/5 compressed images for now. 1D/3D Textures may not be compressed according to spec." );
 | |
|         #endregion Constants
 | |
| 
 | |
|         #region Simplified In-Memory representation of the Image
 | |
|         private static bool _IsCompressed;
 | |
|         private static int _Width, _Height, _Depth, _MipMapCount;
 | |
|         private static int _BytesForMainSurface; // must be handled with care when implementing uncompressed formats!
 | |
|         private static byte _BytesPerBlock;
 | |
|         private static PixelInternalFormat _PixelInternalFormat;
 | |
|         #endregion Simplified In-Memory representation of the Image
 | |
| 
 | |
|         #region Flag Enums
 | |
|         [Flags] // Surface Description
 | |
|         private enum eDDSD: uint
 | |
|         {
 | |
|             CAPS = 0x00000001, // is always present
 | |
|             HEIGHT = 0x00000002, // is always present
 | |
|             WIDTH = 0x00000004, // is always present
 | |
|             PITCH = 0x00000008, // is set if the image is uncompressed
 | |
|             PIXELFORMAT = 0x00001000, // is always present
 | |
|             MIPMAPCOUNT = 0x00020000, // is set if the image contains MipMaps
 | |
|             LINEARSIZE = 0x00080000, // is set if the image is compressed
 | |
|             DEPTH = 0x00800000 // is set for 3D Volume Textures
 | |
|         }
 | |
| 
 | |
|         [Flags] // Pixelformat 
 | |
|         private enum eDDPF: uint
 | |
|         {
 | |
|             NONE = 0x00000000, // not part of DX, added for convenience
 | |
|             ALPHAPIXELS = 0x00000001,
 | |
|             FOURCC = 0x00000004,
 | |
|             RGB = 0x00000040,
 | |
|             RGBA = 0x00000041
 | |
|         }
 | |
| 
 | |
|         /// <summary>This list was derived from nVidia OpenGL SDK</summary>
 | |
|         [Flags] // Texture types
 | |
|         private enum eFOURCC: uint
 | |
|         {
 | |
|             UNKNOWN = 0,
 | |
| #if READALL
 | |
|             R8G8B8 = 20,
 | |
|             A8R8G8B8 = 21,
 | |
|             X8R8G8B8 = 22,
 | |
|             R5G6B5 = 23,
 | |
|             X1R5G5B5 = 24,
 | |
|             A1R5G5B5 = 25,
 | |
|             A4R4G4B4 = 26,
 | |
|             R3G3B2 = 27,
 | |
|             A8 = 28,
 | |
|             A8R3G3B2 = 29,
 | |
|             X4R4G4B4 = 30,
 | |
|             A2B10G10R10 = 31,
 | |
|             A8B8G8R8 = 32,
 | |
|             X8B8G8R8 = 33,
 | |
|             G16R16 = 34,
 | |
|             A2R10G10B10 = 35,
 | |
|             A16B16G16R16 = 36,
 | |
| 
 | |
|             L8 = 50,
 | |
|             A8L8 = 51,
 | |
|             A4L4 = 52,
 | |
| 
 | |
|             D16_LOCKABLE = 70,
 | |
|             D32 = 71,
 | |
|             D24X8 = 77,
 | |
|             D16 = 80,
 | |
| 
 | |
|             D32F_LOCKABLE = 82,
 | |
|             L16 = 81,
 | |
| 
 | |
|             // s10e5 formats (16-bits per channel)
 | |
|             R16F = 111,
 | |
|             G16R16F = 112,
 | |
|             A16B16G16R16F = 113,
 | |
| 
 | |
|             // IEEE s23e8 formats (32-bits per channel)
 | |
|             R32F = 114,
 | |
|             G32R32F = 115,
 | |
|             A32B32G32R32F = 116
 | |
| #endif
 | |
|             DXT1 = 0x31545844,
 | |
|             DXT2 = 0x32545844,
 | |
|             DXT3 = 0x33545844,
 | |
|             DXT4 = 0x34545844,
 | |
|             DXT5 = 0x35545844,
 | |
|         }
 | |
| 
 | |
|         [Flags] // dwCaps1
 | |
|         private enum eDDSCAPS: uint
 | |
|         {
 | |
|             NONE = 0x00000000, // not part of DX, added for convenience
 | |
|             COMPLEX = 0x00000008, // should be set for any DDS file with more than one main surface
 | |
|             TEXTURE = 0x00001000, // should always be set
 | |
|             MIPMAP = 0x00400000 // only for files with MipMaps
 | |
|         }
 | |
| 
 | |
|         [Flags]  // dwCaps2
 | |
|         private enum eDDSCAPS2: uint
 | |
|         {
 | |
|             NONE = 0x00000000, // not part of DX, added for convenience
 | |
|             CUBEMAP = 0x00000200,
 | |
|             CUBEMAP_POSITIVEX = 0x00000400,
 | |
|             CUBEMAP_NEGATIVEX = 0x00000800,
 | |
|             CUBEMAP_POSITIVEY = 0x00001000,
 | |
|             CUBEMAP_NEGATIVEY = 0x00002000,
 | |
|             CUBEMAP_POSITIVEZ = 0x00004000,
 | |
|             CUBEMAP_NEGATIVEZ = 0x00008000,
 | |
|             CUBEMAP_ALL_FACES = 0x0000FC00,
 | |
|             VOLUME = 0x00200000 // for 3D Textures
 | |
|         }
 | |
|         #endregion Flag Enums
 | |
| 
 | |
|         #region Private Members
 | |
|         private static string idString; // 4 bytes, must be "DDS "
 | |
|         private static UInt32 dwSize; // Size of structure is 124 bytes, 128 including all sub-structs and the header
 | |
|         private static UInt32 dwFlags; // Flags to indicate valid fields.
 | |
|         private static UInt32 dwHeight; // Height of the main image in pixels
 | |
|         private static UInt32 dwWidth; // Width of the main image in pixels
 | |
|         private static UInt32 dwPitchOrLinearSize; // For compressed formats, this is the total number of bytes for the main image.
 | |
|         private static UInt32 dwDepth; // For volume textures, this is the depth of the volume.
 | |
|         private static UInt32 dwMipMapCount; // total number of levels in the mipmap chain of the main image.
 | |
| #if READALL
 | |
|         private static UInt32[] dwReserved1; // 11 UInt32s
 | |
| #endif
 | |
|         // Pixelformat sub-struct, 32 bytes
 | |
|         private static UInt32 pfSize; // Size of Pixelformat structure. This member must be set to 32.
 | |
|         private static UInt32 pfFlags; // Flags to indicate valid fields.
 | |
|         private static UInt32 pfFourCC; // This is the four-character code for compressed formats.
 | |
| #if READALL
 | |
|         private static UInt32 pfRGBBitCount; // For RGB formats, this is the total number of bits in the format. dwFlags should include DDpf_RGB in this case. This value is usually 16, 24, or 32. For A8R8G8B8, this value would be 32.
 | |
|         private static UInt32 pfRBitMask; // For RGB formats, these three fields contain the masks for the red, green, and blue channels. For A8R8G8B8, these values would be 0x00ff0000, 0x0000ff00, and 0x000000ff respectively.
 | |
|         private static UInt32 pfGBitMask; // ..
 | |
|         private static UInt32 pfBBitMask; // ..
 | |
|         private static UInt32 pfABitMask; // For RGB formats, this contains the mask for the alpha channel, if any. dwFlags should include DDpf_ALPHAPIXELS in this case. For A8R8G8B8, this value would be 0xff000000.
 | |
| #endif
 | |
|         // Capabilities sub-struct, 16 bytes
 | |
|         private static UInt32 dwCaps1; // always includes DDSCAPS_TEXTURE. with more than one main surface DDSCAPS_COMPLEX should also be set.
 | |
|         private static UInt32 dwCaps2; // For cubic environment maps, DDSCAPS2_CUBEMAP should be included as well as one or more faces of the map (DDSCAPS2_CUBEMAP_POSITIVEX, DDSCAPS2_CUBEMAP_NEGATIVEX, DDSCAPS2_CUBEMAP_POSITIVEY, DDSCAPS2_CUBEMAP_NEGATIVEY, DDSCAPS2_CUBEMAP_POSITIVEZ, DDSCAPS2_CUBEMAP_NEGATIVEZ). For volume textures, DDSCAPS2_VOLUME should be included.
 | |
| #if READALL
 | |
|         private static UInt32[] dwReserved2; // 3 = 2 + 1 UInt32
 | |
| #endif
 | |
|         #endregion Private Members
 | |
| 
 | |
|         /// <summary>
 | |
|         /// This function will generate, bind and fill a Texture Object with a DXT1/3/5 compressed Texture in .dds Format.
 | |
|         /// MipMaps below 4x4 Pixel Size are discarded, because DXTn's smallest unit is a 4x4 block of Pixel data.
 | |
|         /// It will set correct MipMap parameters, Filtering, Wrapping and EnvMode for the Texture. 
 | |
|         /// The only call inside this function affecting OpenGL State is GL.BindTexture();
 | |
|         /// </summary>
 | |
|         /// <param name="filename">The name of the file you wish to load, including path and file extension.</param>
 | |
|         /// <param name="texturehandle">0 if invalid, otherwise a Texture Object usable with GL.BindTexture().</param>
 | |
|         /// <param name="dimension">0 if invalid, will output what was loaded (typically Texture1D/2D/3D or Cubemap)</param>
 | |
|         public static void LoadFromDisk( string filename, out uint texturehandle, out TextureTarget dimension )
 | |
|         {
 | |
|             #region Prep data
 | |
|             // invalidate whatever it was before
 | |
|             dimension = (TextureTarget) 0;
 | |
|             texturehandle = TextureLoaderParameters.OpenGLDefaultTexture;
 | |
|             ErrorCode GLError = ErrorCode.NoError;
 | |
| 
 | |
|             _IsCompressed = false;
 | |
|             _Width = 0;
 | |
|             _Height = 0;
 | |
|             _Depth = 0;
 | |
|             _MipMapCount = 0;
 | |
|             _BytesForMainSurface = 0;
 | |
|             _BytesPerBlock = 0;
 | |
|             _PixelInternalFormat = PixelInternalFormat.Rgba8;
 | |
|             byte[] _RawDataFromFile;
 | |
|             #endregion
 | |
| 
 | |
|             #region Try
 | |
|             try // Exceptions will be thrown if any Problem occurs while working on the file. 
 | |
|             {
 | |
|                 _RawDataFromFile = File.ReadAllBytes( @filename );
 | |
| 
 | |
|                 #region Translate Header to less cryptic representation
 | |
|                 ConvertDX9Header( ref _RawDataFromFile ); // The first 128 Bytes of the file is non-image data
 | |
| 
 | |
|                 // start by checking if all forced flags are present. Flags indicate valid fields, but aren't written by every tool .....
 | |
|                 if ( idString != "DDS " || // magic key
 | |
|                      dwSize != 124 || // constant size of struct, never reused
 | |
|                      pfSize != 32 || // constant size of struct, never reused
 | |
|                      !CheckFlag( dwFlags, (uint) eDDSD.CAPS ) ||        // must know it's caps
 | |
|                      !CheckFlag( dwFlags, (uint) eDDSD.PIXELFORMAT ) || // must know it's format
 | |
|                      !CheckFlag( dwCaps1, (uint) eDDSCAPS.TEXTURE )     // must be a Texture
 | |
|                     )
 | |
|                     throw new ArgumentException( "ERROR: File has invalid signature or missing Flags." );
 | |
| 
 | |
|                 #region Examine Flags
 | |
|                 if ( CheckFlag( dwFlags, (uint) eDDSD.WIDTH ) )
 | |
|                     _Width = (int) dwWidth;
 | |
|                 else
 | |
|                     throw new ArgumentException( "ERROR: Flag for Width not set." );
 | |
| 
 | |
|                 if ( CheckFlag( dwFlags, (uint) eDDSD.HEIGHT ) )
 | |
|                     _Height = (int) dwHeight;
 | |
|                 else
 | |
|                     throw new ArgumentException( "ERROR: Flag for Height not set." );
 | |
| 
 | |
|                 if ( CheckFlag( dwFlags, (uint) eDDSD.DEPTH ) && CheckFlag( dwCaps2, (uint) eDDSCAPS2.VOLUME ) )
 | |
|                 {
 | |
|                     dimension = TextureTarget.Texture3D; // image is 3D Volume
 | |
|                     _Depth = (int) dwDepth;
 | |
|                     throw Unfinished;
 | |
|                 } else
 | |
|                 {// image is 2D or Cube
 | |
|                     if ( CheckFlag( dwCaps2, (uint) eDDSCAPS2.CUBEMAP ) )
 | |
|                     {
 | |
|                         dimension = TextureTarget.TextureCubeMap;
 | |
|                         _Depth = 6;
 | |
|                     } else
 | |
|                     {
 | |
|                         dimension = TextureTarget.Texture2D;
 | |
|                         _Depth = 1;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 // these flags must be set for mipmaps to be included
 | |
|                 if ( CheckFlag( dwCaps1, (uint) eDDSCAPS.MIPMAP ) && CheckFlag( dwFlags, (uint) eDDSD.MIPMAPCOUNT ) )
 | |
|                     _MipMapCount = (int) dwMipMapCount; // image contains MipMaps
 | |
|                 else
 | |
|                     _MipMapCount = 1; // only 1 main image
 | |
| 
 | |
|                 // Should never happen
 | |
|                 if ( CheckFlag( dwFlags, (uint) eDDSD.PITCH ) && CheckFlag( dwFlags, (uint) eDDSD.LINEARSIZE ) )
 | |
|                     throw new ArgumentException( "INVALID: Pitch AND Linear Flags both set. Image cannot be uncompressed and DTXn compressed at the same time." );
 | |
| 
 | |
|                 // This flag is set if format is uncompressed RGB RGBA etc.
 | |
|                 if ( CheckFlag( dwFlags, (uint) eDDSD.PITCH ) )
 | |
|                 {
 | |
|                     // _BytesForMainSurface = (int) dwPitchOrLinearSize; // holds bytes-per-scanline for uncompressed
 | |
|                     _IsCompressed = false;
 | |
|                     throw Unfinished;
 | |
|                 }
 | |
| 
 | |
|                 // This flag is set if format is compressed DXTn.
 | |
|                 if ( CheckFlag( dwFlags, (uint) eDDSD.LINEARSIZE ) )
 | |
|                 {
 | |
|                     _BytesForMainSurface = (int) dwPitchOrLinearSize;
 | |
|                     _IsCompressed = true;
 | |
|                 }
 | |
|                 #endregion Examine Flags
 | |
| 
 | |
|                 #region Examine Pixel Format, anything but DXTn will fail atm.
 | |
|                 if ( CheckFlag( pfFlags, (uint) eDDPF.FOURCC ) )
 | |
|                     switch ( (eFOURCC) pfFourCC )
 | |
|                     {
 | |
|                     case eFOURCC.DXT1:
 | |
|                         _PixelInternalFormat = (PixelInternalFormat) ExtTextureCompressionS3tc.CompressedRgbS3tcDxt1Ext;
 | |
|                         _BytesPerBlock = 8;
 | |
|                         _IsCompressed = true;
 | |
|                         break;
 | |
|                     //case eFOURCC.DXT2:
 | |
|                     case eFOURCC.DXT3:
 | |
|                         _PixelInternalFormat = (PixelInternalFormat) ExtTextureCompressionS3tc.CompressedRgbaS3tcDxt3Ext;
 | |
|                         _BytesPerBlock = 16;
 | |
|                         _IsCompressed = true;
 | |
|                         break;
 | |
|                     //case eFOURCC.DXT4:
 | |
|                     case eFOURCC.DXT5:
 | |
|                         _PixelInternalFormat = (PixelInternalFormat) ExtTextureCompressionS3tc.CompressedRgbaS3tcDxt5Ext;
 | |
|                         _BytesPerBlock = 16;
 | |
|                         _IsCompressed = true;
 | |
|                         break;
 | |
|                     default:
 | |
|                         throw Unfinished; // handle uncompressed formats 
 | |
|                     } else
 | |
|                     throw Unfinished;
 | |
|                 // pf*Bitmasks should be examined here
 | |
|                 #endregion
 | |
| 
 | |
|                 // Works, but commented out because some texture authoring tools don't set this flag.
 | |
|                 /* Safety Check, if file is only 1x 2D surface without mipmaps, eDDSCAPS.COMPLEX should not be set
 | |
|                 if ( CheckFlag( dwCaps1, (uint) eDDSCAPS.COMPLEX ) )
 | |
|                 {
 | |
|                     if ( result == eTextureDimension.Texture2D && _MipMapCount == 1 ) // catch potential problem
 | |
|                         Trace.WriteLine( "Warning: Image is declared complex, but contains only 1 surface." );
 | |
|                 }*/
 | |
| 
 | |
|                 if ( TextureLoaderParameters.Verbose )
 | |
|                     Trace.WriteLine( "\n" + GetDescriptionFromMemory( filename, dimension ) );
 | |
|                 #endregion Translate Header to less cryptic representation
 | |
| 
 | |
|                 #region send the Texture to GL
 | |
|                 #region Generate and Bind Handle
 | |
|                 GL.GenTextures( 1, out texturehandle );
 | |
|                 GL.BindTexture( dimension, texturehandle );
 | |
|                 #endregion Generate and Bind Handle
 | |
| 
 | |
|                 int Cursor = HeaderSizeInBytes;
 | |
|                 // foreach face in the cubemap, get all it's mipmaps levels. Only one iteration for Texture2D
 | |
|                 for ( int Slices = 0 ; Slices < _Depth ; Slices++ )
 | |
|                 {
 | |
|                     int trueMipMapCount = _MipMapCount - 1; // TODO: triplecheck correctness
 | |
|                     int Width = _Width;
 | |
|                     int Height = _Height;
 | |
|                     for ( int Level = 0 ; Level < _MipMapCount ; Level++ ) // start at base image
 | |
|                     {
 | |
|                         #region determine Dimensions
 | |
|                         int BlocksPerRow = ( Width + 3 ) >> 2;
 | |
|                         int BlocksPerColumn = ( Height + 3 ) >> 2;
 | |
|                         int SurfaceBlockCount = BlocksPerRow * BlocksPerColumn; //   // DXTn stores Texels in 4x4 blocks, a Color block is 8 Bytes, an Alpha block is 8 Bytes for DXT3/5
 | |
|                         int SurfaceSizeInBytes = SurfaceBlockCount * _BytesPerBlock;
 | |
| 
 | |
|                         // this check must evaluate to false for 2D and Cube maps, or it's impossible to determine MipMap sizes.
 | |
|                         if ( TextureLoaderParameters.Verbose && Level == 0 && _IsCompressed && _BytesForMainSurface != SurfaceSizeInBytes )
 | |
|                             Trace.WriteLine( "Warning: Calculated byte-count of main image differs from what was read from file." );
 | |
|                         #endregion determine Dimensions
 | |
| 
 | |
|                         // skip mipmaps smaller than a 4x4 Pixels block, which is the smallest DXTn unit.
 | |
|                         if ( Width > 2 && Height > 2 )
 | |
|                         { // Note: there could be a potential problem with non-power-of-two cube maps
 | |
|                             #region Prepare Array for TexImage
 | |
|                             byte[] RawDataOfSurface = new byte[SurfaceSizeInBytes];
 | |
|                             if ( !TextureLoaderParameters.FlipImages )
 | |
|                             { // no changes to the image, copy as is
 | |
|                                 Array.Copy( _RawDataFromFile, Cursor, RawDataOfSurface, 0, SurfaceSizeInBytes );
 | |
|                             } else
 | |
|                             {  // Turn the blocks upside down and the rows aswell, done in a single pass through all blocks
 | |
|                                 for ( int sourceColumn = 0 ; sourceColumn < BlocksPerColumn ; sourceColumn++ )
 | |
|                                 {
 | |
|                                     int targetColumn = BlocksPerColumn - sourceColumn - 1;
 | |
|                                     for ( int row = 0 ; row < BlocksPerRow ; row++ )
 | |
|                                     {
 | |
|                                         int target = ( targetColumn * BlocksPerRow + row ) * _BytesPerBlock;
 | |
|                                         int source = ( sourceColumn * BlocksPerRow + row ) * _BytesPerBlock + Cursor;
 | |
|                                         #region Swap Bytes
 | |
|                                         switch ( _PixelInternalFormat )
 | |
|                                         {
 | |
|                                         case (PixelInternalFormat) ExtTextureCompressionS3tc.CompressedRgbS3tcDxt1Ext:
 | |
|                                             // Color only
 | |
|                                             RawDataOfSurface[target + 0] = _RawDataFromFile[source + 0];
 | |
|                                             RawDataOfSurface[target + 1] = _RawDataFromFile[source + 1];
 | |
|                                             RawDataOfSurface[target + 2] = _RawDataFromFile[source + 2];
 | |
|                                             RawDataOfSurface[target + 3] = _RawDataFromFile[source + 3];
 | |
|                                             RawDataOfSurface[target + 4] = _RawDataFromFile[source + 7];
 | |
|                                             RawDataOfSurface[target + 5] = _RawDataFromFile[source + 6];
 | |
|                                             RawDataOfSurface[target + 6] = _RawDataFromFile[source + 5];
 | |
|                                             RawDataOfSurface[target + 7] = _RawDataFromFile[source + 4];
 | |
|                                             break;
 | |
|                                         case (PixelInternalFormat) ExtTextureCompressionS3tc.CompressedRgbaS3tcDxt3Ext:
 | |
|                                             // Alpha
 | |
|                                             RawDataOfSurface[target + 0] = _RawDataFromFile[source + 6];
 | |
|                                             RawDataOfSurface[target + 1] = _RawDataFromFile[source + 7];
 | |
|                                             RawDataOfSurface[target + 2] = _RawDataFromFile[source + 4];
 | |
|                                             RawDataOfSurface[target + 3] = _RawDataFromFile[source + 5];
 | |
|                                             RawDataOfSurface[target + 4] = _RawDataFromFile[source + 2];
 | |
|                                             RawDataOfSurface[target + 5] = _RawDataFromFile[source + 3];
 | |
|                                             RawDataOfSurface[target + 6] = _RawDataFromFile[source + 0];
 | |
|                                             RawDataOfSurface[target + 7] = _RawDataFromFile[source + 1];
 | |
| 
 | |
|                                             // Color
 | |
|                                             RawDataOfSurface[target + 8] = _RawDataFromFile[source + 8];
 | |
|                                             RawDataOfSurface[target + 9] = _RawDataFromFile[source + 9];
 | |
|                                             RawDataOfSurface[target + 10] = _RawDataFromFile[source + 10];
 | |
|                                             RawDataOfSurface[target + 11] = _RawDataFromFile[source + 11];
 | |
|                                             RawDataOfSurface[target + 12] = _RawDataFromFile[source + 15];
 | |
|                                             RawDataOfSurface[target + 13] = _RawDataFromFile[source + 14];
 | |
|                                             RawDataOfSurface[target + 14] = _RawDataFromFile[source + 13];
 | |
|                                             RawDataOfSurface[target + 15] = _RawDataFromFile[source + 12];
 | |
|                                             break;
 | |
|                                         case (PixelInternalFormat) ExtTextureCompressionS3tc.CompressedRgbaS3tcDxt5Ext:
 | |
|                                             // Alpha, the first 2 bytes remain 
 | |
|                                             RawDataOfSurface[target + 0] = _RawDataFromFile[source + 0];
 | |
|                                             RawDataOfSurface[target + 1] = _RawDataFromFile[source + 1];
 | |
| 
 | |
|                                             // extract 3 bits each and flip them
 | |
|                                             GetBytesFromUInt24( ref RawDataOfSurface, (uint) target + 5, FlipUInt24( GetUInt24( ref _RawDataFromFile, (uint) source + 2 ) ) );
 | |
|                                             GetBytesFromUInt24( ref RawDataOfSurface, (uint) target + 2, FlipUInt24( GetUInt24( ref _RawDataFromFile, (uint) source + 5 ) ) );
 | |
| 
 | |
|                                             // Color
 | |
|                                             RawDataOfSurface[target + 8] = _RawDataFromFile[source + 8];
 | |
|                                             RawDataOfSurface[target + 9] = _RawDataFromFile[source + 9];
 | |
|                                             RawDataOfSurface[target + 10] = _RawDataFromFile[source + 10];
 | |
|                                             RawDataOfSurface[target + 11] = _RawDataFromFile[source + 11];
 | |
|                                             RawDataOfSurface[target + 12] = _RawDataFromFile[source + 15];
 | |
|                                             RawDataOfSurface[target + 13] = _RawDataFromFile[source + 14];
 | |
|                                             RawDataOfSurface[target + 14] = _RawDataFromFile[source + 13];
 | |
|                                             RawDataOfSurface[target + 15] = _RawDataFromFile[source + 12];
 | |
|                                             break;
 | |
|                                         default:
 | |
|                                             throw new ArgumentException( "ERROR: Should have never arrived here! Bad _PixelInternalFormat! Should have been dealt with much earlier." );
 | |
|                                         }
 | |
|                                         #endregion Swap Bytes
 | |
|                                     }
 | |
|                                 }
 | |
|                             }
 | |
|                             #endregion Prepare Array for TexImage
 | |
| 
 | |
|                             #region Create TexImage
 | |
|                             switch ( dimension )
 | |
|                             {
 | |
|                             case TextureTarget.Texture2D:
 | |
|                                 GL.CompressedTexImage2D( TextureTarget.Texture2D,
 | |
|                                                          Level,
 | |
|                                                          _PixelInternalFormat,
 | |
|                                                          Width,
 | |
|                                                          Height,
 | |
|                                                         TextureLoaderParameters.Border,
 | |
|                                                          SurfaceSizeInBytes,
 | |
|                                                          RawDataOfSurface );
 | |
|                                 break;
 | |
|                             case TextureTarget.TextureCubeMap:
 | |
|                                 GL.CompressedTexImage2D( TextureTarget.TextureCubeMapPositiveX + Slices,
 | |
|                                                          Level,
 | |
|                                                          _PixelInternalFormat,
 | |
|                                                          Width,
 | |
|                                                          Height,
 | |
|                                                          TextureLoaderParameters.Border,
 | |
|                                                          SurfaceSizeInBytes,
 | |
|                                                          RawDataOfSurface );
 | |
|                                 break;
 | |
|                             case TextureTarget.Texture1D: // Untested
 | |
|                             case TextureTarget.Texture3D: // Untested
 | |
|                             default:
 | |
|                                 throw new ArgumentException( "ERROR: Use DXT for 2D Images only. Cannot evaluate " + dimension );
 | |
|                             }
 | |
|                             GL.Finish( );
 | |
|                             #endregion Create TexImage
 | |
| 
 | |
|                             #region Query Success
 | |
|                             int width, height, internalformat, compressed;
 | |
|                             switch ( dimension )
 | |
|                             {
 | |
|                             case TextureTarget.Texture1D:
 | |
|                             case TextureTarget.Texture2D:
 | |
|                             case TextureTarget.Texture3D:
 | |
|                                 GL.GetTexLevelParameter( dimension, Level, GetTextureParameter.TextureWidth, out width );
 | |
|                                 GL.GetTexLevelParameter( dimension, Level, GetTextureParameter.TextureHeight, out height );
 | |
|                                 GL.GetTexLevelParameter( dimension, Level, GetTextureParameter.TextureInternalFormat, out internalformat );
 | |
|                                 GL.GetTexLevelParameter( dimension, Level, GetTextureParameter.TextureCompressed, out compressed );
 | |
|                                 break;
 | |
|                             case TextureTarget.TextureCubeMap:
 | |
|                                 GL.GetTexLevelParameter( TextureTarget.TextureCubeMapPositiveX + Slices, Level, GetTextureParameter.TextureWidth, out width );
 | |
|                                 GL.GetTexLevelParameter( TextureTarget.TextureCubeMapPositiveX + Slices, Level, GetTextureParameter.TextureHeight, out height );
 | |
|                                 GL.GetTexLevelParameter( TextureTarget.TextureCubeMapPositiveX + Slices, Level, GetTextureParameter.TextureInternalFormat, out internalformat );
 | |
|                                 GL.GetTexLevelParameter( TextureTarget.TextureCubeMapPositiveX + Slices, Level, GetTextureParameter.TextureCompressed, out compressed );
 | |
|                                 break;
 | |
|                             default:
 | |
|                                 throw Unfinished;
 | |
|                             }
 | |
|                             GLError = GL.GetError( );
 | |
|                             if ( TextureLoaderParameters.Verbose )
 | |
|                                 Trace.WriteLine( "GL: " + GLError.ToString( ) + " Level: " + Level + " DXTn: " + ( ( compressed == 1 ) ? "Yes" : "No" ) + " Frmt:" + (ExtTextureCompressionS3tc) internalformat + " " + width + "*" + height );
 | |
|                             if ( GLError != ErrorCode.NoError || compressed == 0 || width == 0 || height == 0 || internalformat == 0 )
 | |
|                             {
 | |
|                                 GL.DeleteTextures( 1, ref texturehandle );
 | |
|                                 throw new ArgumentException( "ERROR: Something went wrong after GL.CompressedTexImage(); Last GL Error: " + GLError.ToString( ) );
 | |
|                             }
 | |
|                             #endregion Query Success
 | |
|                         } else
 | |
|                         {
 | |
|                             if ( trueMipMapCount > Level )
 | |
|                                 trueMipMapCount = Level - 1; // The current Level is invalid
 | |
|                         }
 | |
| 
 | |
|                         #region Prepare the next MipMap level
 | |
|                         Width /= 2;
 | |
|                         if ( Width < 1 )
 | |
|                             Width = 1;
 | |
|                         Height /= 2;
 | |
|                         if ( Height < 1 )
 | |
|                             Height = 1;
 | |
|                         Cursor += SurfaceSizeInBytes;
 | |
|                         #endregion Prepare the next MipMap level
 | |
|                     }
 | |
| 
 | |
|                     #region Set States properly
 | |
|                     GL.TexParameter( dimension, (TextureParameterName) All.TextureBaseLevel, 0 );
 | |
|                     GL.TexParameter( dimension, (TextureParameterName) All.TextureMaxLevel, trueMipMapCount );
 | |
| 
 | |
|                     int TexMaxLevel;
 | |
|                     GL.GetTexParameter( dimension, GetTextureParameter.TextureMaxLevel, out TexMaxLevel );
 | |
| 
 | |
|                     if ( TextureLoaderParameters.Verbose )
 | |
|                         Trace.WriteLine( "Verification: GL: " + GL.GetError( ).ToString( ) + " TextureMaxLevel: " + TexMaxLevel + ( ( TexMaxLevel == trueMipMapCount ) ? " (Correct.)" : " (Wrong!)" ) );
 | |
|                     #endregion Set States properly
 | |
|                 }
 | |
| 
 | |
|                 #region Set Texture Parameters
 | |
|                 GL.TexParameter( dimension, TextureParameterName.TextureMinFilter, (int) TextureLoaderParameters.MinificationFilter );
 | |
|                 GL.TexParameter( dimension, TextureParameterName.TextureMagFilter, (int) TextureLoaderParameters.MagnificationFilter );
 | |
| 
 | |
|                 GL.TexParameter( dimension, TextureParameterName.TextureWrapS, (int) TextureLoaderParameters.WrapModeS );
 | |
|                 GL.TexParameter( dimension, TextureParameterName.TextureWrapT, (int) TextureLoaderParameters.WrapModeT );
 | |
| 
 | |
|                 GL.TexEnv( TextureEnvTarget.TextureEnv, TextureEnvParameter.TextureEnvMode, (int) TextureLoaderParameters.EnvMode );
 | |
| 
 | |
|                 GLError = GL.GetError( );
 | |
|                 if ( GLError != ErrorCode.NoError )
 | |
|                 {
 | |
|                     throw new ArgumentException( "Error setting Texture Parameters. GL Error: " + GLError );
 | |
|                 }
 | |
|                 #endregion Set Texture Parameters
 | |
| 
 | |
|                 // If it made it here without throwing any Exception the result is a valid Texture.
 | |
|                 return; // success
 | |
|                 #endregion send the Texture to GL
 | |
|             } catch ( Exception e )
 | |
|             {
 | |
|                 dimension = (TextureTarget) 0;
 | |
|                 texturehandle = TextureLoaderParameters.OpenGLDefaultTexture;
 | |
|                 throw new ArgumentException( "ERROR: Exception caught when attempting to load file " + filename + ".\n" + e + "\n" + GetDescriptionFromFile( filename ) );
 | |
|                 // return; // failure
 | |
|             } finally
 | |
|             {
 | |
|                 _RawDataFromFile = null; // clarity, not really needed
 | |
|             }
 | |
|             #endregion Try
 | |
|         }
 | |
| 
 | |
|         #region Helpers
 | |
|         private static void ConvertDX9Header( ref byte[] input )
 | |
|         {
 | |
|             UInt32 offset = 0;
 | |
|             idString = GetString( ref input, offset );
 | |
|             offset += 4;
 | |
|             dwSize = GetUInt32( ref input, offset );
 | |
|             offset += 4;
 | |
|             dwFlags = GetUInt32( ref input, offset );
 | |
|             offset += 4;
 | |
|             dwHeight = GetUInt32( ref input, offset );
 | |
|             offset += 4;
 | |
|             dwWidth = GetUInt32( ref input, offset );
 | |
|             offset += 4;
 | |
|             dwPitchOrLinearSize = GetUInt32( ref input, offset );
 | |
|             offset += 4;
 | |
|             dwDepth = GetUInt32( ref input, offset );
 | |
|             offset += 4;
 | |
|             dwMipMapCount = GetUInt32( ref input, offset );
 | |
|             offset += 4;
 | |
| #if READALL
 | |
|             dwReserved1 = new UInt32[11]; // reserved
 | |
| #endif
 | |
|             offset += 4 * 11;
 | |
|             pfSize = GetUInt32( ref input, offset );
 | |
|             offset += 4;
 | |
|             pfFlags = GetUInt32( ref input, offset );
 | |
|             offset += 4;
 | |
|             pfFourCC = GetUInt32( ref input, offset );
 | |
|             offset += 4;
 | |
| #if READALL
 | |
|             pfRGBBitCount = GetUInt32( ref input, offset );
 | |
|             offset += 4;
 | |
|             pfRBitMask = GetUInt32( ref input, offset );
 | |
|             offset += 4;
 | |
|             pfGBitMask = GetUInt32( ref input, offset );
 | |
|             offset += 4;
 | |
|             pfBBitMask = GetUInt32( ref input, offset );
 | |
|             offset += 4;
 | |
|             pfABitMask = GetUInt32( ref input, offset );
 | |
|             offset += 4;
 | |
| #else
 | |
|             offset += 20;
 | |
| #endif
 | |
|             dwCaps1 = GetUInt32( ref input, offset );
 | |
|             offset += 4;
 | |
|             dwCaps2 = GetUInt32( ref input, offset );
 | |
|             offset += 4;
 | |
| #if READALL
 | |
|             dwReserved2 = new UInt32[3]; // offset is 4+112 here, + 12 = 4+124 
 | |
| #endif
 | |
|             offset += 4 * 3;
 | |
|         }
 | |
| 
 | |
|         /// <summary> Returns true if the flag is set, false otherwise</summary>
 | |
|         private static bool CheckFlag( uint variable, uint flag )
 | |
|         {
 | |
|             return ( variable & flag ) > 0 ? true : false;
 | |
|         }
 | |
| 
 | |
|         private static string GetString( ref byte[] input, uint offset )
 | |
|         {
 | |
|             return "" + (char) input[offset + 0] + (char) input[offset + 1] + (char) input[offset + 2] + (char) input[offset + 3];
 | |
|         }
 | |
| 
 | |
|         private static uint GetUInt32( ref byte[] input, uint offset )
 | |
|         {
 | |
|             return (uint) ( ( ( input[offset + 3] * 256 + input[offset + 2] ) * 256 + input[offset + 1] ) * 256 + input[offset + 0] );
 | |
|         }
 | |
| 
 | |
|         private static uint GetUInt24( ref byte[] input, uint offset )
 | |
|         {
 | |
|             return (uint) ( ( input[offset + 2] * 256 + input[offset + 1] ) * 256 + input[offset + 0] );
 | |
|         }
 | |
| 
 | |
|         private static void GetBytesFromUInt24( ref byte[] input, uint offset, uint splitme )
 | |
|         {
 | |
|             input[offset + 0] = (byte) ( splitme & 0x000000ff );
 | |
|             input[offset + 1] = (byte) ( ( splitme & 0x0000ff00 ) >> 8 );
 | |
|             input[offset + 2] = (byte) ( ( splitme & 0x00ff0000 ) >> 16 );
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         /// <summary>DXT5 Alpha block flipping, inspired by code from Evan Hart (nVidia SDK)</summary>
 | |
|         private static uint FlipUInt24( uint inputUInt24 )
 | |
|         {
 | |
|             byte[][] ThreeBits = new byte[2][];
 | |
|             for ( int i = 0 ; i < 2 ; i++ )
 | |
|                 ThreeBits[i] = new byte[4];
 | |
| 
 | |
|             // extract 3 bits each into the array
 | |
|             ThreeBits[0][0] = (byte) ( inputUInt24 & BitMask );
 | |
|             inputUInt24 >>= 3;
 | |
|             ThreeBits[0][1] = (byte) ( inputUInt24 & BitMask );
 | |
|             inputUInt24 >>= 3;
 | |
|             ThreeBits[0][2] = (byte) ( inputUInt24 & BitMask );
 | |
|             inputUInt24 >>= 3;
 | |
|             ThreeBits[0][3] = (byte) ( inputUInt24 & BitMask );
 | |
|             inputUInt24 >>= 3;
 | |
|             ThreeBits[1][0] = (byte) ( inputUInt24 & BitMask );
 | |
|             inputUInt24 >>= 3;
 | |
|             ThreeBits[1][1] = (byte) ( inputUInt24 & BitMask );
 | |
|             inputUInt24 >>= 3;
 | |
|             ThreeBits[1][2] = (byte) ( inputUInt24 & BitMask );
 | |
|             inputUInt24 >>= 3;
 | |
|             ThreeBits[1][3] = (byte) ( inputUInt24 & BitMask );
 | |
| 
 | |
|             // stuff 8x 3bits into 3 bytes
 | |
|             uint Result = 0;
 | |
|             Result = Result | (uint) ( ThreeBits[1][0] << 0 );
 | |
|             Result = Result | (uint) ( ThreeBits[1][1] << 3 );
 | |
|             Result = Result | (uint) ( ThreeBits[1][2] << 6 );
 | |
|             Result = Result | (uint) ( ThreeBits[1][3] << 9 );
 | |
|             Result = Result | (uint) ( ThreeBits[0][0] << 12 );
 | |
|             Result = Result | (uint) ( ThreeBits[0][1] << 15 );
 | |
|             Result = Result | (uint) ( ThreeBits[0][2] << 18 );
 | |
|             Result = Result | (uint) ( ThreeBits[0][3] << 21 );
 | |
|             return Result;
 | |
|         }
 | |
|         #endregion Helpers
 | |
| 
 | |
|         #region String Representations
 | |
|         private static string GetDescriptionFromFile( string filename )
 | |
|         {
 | |
|             return "\n--> Header of " + filename +
 | |
|                    "\nID: " + idString +
 | |
|                    "\nSize: " + dwSize +
 | |
|                    "\nFlags: " + dwFlags + " (" + (eDDSD) dwFlags + ")" +
 | |
|                    "\nHeight: " + dwHeight +
 | |
|                    "\nWidth: " + dwWidth +
 | |
|                    "\nPitch: " + dwPitchOrLinearSize +
 | |
|                    "\nDepth: " + dwDepth +
 | |
|                    "\nMipMaps: " + dwMipMapCount +
 | |
|                    "\n\n---PixelFormat---" + filename +
 | |
|                    "\nSize: " + pfSize +
 | |
|                    "\nFlags: " + pfFlags + " (" + (eDDPF) pfFlags + ")" +
 | |
|                    "\nFourCC: " + pfFourCC + " (" + (eFOURCC) pfFourCC + ")" +
 | |
| #if READALL
 | |
|                    "\nBitcount: " + pfRGBBitCount +
 | |
|                    "\nBitMask Red: " + pfRBitMask +
 | |
|                    "\nBitMask Green: " + pfGBitMask +
 | |
|                    "\nBitMask Blue: " + pfBBitMask +
 | |
|                    "\nBitMask Alpha: " + pfABitMask +
 | |
| #endif
 | |
|  "\n\n---Capabilities---" + filename +
 | |
|                    "\nCaps1: " + dwCaps1 + " (" + (eDDSCAPS) dwCaps1 + ")" +
 | |
|                    "\nCaps2: " + dwCaps2 + " (" + (eDDSCAPS2) dwCaps2 + ")";
 | |
|         }
 | |
| 
 | |
|         private static string GetDescriptionFromMemory( string filename, TextureTarget Dimension )
 | |
|         {
 | |
|             return "\nFile: " + filename +
 | |
|                    "\nDimension: " + Dimension +
 | |
|                    "\nSize: " + _Width + " * " + _Height + " * " + _Depth +
 | |
|                    "\nCompressed: " + _IsCompressed +
 | |
|                    "\nBytes for Main Image: " + _BytesForMainSurface +
 | |
|                    "\nMipMaps: " + _MipMapCount;
 | |
|         }
 | |
|         #endregion String Representations
 | |
|     }
 | |
| }
 |