forked from eden-emu/eden
		
	Shaders: Implemented a stack for the SSY/SYNC instructions.
The SSY instruction pushes an address into the stack, and the SYNC instruction pops it. The current stack depth is 20, we should figure out if this is enough or not.
This commit is contained in:
		
							parent
							
								
									6eba539f4a
								
							
						
					
					
						commit
						ff358d97e8
					
				
					 1 changed files with 36 additions and 3 deletions
				
			
		|  | @ -815,6 +815,33 @@ private: | ||||||
|         shader.AddLine('}'); |         shader.AddLine('}'); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /*
 | ||||||
|  |      * Emits code to push the input target address to the SSY address stack, incrementing the stack | ||||||
|  |      * top. | ||||||
|  |      */ | ||||||
|  |     void EmitPushToSSYStack(u32 target) { | ||||||
|  |         shader.AddLine('{'); | ||||||
|  |         ++shader.scope; | ||||||
|  |         shader.AddLine("ssy_stack[ssy_stack_top] = " + std::to_string(target) + "u;"); | ||||||
|  |         shader.AddLine("ssy_stack_top++;"); | ||||||
|  |         --shader.scope; | ||||||
|  |         shader.AddLine('}'); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /*
 | ||||||
|  |      * Emits code to pop an address from the SSY address stack, setting the jump address to the | ||||||
|  |      * popped address and decrementing the stack top. | ||||||
|  |      */ | ||||||
|  |     void EmitPopFromSSYStack() { | ||||||
|  |         shader.AddLine('{'); | ||||||
|  |         ++shader.scope; | ||||||
|  |         shader.AddLine("ssy_stack_top--;"); | ||||||
|  |         shader.AddLine("jmp_to = ssy_stack[ssy_stack_top];"); | ||||||
|  |         shader.AddLine("break;"); | ||||||
|  |         --shader.scope; | ||||||
|  |         shader.AddLine('}'); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     /**
 |     /**
 | ||||||
|      * Compiles a single instruction from Tegra to GLSL. |      * Compiles a single instruction from Tegra to GLSL. | ||||||
|      * @param offset the offset of the Tegra shader instruction. |      * @param offset the offset of the Tegra shader instruction. | ||||||
|  | @ -1843,13 +1870,13 @@ private: | ||||||
|                 ASSERT_MSG(instr.bra.constant_buffer == 0, "Constant buffer SSY is not supported"); |                 ASSERT_MSG(instr.bra.constant_buffer == 0, "Constant buffer SSY is not supported"); | ||||||
| 
 | 
 | ||||||
|                 u32 target = offset + instr.bra.GetBranchTarget(); |                 u32 target = offset + instr.bra.GetBranchTarget(); | ||||||
|                 shader.AddLine("ssy_target = " + std::to_string(target) + "u;"); |                 EmitPushToSSYStack(target); | ||||||
|                 break; |                 break; | ||||||
|             } |             } | ||||||
|             case OpCode::Id::SYNC: { |             case OpCode::Id::SYNC: { | ||||||
|                 // The SYNC opcode jumps to the address previously set by the SSY opcode
 |                 // The SYNC opcode jumps to the address previously set by the SSY opcode
 | ||||||
|                 ASSERT(instr.flow.cond == Tegra::Shader::FlowCondition::Always); |                 ASSERT(instr.flow.cond == Tegra::Shader::FlowCondition::Always); | ||||||
|                 shader.AddLine("{ jmp_to = ssy_target; break; }"); |                 EmitPopFromSSYStack(); | ||||||
|                 break; |                 break; | ||||||
|             } |             } | ||||||
|             case OpCode::Id::DEPBAR: { |             case OpCode::Id::DEPBAR: { | ||||||
|  | @ -1920,7 +1947,13 @@ private: | ||||||
|             } else { |             } else { | ||||||
|                 labels.insert(subroutine.begin); |                 labels.insert(subroutine.begin); | ||||||
|                 shader.AddLine("uint jmp_to = " + std::to_string(subroutine.begin) + "u;"); |                 shader.AddLine("uint jmp_to = " + std::to_string(subroutine.begin) + "u;"); | ||||||
|                 shader.AddLine("uint ssy_target = 0u;"); | 
 | ||||||
|  |                 // TODO(Subv): Figure out the actual depth of the SSY stack, for now it seems
 | ||||||
|  |                 // unlikely that shaders will use 20 nested SSYs.
 | ||||||
|  |                 constexpr u32 SSY_STACK_SIZE = 20; | ||||||
|  |                 shader.AddLine("uint ssy_stack[" + std::to_string(SSY_STACK_SIZE) + "];"); | ||||||
|  |                 shader.AddLine("uint ssy_stack_top = 0u;"); | ||||||
|  | 
 | ||||||
|                 shader.AddLine("while (true) {"); |                 shader.AddLine("while (true) {"); | ||||||
|                 ++shader.scope; |                 ++shader.scope; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Subv
						Subv