| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | // Copyright 2015 Citra Emulator Project
 | 
					
						
							| 
									
										
										
										
											2015-01-02 15:03:40 -03:00
										 |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | #include <cstring>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-02 15:03:40 -03:00
										 |  |  | #include "core/hle/hle.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/event.h"
 | 
					
						
							|  |  |  | #include "core/hle/service/y2r_u.h"
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  | #include "core/hw/y2r.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-22 14:49:42 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-21 14:58:59 +01:00
										 |  |  | #include "video_core/renderer_base.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | #include "video_core/utils.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-22 14:49:42 -07:00
										 |  |  | #include "video_core/video_core.h"
 | 
					
						
							| 
									
										
										
										
											2015-01-02 15:03:40 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | ////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | // Namespace Y2R_U
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Y2R_U { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | struct ConversionParameters { | 
					
						
							|  |  |  |     InputFormat input_format; | 
					
						
							|  |  |  |     OutputFormat output_format; | 
					
						
							|  |  |  |     Rotation rotation; | 
					
						
							| 
									
										
										
										
											2015-06-03 17:54:24 -03:00
										 |  |  |     BlockAlignment block_alignment; | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  |     u16 input_line_width; | 
					
						
							|  |  |  |     u16 input_lines; | 
					
						
							| 
									
										
										
										
											2015-06-03 17:54:24 -03:00
										 |  |  |     StandardCoefficient standard_coefficient; | 
					
						
							|  |  |  |     u8 reserved; | 
					
						
							|  |  |  |     u16 alpha; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | static_assert(sizeof(ConversionParameters) == 12, "ConversionParameters struct has incorrect size"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  | static Kernel::SharedPtr<Kernel::Event> completion_event; | 
					
						
							|  |  |  | static ConversionConfiguration conversion; | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  | static const CoefficientSet standard_coefficients[4] = { | 
					
						
							|  |  |  |     {{ 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
 | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  | ResultCode ConversionConfiguration::SetInputLineWidth(u16 width) { | 
					
						
							|  |  |  |     if (width == 0 || width > 1024 || width % 8 != 0) { | 
					
						
							|  |  |  |         return ResultCode(ErrorDescription::OutOfRange, ErrorModule::CAM, | 
					
						
							|  |  |  |             ErrorSummary::InvalidArgument, ErrorLevel::Usage); // 0xE0E053FD
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Note: The hardware uses the register value 0 to represent a width of 1024, so for a width of
 | 
					
						
							|  |  |  |     // 1024 the `camera` module would set the value 0 here, but we don't need to emulate this
 | 
					
						
							|  |  |  |     // internal detail.
 | 
					
						
							|  |  |  |     this->input_line_width = width; | 
					
						
							|  |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ResultCode ConversionConfiguration::SetInputLines(u16 lines) { | 
					
						
							|  |  |  |     if (lines == 0 || lines > 1024) { | 
					
						
							|  |  |  |         return ResultCode(ErrorDescription::OutOfRange, ErrorModule::CAM, | 
					
						
							|  |  |  |             ErrorSummary::InvalidArgument, ErrorLevel::Usage); // 0xE0E053FD
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Note: In what appears to be a bug, the `camera` module does not set the hardware register at
 | 
					
						
							|  |  |  |     // all if `lines` is 1024, so the conversion uses the last value that was set. The intention
 | 
					
						
							|  |  |  |     // was probably to set it to 0 like in SetInputLineWidth.
 | 
					
						
							|  |  |  |     if (lines != 1024) { | 
					
						
							|  |  |  |         this->input_lines = lines; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ResultCode ConversionConfiguration::SetStandardCoefficient(StandardCoefficient standard_coefficient) { | 
					
						
							|  |  |  |     size_t index = static_cast<size_t>(standard_coefficient); | 
					
						
							|  |  |  |     if (index >= 4) { | 
					
						
							|  |  |  |         return ResultCode(ErrorDescription::InvalidEnumValue, ErrorModule::CAM, | 
					
						
							|  |  |  |             ErrorSummary::InvalidArgument, ErrorLevel::Usage); // 0xE0E053ED
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::memcpy(coefficients.data(), standard_coefficients[index].data(), sizeof(coefficients)); | 
					
						
							|  |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void SetInputFormat(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2015-01-10 22:46:44 -08:00
										 |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     conversion.input_format = static_cast<InputFormat>(cmd_buff[1]); | 
					
						
							|  |  |  |     LOG_DEBUG(Service_Y2R, "called input_format=%hhu", conversion.input_format); | 
					
						
							| 
									
										
										
										
											2015-01-10 22:46:44 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void SetOutputFormat(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     conversion.output_format = static_cast<OutputFormat>(cmd_buff[1]); | 
					
						
							|  |  |  |     LOG_DEBUG(Service_Y2R, "called output_format=%hhu", conversion.output_format); | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void SetRotation(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     conversion.rotation = static_cast<Rotation>(cmd_buff[1]); | 
					
						
							|  |  |  |     LOG_DEBUG(Service_Y2R, "called rotation=%hhu", conversion.rotation); | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void SetBlockAlignment(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     conversion.block_alignment = static_cast<BlockAlignment>(cmd_buff[1]); | 
					
						
							|  |  |  |     LOG_DEBUG(Service_Y2R, "called alignment=%hhu", conversion.block_alignment); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void SetTransferEndInterrupt(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-21 22:47:55 -03:00
										 |  |  |     cmd_buff[0] = IPC::MakeHeader(0xD, 1, 0); | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     LOG_DEBUG(Service_Y2R, "(STUBBED) called"); | 
					
						
							| 
									
										
										
										
											2015-01-10 22:46:44 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-24 08:28:36 -05:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2015-06-03 17:54:24 -03:00
										 |  |  |  * Y2R_U::GetTransferEndEvent service function | 
					
						
							|  |  |  |  *  Outputs: | 
					
						
							|  |  |  |  *      1 : Result of function, 0 on success, otherwise error code | 
					
						
							|  |  |  |  *      3 : The handle of the completion event | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2015-02-24 08:28:36 -05:00
										 |  |  | static void GetTransferEndEvent(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-24 08:28:36 -05:00
										 |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							|  |  |  |     cmd_buff[3] = Kernel::g_handle_table.Create(completion_event).MoveFrom(); | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  |     LOG_DEBUG(Service_Y2R, "called"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void SetSendingY(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-03 17:54:24 -03:00
										 |  |  |     conversion.src_Y.address = cmd_buff[1]; | 
					
						
							|  |  |  |     conversion.src_Y.image_size = cmd_buff[2]; | 
					
						
							|  |  |  |     conversion.src_Y.transfer_unit = cmd_buff[3]; | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     conversion.src_Y.gap = cmd_buff[4]; | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  |     u32 src_process_handle = cmd_buff[6]; | 
					
						
							|  |  |  |     LOG_DEBUG(Service_Y2R, "called image_size=0x%08X, transfer_unit=%hu, transfer_stride=%hu, " | 
					
						
							| 
									
										
										
										
											2015-06-03 17:54:24 -03:00
										 |  |  |         "src_process_handle=0x%08X", conversion.src_Y.image_size, | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |         conversion.src_Y.transfer_unit, conversion.src_Y.gap, src_process_handle); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void SetSendingU(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     conversion.src_U.address = cmd_buff[1]; | 
					
						
							|  |  |  |     conversion.src_U.image_size = cmd_buff[2]; | 
					
						
							|  |  |  |     conversion.src_U.transfer_unit = cmd_buff[3]; | 
					
						
							|  |  |  |     conversion.src_U.gap = cmd_buff[4]; | 
					
						
							|  |  |  |     u32 src_process_handle = cmd_buff[6]; | 
					
						
							|  |  |  |     LOG_DEBUG(Service_Y2R, "called image_size=0x%08X, transfer_unit=%hu, transfer_stride=%hu, " | 
					
						
							|  |  |  |         "src_process_handle=0x%08X", conversion.src_U.image_size, | 
					
						
							|  |  |  |         conversion.src_U.transfer_unit, conversion.src_U.gap, src_process_handle); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void SetSendingV(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     conversion.src_V.address = cmd_buff[1]; | 
					
						
							|  |  |  |     conversion.src_V.image_size = cmd_buff[2]; | 
					
						
							|  |  |  |     conversion.src_V.transfer_unit = cmd_buff[3]; | 
					
						
							|  |  |  |     conversion.src_V.gap = cmd_buff[4]; | 
					
						
							|  |  |  |     u32 src_process_handle = cmd_buff[6]; | 
					
						
							|  |  |  |     LOG_DEBUG(Service_Y2R, "called image_size=0x%08X, transfer_unit=%hu, transfer_stride=%hu, " | 
					
						
							|  |  |  |         "src_process_handle=0x%08X", conversion.src_V.image_size, | 
					
						
							|  |  |  |         conversion.src_V.transfer_unit, conversion.src_V.gap, src_process_handle); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void SetSendingYUYV(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     conversion.src_YUYV.address = cmd_buff[1]; | 
					
						
							|  |  |  |     conversion.src_YUYV.image_size = cmd_buff[2]; | 
					
						
							|  |  |  |     conversion.src_YUYV.transfer_unit = cmd_buff[3]; | 
					
						
							|  |  |  |     conversion.src_YUYV.gap = cmd_buff[4]; | 
					
						
							|  |  |  |     u32 src_process_handle = cmd_buff[6]; | 
					
						
							|  |  |  |     LOG_DEBUG(Service_Y2R, "called image_size=0x%08X, transfer_unit=%hu, transfer_stride=%hu, " | 
					
						
							|  |  |  |         "src_process_handle=0x%08X", conversion.src_YUYV.image_size, | 
					
						
							|  |  |  |         conversion.src_YUYV.transfer_unit, conversion.src_YUYV.gap, src_process_handle); | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void SetReceiving(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-03 17:54:24 -03:00
										 |  |  |     conversion.dst.address = cmd_buff[1]; | 
					
						
							|  |  |  |     conversion.dst.image_size = cmd_buff[2]; | 
					
						
							|  |  |  |     conversion.dst.transfer_unit = cmd_buff[3]; | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     conversion.dst.gap = cmd_buff[4]; | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  |     u32 dst_process_handle = cmd_buff[6]; | 
					
						
							|  |  |  |     LOG_DEBUG(Service_Y2R, "called image_size=0x%08X, transfer_unit=%hu, transfer_stride=%hu, " | 
					
						
							| 
									
										
										
										
											2015-06-03 17:54:24 -03:00
										 |  |  |         "dst_process_handle=0x%08X", conversion.dst.image_size, | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |         conversion.dst.transfer_unit, conversion.dst.gap, | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  |         dst_process_handle); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void SetInputLineWidth(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     LOG_DEBUG(Service_Y2R, "called input_line_width=%u", cmd_buff[1]); | 
					
						
							|  |  |  |     cmd_buff[1] = conversion.SetInputLineWidth(cmd_buff[1]).raw; | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void SetInputLines(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     LOG_DEBUG(Service_Y2R, "called input_line_number=%u", cmd_buff[1]); | 
					
						
							|  |  |  |     cmd_buff[1] = conversion.SetInputLines(cmd_buff[1]).raw; | 
					
						
							| 
									
										
										
										
											2015-02-24 08:28:36 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  | static void SetCoefficient(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2015-05-19 21:11:32 -04:00
										 |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     const u16* coefficients = reinterpret_cast<const u16*>(&cmd_buff[1]); | 
					
						
							|  |  |  |     std::memcpy(conversion.coefficients.data(), coefficients, sizeof(CoefficientSet)); | 
					
						
							|  |  |  |     LOG_DEBUG(Service_Y2R, "called coefficients=[%hX, %hX, %hX, %hX, %hX, %hX, %hX, %hX]", | 
					
						
							|  |  |  |             coefficients[0], coefficients[1], coefficients[2], coefficients[3], | 
					
						
							|  |  |  |             coefficients[4], coefficients[5], coefficients[6], coefficients[7]); | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  | static void SetStandardCoefficient(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     LOG_DEBUG(Service_Y2R, "called standard_coefficient=%u", cmd_buff[1]); | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     cmd_buff[1] = conversion.SetStandardCoefficient((StandardCoefficient)cmd_buff[1]).raw; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  | static void SetAlpha(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     conversion.alpha = cmd_buff[1]; | 
					
						
							|  |  |  |     LOG_DEBUG(Service_Y2R, "called alpha=%hu", conversion.alpha); | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  | static void StartConversion(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     HW::Y2R::PerformConversion(conversion); | 
					
						
							| 
									
										
										
										
											2015-05-22 14:49:42 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     // dst_image_size would seem to be perfect for this, but it doesn't include the gap :(
 | 
					
						
							|  |  |  |     u32 total_output_size = conversion.input_lines * | 
					
						
							|  |  |  |         (conversion.dst.transfer_unit + conversion.dst.gap); | 
					
						
							| 
									
										
										
										
											2015-05-22 14:49:42 -07:00
										 |  |  |     VideoCore::g_renderer->hw_rasterizer->NotifyFlush( | 
					
						
							| 
									
										
										
										
											2015-06-03 17:54:24 -03:00
										 |  |  |         Memory::VirtualToPhysicalAddress(conversion.dst.address), total_output_size); | 
					
						
							| 
									
										
										
										
											2015-05-22 14:49:42 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  |     LOG_DEBUG(Service_Y2R, "called"); | 
					
						
							| 
									
										
										
										
											2015-05-19 21:11:32 -04:00
										 |  |  |     completion_event->Signal(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  | static void StopConversion(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-21 22:47:55 -03:00
										 |  |  |     cmd_buff[0] = IPC::MakeHeader(0x27, 1, 0); | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							|  |  |  |     LOG_DEBUG(Service_Y2R, "called"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2015-06-03 17:54:24 -03:00
										 |  |  |  * Y2R_U::IsBusyConversion service function | 
					
						
							|  |  |  |  *  Outputs: | 
					
						
							|  |  |  |  *      1 : Result of function, 0 on success, otherwise error code | 
					
						
							|  |  |  |  *      2 : 1 if there's a conversion running, otherwise 0. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | static void IsBusyConversion(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							| 
									
										
										
										
											2015-05-19 21:11:32 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  |     cmd_buff[2] = 0; // StartConversion always finishes immediately
 | 
					
						
							|  |  |  |     LOG_DEBUG(Service_Y2R, "called"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-03 17:54:24 -03:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Y2R_U::SetConversionParams service function | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void SetConversionParams(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto params = reinterpret_cast<const ConversionParameters*>(&cmd_buff[1]); | 
					
						
							|  |  |  |     LOG_DEBUG(Service_Y2R, | 
					
						
							|  |  |  |         "called input_format=%hhu output_format=%hhu rotation=%hhu block_alignment=%hhu " | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |         "input_line_width=%hu input_lines=%hu standard_coefficient=%hhu " | 
					
						
							|  |  |  |         "reserved=%hhu alpha=%hX", | 
					
						
							| 
									
										
										
										
											2015-06-03 17:54:24 -03:00
										 |  |  |         params->input_format, params->output_format, params->rotation, params->block_alignment, | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |         params->input_line_width, params->input_lines, params->standard_coefficient, | 
					
						
							|  |  |  |         params->reserved, params->alpha); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ResultCode result = RESULT_SUCCESS; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     conversion.input_format = params->input_format; | 
					
						
							|  |  |  |     conversion.output_format = params->output_format; | 
					
						
							|  |  |  |     conversion.rotation = params->rotation; | 
					
						
							|  |  |  |     conversion.block_alignment = params->block_alignment; | 
					
						
							|  |  |  |     result = conversion.SetInputLineWidth(params->input_line_width); | 
					
						
							|  |  |  |     if (result.IsError()) goto cleanup; | 
					
						
							|  |  |  |     result = conversion.SetInputLines(params->input_lines); | 
					
						
							|  |  |  |     if (result.IsError()) goto cleanup; | 
					
						
							|  |  |  |     result = conversion.SetStandardCoefficient(params->standard_coefficient); | 
					
						
							|  |  |  |     if (result.IsError()) goto cleanup; | 
					
						
							|  |  |  |     conversion.alpha = params->alpha; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | cleanup: | 
					
						
							| 
									
										
										
										
											2015-06-21 22:47:55 -03:00
										 |  |  |     cmd_buff[0] = IPC::MakeHeader(0x29, 1, 0); | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     cmd_buff[1] = result.raw; | 
					
						
							| 
									
										
										
										
											2015-06-03 17:54:24 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | static void PingProcess(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							|  |  |  |     cmd_buff[2] = 0; | 
					
						
							|  |  |  |     LOG_WARNING(Service_Y2R, "(STUBBED) called"); | 
					
						
							| 
									
										
										
										
											2015-05-19 21:11:32 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  | static void DriverInitialize(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     conversion.input_format = InputFormat::YUV422_Indiv8; | 
					
						
							|  |  |  |     conversion.output_format = OutputFormat::RGBA8; | 
					
						
							|  |  |  |     conversion.rotation = Rotation::None; | 
					
						
							|  |  |  |     conversion.block_alignment = BlockAlignment::Linear; | 
					
						
							|  |  |  |     conversion.coefficients.fill(0); | 
					
						
							|  |  |  |     conversion.SetInputLineWidth(1024); | 
					
						
							|  |  |  |     conversion.SetInputLines(1024); | 
					
						
							|  |  |  |     conversion.alpha = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ConversionBuffer zero_buffer = {}; | 
					
						
							|  |  |  |     conversion.src_Y = zero_buffer; | 
					
						
							|  |  |  |     conversion.src_U = zero_buffer; | 
					
						
							|  |  |  |     conversion.src_V = zero_buffer; | 
					
						
							|  |  |  |     conversion.dst = zero_buffer; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     completion_event->Clear(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-21 22:47:55 -03:00
										 |  |  |     cmd_buff[0] = IPC::MakeHeader(0x2B, 1, 0); | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							|  |  |  |     LOG_DEBUG(Service_Y2R, "called"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void DriverFinalize(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-21 22:47:55 -03:00
										 |  |  |     cmd_buff[0] = IPC::MakeHeader(0x2C, 1, 0); | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							|  |  |  |     LOG_DEBUG(Service_Y2R, "called"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-02 15:03:40 -03:00
										 |  |  | const Interface::FunctionInfo FunctionTable[] = { | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  |     {0x00010040, SetInputFormat,          "SetInputFormat"}, | 
					
						
							|  |  |  |     {0x00030040, SetOutputFormat,         "SetOutputFormat"}, | 
					
						
							|  |  |  |     {0x00050040, SetRotation,             "SetRotation"}, | 
					
						
							|  |  |  |     {0x00070040, SetBlockAlignment,       "SetBlockAlignment"}, | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     {0x000D0040, SetTransferEndInterrupt, "SetTransferEndInterrupt"}, | 
					
						
							| 
									
										
										
										
											2015-02-24 08:28:36 -05:00
										 |  |  |     {0x000F0000, GetTransferEndEvent,     "GetTransferEndEvent"}, | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  |     {0x00100102, SetSendingY,             "SetSendingY"}, | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     {0x00110102, SetSendingU,             "SetSendingU"}, | 
					
						
							|  |  |  |     {0x00120102, SetSendingV,             "SetSendingV"}, | 
					
						
							|  |  |  |     {0x00130102, SetSendingYUYV,          "SetSendingYUYV"}, | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  |     {0x00180102, SetReceiving,            "SetReceiving"}, | 
					
						
							|  |  |  |     {0x001A0040, SetInputLineWidth,       "SetInputLineWidth"}, | 
					
						
							|  |  |  |     {0x001C0040, SetInputLines,           "SetInputLines"}, | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     {0x001E0100, SetCoefficient,          "SetCoefficient"}, | 
					
						
							|  |  |  |     {0x00200040, SetStandardCoefficient,  "SetStandardCoefficient"}, | 
					
						
							|  |  |  |     {0x00220040, SetAlpha,                "SetAlpha"}, | 
					
						
							| 
									
										
										
										
											2015-05-19 21:11:32 -04:00
										 |  |  |     {0x00260000, StartConversion,         "StartConversion"}, | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     {0x00270000, StopConversion,          "StopConversion"}, | 
					
						
							| 
									
										
										
										
											2015-01-10 22:46:44 -08:00
										 |  |  |     {0x00280000, IsBusyConversion,        "IsBusyConversion"}, | 
					
						
							| 
									
										
										
										
											2015-06-03 17:54:24 -03:00
										 |  |  |     {0x002901C0, SetConversionParams,     "SetConversionParams"}, | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  |     {0x002A0000, PingProcess,             "PingProcess"}, | 
					
						
							| 
									
										
										
										
											2015-06-07 22:24:03 -03:00
										 |  |  |     {0x002B0000, DriverInitialize,        "DriverInitialize"}, | 
					
						
							|  |  |  |     {0x002C0000, DriverFinalize,          "DriverFinalize"}, | 
					
						
							| 
									
										
										
										
											2015-01-02 15:03:40 -03:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | // Interface class
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Interface::Interface() { | 
					
						
							| 
									
										
										
										
											2015-02-24 08:28:36 -05:00
										 |  |  |     completion_event = Kernel::Event::Create(RESETTYPE_ONESHOT, "Y2R:Completed"); | 
					
						
							| 
									
										
										
										
											2015-06-03 17:54:24 -03:00
										 |  |  |     std::memset(&conversion, 0, sizeof(conversion)); | 
					
						
							| 
									
										
										
										
											2015-02-24 08:28:36 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-30 16:56:49 -02:00
										 |  |  |     Register(FunctionTable); | 
					
						
							| 
									
										
										
										
											2015-01-02 15:03:40 -03:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-05-21 23:27:48 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-17 02:24:13 -03:00
										 |  |  | Interface::~Interface() { | 
					
						
							|  |  |  |     completion_event = nullptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-02 15:03:40 -03:00
										 |  |  | } // namespace
 |