aboutsummaryrefslogtreecommitdiffhomepage
path: root/video/vdpau.h
blob: b0f402c3f2af373deb2728e741a7caa185c3a6a0 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
#ifndef MPV_VDPAU_H
#define MPV_VDPAU_H

#include <stdbool.h>
#include <inttypes.h>

#include <pthread.h>

#include <vdpau/vdpau.h>
#include <vdpau/vdpau_x11.h>

#include "common/msg.h"
#include "hwdec.h"

#include "config.h"
#if !HAVE_GPL
#error GPL only
#endif

#define CHECK_VDP_ERROR_ST(ctx, message, statement) \
    do { \
        if (vdp_st != VDP_STATUS_OK) { \
            MP_ERR(ctx, "%s: %s\n", message, vdp->get_error_string(vdp_st)); \
            statement \
        } \
    } while (0)

#define CHECK_VDP_ERROR(ctx, message) \
    CHECK_VDP_ERROR_ST(ctx, message, return -1;)

#define CHECK_VDP_ERROR_NORETURN(ctx, message) \
    CHECK_VDP_ERROR_ST(ctx, message, ;)

#define CHECK_VDP_WARNING(ctx, message) \
    do { \
        if (vdp_st != VDP_STATUS_OK) \
            MP_WARN(ctx, "%s: %s\n", message, vdp->get_error_string(vdp_st)); \
    } while (0)

struct vdp_functions {
#define VDP_FUNCTION(vdp_type, _, mp_name) vdp_type *mp_name;
#include "video/vdpau_functions.inc"
#undef VDP_FUNCTION
};


#define MAX_VIDEO_SURFACES 50

// Shared state. Objects created from different VdpDevices are often (always?)
// incompatible to each other, so all code must use a shared VdpDevice.
struct mp_vdpau_ctx {
    struct mp_log *log;
    Display *x11;
    bool close_display;

    struct mp_hwdec_ctx hwctx;
    struct AVBufferRef *av_device_ref;

    // These are mostly immutable, except on preemption. We don't really care
    // to synchronize the preemption case fully correctly, because it's an
    // extremely obscure corner case, and basically a vdpau API design bug.
    // What we do will sort-of work anyway (no memory errors are possible).
    struct vdp_functions vdp;
    VdpGetProcAddress *get_proc_address;
    VdpDevice vdp_device;

    pthread_mutex_t preempt_lock;
    bool is_preempted;                  // set to true during unavailability
    uint64_t preemption_counter;        // incremented after _restoring_
    bool preemption_user_notified;
    double last_preemption_retry_fail;
    VdpOutputSurface preemption_obj;    // dummy for reliable preempt. check

    // Surface pool
    pthread_mutex_t pool_lock;
    int64_t age_counter;
    struct surface_entry {
        VdpVideoSurface surface;
        VdpOutputSurface osurface;
        bool allocated;
        int w, h;
        VdpRGBAFormat rgb_format;
        VdpChromaType chroma;
        bool rgb;
        bool in_use;
        int64_t age;
    } video_surfaces[MAX_VIDEO_SURFACES];
};

struct mp_vdpau_ctx *mp_vdpau_create_device_x11(struct mp_log *log, Display *x11,
                                                bool probing);
void mp_vdpau_destroy(struct mp_vdpau_ctx *ctx);

int mp_vdpau_handle_preemption(struct mp_vdpau_ctx *ctx, uint64_t *counter);

struct mp_image *mp_vdpau_get_video_surface(struct mp_vdpau_ctx *ctx,
                                            VdpChromaType chroma, int w, int h);

bool mp_vdpau_get_format(int imgfmt, VdpChromaType *out_chroma_type,
                         VdpYCbCrFormat *out_pixel_format);
bool mp_vdpau_get_rgb_format(int imgfmt, VdpRGBAFormat *out_rgba_format);

struct mp_image *mp_vdpau_upload_video_surface(struct mp_vdpau_ctx *ctx,
                                               struct mp_image *mpi);

struct mp_vdpau_ctx *mp_vdpau_get_ctx_from_av(struct AVBufferRef *hw_device_ctx);

bool mp_vdpau_guess_if_emulated(struct mp_vdpau_ctx *ctx);

#endif