| 
									
										
										
										
											2025-09-21 21:58:59 +02:00
										 |  |  | // SPDX-FileCopyrightText: Copyright 2025 Eden Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-3.0-or-later
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-23 04:59:50 -04:00
										 |  |  | // SPDX-FileCopyrightText: Copyright 2021 yuzu Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2021-05-21 17:19:35 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | #include <array>
 | 
					
						
							|  |  |  | #include <vector>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "common/alignment.h"
 | 
					
						
							|  |  |  | #include "common/assert.h"
 | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  | #include "common/polyfill_ranges.h"
 | 
					
						
							| 
									
										
										
										
											2021-05-21 17:19:35 -03:00
										 |  |  | #include "shader_recompiler/shader_info.h"
 | 
					
						
							|  |  |  | #include "video_core/transform_feedback.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace VideoCommon { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-23 14:45:54 +01:00
										 |  |  | std::pair<std::array<Shader::TransformFeedbackVarying, 256>, u32> MakeTransformFeedbackVaryings( | 
					
						
							| 
									
										
										
										
											2021-05-21 17:19:35 -03:00
										 |  |  |     const TransformFeedbackState& state) { | 
					
						
							|  |  |  |     static constexpr std::array VECTORS{ | 
					
						
							| 
									
										
										
										
											2022-08-12 10:58:09 +01:00
										 |  |  |         28U,  // gl_Position
 | 
					
						
							|  |  |  |         32U,  // Generic 0
 | 
					
						
							|  |  |  |         36U,  // Generic 1
 | 
					
						
							|  |  |  |         40U,  // Generic 2
 | 
					
						
							|  |  |  |         44U,  // Generic 3
 | 
					
						
							|  |  |  |         48U,  // Generic 4
 | 
					
						
							|  |  |  |         52U,  // Generic 5
 | 
					
						
							|  |  |  |         56U,  // Generic 6
 | 
					
						
							|  |  |  |         60U,  // Generic 7
 | 
					
						
							|  |  |  |         64U,  // Generic 8
 | 
					
						
							|  |  |  |         68U,  // Generic 9
 | 
					
						
							|  |  |  |         72U,  // Generic 10
 | 
					
						
							|  |  |  |         76U,  // Generic 11
 | 
					
						
							|  |  |  |         80U,  // Generic 12
 | 
					
						
							|  |  |  |         84U,  // Generic 13
 | 
					
						
							|  |  |  |         88U,  // Generic 14
 | 
					
						
							|  |  |  |         92U,  // Generic 15
 | 
					
						
							|  |  |  |         96U,  // Generic 16
 | 
					
						
							|  |  |  |         100U, // Generic 17
 | 
					
						
							|  |  |  |         104U, // Generic 18
 | 
					
						
							|  |  |  |         108U, // Generic 19
 | 
					
						
							|  |  |  |         112U, // Generic 20
 | 
					
						
							|  |  |  |         116U, // Generic 21
 | 
					
						
							|  |  |  |         120U, // Generic 22
 | 
					
						
							|  |  |  |         124U, // Generic 23
 | 
					
						
							|  |  |  |         128U, // Generic 24
 | 
					
						
							|  |  |  |         132U, // Generic 25
 | 
					
						
							|  |  |  |         136U, // Generic 26
 | 
					
						
							|  |  |  |         140U, // Generic 27
 | 
					
						
							|  |  |  |         144U, // Generic 28
 | 
					
						
							|  |  |  |         148U, // Generic 29
 | 
					
						
							|  |  |  |         152U, // Generic 30
 | 
					
						
							|  |  |  |         156U, // Generic 31
 | 
					
						
							|  |  |  |         160U, // gl_FrontColor
 | 
					
						
							|  |  |  |         164U, // gl_FrontSecondaryColor
 | 
					
						
							|  |  |  |         160U, // gl_BackColor
 | 
					
						
							|  |  |  |         164U, // gl_BackSecondaryColor
 | 
					
						
							|  |  |  |         192U, // gl_TexCoord[0]
 | 
					
						
							|  |  |  |         196U, // gl_TexCoord[1]
 | 
					
						
							|  |  |  |         200U, // gl_TexCoord[2]
 | 
					
						
							|  |  |  |         204U, // gl_TexCoord[3]
 | 
					
						
							|  |  |  |         208U, // gl_TexCoord[4]
 | 
					
						
							|  |  |  |         212U, // gl_TexCoord[5]
 | 
					
						
							|  |  |  |         216U, // gl_TexCoord[6]
 | 
					
						
							|  |  |  |         220U, // gl_TexCoord[7]
 | 
					
						
							| 
									
										
										
										
											2021-05-21 17:19:35 -03:00
										 |  |  |     }; | 
					
						
							| 
									
										
										
										
											2023-05-23 14:45:54 +01:00
										 |  |  |     std::array<Shader::TransformFeedbackVarying, 256> xfb{}; | 
					
						
							|  |  |  |     u32 count{0}; | 
					
						
							| 
									
										
										
										
											2021-05-21 17:19:35 -03:00
										 |  |  |     for (size_t buffer = 0; buffer < state.layouts.size(); ++buffer) { | 
					
						
							|  |  |  |         const auto& locations = state.varyings[buffer]; | 
					
						
							|  |  |  |         const auto& layout = state.layouts[buffer]; | 
					
						
							|  |  |  |         const u32 varying_count = layout.varying_count; | 
					
						
							|  |  |  |         u32 highest = 0; | 
					
						
							|  |  |  |         for (u32 offset = 0; offset < varying_count; ++offset) { | 
					
						
							| 
									
										
										
										
											2022-08-12 10:58:09 +01:00
										 |  |  |             const auto get_attribute = [&locations](u32 index) -> u32 { | 
					
						
							|  |  |  |                 switch (index % 4) { | 
					
						
							|  |  |  |                 case 0: | 
					
						
							|  |  |  |                     return locations[index / 4].attribute0.Value(); | 
					
						
							|  |  |  |                 case 1: | 
					
						
							|  |  |  |                     return locations[index / 4].attribute1.Value(); | 
					
						
							|  |  |  |                 case 2: | 
					
						
							|  |  |  |                     return locations[index / 4].attribute2.Value(); | 
					
						
							|  |  |  |                 case 3: | 
					
						
							|  |  |  |                     return locations[index / 4].attribute3.Value(); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 UNREACHABLE(); | 
					
						
							|  |  |  |                 return 0; | 
					
						
							|  |  |  |             }; | 
					
						
							| 
									
										
										
										
											2021-05-21 17:19:35 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-02 01:10:04 -03:00
										 |  |  |             UNIMPLEMENTED_IF_MSG(layout.stream != 0, "Stream is not zero: {}", layout.stream); | 
					
						
							| 
									
										
										
										
											2021-05-21 17:19:35 -03:00
										 |  |  |             Shader::TransformFeedbackVarying varying{ | 
					
						
							| 
									
										
										
										
											2021-06-02 01:10:04 -03:00
										 |  |  |                 .buffer = static_cast<u32>(buffer), | 
					
						
							| 
									
										
										
										
											2021-05-21 17:19:35 -03:00
										 |  |  |                 .stride = layout.stride, | 
					
						
							|  |  |  |                 .offset = offset * 4, | 
					
						
							|  |  |  |                 .components = 1, | 
					
						
							|  |  |  |             }; | 
					
						
							| 
									
										
										
										
											2022-08-12 10:58:09 +01:00
										 |  |  |             const u32 base_offset = offset; | 
					
						
							|  |  |  |             const auto attribute{get_attribute(offset)}; | 
					
						
							|  |  |  |             if (std::ranges::find(VECTORS, Common::AlignDown(attribute, 4)) != VECTORS.end()) { | 
					
						
							|  |  |  |                 UNIMPLEMENTED_IF_MSG(attribute % 4 != 0, "Unaligned TFB {}", attribute); | 
					
						
							| 
									
										
										
										
											2021-05-21 17:19:35 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-12 10:58:09 +01:00
										 |  |  |                 const auto base_index = attribute / 4; | 
					
						
							|  |  |  |                 while (offset + 1 < varying_count && base_index == get_attribute(offset + 1) / 4) { | 
					
						
							| 
									
										
										
										
											2021-05-21 17:19:35 -03:00
										 |  |  |                     ++offset; | 
					
						
							|  |  |  |                     ++varying.components; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-08-12 10:58:09 +01:00
										 |  |  |             xfb[attribute] = varying; | 
					
						
							| 
									
										
										
										
											2025-09-09 20:47:49 +02:00
										 |  |  |             count = (std::max)(count, attribute); | 
					
						
							|  |  |  |             highest = (std::max)(highest, (base_offset + varying.components) * 4); | 
					
						
							| 
									
										
										
										
											2021-05-21 17:19:35 -03:00
										 |  |  |         } | 
					
						
							|  |  |  |         UNIMPLEMENTED_IF(highest != layout.stride); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-05-23 14:45:54 +01:00
										 |  |  |     return {xfb, count + 1}; | 
					
						
							| 
									
										
										
										
											2021-05-21 17:19:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace VideoCommon
 |