Mercurial > mplayer.hg
annotate libvo/vo_direct3d.c @ 32423:4a3bfd196e17
Add support for AOPLAY_FINAL_CHUNK, reportedly also fixes a possible
hang at the end of playback.
author | reimar |
---|---|
date | Sun, 17 Oct 2010 17:39:53 +0000 |
parents | 5f8cbd711a04 |
children | 9e627a1793b1 |
rev | line source |
---|---|
27921 | 1 /* |
2 * Copyright (c) 2008 Georgi Petrov (gogothebee) <gogothebee@gmail.com> | |
3 * | |
4 * This file is part of MPlayer. | |
5 * | |
6 * MPlayer is free software; you can redistribute it and/or modify | |
7 * it under the terms of the GNU General Public License as published by | |
8 * the Free Software Foundation; either version 2 of the License, or | |
9 * (at your option) any later version. | |
10 * | |
11 * MPlayer is distributed in the hope that it will be useful, | |
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
14 * GNU General Public License for more details. | |
15 * | |
16 * You should have received a copy of the GNU General Public License along | |
17 * with MPlayer; if not, write to the Free Software Foundation, Inc., | |
18 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. | |
19 */ | |
20 | |
21 #include <windows.h> | |
22 #include <errno.h> | |
23 #include <stdio.h> | |
24 #include <d3d9.h> | |
25 #include "config.h" | |
26 #include "video_out.h" | |
27 #include "video_out_internal.h" | |
28 #include "fastmemcpy.h" | |
29 #include "mp_msg.h" | |
30 #include "aspect.h" | |
31 #include "w32_common.h" | |
27937 | 32 #include "libavutil/common.h" |
28084 | 33 #include "font_load.h" |
34 #include "sub.h" | |
27921 | 35 |
36 static const vo_info_t info = | |
37 { | |
38 "Direct3D 9 Renderer", | |
39 "direct3d", | |
40 "Georgi Petrov (gogothebee) <gogothebee@gmail.com>", | |
41 "" | |
42 }; | |
43 | |
44 /* | |
45 * Link essential libvo functions: preinit, config, control, draw_frame, | |
46 * draw_slice, draw_osd, flip_page, check_events, uninit and | |
47 * the structure info. | |
48 */ | |
49 const LIBVO_EXTERN(direct3d) | |
50 | |
51 | |
27928 | 52 /* Global variables "priv" structure. I try to keep their count low. |
27921 | 53 */ |
27928 | 54 static struct global_priv { |
55 int is_paused; /**< 1 = Movie is paused, | |
56 0 = Movie is not paused */ | |
27984
21221fd6d994
Make sure the backbuffer is cleared when the border size might have changed.
reimar
parents:
27971
diff
changeset
|
57 int is_clear_needed; /**< 1 = Clear the backbuffer before StretchRect |
21221fd6d994
Make sure the backbuffer is cleared when the border size might have changed.
reimar
parents:
27971
diff
changeset
|
58 0 = (default) Don't clear it */ |
28418 | 59 D3DLOCKED_RECT locked_rect; /**< The locked offscreen surface */ |
27928 | 60 RECT fs_movie_rect; /**< Rect (upscaled) of the movie when displayed |
61 in fullscreen */ | |
62 RECT fs_panscan_rect; /**< PanScan source surface cropping in | |
63 fullscreen */ | |
64 int src_width; /**< Source (movie) width */ | |
65 int src_height; /**< Source (movie) heigth */ | |
28598
4172d83194f2
Extend calc_src_dst_rects to also calculate the border values needed for
reimar
parents:
28511
diff
changeset
|
66 int border_x; /**< horizontal border value for OSD */ |
4172d83194f2
Extend calc_src_dst_rects to also calculate the border values needed for
reimar
parents:
28511
diff
changeset
|
67 int border_y; /**< vertical border value for OSD */ |
27921 | 68 |
27928 | 69 D3DFORMAT movie_src_fmt; /**< Movie colorspace format (depends on |
70 the movie's codec) */ | |
71 D3DFORMAT desktop_fmt; /**< Desktop (screen) colorspace format. | |
72 Usually XRGB */ | |
30148
517a800e940e
Modify -vo direct3d so we do not have to link against d3d9.dll, it might
reimar
parents:
28598
diff
changeset
|
73 |
517a800e940e
Modify -vo direct3d so we do not have to link against d3d9.dll, it might
reimar
parents:
28598
diff
changeset
|
74 HANDLE d3d9_dll; /**< d3d9 Library HANDLE */ |
517a800e940e
Modify -vo direct3d so we do not have to link against d3d9.dll, it might
reimar
parents:
28598
diff
changeset
|
75 IDirect3D9 * (WINAPI *pDirect3DCreate9)(UINT); /**< pointer to Direct3DCreate9 function */ |
517a800e940e
Modify -vo direct3d so we do not have to link against d3d9.dll, it might
reimar
parents:
28598
diff
changeset
|
76 |
27928 | 77 LPDIRECT3D9 d3d_handle; /**< Direct3D Handle */ |
78 LPDIRECT3DDEVICE9 d3d_device; /**< The Direct3D Adapter */ | |
79 IDirect3DSurface9 *d3d_surface; /**< Offscreen Direct3D Surface. MPlayer | |
80 renders inside it. Uses colorspace | |
81 priv->movie_src_fmt */ | |
28084 | 82 IDirect3DTexture9 *d3d_texture_osd; /**< Direct3D Texture. Uses RGBA */ |
83 IDirect3DTexture9 *d3d_texture_system; /**< Direct3D Texture. System memory | |
84 cannot lock a normal texture. Uses RGBA */ | |
27928 | 85 IDirect3DSurface9 *d3d_backbuf; /**< Video card's back buffer (used to |
86 display next frame) */ | |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
87 int cur_backbuf_width; /**< Current backbuffer width */ |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
88 int cur_backbuf_height; /**< Current backbuffer height */ |
28084 | 89 int is_osd_populated; /**< 1 = OSD texture has something to display, |
90 0 = OSD texture is clear */ | |
91 int device_caps_power2_only; /**< 1 = texture sizes have to be power 2 | |
92 0 = texture sizes can be anything */ | |
93 int device_caps_square_only; /**< 1 = textures have to be square | |
94 0 = textures do not have to be square */ | |
95 int device_texture_sys; /**< 1 = device can texture from system memory | |
96 0 = device requires shadow */ | |
97 int max_texture_width; /**< from the device capabilities */ | |
98 int max_texture_height; /**< from the device capabilities */ | |
99 int osd_width; /**< current width of the OSD */ | |
100 int osd_height; /**< current height of the OSD */ | |
101 int osd_texture_width; /**< current width of the OSD texture */ | |
102 int osd_texture_height; /**< current height of the OSD texture */ | |
27928 | 103 } *priv; |
104 | |
105 typedef struct { | |
106 const unsigned int mplayer_fmt; /**< Given by MPlayer */ | |
107 const D3DFORMAT fourcc; /**< Required by D3D's test function */ | |
108 } struct_fmt_table; | |
27921 | 109 |
110 /* Map table from reported MPlayer format to the required | |
27928 | 111 fourcc. This is needed to perform the format query. */ |
27921 | 112 |
27928 | 113 static const struct_fmt_table fmt_table[] = { |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
114 {IMGFMT_YV12, MAKEFOURCC('Y','V','1','2')}, |
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
115 {IMGFMT_I420, MAKEFOURCC('I','4','2','0')}, |
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
116 {IMGFMT_IYUV, MAKEFOURCC('I','Y','U','V')}, |
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
117 {IMGFMT_YVU9, MAKEFOURCC('Y','V','U','9')}, |
28007
16b39ef63bb5
Use D3DFMT_ constants where possible instead of MAKEFOURCC
reimar
parents:
28006
diff
changeset
|
118 {IMGFMT_YUY2, D3DFMT_YUY2}, |
16b39ef63bb5
Use D3DFMT_ constants where possible instead of MAKEFOURCC
reimar
parents:
28006
diff
changeset
|
119 {IMGFMT_UYVY, D3DFMT_UYVY}, |
28006 | 120 {IMGFMT_BGR32, D3DFMT_X8R8G8B8}, |
121 {IMGFMT_RGB32, D3DFMT_X8B8G8R8}, | |
122 {IMGFMT_BGR24, D3DFMT_R8G8B8}, //untested | |
123 {IMGFMT_BGR16, D3DFMT_R5G6B5}, | |
124 {IMGFMT_BGR15, D3DFMT_X1R5G5B5}, | |
125 {IMGFMT_BGR8 , D3DFMT_R3G3B2}, //untested | |
27921 | 126 }; |
127 | |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
128 #define DISPLAY_FORMAT_TABLE_ENTRIES (sizeof(fmt_table) / sizeof(fmt_table[0])) |
27921 | 129 |
28084 | 130 #define D3DFVF_MY_VERTEX (D3DFVF_XYZ | D3DFVF_TEX1) |
131 | |
132 typedef struct { | |
133 float x, y, z; /* Position of vertex in 3D space */ | |
134 float tu, tv; /* Texture coordinates */ | |
135 } struct_vertex; | |
136 | |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
137 typedef enum back_buffer_action { |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
138 BACKBUFFER_CREATE, |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
139 BACKBUFFER_RESET |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
140 } back_buffer_action_e; |
27921 | 141 /**************************************************************************** |
142 * * | |
143 * * | |
144 * * | |
145 * Direct3D specific implementation functions * | |
146 * * | |
147 * * | |
148 * * | |
149 ****************************************************************************/ | |
150 | |
151 /** @brief Calculate scaled fullscreen movie rectangle with | |
152 * preserved aspect ratio. | |
153 */ | |
27928 | 154 static void calc_fs_rect(void) |
27921 | 155 { |
28511
db19e31a2c7c
Add a calc_src_dst_rects that calculates from window size, panscan etc.
reimar
parents:
28434
diff
changeset
|
156 struct vo_rect src_rect; |
db19e31a2c7c
Add a calc_src_dst_rects that calculates from window size, panscan etc.
reimar
parents:
28434
diff
changeset
|
157 struct vo_rect dst_rect; |
28598
4172d83194f2
Extend calc_src_dst_rects to also calculate the border values needed for
reimar
parents:
28511
diff
changeset
|
158 struct vo_rect borders; |
4172d83194f2
Extend calc_src_dst_rects to also calculate the border values needed for
reimar
parents:
28511
diff
changeset
|
159 calc_src_dst_rects(priv->src_width, priv->src_height, &src_rect, &dst_rect, &borders, NULL); |
27921 | 160 |
28511
db19e31a2c7c
Add a calc_src_dst_rects that calculates from window size, panscan etc.
reimar
parents:
28434
diff
changeset
|
161 priv->fs_movie_rect.left = dst_rect.left; |
db19e31a2c7c
Add a calc_src_dst_rects that calculates from window size, panscan etc.
reimar
parents:
28434
diff
changeset
|
162 priv->fs_movie_rect.right = dst_rect.right; |
db19e31a2c7c
Add a calc_src_dst_rects that calculates from window size, panscan etc.
reimar
parents:
28434
diff
changeset
|
163 priv->fs_movie_rect.top = dst_rect.top; |
db19e31a2c7c
Add a calc_src_dst_rects that calculates from window size, panscan etc.
reimar
parents:
28434
diff
changeset
|
164 priv->fs_movie_rect.bottom = dst_rect.bottom; |
db19e31a2c7c
Add a calc_src_dst_rects that calculates from window size, panscan etc.
reimar
parents:
28434
diff
changeset
|
165 priv->fs_panscan_rect.left = src_rect.left; |
db19e31a2c7c
Add a calc_src_dst_rects that calculates from window size, panscan etc.
reimar
parents:
28434
diff
changeset
|
166 priv->fs_panscan_rect.right = src_rect.right; |
db19e31a2c7c
Add a calc_src_dst_rects that calculates from window size, panscan etc.
reimar
parents:
28434
diff
changeset
|
167 priv->fs_panscan_rect.top = src_rect.top; |
db19e31a2c7c
Add a calc_src_dst_rects that calculates from window size, panscan etc.
reimar
parents:
28434
diff
changeset
|
168 priv->fs_panscan_rect.bottom = src_rect.bottom; |
28598
4172d83194f2
Extend calc_src_dst_rects to also calculate the border values needed for
reimar
parents:
28511
diff
changeset
|
169 priv->border_x = borders.left; |
4172d83194f2
Extend calc_src_dst_rects to also calculate the border values needed for
reimar
parents:
28511
diff
changeset
|
170 priv->border_y = borders.top; |
27921 | 171 |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
172 mp_msg(MSGT_VO, MSGL_V, |
28433 | 173 "<vo_direct3d>Fullscreen movie rectangle: t: %ld, l: %ld, r: %ld, b:%ld\n", |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
174 priv->fs_movie_rect.top, priv->fs_movie_rect.left, |
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
175 priv->fs_movie_rect.right, priv->fs_movie_rect.bottom); |
27984
21221fd6d994
Make sure the backbuffer is cleared when the border size might have changed.
reimar
parents:
27971
diff
changeset
|
176 |
21221fd6d994
Make sure the backbuffer is cleared when the border size might have changed.
reimar
parents:
27971
diff
changeset
|
177 /* The backbuffer should be cleared before next StretchRect. This is |
21221fd6d994
Make sure the backbuffer is cleared when the border size might have changed.
reimar
parents:
27971
diff
changeset
|
178 * necessary because our new draw area could be smaller than the |
21221fd6d994
Make sure the backbuffer is cleared when the border size might have changed.
reimar
parents:
27971
diff
changeset
|
179 * previous one used by StretchRect and without it, leftovers from the |
21221fd6d994
Make sure the backbuffer is cleared when the border size might have changed.
reimar
parents:
27971
diff
changeset
|
180 * previous frame will be left. */ |
21221fd6d994
Make sure the backbuffer is cleared when the border size might have changed.
reimar
parents:
27971
diff
changeset
|
181 priv->is_clear_needed = 1; |
27921 | 182 } |
183 | |
28026 | 184 /** @brief Destroy D3D Offscreen and Backbuffer surfaces. |
27921 | 185 */ |
28026 | 186 static void destroy_d3d_surfaces(void) |
27921 | 187 { |
28433 | 188 mp_msg(MSGT_VO, MSGL_V, "<vo_direct3d>destroy_d3d_surfaces called.\n"); |
28026 | 189 /* Let's destroy the old (if any) D3D Surfaces */ |
27921 | 190 |
28085
f2606c3433ef
Slightly simplify the conditional release/free code
reimar
parents:
28084
diff
changeset
|
191 if (priv->locked_rect.pBits) |
27969
7ddd69cf214f
Lock/unlock surface only once even when drawing many slices.
reimar
parents:
27968
diff
changeset
|
192 IDirect3DSurface9_UnlockRect(priv->d3d_surface); |
28085
f2606c3433ef
Slightly simplify the conditional release/free code
reimar
parents:
28084
diff
changeset
|
193 priv->locked_rect.pBits = NULL; |
27969
7ddd69cf214f
Lock/unlock surface only once even when drawing many slices.
reimar
parents:
27968
diff
changeset
|
194 |
28085
f2606c3433ef
Slightly simplify the conditional release/free code
reimar
parents:
28084
diff
changeset
|
195 if (priv->d3d_surface) |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
196 IDirect3DSurface9_Release(priv->d3d_surface); |
28085
f2606c3433ef
Slightly simplify the conditional release/free code
reimar
parents:
28084
diff
changeset
|
197 priv->d3d_surface = NULL; |
27921 | 198 |
28084 | 199 /* kill the OSD texture and its shadow copy */ |
28085
f2606c3433ef
Slightly simplify the conditional release/free code
reimar
parents:
28084
diff
changeset
|
200 if (priv->d3d_texture_osd) |
28084 | 201 IDirect3DTexture9_Release(priv->d3d_texture_osd); |
28085
f2606c3433ef
Slightly simplify the conditional release/free code
reimar
parents:
28084
diff
changeset
|
202 priv->d3d_texture_osd = NULL; |
28084 | 203 |
28085
f2606c3433ef
Slightly simplify the conditional release/free code
reimar
parents:
28084
diff
changeset
|
204 if (priv->d3d_texture_system) |
28084 | 205 IDirect3DTexture9_Release(priv->d3d_texture_system); |
28085
f2606c3433ef
Slightly simplify the conditional release/free code
reimar
parents:
28084
diff
changeset
|
206 priv->d3d_texture_system = NULL; |
28084 | 207 |
28085
f2606c3433ef
Slightly simplify the conditional release/free code
reimar
parents:
28084
diff
changeset
|
208 if (priv->d3d_backbuf) |
28026 | 209 IDirect3DSurface9_Release(priv->d3d_backbuf); |
28085
f2606c3433ef
Slightly simplify the conditional release/free code
reimar
parents:
28084
diff
changeset
|
210 priv->d3d_backbuf = NULL; |
28026 | 211 } |
212 | |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
213 /** @brief Create D3D Offscreen and Backbuffer surfaces. Each |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
214 * surface is created only if it's not already present. |
28026 | 215 * @return 1 on success, 0 on failure |
216 */ | |
217 static int create_d3d_surfaces(void) | |
218 { | |
28084 | 219 int osd_width = vo_dwidth, osd_height = vo_dheight; |
220 int tex_width = osd_width, tex_height = osd_height; | |
28433 | 221 mp_msg(MSGT_VO, MSGL_V, "<vo_direct3d>create_d3d_surfaces called.\n"); |
28026 | 222 |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
223 if (!priv->d3d_surface && |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
224 FAILED(IDirect3DDevice9_CreateOffscreenPlainSurface( |
28026 | 225 priv->d3d_device, priv->src_width, priv->src_height, |
226 priv->movie_src_fmt, D3DPOOL_DEFAULT, &priv->d3d_surface, NULL))) { | |
227 mp_msg(MSGT_VO, MSGL_ERR, | |
28433 | 228 "<vo_direct3d>Allocating offscreen surface failed.\n"); |
28026 | 229 return 0; |
230 } | |
231 | |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
232 if (!priv->d3d_backbuf && |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
233 FAILED(IDirect3DDevice9_GetBackBuffer(priv->d3d_device, 0, 0, |
28026 | 234 D3DBACKBUFFER_TYPE_MONO, |
28029 | 235 &priv->d3d_backbuf))) { |
28433 | 236 mp_msg(MSGT_VO, MSGL_ERR, "<vo_direct3d>Allocating backbuffer failed.\n"); |
28026 | 237 return 0; |
27921 | 238 } |
239 | |
28084 | 240 /* calculate the best size for the OSD depending on the factors from the device */ |
241 if (priv->device_caps_power2_only) { | |
242 tex_width = 1; | |
243 tex_height = 1; | |
244 while (tex_width < osd_width ) tex_width <<= 1; | |
245 while (tex_height < osd_height) tex_height <<= 1; | |
246 } | |
247 if (priv->device_caps_square_only) | |
248 /* device only supports square textures */ | |
249 tex_width = tex_height = tex_width > tex_height ? tex_width : tex_height; | |
250 // better round up to a multiple of 16 | |
251 tex_width = (tex_width + 15) & ~15; | |
252 tex_height = (tex_height + 15) & ~15; | |
253 | |
254 // make sure we respect the size limits without breaking aspect or pow2-requirements | |
255 while (tex_width > priv->max_texture_width || tex_height > priv->max_texture_height) { | |
28418 | 256 osd_width >>= 1; |
257 osd_height >>= 1; | |
258 tex_width >>= 1; | |
259 tex_height >>= 1; | |
28084 | 260 } |
261 | |
262 priv->osd_width = osd_width; | |
263 priv->osd_height = osd_height; | |
264 priv->osd_texture_width = tex_width; | |
265 priv->osd_texture_height = tex_height; | |
266 | |
28433 | 267 mp_msg(MSGT_VO, MSGL_V, "<vo_direct3d>OSD texture size (%dx%d), requested (%dx%d).\n", |
28084 | 268 vo_dwidth, vo_dheight, priv->osd_texture_width, priv->osd_texture_height); |
269 | |
270 /* create OSD */ | |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
271 if (!priv->d3d_texture_system && |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
272 FAILED(IDirect3DDevice9_CreateTexture(priv->d3d_device, |
28084 | 273 priv->osd_texture_width, |
274 priv->osd_texture_height, | |
275 1, | |
276 D3DUSAGE_DYNAMIC, | |
277 D3DFMT_A8L8, | |
278 D3DPOOL_SYSTEMMEM, | |
279 &priv->d3d_texture_system, | |
280 NULL))) { | |
281 mp_msg(MSGT_VO,MSGL_ERR, | |
28433 | 282 "<vo_direct3d>Allocating OSD texture in system RAM failed.\n"); |
28084 | 283 return 0; |
284 } | |
285 | |
286 if (!priv->device_texture_sys) { | |
287 /* only create if we need a shadow version on the external device */ | |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
288 if (!priv->d3d_texture_osd && |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
289 FAILED(IDirect3DDevice9_CreateTexture(priv->d3d_device, |
28084 | 290 priv->osd_texture_width, |
291 priv->osd_texture_height, | |
292 1, | |
293 D3DUSAGE_DYNAMIC, | |
294 D3DFMT_A8L8, | |
295 D3DPOOL_DEFAULT, | |
296 &priv->d3d_texture_osd, | |
297 NULL))) { | |
298 mp_msg(MSGT_VO,MSGL_ERR, | |
28433 | 299 "<vo_direct3d>Allocating OSD texture in video RAM failed.\n"); |
28084 | 300 return 0; |
301 } | |
302 } | |
303 | |
304 /* setup default renderstate */ | |
305 IDirect3DDevice9_SetRenderState(priv->d3d_device, D3DRS_SRCBLEND, D3DBLEND_ONE); | |
306 IDirect3DDevice9_SetRenderState(priv->d3d_device, D3DRS_DESTBLEND, D3DBLEND_INVSRCALPHA); | |
307 IDirect3DDevice9_SetRenderState(priv->d3d_device, D3DRS_ALPHAFUNC, D3DCMP_GREATER); | |
308 IDirect3DDevice9_SetRenderState(priv->d3d_device, D3DRS_ALPHAREF, (DWORD)0x0); | |
309 IDirect3DDevice9_SetRenderState(priv->d3d_device, D3DRS_LIGHTING, FALSE); | |
310 IDirect3DDevice9_SetSamplerState(priv->d3d_device, 0, D3DSAMP_MINFILTER, D3DTEXF_LINEAR); | |
311 IDirect3DDevice9_SetSamplerState(priv->d3d_device, 0, D3DSAMP_MAGFILTER, D3DTEXF_LINEAR); | |
312 | |
28026 | 313 return 1; |
27921 | 314 } |
315 | |
28026 | 316 /** @brief Fill D3D Presentation parameters |
317 */ | |
318 static void fill_d3d_presentparams(D3DPRESENT_PARAMETERS *present_params) | |
319 { | |
320 /* Prepare Direct3D initialization parameters. */ | |
321 memset(present_params, 0, sizeof(D3DPRESENT_PARAMETERS)); | |
322 present_params->Windowed = TRUE; | |
323 present_params->SwapEffect = D3DSWAPEFFECT_COPY; | |
324 present_params->Flags = D3DPRESENTFLAG_VIDEO; | |
325 present_params->hDeviceWindow = vo_w32_window; /* w32_common var */ | |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
326 present_params->BackBufferWidth = priv->cur_backbuf_width; |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
327 present_params->BackBufferHeight = priv->cur_backbuf_height; |
28026 | 328 present_params->MultiSampleType = D3DMULTISAMPLE_NONE; |
329 present_params->PresentationInterval = D3DPRESENT_INTERVAL_ONE; | |
330 present_params->BackBufferFormat = priv->desktop_fmt; | |
331 present_params->BackBufferCount = 1; | |
332 present_params->EnableAutoDepthStencil = FALSE; | |
333 } | |
27921 | 334 |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
335 |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
336 /** @brief Create a new backbuffer. Create or Reset the D3D |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
337 * device. |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
338 * @return 1 on success, 0 on failure |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
339 */ |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
340 static int change_d3d_backbuffer(back_buffer_action_e action) |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
341 { |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
342 D3DPRESENT_PARAMETERS present_params; |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
343 |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
344 destroy_d3d_surfaces(); |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
345 |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
346 /* Grow the backbuffer in the required dimension. */ |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
347 if (vo_dwidth > priv->cur_backbuf_width) |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
348 priv->cur_backbuf_width = vo_dwidth; |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
349 |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
350 if (vo_dheight > priv->cur_backbuf_height) |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
351 priv->cur_backbuf_height = vo_dheight; |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
352 |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
353 /* The grown backbuffer dimensions are ready and fill_d3d_presentparams |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
354 * will use them, so we can reset the device. |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
355 */ |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
356 fill_d3d_presentparams(&present_params); |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
357 |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
358 /* vo_w32_window is w32_common variable. It's a handle to the window. */ |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
359 if (action == BACKBUFFER_CREATE && |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
360 FAILED(IDirect3D9_CreateDevice(priv->d3d_handle, |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
361 D3DADAPTER_DEFAULT, |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
362 D3DDEVTYPE_HAL, vo_w32_window, |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
363 D3DCREATE_SOFTWARE_VERTEXPROCESSING, |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
364 &present_params, &priv->d3d_device))) { |
28434
6d926d2693ec
Avoid message spam during video adapter uncooperative state.
gogothebee
parents:
28433
diff
changeset
|
365 mp_msg(MSGT_VO, MSGL_V, |
28433 | 366 "<vo_direct3d>Creating Direct3D device failed.\n"); |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
367 return 0; |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
368 } |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
369 |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
370 if (action == BACKBUFFER_RESET && |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
371 FAILED(IDirect3DDevice9_Reset(priv->d3d_device, &present_params))) { |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
372 mp_msg(MSGT_VO, MSGL_ERR, |
28433 | 373 "<vo_direct3d>Reseting Direct3D device failed.\n"); |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
374 return 0; |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
375 } |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
376 |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
377 mp_msg(MSGT_VO, MSGL_V, |
28433 | 378 "<vo_direct3d>New backbuffer (%dx%d), VO (%dx%d)\n", |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
379 present_params.BackBufferWidth, present_params.BackBufferHeight, |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
380 vo_dwidth, vo_dheight); |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
381 |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
382 return 1; |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
383 } |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
384 |
28026 | 385 /** @brief Configure initial Direct3D context. The first |
386 * function called to initialize the D3D context. | |
27921 | 387 * @return 1 on success, 0 on failure |
388 */ | |
28026 | 389 static int configure_d3d(void) |
27921 | 390 { |
27928 | 391 D3DDISPLAYMODE disp_mode; |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
392 D3DVIEWPORT9 vp = {0, 0, vo_dwidth, vo_dheight, 0, 1}; |
27921 | 393 |
28433 | 394 mp_msg(MSGT_VO, MSGL_V, "<vo_direct3d>configure_d3d called.\n"); |
27921 | 395 |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
396 destroy_d3d_surfaces(); |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
397 |
27921 | 398 /* Get the current desktop display mode, so we can set up a back buffer |
399 * of the same format. */ | |
27928 | 400 if (FAILED(IDirect3D9_GetAdapterDisplayMode(priv->d3d_handle, |
401 D3DADAPTER_DEFAULT, | |
402 &disp_mode))) { | |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
403 mp_msg(MSGT_VO, MSGL_ERR, |
28433 | 404 "<vo_direct3d>Reading adapter display mode failed.\n"); |
27921 | 405 return 0; |
406 } | |
407 | |
408 /* Write current Desktop's colorspace format in the global storage. */ | |
27928 | 409 priv->desktop_fmt = disp_mode.Format; |
27921 | 410 |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
411 if (!change_d3d_backbuffer(BACKBUFFER_CREATE)) |
27921 | 412 return 0; |
413 | |
28026 | 414 if (!create_d3d_surfaces()) |
415 return 0; | |
416 | |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
417 if (FAILED(IDirect3DDevice9_SetViewport(priv->d3d_device, |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
418 &vp))) { |
28433 | 419 mp_msg(MSGT_VO, MSGL_ERR, "<vo_direct3d>Setting viewport failed.\n"); |
28416 | 420 return 0; |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
421 } |
27921 | 422 |
28026 | 423 calc_fs_rect(); |
424 | |
425 return 1; | |
426 } | |
427 | |
428 /** @brief Reconfigure the whole Direct3D. Called only | |
429 * when the video adapter becomes uncooperative. | |
430 * @return 1 on success, 0 on failure | |
431 */ | |
432 static int reconfigure_d3d(void) | |
433 { | |
28433 | 434 mp_msg(MSGT_VO, MSGL_V, "<vo_direct3d>reconfigure_d3d called.\n"); |
28026 | 435 |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
436 /* Destroy the offscreen, OSD and backbuffer surfaces */ |
28026 | 437 destroy_d3d_surfaces(); |
438 | |
439 /* Destroy the D3D Device */ | |
28085
f2606c3433ef
Slightly simplify the conditional release/free code
reimar
parents:
28084
diff
changeset
|
440 if (priv->d3d_device) |
28026 | 441 IDirect3DDevice9_Release(priv->d3d_device); |
28085
f2606c3433ef
Slightly simplify the conditional release/free code
reimar
parents:
28084
diff
changeset
|
442 priv->d3d_device = NULL; |
28026 | 443 |
444 /* Stop the whole Direct3D */ | |
445 IDirect3D9_Release(priv->d3d_handle); | |
446 | |
447 /* Initialize Direct3D from the beginning */ | |
30148
517a800e940e
Modify -vo direct3d so we do not have to link against d3d9.dll, it might
reimar
parents:
28598
diff
changeset
|
448 priv->d3d_handle = priv->pDirect3DCreate9(D3D_SDK_VERSION); |
28026 | 449 if (!priv->d3d_handle) { |
28433 | 450 mp_msg(MSGT_VO, MSGL_ERR, "<vo_direct3d>Initializing Direct3D failed.\n"); |
28416 | 451 return 0; |
28026 | 452 } |
453 | |
454 /* Configure Direct3D */ | |
455 if (!configure_d3d()) | |
456 return 0; | |
457 | |
458 return 1; | |
459 } | |
460 | |
461 /** @brief Resize Direct3D context on window resize. | |
462 * @return 1 on success, 0 on failure | |
463 */ | |
464 static int resize_d3d(void) | |
465 { | |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
466 D3DVIEWPORT9 vp = {0, 0, vo_dwidth, vo_dheight, 0, 1}; |
28026 | 467 |
28433 | 468 mp_msg(MSGT_VO, MSGL_V, "<vo_direct3d>resize_d3d called.\n"); |
28026 | 469 |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
470 /* Make sure that backbuffer is large enough to accomodate the new |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
471 viewport dimensions. Grow it if necessary. */ |
28026 | 472 |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
473 if (vo_dwidth > priv->cur_backbuf_width || |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
474 vo_dheight > priv->cur_backbuf_height) { |
28417 | 475 if (!change_d3d_backbuffer(BACKBUFFER_RESET)) |
476 return 0; | |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
477 } |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
478 |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
479 /* Destroy the OSD textures. They should always match the new dimensions |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
480 * of the onscreen window, so on each resize we need new OSD dimensions. |
28026 | 481 */ |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
482 |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
483 if (priv->d3d_texture_osd) |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
484 IDirect3DTexture9_Release(priv->d3d_texture_osd); |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
485 priv->d3d_texture_osd = NULL; |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
486 |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
487 if (priv->d3d_texture_system) |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
488 IDirect3DTexture9_Release(priv->d3d_texture_system); |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
489 priv->d3d_texture_system = NULL; |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
490 |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
491 |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
492 /* Recreate the OSD. The function will observe that the offscreen plain |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
493 * surface and the backbuffer are not destroyed and will skip their creation, |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
494 * effectively recreating only the OSD. |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
495 */ |
27921 | 496 |
28026 | 497 if (!create_d3d_surfaces()) |
27921 | 498 return 0; |
28026 | 499 |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
500 if (FAILED(IDirect3DDevice9_SetViewport(priv->d3d_device, |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
501 &vp))) { |
28433 | 502 mp_msg(MSGT_VO, MSGL_ERR, "<vo_direct3d>Setting viewport failed.\n"); |
28416 | 503 return 0; |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
504 } |
27921 | 505 |
27937 | 506 calc_fs_rect(); |
27921 | 507 |
28084 | 508 #ifdef CONFIG_FREETYPE |
509 // font needs to be adjusted | |
510 force_load_font = 1; | |
511 #endif | |
512 // OSD needs to be drawn fresh for new size | |
513 vo_osd_changed(OSDTYPE_OSD); | |
514 | |
27921 | 515 return 1; |
516 } | |
517 | |
518 /** @brief Uninitialize Direct3D and close the window. | |
519 */ | |
27928 | 520 static void uninit_d3d(void) |
27921 | 521 { |
28433 | 522 mp_msg(MSGT_VO, MSGL_V, "<vo_direct3d>uninit_d3d called.\n"); |
27921 | 523 |
28026 | 524 destroy_d3d_surfaces(); |
525 | |
526 /* Destroy the D3D Device */ | |
28086 | 527 if (priv->d3d_device) |
28026 | 528 IDirect3DDevice9_Release(priv->d3d_device); |
28086 | 529 priv->d3d_device = NULL; |
27921 | 530 |
531 /* Stop the whole D3D. */ | |
28028
62ccb6c80212
Consistency cosmetics: do not compare against NULL in ifs
reimar
parents:
28027
diff
changeset
|
532 if (priv->d3d_handle) { |
28433 | 533 mp_msg(MSGT_VO, MSGL_V, "<vo_direct3d>Stopping Direct3D.\n"); |
27928 | 534 IDirect3D9_Release(priv->d3d_handle); |
27921 | 535 } |
28087 | 536 priv->d3d_handle = NULL; |
27921 | 537 } |
538 | |
539 /** @brief Render a frame on the screen. | |
540 * @param mpi mpi structure with the decoded frame inside | |
541 * @return VO_TRUE on success, VO_ERROR on failure | |
542 */ | |
27928 | 543 static uint32_t render_d3d_frame(mp_image_t *mpi) |
27921 | 544 { |
545 /* Uncomment when direct rendering is implemented. | |
546 * if (mpi->flags & MP_IMGFLAG_DIRECT) ... | |
547 */ | |
548 | |
28422
f7ba65621809
Add checks that a D3D device is available before attempting rendering.
reimar
parents:
28421
diff
changeset
|
549 /* If the D3D device is uncooperative (not initialized), return success. |
f7ba65621809
Add checks that a D3D device is available before attempting rendering.
reimar
parents:
28421
diff
changeset
|
550 The device will be probed for reinitialization in the next flip_page() */ |
f7ba65621809
Add checks that a D3D device is available before attempting rendering.
reimar
parents:
28421
diff
changeset
|
551 if (!priv->d3d_device) |
f7ba65621809
Add checks that a D3D device is available before attempting rendering.
reimar
parents:
28421
diff
changeset
|
552 return VO_TRUE; |
f7ba65621809
Add checks that a D3D device is available before attempting rendering.
reimar
parents:
28421
diff
changeset
|
553 |
27921 | 554 if (mpi->flags & MP_IMGFLAG_DRAW_CALLBACK) |
27967
ad71ce76c6ce
Move the StretchRect call from draw_slices to render_d3d_frame.
reimar
parents:
27966
diff
changeset
|
555 goto skip_upload; |
27921 | 556 |
27928 | 557 if (mpi->flags & MP_IMGFLAG_PLANAR) { /* Copy a planar frame. */ |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
558 draw_slice(mpi->planes, mpi->stride, mpi->w, mpi->h, 0, 0); |
27967
ad71ce76c6ce
Move the StretchRect call from draw_slices to render_d3d_frame.
reimar
parents:
27966
diff
changeset
|
559 goto skip_upload; |
27921 | 560 } |
561 | |
27969
7ddd69cf214f
Lock/unlock surface only once even when drawing many slices.
reimar
parents:
27968
diff
changeset
|
562 /* If we're here, then we should lock the rect and copy a packed frame */ |
7ddd69cf214f
Lock/unlock surface only once even when drawing many slices.
reimar
parents:
27968
diff
changeset
|
563 if (!priv->locked_rect.pBits) { |
27970 | 564 if (FAILED(IDirect3DSurface9_LockRect(priv->d3d_surface, |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
565 &priv->locked_rect, NULL, 0))) { |
28433 | 566 mp_msg(MSGT_VO, MSGL_ERR, "<vo_direct3d>Surface lock failed.\n"); |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
567 return VO_ERROR; |
27970 | 568 } |
27969
7ddd69cf214f
Lock/unlock surface only once even when drawing many slices.
reimar
parents:
27968
diff
changeset
|
569 } |
27921 | 570 |
27968
1081658aa871
Move locked_rect from stack to priv struct in preparation for following patch.
reimar
parents:
27967
diff
changeset
|
571 memcpy_pic(priv->locked_rect.pBits, mpi->planes[0], mpi->stride[0], |
1081658aa871
Move locked_rect from stack to priv struct in preparation for following patch.
reimar
parents:
27967
diff
changeset
|
572 mpi->height, priv->locked_rect.Pitch, mpi->stride[0]); |
27921 | 573 |
27969
7ddd69cf214f
Lock/unlock surface only once even when drawing many slices.
reimar
parents:
27968
diff
changeset
|
574 skip_upload: |
7ddd69cf214f
Lock/unlock surface only once even when drawing many slices.
reimar
parents:
27968
diff
changeset
|
575 /* This unlock is used for both slice_draw path and render_d3d_frame path. */ |
27928 | 576 if (FAILED(IDirect3DSurface9_UnlockRect(priv->d3d_surface))) { |
28433 | 577 mp_msg(MSGT_VO, MSGL_V, "<vo_direct3d>Surface unlock failed.\n"); |
27921 | 578 return VO_ERROR; |
579 } | |
27969
7ddd69cf214f
Lock/unlock surface only once even when drawing many slices.
reimar
parents:
27968
diff
changeset
|
580 priv->locked_rect.pBits = NULL; |
27921 | 581 |
27967
ad71ce76c6ce
Move the StretchRect call from draw_slices to render_d3d_frame.
reimar
parents:
27966
diff
changeset
|
582 if (FAILED(IDirect3DDevice9_BeginScene(priv->d3d_device))) { |
28433 | 583 mp_msg(MSGT_VO, MSGL_ERR, "<vo_direct3d>BeginScene failed.\n"); |
27970 | 584 return VO_ERROR; |
27967
ad71ce76c6ce
Move the StretchRect call from draw_slices to render_d3d_frame.
reimar
parents:
27966
diff
changeset
|
585 } |
ad71ce76c6ce
Move the StretchRect call from draw_slices to render_d3d_frame.
reimar
parents:
27966
diff
changeset
|
586 |
27984
21221fd6d994
Make sure the backbuffer is cleared when the border size might have changed.
reimar
parents:
27971
diff
changeset
|
587 if (priv->is_clear_needed) { |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
588 IDirect3DDevice9_Clear(priv->d3d_device, 0, NULL, |
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
589 D3DCLEAR_TARGET, 0, 0, 0); |
27984
21221fd6d994
Make sure the backbuffer is cleared when the border size might have changed.
reimar
parents:
27971
diff
changeset
|
590 priv->is_clear_needed = 0; |
21221fd6d994
Make sure the backbuffer is cleared when the border size might have changed.
reimar
parents:
27971
diff
changeset
|
591 } |
21221fd6d994
Make sure the backbuffer is cleared when the border size might have changed.
reimar
parents:
27971
diff
changeset
|
592 |
27928 | 593 if (FAILED(IDirect3DDevice9_StretchRect(priv->d3d_device, |
594 priv->d3d_surface, | |
27937 | 595 &priv->fs_panscan_rect, |
27928 | 596 priv->d3d_backbuf, |
27937 | 597 &priv->fs_movie_rect, |
27928 | 598 D3DTEXF_LINEAR))) { |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
599 mp_msg(MSGT_VO, MSGL_ERR, |
28433 | 600 "<vo_direct3d>Copying frame to the backbuffer failed.\n"); |
27921 | 601 return VO_ERROR; |
602 } | |
603 | |
27928 | 604 if (FAILED(IDirect3DDevice9_EndScene(priv->d3d_device))) { |
28433 | 605 mp_msg(MSGT_VO, MSGL_ERR, "<vo_direct3d>EndScene failed.\n"); |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
606 return VO_ERROR; |
27921 | 607 } |
608 | |
609 return VO_TRUE; | |
610 } | |
611 | |
612 | |
613 /** @brief Query if movie colorspace is supported by the HW. | |
614 * @return 0 on failure, device capabilities (not probed | |
615 * currently) on success. | |
616 */ | |
27928 | 617 static int query_format(uint32_t movie_fmt) |
27921 | 618 { |
619 int i; | |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
620 for (i = 0; i < DISPLAY_FORMAT_TABLE_ENTRIES; i++) { |
27928 | 621 if (fmt_table[i].mplayer_fmt == movie_fmt) { |
27921 | 622 /* Test conversion from Movie colorspace to |
623 * display's target colorspace. */ | |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
624 if (FAILED(IDirect3D9_CheckDeviceFormatConversion(priv->d3d_handle, |
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
625 D3DADAPTER_DEFAULT, |
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
626 D3DDEVTYPE_HAL, |
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
627 fmt_table[i].fourcc, |
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
628 priv->desktop_fmt))) { |
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
629 mp_msg(MSGT_VO, MSGL_V, "<vo_direct3d>Rejected image format: %s\n", |
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
630 vo_format_name(fmt_table[i].mplayer_fmt)); |
27928 | 631 return 0; |
632 } | |
27921 | 633 |
27928 | 634 priv->movie_src_fmt = fmt_table[i].fourcc; |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
635 mp_msg(MSGT_VO, MSGL_V, "<vo_direct3d>Accepted image format: %s\n", |
27928 | 636 vo_format_name(fmt_table[i].mplayer_fmt)); |
27921 | 637 return (VFCAP_CSP_SUPPORTED | VFCAP_CSP_SUPPORTED_BY_HW |
28084 | 638 | VFCAP_OSD | VFCAP_HWSCALE_UP | VFCAP_HWSCALE_DOWN); |
27921 | 639 |
640 } | |
641 } | |
642 | |
643 return 0; | |
644 } | |
645 | |
646 /**************************************************************************** | |
647 * * | |
648 * * | |
649 * * | |
650 * libvo Control / Callback functions * | |
651 * * | |
652 * * | |
653 * * | |
654 ****************************************************************************/ | |
655 | |
656 | |
657 | |
658 | |
659 /** @brief libvo Callback: Preinitialize the video card. | |
660 * Preinit the hardware just enough to be queried about | |
661 * supported formats. | |
662 * | |
663 * @return 0 on success, -1 on failure | |
664 */ | |
27928 | 665 |
27921 | 666 static int preinit(const char *arg) |
667 { | |
27928 | 668 D3DDISPLAYMODE disp_mode; |
28084 | 669 D3DCAPS9 disp_caps; |
670 DWORD texture_caps; | |
671 DWORD dev_caps; | |
27928 | 672 |
27921 | 673 /* Set to zero all global variables. */ |
28027 | 674 priv = calloc(1, sizeof(struct global_priv)); |
27928 | 675 if (!priv) { |
28433 | 676 mp_msg(MSGT_VO, MSGL_ERR, "<vo_direct3d>Allocating private memory failed.\n"); |
30149
5f8cbd711a04
Avoid memleaks when vo_direct3d initialization fails.
reimar
parents:
30148
diff
changeset
|
677 goto err_out; |
27928 | 678 } |
27921 | 679 |
680 /* FIXME | |
681 > Please use subopt-helper.h for this, see vo_gl.c:preinit for | |
682 > an example of how to use it. | |
683 */ | |
684 | |
30148
517a800e940e
Modify -vo direct3d so we do not have to link against d3d9.dll, it might
reimar
parents:
28598
diff
changeset
|
685 priv->d3d9_dll = LoadLibraryA("d3d9.dll"); |
517a800e940e
Modify -vo direct3d so we do not have to link against d3d9.dll, it might
reimar
parents:
28598
diff
changeset
|
686 if (!priv->d3d9_dll) { |
517a800e940e
Modify -vo direct3d so we do not have to link against d3d9.dll, it might
reimar
parents:
28598
diff
changeset
|
687 mp_msg(MSGT_VO, MSGL_ERR, "<vo_direct3d>Unable to dynamically load d3d9.dll\n"); |
30149
5f8cbd711a04
Avoid memleaks when vo_direct3d initialization fails.
reimar
parents:
30148
diff
changeset
|
688 goto err_out; |
30148
517a800e940e
Modify -vo direct3d so we do not have to link against d3d9.dll, it might
reimar
parents:
28598
diff
changeset
|
689 } |
517a800e940e
Modify -vo direct3d so we do not have to link against d3d9.dll, it might
reimar
parents:
28598
diff
changeset
|
690 |
517a800e940e
Modify -vo direct3d so we do not have to link against d3d9.dll, it might
reimar
parents:
28598
diff
changeset
|
691 priv->pDirect3DCreate9 = (void *)GetProcAddress(priv->d3d9_dll, "Direct3DCreate9"); |
517a800e940e
Modify -vo direct3d so we do not have to link against d3d9.dll, it might
reimar
parents:
28598
diff
changeset
|
692 if (!priv->pDirect3DCreate9) { |
517a800e940e
Modify -vo direct3d so we do not have to link against d3d9.dll, it might
reimar
parents:
28598
diff
changeset
|
693 mp_msg(MSGT_VO, MSGL_ERR, "<vo_direct3d>Unable to find entry point of Direct3DCreate9\n"); |
30149
5f8cbd711a04
Avoid memleaks when vo_direct3d initialization fails.
reimar
parents:
30148
diff
changeset
|
694 goto err_out; |
30148
517a800e940e
Modify -vo direct3d so we do not have to link against d3d9.dll, it might
reimar
parents:
28598
diff
changeset
|
695 } |
517a800e940e
Modify -vo direct3d so we do not have to link against d3d9.dll, it might
reimar
parents:
28598
diff
changeset
|
696 |
517a800e940e
Modify -vo direct3d so we do not have to link against d3d9.dll, it might
reimar
parents:
28598
diff
changeset
|
697 priv->d3d_handle = priv->pDirect3DCreate9(D3D_SDK_VERSION); |
27928 | 698 if (!priv->d3d_handle) { |
28433 | 699 mp_msg(MSGT_VO, MSGL_ERR, "<vo_direct3d>Initializing Direct3D failed.\n"); |
30149
5f8cbd711a04
Avoid memleaks when vo_direct3d initialization fails.
reimar
parents:
30148
diff
changeset
|
700 goto err_out; |
27921 | 701 } |
702 | |
27928 | 703 if (FAILED(IDirect3D9_GetAdapterDisplayMode(priv->d3d_handle, |
704 D3DADAPTER_DEFAULT, | |
705 &disp_mode))) { | |
28433 | 706 mp_msg(MSGT_VO, MSGL_ERR, "<vo_direct3d>Reading display mode failed.\n"); |
30149
5f8cbd711a04
Avoid memleaks when vo_direct3d initialization fails.
reimar
parents:
30148
diff
changeset
|
707 goto err_out; |
27921 | 708 } |
709 | |
27928 | 710 /* Store in priv->desktop_fmt the user desktop's colorspace. Usually XRGB. */ |
711 priv->desktop_fmt = disp_mode.Format; | |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
712 priv->cur_backbuf_width = disp_mode.Width; |
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
713 priv->cur_backbuf_height = disp_mode.Height; |
27921 | 714 |
28433 | 715 mp_msg(MSGT_VO, MSGL_V, "<vo_direct3d>Setting backbuffer dimensions to (%dx%d).\n", |
27928 | 716 disp_mode.Width, disp_mode.Height); |
27921 | 717 |
28084 | 718 if (FAILED(IDirect3D9_GetDeviceCaps(priv->d3d_handle, |
719 D3DADAPTER_DEFAULT, | |
720 D3DDEVTYPE_HAL, | |
721 &disp_caps))) { | |
28433 | 722 mp_msg(MSGT_VO, MSGL_ERR, "<vo_direct3d>Reading display capabilities failed.\n"); |
30149
5f8cbd711a04
Avoid memleaks when vo_direct3d initialization fails.
reimar
parents:
30148
diff
changeset
|
723 goto err_out; |
28084 | 724 } |
725 | |
726 /* Store relevant information reguarding caps of device */ | |
727 texture_caps = disp_caps.TextureCaps; | |
728 dev_caps = disp_caps.DevCaps; | |
729 priv->device_caps_power2_only = (texture_caps & D3DPTEXTURECAPS_POW2) && | |
730 !(texture_caps & D3DPTEXTURECAPS_NONPOW2CONDITIONAL); | |
731 priv->device_caps_square_only = texture_caps & D3DPTEXTURECAPS_SQUAREONLY; | |
732 priv->device_texture_sys = dev_caps & D3DDEVCAPS_TEXTURESYSTEMMEMORY; | |
733 priv->max_texture_width = disp_caps.MaxTextureWidth; | |
734 priv->max_texture_height = disp_caps.MaxTextureHeight; | |
735 | |
736 mp_msg(MSGT_VO, MSGL_V, "<vo_direct3d>device_caps_power2_only %d, device_caps_square_only %d\n" | |
737 "<vo_direct3d>device_texture_sys %d\n" | |
738 "<vo_direct3d>max_texture_width %d, max_texture_height %d\n", | |
739 priv->device_caps_power2_only, priv->device_caps_square_only, | |
740 priv->device_texture_sys, priv->max_texture_width, | |
741 priv->max_texture_height); | |
742 | |
27921 | 743 /* w32_common framework call. Configures window on the screen, gets |
744 * fullscreen dimensions and does other useful stuff. | |
745 */ | |
27928 | 746 if (!vo_w32_init()) { |
28433 | 747 mp_msg(MSGT_VO, MSGL_V, "<vo_direct3d>Configuring onscreen window failed.\n"); |
30149
5f8cbd711a04
Avoid memleaks when vo_direct3d initialization fails.
reimar
parents:
30148
diff
changeset
|
748 goto err_out; |
27921 | 749 } |
750 | |
751 return 0; | |
30149
5f8cbd711a04
Avoid memleaks when vo_direct3d initialization fails.
reimar
parents:
30148
diff
changeset
|
752 |
5f8cbd711a04
Avoid memleaks when vo_direct3d initialization fails.
reimar
parents:
30148
diff
changeset
|
753 err_out: |
5f8cbd711a04
Avoid memleaks when vo_direct3d initialization fails.
reimar
parents:
30148
diff
changeset
|
754 uninit(); |
5f8cbd711a04
Avoid memleaks when vo_direct3d initialization fails.
reimar
parents:
30148
diff
changeset
|
755 return -1; |
27921 | 756 } |
757 | |
758 | |
759 | |
760 /** @brief libvo Callback: Handle control requests. | |
761 * @return VO_TRUE on success, VO_NOTIMPL when not implemented | |
762 */ | |
763 static int control(uint32_t request, void *data, ...) | |
764 { | |
27928 | 765 switch (request) { |
27921 | 766 case VOCTRL_QUERY_FORMAT: |
27928 | 767 return query_format(*(uint32_t*) data); |
27921 | 768 case VOCTRL_GET_IMAGE: /* Direct Rendering. Not implemented yet. */ |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
769 mp_msg(MSGT_VO, MSGL_V, |
28433 | 770 "<vo_direct3d>Direct Rendering request. Not implemented yet.\n"); |
27921 | 771 return VO_NOTIMPL; |
772 case VOCTRL_DRAW_IMAGE: | |
27928 | 773 return render_d3d_frame(data); |
27921 | 774 case VOCTRL_FULLSCREEN: |
775 vo_w32_fullscreen(); | |
28026 | 776 resize_d3d(); |
27921 | 777 return VO_TRUE; |
778 case VOCTRL_RESET: | |
779 return VO_NOTIMPL; | |
780 case VOCTRL_PAUSE: | |
27928 | 781 priv->is_paused = 1; |
27921 | 782 return VO_TRUE; |
783 case VOCTRL_RESUME: | |
27928 | 784 priv->is_paused = 0; |
27921 | 785 return VO_TRUE; |
786 case VOCTRL_GUISUPPORT: | |
787 return VO_NOTIMPL; | |
788 case VOCTRL_SET_EQUALIZER: | |
789 return VO_NOTIMPL; | |
790 case VOCTRL_GET_EQUALIZER: | |
791 return VO_NOTIMPL; | |
792 case VOCTRL_ONTOP: | |
793 vo_w32_ontop(); | |
794 return VO_TRUE; | |
795 case VOCTRL_BORDER: | |
796 vo_w32_border(); | |
28026 | 797 resize_d3d(); |
27921 | 798 return VO_TRUE; |
799 case VOCTRL_UPDATE_SCREENINFO: | |
800 w32_update_xinerama_info(); | |
801 return VO_TRUE; | |
802 case VOCTRL_SET_PANSCAN: | |
28027 | 803 calc_fs_rect(); |
27921 | 804 return VO_TRUE; |
805 case VOCTRL_GET_PANSCAN: | |
806 return VO_TRUE; | |
807 } | |
808 return VO_FALSE; | |
809 } | |
810 | |
811 /** @brief libvo Callback: Configre the Direct3D adapter. | |
812 * @param width Movie source width | |
813 * @param height Movie source height | |
814 * @param d_width Screen (destination) width | |
815 * @param d_height Screen (destination) height | |
816 * @param options Options bitmap | |
817 * @param title Window title | |
818 * @param format Movie colorspace format (using MPlayer's | |
819 * defines, e.g. IMGFMT_YUY2) | |
820 * @return 0 on success, VO_ERROR on failure | |
821 */ | |
822 static int config(uint32_t width, uint32_t height, uint32_t d_width, | |
823 uint32_t d_height, uint32_t options, char *title, | |
824 uint32_t format) | |
825 { | |
27928 | 826 |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
827 priv->src_width = width; |
27928 | 828 priv->src_height = height; |
27921 | 829 |
830 /* w32_common framework call. Creates window on the screen with | |
831 * the given coordinates. | |
832 */ | |
27928 | 833 if (!vo_w32_config(d_width, d_height, options)) { |
28433 | 834 mp_msg(MSGT_VO, MSGL_V, "<vo_direct3d>Creating onscreen window failed.\n"); |
27921 | 835 return VO_ERROR; |
836 } | |
837 | |
28026 | 838 /* "config" may be called several times, so if this is not the first |
839 * call, we should destroy Direct3D adapter and surfaces before | |
840 * calling configure_d3d, which will create them again. | |
841 */ | |
842 | |
843 destroy_d3d_surfaces(); | |
844 | |
845 /* Destroy the D3D Device */ | |
28085
f2606c3433ef
Slightly simplify the conditional release/free code
reimar
parents:
28084
diff
changeset
|
846 if (priv->d3d_device) |
28026 | 847 IDirect3DDevice9_Release(priv->d3d_device); |
28085
f2606c3433ef
Slightly simplify the conditional release/free code
reimar
parents:
28084
diff
changeset
|
848 priv->d3d_device = NULL; |
28026 | 849 |
850 if (!configure_d3d()) | |
27966 | 851 return VO_ERROR; |
852 | |
27921 | 853 return 0; /* Success */ |
854 } | |
855 | |
856 /** @brief libvo Callback: Flip next already drawn frame on the | |
857 * screen. | |
858 */ | |
859 static void flip_page(void) | |
860 { | |
28344
a69d51cd4ee3
Allocate a larger backbuffer to allow resizing without reinit.
reimar
parents:
28087
diff
changeset
|
861 RECT rect = {0, 0, vo_dwidth, vo_dheight}; |
28422
f7ba65621809
Add checks that a D3D device is available before attempting rendering.
reimar
parents:
28421
diff
changeset
|
862 if (!priv->d3d_device || |
f7ba65621809
Add checks that a D3D device is available before attempting rendering.
reimar
parents:
28421
diff
changeset
|
863 FAILED(IDirect3DDevice9_Present(priv->d3d_device, &rect, 0, 0, 0))) { |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
864 mp_msg(MSGT_VO, MSGL_V, |
28419
d4fbeeaa1c76
Cosmetics: remove empty line, improve some messages.
reimar
parents:
28418
diff
changeset
|
865 "<vo_direct3d>Trying to reinitialize uncooperative video adapter.\n"); |
27928 | 866 if (!reconfigure_d3d()) { |
28419
d4fbeeaa1c76
Cosmetics: remove empty line, improve some messages.
reimar
parents:
28418
diff
changeset
|
867 mp_msg(MSGT_VO, MSGL_V, "<vo_direct3d>Reinitialization failed.\n"); |
27921 | 868 return; |
869 } | |
870 else | |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
871 mp_msg(MSGT_VO, MSGL_V, "<vo_direct3d>Video adapter reinitialized.\n"); |
27921 | 872 } |
873 } | |
874 | |
875 /** @brief libvo Callback: Uninitializes all pointers and closes | |
876 * all D3D related stuff, | |
877 */ | |
878 static void uninit(void) | |
879 { | |
28433 | 880 mp_msg(MSGT_VO, MSGL_V, "<vo_direct3d>uninit called.\n"); |
27921 | 881 |
27928 | 882 uninit_d3d(); |
27921 | 883 vo_w32_uninit(); /* w32_common framework call */ |
30149
5f8cbd711a04
Avoid memleaks when vo_direct3d initialization fails.
reimar
parents:
30148
diff
changeset
|
884 if (priv->d3d9_dll) |
5f8cbd711a04
Avoid memleaks when vo_direct3d initialization fails.
reimar
parents:
30148
diff
changeset
|
885 FreeLibrary(priv->d3d9_dll); |
5f8cbd711a04
Avoid memleaks when vo_direct3d initialization fails.
reimar
parents:
30148
diff
changeset
|
886 priv->d3d9_dll = NULL; |
28027 | 887 free(priv); |
27928 | 888 priv = NULL; |
27921 | 889 } |
890 | |
891 /** @brief libvo Callback: Handles video window events. | |
892 */ | |
893 static void check_events(void) | |
894 { | |
895 int flags; | |
896 /* w32_common framework call. Handles video window events. | |
897 * Updates global libvo's vo_dwidth/vo_dheight upon resize | |
898 * with the new window width/height. | |
899 */ | |
900 flags = vo_w32_check_events(); | |
901 if (flags & VO_EVENT_RESIZE) | |
28026 | 902 resize_d3d(); |
27921 | 903 |
27928 | 904 if ((flags & VO_EVENT_EXPOSE) && priv->is_paused) |
27921 | 905 flip_page(); |
906 } | |
907 | |
908 /** @brief libvo Callback: Draw slice | |
909 * @return 0 on success | |
910 */ | |
911 static int draw_slice(uint8_t *src[], int stride[], int w,int h,int x,int y ) | |
912 { | |
27971 | 913 char *my_src; /**< Pointer to the source image */ |
914 char *dst; /**< Pointer to the destination image */ | |
915 int uv_stride; /**< Stride of the U/V planes */ | |
27921 | 916 |
28422
f7ba65621809
Add checks that a D3D device is available before attempting rendering.
reimar
parents:
28421
diff
changeset
|
917 /* If the D3D device is uncooperative (not initialized), return success. |
f7ba65621809
Add checks that a D3D device is available before attempting rendering.
reimar
parents:
28421
diff
changeset
|
918 The device will be probed for reinitialization in the next flip_page() */ |
f7ba65621809
Add checks that a D3D device is available before attempting rendering.
reimar
parents:
28421
diff
changeset
|
919 if (!priv->d3d_device) |
f7ba65621809
Add checks that a D3D device is available before attempting rendering.
reimar
parents:
28421
diff
changeset
|
920 return 0; |
f7ba65621809
Add checks that a D3D device is available before attempting rendering.
reimar
parents:
28421
diff
changeset
|
921 |
27969
7ddd69cf214f
Lock/unlock surface only once even when drawing many slices.
reimar
parents:
27968
diff
changeset
|
922 /* Lock the offscreen surface if it's not already locked. */ |
7ddd69cf214f
Lock/unlock surface only once even when drawing many slices.
reimar
parents:
27968
diff
changeset
|
923 if (!priv->locked_rect.pBits) { |
27970 | 924 if (FAILED(IDirect3DSurface9_LockRect(priv->d3d_surface, |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
925 &priv->locked_rect, NULL, 0))) { |
28433 | 926 mp_msg(MSGT_VO, MSGL_V, "<vo_direct3d>Surface lock failure.\n"); |
27970 | 927 return VO_FALSE; |
928 } | |
27969
7ddd69cf214f
Lock/unlock surface only once even when drawing many slices.
reimar
parents:
27968
diff
changeset
|
929 } |
27921 | 930 |
27971 | 931 uv_stride = priv->locked_rect.Pitch / 2; |
27921 | 932 |
933 /* Copy Y */ | |
27971 | 934 dst = priv->locked_rect.pBits; |
935 dst = dst + priv->locked_rect.Pitch * y + x; | |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
936 my_src = src[0]; |
27971 | 937 memcpy_pic(dst, my_src, w, h, priv->locked_rect.Pitch, stride[0]); |
27921 | 938 |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
939 w /= 2; |
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
940 h /= 2; |
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
941 x /= 2; |
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
942 y /= 2; |
27921 | 943 |
944 /* Copy U */ | |
27971 | 945 dst = priv->locked_rect.pBits; |
946 dst = dst + priv->locked_rect.Pitch * priv->src_height | |
947 + uv_stride * y + x; | |
27928 | 948 if (priv->movie_src_fmt == MAKEFOURCC('Y','V','1','2')) |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
949 my_src = src[2]; |
27921 | 950 else |
28011
7f7ffa0eb45a
whitespace cosmetics: prettyprinting and indentation
diego
parents:
28007
diff
changeset
|
951 my_src = src[1]; |
27921 | 952 |
27971 | 953 memcpy_pic(dst, my_src, w, h, uv_stride, stride[1]); |
27921 | 954 |
955 /* Copy V */ | |
27971 | 956 dst = priv->locked_rect.pBits; |
957 dst = dst + priv->locked_rect.Pitch * priv->src_height | |
958 + uv_stride * (priv->src_height / 2) + uv_stride * y + x; | |
27928 | 959 if (priv->movie_src_fmt == MAKEFOURCC('Y','V','1','2')) |
27971 | 960 my_src=src[1]; |
27921 | 961 else |
27971 | 962 my_src=src[2]; |
27921 | 963 |
27971 | 964 memcpy_pic(dst, my_src, w, h, uv_stride, stride[2]); |
27921 | 965 |
966 return 0; /* Success */ | |
967 } | |
968 | |
969 /** @brief libvo Callback: Unused function | |
970 */ | |
971 static int draw_frame(uint8_t *src[]) | |
972 { | |
28433 | 973 mp_msg(MSGT_VO, MSGL_V, "<vo_direct3d>draw_frame called.\n"); |
27921 | 974 return VO_FALSE; |
975 } | |
28084 | 976 |
977 /** @brief Maps MPlayer alpha to D3D | |
978 * 0x0 -> transparent and discarded by alpha test | |
979 * 0x1 -> 0xFF to become opaque | |
980 * other alpha values are inverted +1 (2 = -2) | |
981 * These values are then inverted again with | |
982 the texture filter D3DBLEND_INVSRCALPHA | |
983 */ | |
984 void vo_draw_alpha_l8a8(int w, int h, unsigned char* src, unsigned char *srca, | |
985 int srcstride, unsigned char* dstbase, int dststride) | |
986 { | |
987 int y; | |
988 for (y = 0; y < h; y++) { | |
989 unsigned short *dst = (unsigned short*)dstbase; | |
990 int x; | |
991 for (x = 0; x < w; x++) { | |
992 dst[x] = (-srca[x] << 8) | src[x]; | |
993 } | |
994 src += srcstride; | |
995 srca += srcstride; | |
996 dstbase += dststride; | |
997 } | |
998 } | |
999 | |
1000 /** @brief Callback function to render the OSD to the texture | |
1001 */ | |
1002 static void draw_alpha(int x0, int y0, int w, int h, unsigned char *src, | |
1003 unsigned char *srca, int stride) | |
1004 { | |
1005 D3DLOCKED_RECT locked_rect; /**< Offscreen surface we lock in order | |
1006 to copy MPlayer's frame inside it.*/ | |
1007 | |
1008 if (FAILED(IDirect3DTexture9_LockRect(priv->d3d_texture_system, 0, | |
1009 &locked_rect, NULL, 0))) { | |
28433 | 1010 mp_msg(MSGT_VO,MSGL_ERR,"<vo_direct3d>OSD texture lock failed.\n"); |
28084 | 1011 return; |
1012 } | |
1013 | |
1014 vo_draw_alpha_l8a8(w, h, src, srca, stride, | |
1015 (unsigned char *)locked_rect.pBits + locked_rect.Pitch*y0 + 2*x0, locked_rect.Pitch); | |
1016 | |
1017 /* this unlock is used for both slice_draw path and D3DRenderFrame path */ | |
1018 if (FAILED(IDirect3DTexture9_UnlockRect(priv->d3d_texture_system, 0))) { | |
28433 | 1019 mp_msg(MSGT_VO,MSGL_ERR,"<vo_direct3d>OSD texture unlock failed.\n"); |
28084 | 1020 return; |
1021 } | |
1022 | |
1023 priv->is_osd_populated = 1; | |
1024 } | |
1025 | |
1026 /** @brief libvo Callback: Draw OSD/Subtitles, | |
1027 */ | |
1028 static void draw_osd(void) | |
1029 { | |
28422
f7ba65621809
Add checks that a D3D device is available before attempting rendering.
reimar
parents:
28421
diff
changeset
|
1030 // we can not render OSD if we lost the device e.g. because it was uncooperative |
f7ba65621809
Add checks that a D3D device is available before attempting rendering.
reimar
parents:
28421
diff
changeset
|
1031 if (!priv->d3d_device) |
f7ba65621809
Add checks that a D3D device is available before attempting rendering.
reimar
parents:
28421
diff
changeset
|
1032 return; |
f7ba65621809
Add checks that a D3D device is available before attempting rendering.
reimar
parents:
28421
diff
changeset
|
1033 |
28084 | 1034 if (vo_osd_changed(0)) { |
1035 D3DLOCKED_RECT locked_rect; /**< Offscreen surface we lock in order | |
1036 to copy MPlayer's frame inside it.*/ | |
1037 | |
1038 /* clear the OSD */ | |
1039 if (FAILED(IDirect3DTexture9_LockRect(priv->d3d_texture_system, 0, | |
1040 &locked_rect, NULL, 0))) { | |
28433 | 1041 mp_msg(MSGT_VO,MSGL_ERR, "<vo_direct3d>OSD texture lock failed.\n"); |
28084 | 1042 return; |
1043 } | |
1044 | |
1045 /* clear the whole texture to avoid issues due to interpolation */ | |
1046 memset(locked_rect.pBits, 0, locked_rect.Pitch * priv->osd_texture_height); | |
1047 | |
1048 /* this unlock is used for both slice_draw path and D3DRenderFrame path */ | |
1049 if (FAILED(IDirect3DTexture9_UnlockRect(priv->d3d_texture_system, 0))) { | |
28433 | 1050 mp_msg(MSGT_VO,MSGL_ERR, "<vo_direct3d>OSD texture unlock failed.\n"); |
28084 | 1051 return; |
1052 } | |
1053 | |
1054 priv->is_osd_populated = 0; | |
1055 /* required for if subs are in the boarder region */ | |
1056 priv->is_clear_needed = 1; | |
1057 | |
28598
4172d83194f2
Extend calc_src_dst_rects to also calculate the border values needed for
reimar
parents:
28511
diff
changeset
|
1058 vo_draw_text_ext(priv->osd_width, priv->osd_height, priv->border_x, priv->border_y, |
4172d83194f2
Extend calc_src_dst_rects to also calculate the border values needed for
reimar
parents:
28511
diff
changeset
|
1059 priv->border_x, priv->border_y, priv->src_width, priv->src_height, draw_alpha); |
28084 | 1060 |
1061 if (!priv->device_texture_sys) | |
1062 { | |
1063 /* only DMA to the shadow if its required */ | |
1064 if (FAILED(IDirect3DDevice9_UpdateTexture(priv->d3d_device, | |
1065 (IDirect3DBaseTexture9 *)priv->d3d_texture_system, | |
1066 (IDirect3DBaseTexture9 *)priv->d3d_texture_osd))) { | |
28433 | 1067 mp_msg(MSGT_VO,MSGL_ERR, "<vo_direct3d>OSD texture transfer failed.\n"); |
28084 | 1068 return; |
1069 } | |
1070 } | |
1071 } | |
1072 | |
1073 /* update OSD */ | |
1074 | |
1075 if (priv->is_osd_populated) { | |
1076 | |
1077 struct_vertex osd_quad_vb[] = { | |
1078 {-1.0f, 1.0f, 0.0f, 0, 0 }, | |
1079 { 1.0f, 1.0f, 0.0f, 1, 0 }, | |
1080 {-1.0f,-1.0f, 0.0f, 0, 1 }, | |
1081 { 1.0f,-1.0f, 0.0f, 1, 1 } | |
1082 }; | |
1083 | |
1084 /* calculate the texture coordinates */ | |
1085 osd_quad_vb[1].tu = | |
1086 osd_quad_vb[3].tu = (float)priv->osd_width / priv->osd_texture_width; | |
1087 osd_quad_vb[2].tv = | |
1088 osd_quad_vb[3].tv = (float)priv->osd_height / priv->osd_texture_height; | |
1089 | |
1090 if (FAILED(IDirect3DDevice9_BeginScene(priv->d3d_device))) { | |
28433 | 1091 mp_msg(MSGT_VO,MSGL_ERR,"<vo_direct3d>BeginScene failed.\n"); |
28084 | 1092 return; |
1093 } | |
1094 | |
1095 /* turn on alpha test */ | |
1096 IDirect3DDevice9_SetRenderState(priv->d3d_device, D3DRS_ALPHABLENDENABLE, TRUE); | |
1097 IDirect3DDevice9_SetRenderState(priv->d3d_device, D3DRS_ALPHATESTENABLE, TRUE); | |
1098 | |
1099 /* need to use a texture here (done here as we may be able to texture from system memory) */ | |
1100 IDirect3DDevice9_SetTexture(priv->d3d_device, 0, | |
1101 (IDirect3DBaseTexture9 *)(priv->device_texture_sys | |
1102 ? priv->d3d_texture_system : priv->d3d_texture_osd)); | |
1103 | |
1104 IDirect3DDevice9_SetFVF(priv->d3d_device, D3DFVF_MY_VERTEX); | |
1105 IDirect3DDevice9_DrawPrimitiveUP(priv->d3d_device, D3DPT_TRIANGLESTRIP, 2, osd_quad_vb, sizeof(struct_vertex)); | |
1106 | |
1107 /* turn off alpha test */ | |
1108 IDirect3DDevice9_SetRenderState(priv->d3d_device, D3DRS_ALPHATESTENABLE, FALSE); | |
1109 IDirect3DDevice9_SetRenderState(priv->d3d_device, D3DRS_ALPHABLENDENABLE, FALSE); | |
1110 | |
1111 if (FAILED(IDirect3DDevice9_EndScene(priv->d3d_device))) { | |
28433 | 1112 mp_msg(MSGT_VO,MSGL_ERR,"<vo_direct3d>EndScene failed.\n"); |
28084 | 1113 return; |
1114 } | |
1115 } | |
1116 } |