mirror of
				https://git.eden-emu.dev/eden-emu/eden.git
				synced 2025-10-26 14:53:18 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			113 lines
		
	
	
	
		
			3.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			113 lines
		
	
	
	
		
			3.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // SPDX-FileCopyrightText: Copyright 2021 yuzu Emulator Project
 | |
| // SPDX-License-Identifier: GPL-2.0-or-later
 | |
| 
 | |
| #include <algorithm>
 | |
| #include <array>
 | |
| #include <vector>
 | |
| 
 | |
| #include "common/alignment.h"
 | |
| #include "common/assert.h"
 | |
| #include "common/polyfill_ranges.h"
 | |
| #include "shader_recompiler/shader_info.h"
 | |
| #include "video_core/transform_feedback.h"
 | |
| 
 | |
| namespace VideoCommon {
 | |
| 
 | |
| std::vector<Shader::TransformFeedbackVarying> MakeTransformFeedbackVaryings(
 | |
|     const TransformFeedbackState& state) {
 | |
|     static constexpr std::array VECTORS{
 | |
|         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]
 | |
|     };
 | |
|     std::vector<Shader::TransformFeedbackVarying> xfb(256);
 | |
|     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) {
 | |
|             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;
 | |
|             };
 | |
| 
 | |
|             UNIMPLEMENTED_IF_MSG(layout.stream != 0, "Stream is not zero: {}", layout.stream);
 | |
|             Shader::TransformFeedbackVarying varying{
 | |
|                 .buffer = static_cast<u32>(buffer),
 | |
|                 .stride = layout.stride,
 | |
|                 .offset = offset * 4,
 | |
|                 .components = 1,
 | |
|             };
 | |
|             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);
 | |
| 
 | |
|                 const auto base_index = attribute / 4;
 | |
|                 while (offset + 1 < varying_count && base_index == get_attribute(offset + 1) / 4) {
 | |
|                     ++offset;
 | |
|                     ++varying.components;
 | |
|                 }
 | |
|             }
 | |
|             xfb[attribute] = varying;
 | |
|             highest = std::max(highest, (base_offset + varying.components) * 4);
 | |
|         }
 | |
|         UNIMPLEMENTED_IF(highest != layout.stride);
 | |
|     }
 | |
|     return xfb;
 | |
| }
 | |
| 
 | |
| } // namespace VideoCommon
 | 
