wmiutils: Implement IWbemPath::GetNamespaceAt.
[wine] / dlls / d3dx9_36 / util.c
1 /*
2  * Copyright (C) 2009 Tony Wasserka
3  *
4  * This library is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU Lesser General Public
6  * License as published by the Free Software Foundation; either
7  * version 2.1 of the License, or (at your option) any later version.
8  *
9  * This library is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * Lesser General Public License for more details.
13  *
14  * You should have received a copy of the GNU Lesser General Public
15  * License along with this library; if not, write to the Free Software
16  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
17  *
18  */
19
20 #include "wine/debug.h"
21 #include "d3dx9_36_private.h"
22
23 WINE_DEFAULT_DEBUG_CHANNEL(d3dx);
24
25 static void la_from_rgba(const struct vec4 *rgba, struct vec4 *la)
26 {
27     la->x = rgba->x * 0.2125f + rgba->y * 0.7154f + rgba->z * 0.0721f;
28     la->w = rgba->w;
29 }
30
31 static void la_to_rgba(const struct vec4 *la, struct vec4 *rgba)
32 {
33     rgba->x = la->x;
34     rgba->y = la->x;
35     rgba->z = la->x;
36     rgba->w = la->w;
37 }
38
39 /************************************************************
40  * pixel format table providing info about number of bytes per pixel,
41  * number of bits per channel and format type.
42  *
43  * Call get_format_info to request information about a specific format.
44  */
45 static const struct pixel_format_desc formats[] =
46 {
47     /* format              bpc               shifts             bpp blocks   type            from_rgba     to_rgba */
48     {D3DFMT_R8G8B8,        { 0,  8,  8,  8}, { 0, 16,  8,  0},  3, 1, 1,  3, FORMAT_ARGB,    NULL,         NULL      },
49     {D3DFMT_A8R8G8B8,      { 8,  8,  8,  8}, {24, 16,  8,  0},  4, 1, 1,  4, FORMAT_ARGB,    NULL,         NULL      },
50     {D3DFMT_X8R8G8B8,      { 0,  8,  8,  8}, { 0, 16,  8,  0},  4, 1, 1,  4, FORMAT_ARGB,    NULL,         NULL      },
51     {D3DFMT_A8B8G8R8,      { 8,  8,  8,  8}, {24,  0,  8, 16},  4, 1, 1,  4, FORMAT_ARGB,    NULL,         NULL      },
52     {D3DFMT_X8B8G8R8,      { 0,  8,  8,  8}, { 0,  0,  8, 16},  4, 1, 1,  4, FORMAT_ARGB,    NULL,         NULL      },
53     {D3DFMT_R5G6B5,        { 0,  5,  6,  5}, { 0, 11,  5,  0},  2, 1, 1,  2, FORMAT_ARGB,    NULL,         NULL      },
54     {D3DFMT_X1R5G5B5,      { 0,  5,  5,  5}, { 0, 10,  5,  0},  2, 1, 1,  2, FORMAT_ARGB,    NULL,         NULL      },
55     {D3DFMT_A1R5G5B5,      { 1,  5,  5,  5}, {15, 10,  5,  0},  2, 1, 1,  2, FORMAT_ARGB,    NULL,         NULL      },
56     {D3DFMT_R3G3B2,        { 0,  3,  3,  2}, { 0,  5,  2,  0},  1, 1, 1,  1, FORMAT_ARGB,    NULL,         NULL      },
57     {D3DFMT_A8R3G3B2,      { 8,  3,  3,  2}, { 8,  5,  2,  0},  2, 1, 1,  2, FORMAT_ARGB,    NULL,         NULL      },
58     {D3DFMT_A4R4G4B4,      { 4,  4,  4,  4}, {12,  8,  4,  0},  2, 1, 1,  2, FORMAT_ARGB,    NULL,         NULL      },
59     {D3DFMT_X4R4G4B4,      { 0,  4,  4,  4}, { 0,  8,  4,  0},  2, 1, 1,  2, FORMAT_ARGB,    NULL,         NULL      },
60     {D3DFMT_A2R10G10B10,   { 2, 10, 10, 10}, {30, 20, 10,  0},  4, 1, 1,  4, FORMAT_ARGB,    NULL,         NULL      },
61     {D3DFMT_A2B10G10R10,   { 2, 10, 10, 10}, {30,  0, 10, 20},  4, 1, 1,  4, FORMAT_ARGB,    NULL,         NULL      },
62     {D3DFMT_A16B16G16R16,  {16, 16, 16, 16}, {48,  0, 16, 32},  8, 1, 1,  8, FORMAT_ARGB,    NULL,         NULL      },
63     {D3DFMT_G16R16,        { 0, 16, 16,  0}, { 0,  0, 16,  0},  4, 1, 1,  4, FORMAT_ARGB,    NULL,         NULL      },
64     {D3DFMT_A8,            { 8,  0,  0,  0}, { 0,  0,  0,  0},  1, 1, 1,  1, FORMAT_ARGB,    NULL,         NULL      },
65     {D3DFMT_A8L8,          { 8,  8,  0,  0}, { 8,  0,  0,  0},  2, 1, 1,  2, FORMAT_ARGB,    la_from_rgba, la_to_rgba},
66     {D3DFMT_A4L4,          { 4,  4,  0,  0}, { 4,  0,  0,  0},  1, 1, 1,  1, FORMAT_ARGB,    la_from_rgba, la_to_rgba},
67     {D3DFMT_L8,            { 0,  8,  0,  0}, { 0,  0,  0,  0},  1, 1, 1,  1, FORMAT_ARGB,    la_from_rgba, la_to_rgba},
68     {D3DFMT_L16,           { 0, 16,  0,  0}, { 0,  0,  0,  0},  2, 1, 1,  2, FORMAT_ARGB,    la_from_rgba, la_to_rgba},
69     {D3DFMT_DXT1,          { 0,  0,  0,  0}, { 0,  0,  0,  0},  1, 4, 4,  8, FORMAT_DXT,     NULL,         NULL      },
70     {D3DFMT_DXT2,          { 0,  0,  0,  0}, { 0,  0,  0,  0},  1, 4, 4, 16, FORMAT_DXT,     NULL,         NULL      },
71     {D3DFMT_DXT3,          { 0,  0,  0,  0}, { 0,  0,  0,  0},  1, 4, 4, 16, FORMAT_DXT,     NULL,         NULL      },
72     {D3DFMT_DXT4,          { 0,  0,  0,  0}, { 0,  0,  0,  0},  1, 4, 4, 16, FORMAT_DXT,     NULL,         NULL      },
73     {D3DFMT_DXT5,          { 0,  0,  0,  0}, { 0,  0,  0,  0},  1, 4, 4, 16, FORMAT_DXT,     NULL,         NULL      },
74     {D3DFMT_A16B16G16R16F, {16, 16, 16, 16}, {48,  0, 16, 32},  8, 1, 1,  8, FORMAT_ARGBF16, NULL,         NULL      },
75     {D3DFMT_A32B32G32R32F, {32, 32, 32, 32}, {96,  0, 32, 64}, 16, 1, 1, 16, FORMAT_ARGBF,   NULL,         NULL      },
76     /* marks last element */
77     {D3DFMT_UNKNOWN,       { 0,  0,  0,  0}, { 0,  0,  0,  0},  0, 1, 1,  0, FORMAT_UNKNOWN, NULL,         NULL      },
78 };
79
80
81 /************************************************************
82  * map_view_of_file
83  *
84  * Loads a file into buffer and stores the number of read bytes in length.
85  *
86  * PARAMS
87  *   filename [I] name of the file to be loaded
88  *   buffer   [O] pointer to destination buffer
89  *   length   [O] size of the obtained data
90  *
91  * RETURNS
92  *   Success: D3D_OK
93  *   Failure:
94  *     see error codes for CreateFileW, GetFileSize, CreateFileMapping and MapViewOfFile
95  *
96  * NOTES
97  *   The caller must UnmapViewOfFile when it doesn't need the data anymore
98  *
99  */
100 HRESULT map_view_of_file(LPCWSTR filename, LPVOID *buffer, DWORD *length)
101 {
102     HANDLE hfile, hmapping = NULL;
103
104     hfile = CreateFileW(filename, GENERIC_READ, FILE_SHARE_READ, 0, OPEN_EXISTING, 0, 0);
105     if(hfile == INVALID_HANDLE_VALUE) goto error;
106
107     *length = GetFileSize(hfile, NULL);
108     if(*length == INVALID_FILE_SIZE) goto error;
109
110     hmapping = CreateFileMappingW(hfile, NULL, PAGE_READONLY, 0, 0, NULL);
111     if(!hmapping) goto error;
112
113     *buffer = MapViewOfFile(hmapping, FILE_MAP_READ, 0, 0, 0);
114     if(*buffer == NULL) goto error;
115
116     CloseHandle(hmapping);
117     CloseHandle(hfile);
118
119     return S_OK;
120
121 error:
122     CloseHandle(hmapping);
123     CloseHandle(hfile);
124     return HRESULT_FROM_WIN32(GetLastError());
125 }
126
127 /************************************************************
128  * load_resource_into_memory
129  *
130  * Loads a resource into buffer and stores the number of
131  * read bytes in length.
132  *
133  * PARAMS
134  *   module  [I] handle to the module
135  *   resinfo [I] handle to the resource's information block
136  *   buffer  [O] pointer to destination buffer
137  *   length  [O] size of the obtained data
138  *
139  * RETURNS
140  *   Success: D3D_OK
141  *   Failure:
142  *     See error codes for SizeofResource, LoadResource and LockResource
143  *
144  * NOTES
145  *   The memory doesn't need to be freed by the caller manually
146  *
147  */
148 HRESULT load_resource_into_memory(HMODULE module, HRSRC resinfo, LPVOID *buffer, DWORD *length)
149 {
150     HGLOBAL resource;
151
152     *length = SizeofResource(module, resinfo);
153     if(*length == 0) return HRESULT_FROM_WIN32(GetLastError());
154
155     resource = LoadResource(module, resinfo);
156     if( !resource ) return HRESULT_FROM_WIN32(GetLastError());
157
158     *buffer = LockResource(resource);
159     if(*buffer == NULL) return HRESULT_FROM_WIN32(GetLastError());
160
161     return S_OK;
162 }
163
164 HRESULT write_buffer_to_file(const WCHAR *dst_filename, ID3DXBuffer *buffer)
165 {
166     HRESULT hr = S_OK;
167     void *buffer_pointer;
168     DWORD buffer_size;
169     HANDLE file = CreateFileW(dst_filename, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
170     if (file == INVALID_HANDLE_VALUE)
171         return HRESULT_FROM_WIN32(GetLastError());
172
173     buffer_pointer = ID3DXBuffer_GetBufferPointer(buffer);
174     buffer_size = ID3DXBuffer_GetBufferSize(buffer);
175
176     if (!WriteFile(file, buffer_pointer, buffer_size, NULL, NULL))
177         hr = HRESULT_FROM_WIN32(GetLastError());
178
179     CloseHandle(file);
180     return hr;
181 }
182
183
184 /************************************************************
185  * get_format_info
186  *
187  * Returns information about the specified format.
188  * If the format is unsupported, it's filled with the D3DFMT_UNKNOWN desc.
189  *
190  * PARAMS
191  *   format [I] format whose description is queried
192  *
193  */
194 const struct pixel_format_desc *get_format_info(D3DFORMAT format)
195 {
196     unsigned int i = 0;
197     while(formats[i].format != format && formats[i].format != D3DFMT_UNKNOWN) i++;
198     if (formats[i].format == D3DFMT_UNKNOWN)
199         FIXME("Unknown format %#x (as FOURCC %s).\n", format, debugstr_an((const char *)&format, 4));
200     return &formats[i];
201 }
202
203 const struct pixel_format_desc *get_format_info_idx(int idx)
204 {
205     if(idx >= sizeof(formats) / sizeof(formats[0]))
206         return NULL;
207     if(formats[idx].format == D3DFMT_UNKNOWN)
208         return NULL;
209     return &formats[idx];
210 }
211
212 #define WINE_D3DX_TO_STR(x) case x: return #x
213
214 const char *debug_d3dxparameter_class(D3DXPARAMETER_CLASS c)
215 {
216     switch (c)
217     {
218         WINE_D3DX_TO_STR(D3DXPC_SCALAR);
219         WINE_D3DX_TO_STR(D3DXPC_VECTOR);
220         WINE_D3DX_TO_STR(D3DXPC_MATRIX_ROWS);
221         WINE_D3DX_TO_STR(D3DXPC_MATRIX_COLUMNS);
222         WINE_D3DX_TO_STR(D3DXPC_OBJECT);
223         WINE_D3DX_TO_STR(D3DXPC_STRUCT);
224         default:
225             FIXME("Unrecognized D3DXPARAMETER_CLASS %#x.\n", c);
226             return "unrecognized";
227     }
228 }
229
230 const char *debug_d3dxparameter_type(D3DXPARAMETER_TYPE t)
231 {
232     switch (t)
233     {
234         WINE_D3DX_TO_STR(D3DXPT_VOID);
235         WINE_D3DX_TO_STR(D3DXPT_BOOL);
236         WINE_D3DX_TO_STR(D3DXPT_INT);
237         WINE_D3DX_TO_STR(D3DXPT_FLOAT);
238         WINE_D3DX_TO_STR(D3DXPT_STRING);
239         WINE_D3DX_TO_STR(D3DXPT_TEXTURE);
240         WINE_D3DX_TO_STR(D3DXPT_TEXTURE1D);
241         WINE_D3DX_TO_STR(D3DXPT_TEXTURE2D);
242         WINE_D3DX_TO_STR(D3DXPT_TEXTURE3D);
243         WINE_D3DX_TO_STR(D3DXPT_TEXTURECUBE);
244         WINE_D3DX_TO_STR(D3DXPT_SAMPLER);
245         WINE_D3DX_TO_STR(D3DXPT_SAMPLER1D);
246         WINE_D3DX_TO_STR(D3DXPT_SAMPLER2D);
247         WINE_D3DX_TO_STR(D3DXPT_SAMPLER3D);
248         WINE_D3DX_TO_STR(D3DXPT_SAMPLERCUBE);
249         WINE_D3DX_TO_STR(D3DXPT_PIXELSHADER);
250         WINE_D3DX_TO_STR(D3DXPT_VERTEXSHADER);
251         WINE_D3DX_TO_STR(D3DXPT_PIXELFRAGMENT);
252         WINE_D3DX_TO_STR(D3DXPT_VERTEXFRAGMENT);
253         WINE_D3DX_TO_STR(D3DXPT_UNSUPPORTED);
254         default:
255             FIXME("Unrecognized D3DXPARAMETER_TYP %#x.\n", t);
256             return "unrecognized";
257     }
258 }
259
260 const char *debug_d3dxparameter_registerset(D3DXREGISTER_SET r)
261 {
262     switch (r)
263     {
264         WINE_D3DX_TO_STR(D3DXRS_BOOL);
265         WINE_D3DX_TO_STR(D3DXRS_INT4);
266         WINE_D3DX_TO_STR(D3DXRS_FLOAT4);
267         WINE_D3DX_TO_STR(D3DXRS_SAMPLER);
268         default:
269             FIXME("Unrecognized D3DXREGISTER_SET %#x.\n", r);
270             return "unrecognized";
271     }
272 }
273
274 #undef WINE_D3DX_TO_STR
275
276 /* parameter type conversion helpers */
277 static BOOL get_bool(D3DXPARAMETER_TYPE type, LPCVOID data)
278 {
279     switch (type)
280     {
281         case D3DXPT_FLOAT:
282         case D3DXPT_INT:
283         case D3DXPT_BOOL:
284             return *(DWORD *)data != 0;
285
286         case D3DXPT_VOID:
287             return *(BOOL *)data;
288
289         default:
290             FIXME("Unhandled type %s.\n", debug_d3dxparameter_type(type));
291             return FALSE;
292     }
293 }
294
295 static INT get_int(D3DXPARAMETER_TYPE type, LPCVOID data)
296 {
297     switch (type)
298     {
299         case D3DXPT_FLOAT:
300             return *(FLOAT *)data;
301
302         case D3DXPT_INT:
303         case D3DXPT_VOID:
304             return *(INT *)data;
305
306         case D3DXPT_BOOL:
307             return get_bool(type, data);
308
309         default:
310             FIXME("Unhandled type %s.\n", debug_d3dxparameter_type(type));
311             return 0;
312     }
313 }
314
315 static FLOAT get_float(D3DXPARAMETER_TYPE type, LPCVOID data)
316 {
317     switch (type)
318     {
319         case D3DXPT_FLOAT:
320         case D3DXPT_VOID:
321             return *(FLOAT *)data;
322
323         case D3DXPT_INT:
324             return *(INT *)data;
325
326         case D3DXPT_BOOL:
327             return get_bool(type, data);
328
329         default:
330             FIXME("Unhandled type %s.\n", debug_d3dxparameter_type(type));
331             return 0.0f;
332     }
333 }
334
335 void set_number(LPVOID outdata, D3DXPARAMETER_TYPE outtype, LPCVOID indata, D3DXPARAMETER_TYPE intype)
336 {
337     TRACE("Changing from type %s to type %s\n", debug_d3dxparameter_type(intype), debug_d3dxparameter_type(outtype));
338
339     if (outtype == intype)
340     {
341         *(DWORD *)outdata = *(DWORD *)indata;
342         return;
343     }
344
345     switch (outtype)
346     {
347         case D3DXPT_FLOAT:
348             *(FLOAT *)outdata = get_float(intype, indata);
349             break;
350
351         case D3DXPT_BOOL:
352             *(BOOL *)outdata = get_bool(intype, indata);
353             break;
354
355         case D3DXPT_INT:
356             *(INT *)outdata = get_int(intype, indata);
357             break;
358
359         default:
360             FIXME("Unhandled type %s.\n", debug_d3dxparameter_type(outtype));
361             *(DWORD *)outdata = 0;
362             break;
363     }
364 }