aboutsummaryrefslogtreecommitdiff
path: root/Ryujinx.Graphics/Vic/VideoImageComposer.cs
blob: e05bcfdb65227f042dfb842f009ca2eae57a486f (plain) (blame)
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
using Ryujinx.Graphics.Memory;

namespace Ryujinx.Graphics.Vic
{
    class VideoImageComposer
    {
        private NvGpu _gpu;

        private long _configStructAddress;
        private long _outputSurfaceLumaAddress;
        private long _outputSurfaceChromaUAddress;
        private long _outputSurfaceChromaVAddress;

        public VideoImageComposer(NvGpu gpu)
        {
            _gpu = gpu;
        }

        public void Process(NvGpuVmm vmm, int methodOffset, int[] arguments)
        {
            VideoImageComposerMeth method = (VideoImageComposerMeth)methodOffset;

            switch (method)
            {
                case VideoImageComposerMeth.Execute:
                    Execute(vmm, arguments);
                    break;

                case VideoImageComposerMeth.SetConfigStructOffset:
                    SetConfigStructOffset(vmm, arguments);
                    break;

                case VideoImageComposerMeth.SetOutputSurfaceLumaOffset:
                    SetOutputSurfaceLumaOffset(vmm, arguments);
                    break;

                case VideoImageComposerMeth.SetOutputSurfaceChromaUOffset:
                    SetOutputSurfaceChromaUOffset(vmm, arguments);
                    break;

                case VideoImageComposerMeth.SetOutputSurfaceChromaVOffset:
                    SetOutputSurfaceChromaVOffset(vmm, arguments);
                    break;
            }
        }

        private void Execute(NvGpuVmm vmm, int[] arguments)
        {
            StructUnpacker unpacker = new StructUnpacker(vmm, _configStructAddress + 0x20);

            SurfacePixelFormat pixelFormat = (SurfacePixelFormat)unpacker.Read(7);

            int chromaLocHoriz = unpacker.Read(2);
            int chromaLocVert  = unpacker.Read(2);

            int blockLinearKind       = unpacker.Read(4);
            int blockLinearHeightLog2 = unpacker.Read(4);

            int reserved0 = unpacker.Read(3);
            int reserved1 = unpacker.Read(10);

            int surfaceWidthMinus1  = unpacker.Read(14);
            int surfaceHeightMinus1 = unpacker.Read(14);

            int gobBlockHeight = 1 << blockLinearHeightLog2;

            int surfaceWidth  = surfaceWidthMinus1  + 1;
            int surfaceHeight = surfaceHeightMinus1 + 1;

            SurfaceOutputConfig outputConfig = new SurfaceOutputConfig(
                pixelFormat,
                surfaceWidth,
                surfaceHeight,
                gobBlockHeight,
                _outputSurfaceLumaAddress,
                _outputSurfaceChromaUAddress,
                _outputSurfaceChromaVAddress);

            _gpu.VideoDecoder.CopyPlanes(vmm, outputConfig);
        }

        private void SetConfigStructOffset(NvGpuVmm vmm, int[] arguments)
        {
            _configStructAddress = GetAddress(arguments);
        }

        private void SetOutputSurfaceLumaOffset(NvGpuVmm vmm, int[] arguments)
        {
            _outputSurfaceLumaAddress = GetAddress(arguments);
        }

        private void SetOutputSurfaceChromaUOffset(NvGpuVmm vmm, int[] arguments)
        {
            _outputSurfaceChromaUAddress = GetAddress(arguments);
        }

        private void SetOutputSurfaceChromaVOffset(NvGpuVmm vmm, int[] arguments)
        {
            _outputSurfaceChromaVAddress = GetAddress(arguments);
        }

        private static long GetAddress(int[] arguments)
        {
            return (long)(uint)arguments[0] << 8;
        }
    }
}