aboutsummaryrefslogtreecommitdiffstats
path: root/libavfilter/vsrc_ddagrab.c
diff options
context:
space:
mode:
authorTimo Rothenpieler <timo@rothenpieler.org>2023-10-08 17:41:34 +0200
committerTimo Rothenpieler <timo@rothenpieler.org>2023-10-09 20:17:50 +0200
commit8b6ee74d50708e593c258726c5c29fddbf2765ab (patch)
tree9b703220409a7a88fac460850d94428bfd91ce5e /libavfilter/vsrc_ddagrab.c
parent0e7ae95cc289e5b5264ff2921f361eca3702aab1 (diff)
downloadffmpeg-8b6ee74d50708e593c258726c5c29fddbf2765ab.tar.gz
avfilter/ddagrab: create secondary xor mouse texture
Diffstat (limited to 'libavfilter/vsrc_ddagrab.c')
-rw-r--r--libavfilter/vsrc_ddagrab.c137
1 files changed, 112 insertions, 25 deletions
diff --git a/libavfilter/vsrc_ddagrab.c b/libavfilter/vsrc_ddagrab.c
index 936936eb7f..9c59faf53e 100644
--- a/libavfilter/vsrc_ddagrab.c
+++ b/libavfilter/vsrc_ddagrab.c
@@ -68,7 +68,9 @@ typedef struct DdagrabContext {
int mouse_x, mouse_y;
ID3D11Texture2D *mouse_texture;
- ID3D11ShaderResourceView* mouse_resource_view ;
+ ID3D11ShaderResourceView* mouse_resource_view;
+ ID3D11Texture2D *mouse_xor_texture;
+ ID3D11ShaderResourceView* mouse_xor_resource_view;
AVRational time_base;
int64_t time_frame;
@@ -87,6 +89,7 @@ typedef struct DdagrabContext {
ID3D11Buffer *const_buffer;
ID3D11SamplerState *sampler_state;
ID3D11BlendState *blend_state;
+ ID3D11BlendState *blend_state_xor;
int output_idx;
int draw_mouse;
@@ -140,6 +143,7 @@ static av_cold void ddagrab_uninit(AVFilterContext *avctx)
DdagrabContext *dda = avctx->priv;
release_resource(&dda->blend_state);
+ release_resource(&dda->blend_state_xor);
release_resource(&dda->sampler_state);
release_resource(&dda->pixel_shader);
release_resource(&dda->input_layout);
@@ -151,6 +155,8 @@ static av_cold void ddagrab_uninit(AVFilterContext *avctx)
release_resource(&dda->dxgi_outdupl);
release_resource(&dda->mouse_resource_view);
release_resource(&dda->mouse_texture);
+ release_resource(&dda->mouse_xor_resource_view);
+ release_resource(&dda->mouse_xor_texture);
av_frame_free(&dda->last_frame);
av_buffer_unref(&dda->frames_ref);
@@ -412,6 +418,16 @@ static av_cold int init_render_resources(AVFilterContext *avctx)
return AVERROR_EXTERNAL;
}
+ blend_desc.RenderTarget[0].SrcBlend = D3D11_BLEND_INV_DEST_COLOR;
+ blend_desc.RenderTarget[0].DestBlend = D3D11_BLEND_INV_SRC_COLOR;
+ hr = ID3D11Device_CreateBlendState(dev,
+ &blend_desc,
+ &dda->blend_state_xor);
+ if (FAILED(hr)) {
+ av_log(avctx, AV_LOG_ERROR, "CreateBlendState (xor) failed: %lx\n", hr);
+ return AVERROR_EXTERNAL;
+ }
+
return 0;
}
@@ -482,50 +498,102 @@ static int create_d3d11_pointer_tex(AVFilterContext *avctx,
return 0;
}
-static uint8_t *convert_mono_buffer(uint8_t *input, int *_width, int *_height, int *_pitch)
+static int convert_mono_buffer(uint8_t *input, uint8_t **rgba_out, uint8_t **xor_out, int *_width, int *_height, int *_pitch)
{
int width = *_width, height = *_height, pitch = *_pitch;
int real_height = height / 2;
+ int size = real_height * pitch;
+
uint8_t *output = av_malloc(real_height * width * 4);
+ uint8_t *output_xor = av_malloc(real_height * width * 4);
+
int y, x;
- if (!output)
- return NULL;
+ if (!output || !output_xor) {
+ av_free(output);
+ av_free(output_xor);
+ return AVERROR(ENOMEM);
+ }
- // This simulates drawing the cursor on a full black surface
- // i.e. ignore the AND mask, turn XOR mask into all 4 color channels
for (y = 0; y < real_height; y++) {
for (x = 0; x < width; x++) {
- int v = input[(real_height + y) * pitch + (x / 8)];
- v = (v >> (7 - (x % 8))) & 1;
- memset(&output[4 * ((y*width) + x)], v ? 0xFF : 0, 4);
+ int in_pos = (y * pitch) + (x / 8);
+ int out_pos = 4 * ((y * width) + x);
+ int and_val = (input[in_pos] >> (7 - (x % 8))) & 1;
+ int xor_val = (input[in_pos + size] >> (7 - (x % 8))) & 1;
+
+ if (!and_val && !xor_val) {
+ // solid black
+ memset(&output[out_pos], 0, 4);
+ output[out_pos + 3] = 0xFF;
+
+ // transparent
+ memset(&output_xor[out_pos], 0, 4);
+ } else if (and_val && !xor_val) {
+ // transparent
+ memset(&output[out_pos], 0, 4);
+
+ // transparent
+ memset(&output_xor[out_pos], 0, 4);
+ } else if (!and_val && xor_val) {
+ // solid white
+ memset(&output[out_pos], 0xFF, 4);
+
+ // transparent
+ memset(&output_xor[out_pos], 0, 4);
+ } else if (and_val && xor_val) {
+ // transparent
+ memset(&output[out_pos], 0, 4);
+
+ // solid white -> invert color
+ memset(&output_xor[out_pos], 0xFF, 4);
+ }
}
}
*_pitch = width * 4;
*_height = real_height;
+ *rgba_out = output;
+ *xor_out = output_xor;
- return output;
+ return 0;
}
-static void fixup_color_mask(uint8_t *buf, int width, int height, int pitch)
+static int fixup_color_mask(uint8_t *input, uint8_t **rgba_out, uint8_t **xor_out, int width, int height, int pitch)
{
+ int size = height * pitch;
+ uint8_t *output = av_malloc(size);
+ uint8_t *output_xor = av_malloc(size);
int x, y;
- // There is no good way to replicate XOR'ig parts of the texture with the screen
- // best effort is rendering the non-masked parts, and make the rest transparent
+
+ if (!output || !output_xor) {
+ av_free(output);
+ av_free(output_xor);
+ return AVERROR(ENOMEM);
+ }
+
+ memcpy(output, input, size);
+ memcpy(output_xor, input, size);
+
for (y = 0; y < height; y++) {
for (x = 0; x < width; x++) {
int pos = (y*pitch) + (4*x) + 3;
- buf[pos] = buf[pos] ? 0 : 0xFF;
+ output[pos] = input[pos] ? 0 : 0xFF;
+ output_xor[pos] = input[pos] ? 0xFF : 0;
}
}
+
+ *rgba_out = output;
+ *xor_out = output_xor;
+
+ return 0;
}
static int update_mouse_pointer(AVFilterContext *avctx, DXGI_OUTDUPL_FRAME_INFO *frame_info)
{
DdagrabContext *dda = avctx->priv;
HRESULT hr;
- int ret;
+ int ret, ret2;
if (frame_info->LastMouseUpdateTime.QuadPart == 0)
return 0;
@@ -555,6 +623,7 @@ static int update_mouse_pointer(AVFilterContext *avctx, DXGI_OUTDUPL_FRAME_INFO
if (frame_info->PointerShapeBufferSize) {
UINT size = frame_info->PointerShapeBufferSize;
DXGI_OUTDUPL_POINTER_SHAPE_INFO shape_info;
+ uint8_t *rgba_buf = NULL, *rgb_xor_buf = NULL;
uint8_t *buf = av_malloc(size);
if (!buf)
return AVERROR(ENOMEM);
@@ -571,26 +640,37 @@ static int update_mouse_pointer(AVFilterContext *avctx, DXGI_OUTDUPL_FRAME_INFO
}
if (shape_info.Type == DXGI_OUTDUPL_POINTER_SHAPE_TYPE_MONOCHROME) {
- uint8_t *new_buf = convert_mono_buffer(buf, &shape_info.Width, &shape_info.Height, &shape_info.Pitch);
- av_free(buf);
- if (!new_buf)
- return AVERROR(ENOMEM);
- buf = new_buf;
+ ret = convert_mono_buffer(buf, &rgba_buf, &rgb_xor_buf, &shape_info.Width, &shape_info.Height, &shape_info.Pitch);
+ av_freep(&buf);
+ if (ret < 0)
+ return ret;
} else if (shape_info.Type == DXGI_OUTDUPL_POINTER_SHAPE_TYPE_MASKED_COLOR) {
- fixup_color_mask(buf, shape_info.Width, shape_info.Height, shape_info.Pitch);
- } else if (shape_info.Type != DXGI_OUTDUPL_POINTER_SHAPE_TYPE_COLOR) {
+ ret = fixup_color_mask(buf, &rgba_buf, &rgb_xor_buf, shape_info.Width, shape_info.Height, shape_info.Pitch);
+ av_freep(&buf);
+ if (ret < 0)
+ return ret;
+ } else if (shape_info.Type == DXGI_OUTDUPL_POINTER_SHAPE_TYPE_COLOR) {
+ rgba_buf = buf;
+ buf = NULL;
+ } else {
av_log(avctx, AV_LOG_WARNING, "Unsupported pointer shape type: %d\n", (int)shape_info.Type);
- av_free(buf);
+ av_freep(&buf);
return 0;
}
release_resource(&dda->mouse_resource_view);
release_resource(&dda->mouse_texture);
+ release_resource(&dda->mouse_xor_resource_view);
+ release_resource(&dda->mouse_xor_texture);
- ret = create_d3d11_pointer_tex(avctx, buf, &shape_info, &dda->mouse_texture, &dda->mouse_resource_view);
- av_freep(&buf);
+ ret = create_d3d11_pointer_tex(avctx, rgba_buf, &shape_info, &dda->mouse_texture, &dda->mouse_resource_view);
+ ret2 = rgb_xor_buf ? create_d3d11_pointer_tex(avctx, rgb_xor_buf, &shape_info, &dda->mouse_xor_texture, &dda->mouse_xor_resource_view) : 0;
+ av_freep(&rgba_buf);
+ av_freep(&rgb_xor_buf);
if (ret < 0)
return ret;
+ if (ret2 < 0)
+ return ret2;
av_log(avctx, AV_LOG_VERBOSE, "Updated pointer shape texture\n");
}
@@ -934,6 +1014,13 @@ static int draw_mouse_pointer(AVFilterContext *avctx, AVFrame *frame)
ID3D11DeviceContext_Draw(devctx, num_vertices, 0);
+ if (dda->mouse_xor_resource_view) {
+ ID3D11DeviceContext_PSSetShaderResources(devctx, 0, 1, &dda->mouse_xor_resource_view);
+ ID3D11DeviceContext_OMSetBlendState(devctx, dda->blend_state_xor, NULL, 0xFFFFFFFF);
+
+ ID3D11DeviceContext_Draw(devctx, num_vertices, 0);
+ }
+
end:
release_resource(&mouse_vertex_buffer);
release_resource(&target_view);