#pragma once #include #include #include "helpers.hpp" #include "kernel_types.hpp" #include "logger.hpp" #include "memory.hpp" // Circular dependencies go br class Kernel; class Y2RService { using Handle = HorizonHandle; Handle handle = KernelHandles::Y2R; Memory& mem; Kernel& kernel; MAKE_LOG_FUNCTION(log, y2rLogger) std::optional transferEndEvent; bool transferEndInterruptEnabled; enum class BusyStatus : u32 { NotBusy = 0, Busy = 1, }; enum class InputFormat : u32 { YUV422_Individual8 = 0, YUV420_Individual8 = 1, YUV422_Individual16 = 2, YUV420_Individual16 = 3, YUV422_Batch = 4, }; enum class OutputFormat : u32 { RGB32 = 0, RGB24 = 1, RGB15 = 2, RGB565 = 3, }; // Clockwise rotation enum class Rotation : u32 { None = 0, Rotate90 = 1, Rotate180 = 2, Rotate270 = 3, }; enum class BlockAlignment : u32 { Line = 0, // Output buffer's pixels are arranged linearly. Used when outputting to the framebuffer. Block8x8 = 1, // Output buffer's pixels are morton swizzled. Used when outputting to a GPU texture. }; // https://github.com/citra-emu/citra/blob/ac9d72a95ca9a60de8d39484a14aecf489d6d016/src/core/hle/service/cam/y2r_u.cpp#L33 using CoefficientSet = std::array; static constexpr std::array standardCoefficients{{ {{0x100, 0x166, 0xB6, 0x58, 0x1C5, -0x166F, 0x10EE, -0x1C5B}}, // ITU_Rec601 {{0x100, 0x193, 0x77, 0x2F, 0x1DB, -0x1933, 0xA7C, -0x1D51}}, // ITU_Rec709 {{0x12A, 0x198, 0xD0, 0x64, 0x204, -0x1BDE, 0x10F2, -0x229B}}, // ITU_Rec601_Scaling {{0x12A, 0x1CA, 0x88, 0x36, 0x21C, -0x1F04, 0x99C, -0x2421}}, // ITU_Rec709_Scaling }}; CoefficientSet conversionCoefficients; // Current conversion coefficients InputFormat inputFmt; OutputFormat outputFmt; Rotation rotation; BlockAlignment alignment; bool spacialDithering; bool temporalDithering; u16 alpha; u16 inputLineWidth; u16 inputLines; // Service commands void driverInitialize(u32 messagePointer); void driverFinalize(u32 messagePointer); void getTransferEndEvent(u32 messagePointer); void getBlockAlignment(u32 messagePointer); void getInputLines(u32 messagePointer); void getInputLineWidth(u32 messagePointer); void getOutputFormat(u32 messagePointer); void isBusyConversion(u32 messagePointer); void isFinishedReceiving(u32 messagePointer); void isFinishedSendingY(u32 messagePointer); void isFinishedSendingU(u32 messagePointer); void isFinishedSendingV(u32 messagePointer); void isFinishedSendingYUV(u32 messagePointer); void pingProcess(u32 messagePointer); void setTransferEndInterrupt(u32 messagePointer); void setAlpha(u32 messagePointer); void setBlockAlignment(u32 messagePointer); void setCoefficientParams(u32 messagePointer); void setInputFormat(u32 messagePointer); void setInputLineWidth(u32 messagePointer); void setInputLines(u32 messagePointer); void setOutputFormat(u32 messagePointer); void setPackageParameter(u32 messagePointer); void setReceiving(u32 messagePointer); void setRotation(u32 messagePointer); void setSendingY(u32 messagePointer); void setSendingU(u32 messagePointer); void setSendingV(u32 messagePointer); void setSendingYUV(u32 messagePointer); void setSpacialDithering(u32 messagePointer); void setStandardCoeff(u32 messagePointer); void setTemporalDithering(u32 messagePointer); void getCoefficientParams(u32 messagePointer); void getStandardCoefficientParams(u32 messagePointer); void startConversion(u32 messagePointer); void stopConversion(u32 messagePointer); bool isBusy; public: Y2RService(Memory& mem, Kernel& kernel) : mem(mem), kernel(kernel) {} void reset(); void handleSyncRequest(u32 messagePointer); void signalConversionDone(); };