summaryrefslogtreecommitdiff
path: root/libavutil/hwcontext_dxva2.c
diff options
context:
space:
mode:
authorAaron Levinson <alevinsn@aracnet.com>2017-05-16 05:04:36 -0700
committerMark Thompson <sw@jkqxz.net>2017-05-16 21:45:52 +0100
commit0c1c514643d5e1645160d697fa4c27cd38c7c791 (patch)
tree9f12f38c010f7b4615639a37c9e02ac9374910f5 /libavutil/hwcontext_dxva2.c
parent605c5ca3123a051cfb37ad52f608ffd2f6a91abf (diff)
downloadffmpeg-0c1c514643d5e1645160d697fa4c27cd38c7c791.tar.gz
avutil/hwcontext_dxva2: Don't improperly free IDirect3DSurface9 objects
Add dxva2_pool_release_dummy() and use it in call to av_buffer_create() in dxva2_pool_alloc(). Prior to this change, av_buffer_create() was called with NULL for the third argument, which indicates that av_buffer_default_free() should be used to free the buffer's data. Eventually, it gets to buffer_pool_free() and calls buf->free() on a surface object (which is av_buffer_default_free()). This can result in a crash when the debug version of the C-runtime is used on Windows. While it doesn't appear to result in a crash when the release version of the C-runtime is used on Windows, it likely results in memory corruption, since av_free() is being called on memory that was allocated using IDirectXVideoAccelerationService::CreateSurface(). Signed-off-by: Aaron Levinson <alevinsn@aracnet.com> Reviewed-by: wm4 <nfxjfg@googlemail.com> Reviewed-by: Steven Liu <lingjiujianke@gmail.com> Reviewed-by: Mark Thompson <sw@jkqxz.net>
Diffstat (limited to 'libavutil/hwcontext_dxva2.c')
-rw-r--r--libavutil/hwcontext_dxva2.c9
1 files changed, 8 insertions, 1 deletions
diff --git a/libavutil/hwcontext_dxva2.c b/libavutil/hwcontext_dxva2.c
index 4ed0d56aea..6c41788e2e 100644
--- a/libavutil/hwcontext_dxva2.c
+++ b/libavutil/hwcontext_dxva2.c
@@ -121,6 +121,13 @@ static void dxva2_frames_uninit(AVHWFramesContext *ctx)
}
}
+static void dxva2_pool_release_dummy(void *opaque, uint8_t *data)
+{
+ // important not to free anything here--data is a surface object
+ // associated with the call to CreateSurface(), and these surfaces are
+ // released in dxva2_frames_uninit()
+}
+
static AVBufferRef *dxva2_pool_alloc(void *opaque, int size)
{
AVHWFramesContext *ctx = (AVHWFramesContext*)opaque;
@@ -130,7 +137,7 @@ static AVBufferRef *dxva2_pool_alloc(void *opaque, int size)
if (s->nb_surfaces_used < hwctx->nb_surfaces) {
s->nb_surfaces_used++;
return av_buffer_create((uint8_t*)s->surfaces_internal[s->nb_surfaces_used - 1],
- sizeof(*hwctx->surfaces), NULL, 0, 0);
+ sizeof(*hwctx->surfaces), dxva2_pool_release_dummy, 0, 0);
}
return NULL;