| 
									
										
										
										
											2013-04-17 23:29:41 -04:00
										 |  |  | // Copyright 2013 Dolphin Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							| 
									
										
										
										
											2009-04-03 14:35:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | #include "Globals.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <fstream>
 | 
					
						
							|  |  |  | #include <vector>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-08 01:39:56 +00:00
										 |  |  | #include "Fifo.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-18 12:44:06 -05:00
										 |  |  | #include "DriverDetails.h"
 | 
					
						
							| 
									
										
										
										
											2009-09-13 09:23:30 +00:00
										 |  |  | #include "VideoConfig.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | #include "Statistics.h"
 | 
					
						
							|  |  |  | #include "MemoryUtil.h"
 | 
					
						
							|  |  |  | #include "Render.h"
 | 
					
						
							|  |  |  | #include "ImageWrite.h"
 | 
					
						
							| 
									
										
										
										
											2009-06-22 09:31:30 +00:00
										 |  |  | #include "BPMemory.h"
 | 
					
						
							| 
									
										
										
										
											2010-09-28 02:15:02 +00:00
										 |  |  | #include "TextureCache.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | #include "PixelShaderManager.h"
 | 
					
						
							|  |  |  | #include "VertexShaderManager.h"
 | 
					
						
							| 
									
										
										
										
											2011-11-30 21:00:21 -06:00
										 |  |  | #include "ProgramShaderCache.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-26 17:33:53 +00:00
										 |  |  | #include "VertexShaderGen.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | #include "VertexLoader.h"
 | 
					
						
							|  |  |  | #include "VertexManager.h"
 | 
					
						
							| 
									
										
										
										
											2009-09-26 12:39:12 +00:00
										 |  |  | #include "IndexGenerator.h"
 | 
					
						
							| 
									
										
										
										
											2009-10-02 14:03:07 +00:00
										 |  |  | #include "OpcodeDecoding.h"
 | 
					
						
							| 
									
										
										
										
											2010-02-02 21:56:29 +00:00
										 |  |  | #include "FileUtil.h"
 | 
					
						
							| 
									
										
										
										
											2010-12-05 14:15:36 +00:00
										 |  |  | #include "Debugger.h"
 | 
					
						
							| 
									
										
										
										
											2013-01-31 23:11:53 +01:00
										 |  |  | #include "StreamBuffer.h"
 | 
					
						
							| 
									
										
										
										
											2012-06-17 13:58:29 +02:00
										 |  |  | #include "PerfQueryBase.h"
 | 
					
						
							| 
									
										
										
										
											2013-03-25 15:14:24 +01:00
										 |  |  | #include "Render.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-04 21:02:32 +00:00
										 |  |  | #include "main.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | // internal state for loading vertices
 | 
					
						
							|  |  |  | extern NativeVertexFormat *g_nativeVertexFmt; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-03 00:41:06 +00:00
										 |  |  | namespace OGL | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-04-19 09:21:45 -04:00
										 |  |  | //This are the initially requested size for the buffers expressed in bytes
 | 
					
						
							| 
									
										
										
										
											2013-02-22 10:25:38 +01:00
										 |  |  | const u32 MAX_IBUFFER_SIZE =  2*1024*1024; | 
					
						
							|  |  |  | const u32 MAX_VBUFFER_SIZE = 16*1024*1024; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-31 23:11:53 +01:00
										 |  |  | static StreamBuffer *s_vertexBuffer; | 
					
						
							|  |  |  | static StreamBuffer *s_indexBuffer; | 
					
						
							|  |  |  | static u32 s_baseVertex; | 
					
						
							|  |  |  | static u32 s_offset[3]; | 
					
						
							| 
									
										
										
										
											2010-09-30 15:24:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-03 00:41:06 +00:00
										 |  |  | VertexManager::VertexManager() | 
					
						
							| 
									
										
										
										
											2010-09-28 02:15:02 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-10-26 11:34:02 -03:00
										 |  |  | 	CreateDeviceObjects(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-01-07 13:47:34 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-26 11:34:02 -03:00
										 |  |  | VertexManager::~VertexManager() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	DestroyDeviceObjects(); | 
					
						
							| 
									
										
										
										
											2009-09-26 12:39:12 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-20 10:22:15 -03:00
										 |  |  | void VertexManager::CreateDeviceObjects() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-02-22 10:25:38 +01:00
										 |  |  | 	s_vertexBuffer = new StreamBuffer(GL_ARRAY_BUFFER, MAX_VBUFFER_SIZE); | 
					
						
							| 
									
										
										
										
											2013-01-31 23:11:53 +01:00
										 |  |  | 	m_vertex_buffers = s_vertexBuffer->getBuffer(); | 
					
						
							| 
									
										
										
										
											2013-03-23 15:37:01 -05:00
										 |  |  | 	s_indexBuffer = new StreamBuffer(GL_ELEMENT_ARRAY_BUFFER, MAX_IBUFFER_SIZE, (StreamType)(DETECT_MASK & ~PINNED_MEMORY)); | 
					
						
							| 
									
										
										
										
											2013-01-31 23:11:53 +01:00
										 |  |  | 	m_index_buffers = s_indexBuffer->getBuffer(); | 
					
						
							| 
									
										
										
										
											2012-10-20 10:22:15 -03:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-10-26 23:18:09 -03:00
										 |  |  | 	m_CurrentVertexFmt = NULL; | 
					
						
							| 
									
										
										
										
											2012-12-15 14:43:01 +01:00
										 |  |  | 	m_last_vao = 0; | 
					
						
							| 
									
										
										
										
											2012-10-20 10:22:15 -03:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2013-02-22 10:25:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-20 10:22:15 -03:00
										 |  |  | void VertexManager::DestroyDeviceObjects() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-10-26 11:34:02 -03:00
										 |  |  | 	GL_REPORT_ERRORD(); | 
					
						
							| 
									
										
										
										
											2012-12-15 14:43:01 +01:00
										 |  |  | 	glBindBuffer(GL_ARRAY_BUFFER, 0 ); | 
					
						
							|  |  |  | 	glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0 ); | 
					
						
							| 
									
										
										
										
											2012-10-26 11:34:02 -03:00
										 |  |  | 	GL_REPORT_ERROR(); | 
					
						
							| 
									
										
										
										
											2012-10-20 10:22:15 -03:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2013-01-31 23:11:53 +01:00
										 |  |  | 	delete s_vertexBuffer; | 
					
						
							|  |  |  | 	delete s_indexBuffer; | 
					
						
							| 
									
										
										
										
											2012-12-15 14:43:01 +01:00
										 |  |  | 	GL_REPORT_ERROR(); | 
					
						
							| 
									
										
										
										
											2012-10-20 10:22:15 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-26 11:34:02 -03:00
										 |  |  | void VertexManager::PrepareDrawBuffers(u32 stride) | 
					
						
							| 
									
										
										
										
											2010-06-16 10:12:57 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-02-01 12:30:08 +01:00
										 |  |  | 	u32 vertex_data_size = IndexGenerator::GetNumVerts() * stride; | 
					
						
							|  |  |  | 	u32 triangle_index_size = IndexGenerator::GetTriangleindexLen(); | 
					
						
							|  |  |  | 	u32 line_index_size = IndexGenerator::GetLineindexLen(); | 
					
						
							|  |  |  | 	u32 point_index_size = IndexGenerator::GetPointindexLen(); | 
					
						
							|  |  |  | 	u32 index_size = (triangle_index_size+line_index_size+point_index_size) * sizeof(u16); | 
					
						
							| 
									
										
										
										
											2012-12-15 14:43:01 +01:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2013-02-01 15:15:25 +01:00
										 |  |  | 	s_vertexBuffer->Alloc(vertex_data_size, stride); | 
					
						
							| 
									
										
										
										
											2013-03-06 15:59:29 +01:00
										 |  |  | 	u32 offset = s_vertexBuffer->Upload(GetVertexBuffer(), vertex_data_size); | 
					
						
							| 
									
										
										
										
											2013-01-31 23:11:53 +01:00
										 |  |  | 	s_baseVertex = offset / stride; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-01 12:30:08 +01:00
										 |  |  | 	s_indexBuffer->Alloc(index_size); | 
					
						
							| 
									
										
										
										
											2013-01-31 23:11:53 +01:00
										 |  |  | 	if(triangle_index_size) | 
					
						
							| 
									
										
										
										
											2009-09-29 18:27:41 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2013-03-06 15:59:29 +01:00
										 |  |  | 		s_offset[0] = s_indexBuffer->Upload((u8*)GetTriangleIndexBuffer(), triangle_index_size * sizeof(u16)); | 
					
						
							| 
									
										
										
										
											2009-09-26 12:39:12 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-01-31 23:11:53 +01:00
										 |  |  | 	if(line_index_size) | 
					
						
							| 
									
										
										
										
											2009-09-26 12:39:12 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2013-03-06 15:59:29 +01:00
										 |  |  | 		s_offset[1] = s_indexBuffer->Upload((u8*)GetLineIndexBuffer(), line_index_size * sizeof(u16)); | 
					
						
							| 
									
										
										
										
											2009-09-26 12:39:12 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-01-31 23:11:53 +01:00
										 |  |  | 	if(point_index_size) | 
					
						
							| 
									
										
										
										
											2009-09-29 18:27:41 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2013-03-06 15:59:29 +01:00
										 |  |  | 		s_offset[2] = s_indexBuffer->Upload((u8*)GetPointIndexBuffer(), point_index_size * sizeof(u16)); | 
					
						
							| 
									
										
										
										
											2012-10-26 11:34:02 -03:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-05-23 21:07:01 +02:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	ADDSTAT(stats.thisFrame.bytesVertexStreamed, vertex_data_size); | 
					
						
							|  |  |  | 	ADDSTAT(stats.thisFrame.bytesIndexStreamed, index_size); | 
					
						
							| 
									
										
										
										
											2012-10-26 11:34:02 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-15 14:43:01 +01:00
										 |  |  | void VertexManager::Draw(u32 stride) | 
					
						
							| 
									
										
										
										
											2012-10-26 23:18:09 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-02-01 12:30:08 +01:00
										 |  |  | 	u32 triangle_index_size = IndexGenerator::GetTriangleindexLen(); | 
					
						
							|  |  |  | 	u32 line_index_size = IndexGenerator::GetLineindexLen(); | 
					
						
							|  |  |  | 	u32 point_index_size = IndexGenerator::GetPointindexLen(); | 
					
						
							| 
									
										
										
										
											2013-08-20 15:25:02 +02:00
										 |  |  | 	u32 max_index = IndexGenerator::GetNumVerts(); | 
					
						
							| 
									
										
										
										
											2013-04-10 14:12:35 +02:00
										 |  |  | 	GLenum triangle_mode = g_ActiveConfig.backend_info.bSupportsPrimitiveRestart?GL_TRIANGLE_STRIP:GL_TRIANGLES; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2013-03-25 15:14:24 +01:00
										 |  |  | 	if(g_ogl_config.bSupportsGLBaseVertex) { | 
					
						
							| 
									
										
										
										
											2013-02-26 23:15:55 +01:00
										 |  |  | 		if (triangle_index_size > 0) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2013-08-20 15:25:02 +02:00
										 |  |  | 			glDrawRangeElementsBaseVertex(triangle_mode, 0, max_index, triangle_index_size, GL_UNSIGNED_SHORT, (u8*)NULL+s_offset[0], s_baseVertex); | 
					
						
							| 
									
										
										
										
											2013-02-26 23:15:55 +01:00
										 |  |  | 			INCSTAT(stats.thisFrame.numIndexedDrawCalls); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (line_index_size > 0) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2013-08-20 15:25:02 +02:00
										 |  |  | 			glDrawRangeElementsBaseVertex(GL_LINES, 0, max_index, line_index_size, GL_UNSIGNED_SHORT, (u8*)NULL+s_offset[1], s_baseVertex); | 
					
						
							| 
									
										
										
										
											2013-02-26 23:15:55 +01:00
										 |  |  | 			INCSTAT(stats.thisFrame.numIndexedDrawCalls); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (point_index_size > 0) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2013-08-20 15:25:02 +02:00
										 |  |  | 			glDrawRangeElementsBaseVertex(GL_POINTS, 0, max_index, point_index_size, GL_UNSIGNED_SHORT, (u8*)NULL+s_offset[2], s_baseVertex); | 
					
						
							| 
									
										
										
										
											2013-02-26 23:15:55 +01:00
										 |  |  | 			INCSTAT(stats.thisFrame.numIndexedDrawCalls); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		if (triangle_index_size > 0) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2013-08-20 15:25:02 +02:00
										 |  |  | 			glDrawRangeElements(triangle_mode, 0, max_index, triangle_index_size, GL_UNSIGNED_SHORT, (u8*)NULL+s_offset[0]); | 
					
						
							| 
									
										
										
										
											2013-02-26 23:15:55 +01:00
										 |  |  | 			INCSTAT(stats.thisFrame.numIndexedDrawCalls); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (line_index_size > 0) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2013-08-20 15:25:02 +02:00
										 |  |  | 			glDrawRangeElements(GL_LINES, 0, max_index, line_index_size, GL_UNSIGNED_SHORT, (u8*)NULL+s_offset[1]); | 
					
						
							| 
									
										
										
										
											2013-02-26 23:15:55 +01:00
										 |  |  | 			INCSTAT(stats.thisFrame.numIndexedDrawCalls); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (point_index_size > 0) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2013-08-20 15:25:02 +02:00
										 |  |  | 			glDrawRangeElements(GL_POINTS, 0, max_index, point_index_size, GL_UNSIGNED_SHORT, (u8*)NULL+s_offset[2]); | 
					
						
							| 
									
										
										
										
											2013-02-26 23:15:55 +01:00
										 |  |  | 			INCSTAT(stats.thisFrame.numIndexedDrawCalls); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-06-16 10:12:57 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-03 00:41:06 +00:00
										 |  |  | void VertexManager::vFlush() | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | { | 
					
						
							|  |  |  | #if defined(_DEBUG) || defined(DEBUGFAST) 
 | 
					
						
							| 
									
										
										
										
											2011-02-05 18:25:34 +00:00
										 |  |  | 	PRIM_LOG("frame%d:\n texgen=%d, numchan=%d, dualtex=%d, ztex=%d, cole=%d, alpe=%d, ze=%d", g_ActiveConfig.iSaveTargetId, xfregs.numTexGen.numTexGens, | 
					
						
							|  |  |  | 		xfregs.numChan.numColorChans, xfregs.dualTexTrans.enabled, bpmem.ztex2.op, | 
					
						
							| 
									
										
										
										
											2009-06-22 09:31:30 +00:00
										 |  |  | 		bpmem.blendmode.colorupdate, bpmem.blendmode.alphaupdate, bpmem.zmode.updateenable); | 
					
						
							| 
									
										
										
										
											2009-04-03 14:35:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-05 18:25:34 +00:00
										 |  |  | 	for (unsigned int i = 0; i < xfregs.numChan.numColorChans; ++i)  | 
					
						
							| 
									
										
										
										
											2009-04-03 14:35:49 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2011-02-05 18:25:34 +00:00
										 |  |  | 		LitChannel* ch = &xfregs.color[i]; | 
					
						
							| 
									
										
										
										
											2009-03-21 20:07:56 +00:00
										 |  |  | 		PRIM_LOG("colchan%d: matsrc=%d, light=0x%x, ambsrc=%d, diffunc=%d, attfunc=%d", i, ch->matsource, ch->GetFullLightMask(), ch->ambsource, ch->diffusefunc, ch->attnfunc); | 
					
						
							| 
									
										
										
										
											2011-02-05 18:25:34 +00:00
										 |  |  | 		ch = &xfregs.alpha[i]; | 
					
						
							| 
									
										
										
										
											2009-03-21 20:07:56 +00:00
										 |  |  | 		PRIM_LOG("alpchan%d: matsrc=%d, light=0x%x, ambsrc=%d, diffunc=%d, attfunc=%d", i, ch->matsource, ch->GetFullLightMask(), ch->ambsource, ch->diffusefunc, ch->attnfunc); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-05 18:25:34 +00:00
										 |  |  | 	for (unsigned int i = 0; i < xfregs.numTexGen.numTexGens; ++i)  | 
					
						
							| 
									
										
										
										
											2009-04-03 14:35:49 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2011-02-05 18:25:34 +00:00
										 |  |  | 		TexMtxInfo tinfo = xfregs.texMtxInfo[i]; | 
					
						
							| 
									
										
										
										
											2008-12-26 17:02:46 +00:00
										 |  |  | 		if (tinfo.texgentype != XF_TEXGEN_EMBOSS_MAP) tinfo.hex &= 0x7ff; | 
					
						
							|  |  |  | 		if (tinfo.texgentype != XF_TEXGEN_REGULAR) tinfo.projection = 0; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-21 20:07:56 +00:00
										 |  |  | 		PRIM_LOG("txgen%d: proj=%d, input=%d, gentype=%d, srcrow=%d, embsrc=%d, emblght=%d, postmtx=%d, postnorm=%d", | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 			i, tinfo.projection, tinfo.inputform, tinfo.texgentype, tinfo.sourcerow, tinfo.embosssourceshift, tinfo.embosslightshift, | 
					
						
							| 
									
										
										
										
											2011-02-05 18:25:34 +00:00
										 |  |  | 			xfregs.postMtxInfo[i].index, xfregs.postMtxInfo[i].normalize); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-08 17:18:45 +01:00
										 |  |  | 	PRIM_LOG("pixel: tev=%d, ind=%d, texgen=%d, dstalpha=%d, alphatest=0x%x", bpmem.genMode.numtevstages+1, bpmem.genMode.numindstages, | 
					
						
							|  |  |  | 		bpmem.genMode.numtexgens, (u32)bpmem.dstalpha.enable, (bpmem.alpha_test.hex>>16)&0xff); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-04 14:00:59 +00:00
										 |  |  | 	(void)GL_REPORT_ERROR(); | 
					
						
							| 
									
										
										
										
											2009-09-26 12:39:12 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-12-15 17:28:58 +01:00
										 |  |  | 	GLVertexFormat *nativeVertexFmt = (GLVertexFormat*)g_nativeVertexFmt; | 
					
						
							|  |  |  | 	u32 stride  = nativeVertexFmt->GetVertexStride(); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if(m_last_vao != nativeVertexFmt->VAO) { | 
					
						
							|  |  |  | 		glBindVertexArray(nativeVertexFmt->VAO); | 
					
						
							|  |  |  | 		m_last_vao = nativeVertexFmt->VAO; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-26 11:34:02 -03:00
										 |  |  | 	PrepareDrawBuffers(stride); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 	GL_REPORT_ERRORD(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-22 11:21:44 +00:00
										 |  |  | 	u32 usedtextures = 0; | 
					
						
							| 
									
										
										
										
											2009-06-22 09:31:30 +00:00
										 |  |  | 	for (u32 i = 0; i < (u32)bpmem.genMode.numtevstages + 1; ++i) | 
					
						
							|  |  |  | 		if (bpmem.tevorders[i / 2].getEnable(i & 1)) | 
					
						
							|  |  |  | 			usedtextures |= 1 << bpmem.tevorders[i/2].getTexMap(i & 1); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-22 09:31:30 +00:00
										 |  |  | 	if (bpmem.genMode.numindstages > 0) | 
					
						
							|  |  |  | 		for (u32 i = 0; i < (u32)bpmem.genMode.numtevstages + 1; ++i) | 
					
						
							| 
									
										
										
										
											2010-09-28 02:15:02 +00:00
										 |  |  | 			if (bpmem.tevind[i].IsActive() && bpmem.tevind[i].bt < bpmem.genMode.numindstages) | 
					
						
							| 
									
										
										
										
											2009-06-22 09:31:30 +00:00
										 |  |  | 				usedtextures |= 1 << bpmem.tevindref.getTexMap(bpmem.tevind[i].bt); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-26 11:34:02 -03:00
										 |  |  | 	for (u32 i = 0; i < 8; i++) | 
					
						
							| 
									
										
										
										
											2009-04-03 14:35:49 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2010-06-16 10:12:57 +00:00
										 |  |  | 		if (usedtextures & (1 << i)) | 
					
						
							| 
									
										
										
										
											2009-04-03 14:35:49 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2013-03-06 15:59:29 +01:00
										 |  |  | 			TextureCache::SetNextStage(i); | 
					
						
							| 
									
										
										
										
											2013-02-19 18:22:38 -06:00
										 |  |  | 			g_renderer->SetSamplerState(i % 4, i / 4); | 
					
						
							| 
									
										
										
										
											2009-06-22 09:31:30 +00:00
										 |  |  | 			FourTexUnits &tex = bpmem.tex[i >> 2]; | 
					
						
							| 
									
										
										
										
											2010-10-19 22:24:27 +00:00
										 |  |  | 			TextureCache::TCacheEntryBase* tentry = TextureCache::Load(i,  | 
					
						
							| 
									
										
										
										
											2010-09-28 02:15:02 +00:00
										 |  |  | 				(tex.texImage3[i&3].image_base/* & 0x1FFFFF*/) << 5, | 
					
						
							| 
									
										
										
										
											2009-03-22 11:21:44 +00:00
										 |  |  | 				tex.texImage0[i&3].width + 1, tex.texImage0[i&3].height + 1, | 
					
						
							| 
									
										
										
										
											2010-09-28 02:15:02 +00:00
										 |  |  | 				tex.texImage0[i&3].format, tex.texTlut[i&3].tmem_offset<<9,  | 
					
						
							| 
									
										
										
										
											2010-10-19 22:24:27 +00:00
										 |  |  | 				tex.texTlut[i&3].tlut_format, | 
					
						
							| 
									
										
										
										
											2012-08-10 13:29:19 +02:00
										 |  |  | 				(tex.texMode0[i&3].min_filter & 3), | 
					
						
							| 
									
										
										
										
											2013-02-15 20:46:03 -06:00
										 |  |  | 				(tex.texMode1[i&3].max_lod + 0xf) / 0x10, | 
					
						
							| 
									
										
										
										
											2012-01-29 21:49:50 +01:00
										 |  |  | 				tex.texImage1[i&3].image_type); | 
					
						
							| 
									
										
										
										
											2009-03-22 11:21:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-28 02:15:02 +00:00
										 |  |  | 			if (tentry) | 
					
						
							| 
									
										
										
										
											2009-04-03 14:35:49 +00:00
										 |  |  | 			{ | 
					
						
							| 
									
										
										
										
											2010-08-29 04:56:53 +00:00
										 |  |  | 				// 0s are probably for no manual wrapping needed.
 | 
					
						
							| 
									
										
										
										
											2012-03-24 04:47:28 +01:00
										 |  |  | 				PixelShaderManager::SetTexDims(i, tentry->native_width, tentry->native_height, 0, 0); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-04-03 14:35:49 +00:00
										 |  |  | 			else | 
					
						
							| 
									
										
										
										
											2013-04-01 00:10:54 -04:00
										 |  |  | 				ERROR_LOG(VIDEO, "Error loading texture"); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-23 19:55:19 +00:00
										 |  |  | 	bool useDstAlpha = !g_ActiveConfig.bDstAlphaPass && bpmem.dstalpha.enable && bpmem.blendmode.alphaupdate | 
					
						
							|  |  |  | 		&& bpmem.zcontrol.pixel_format == PIXELFMT_RGBA6_Z24; | 
					
						
							| 
									
										
										
										
											2010-10-24 18:26:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-08 05:09:48 -06:00
										 |  |  | 	// Makes sure we can actually do Dual source blending
 | 
					
						
							| 
									
										
										
										
											2013-01-14 20:00:33 +01:00
										 |  |  | 	bool dualSourcePossible = g_ActiveConfig.backend_info.bSupportsDualSourceBlend; | 
					
						
							| 
									
										
										
										
											2010-10-23 19:55:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 	// finally bind
 | 
					
						
							| 
									
										
										
										
											2010-10-23 19:55:19 +00:00
										 |  |  | 	if (dualSourcePossible) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		if (useDstAlpha) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			// If host supports GL_ARB_blend_func_extended, we can do dst alpha in
 | 
					
						
							|  |  |  | 			// the same pass as regular rendering.
 | 
					
						
							| 
									
										
										
										
											2013-02-13 13:12:19 +01:00
										 |  |  | 			ProgramShaderCache::SetShader(DSTALPHA_DUAL_SOURCE_BLEND, g_nativeVertexFmt->m_components); | 
					
						
							| 
									
										
										
										
											2010-10-23 19:55:19 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2013-02-13 13:12:19 +01:00
										 |  |  | 			ProgramShaderCache::SetShader(DSTALPHA_NONE,g_nativeVertexFmt->m_components); | 
					
						
							| 
									
										
										
										
											2010-10-23 19:55:19 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2013-02-13 13:12:19 +01:00
										 |  |  | 		ProgramShaderCache::SetShader(DSTALPHA_NONE,g_nativeVertexFmt->m_components); | 
					
						
							| 
									
										
										
										
											2010-10-23 19:55:19 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-30 21:00:21 -06:00
										 |  |  | 	// set global constants
 | 
					
						
							|  |  |  | 	VertexShaderManager::SetConstants(); | 
					
						
							| 
									
										
										
										
											2013-03-26 22:16:29 +01:00
										 |  |  | 	PixelShaderManager::SetConstants(g_nativeVertexFmt->m_components); | 
					
						
							| 
									
										
										
										
											2013-01-14 13:58:11 +01:00
										 |  |  | 	ProgramShaderCache::UploadConstants(); | 
					
						
							| 
									
										
										
										
											2011-12-07 22:04:34 -06:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	// setup the pointers
 | 
					
						
							|  |  |  | 	if (g_nativeVertexFmt) | 
					
						
							|  |  |  | 		g_nativeVertexFmt->SetupVertexPointers(); | 
					
						
							|  |  |  | 	GL_REPORT_ERRORD(); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-01 01:14:10 +01:00
										 |  |  | 	g_perf_query->EnableQuery(bpmem.zcontrol.early_ztest ? PQG_ZCOMP_ZCOMPLOC : PQG_ZCOMP); | 
					
						
							| 
									
										
										
										
											2012-12-15 14:43:01 +01:00
										 |  |  | 	Draw(stride); | 
					
						
							| 
									
										
										
										
											2013-03-01 01:14:10 +01:00
										 |  |  | 	g_perf_query->DisableQuery(bpmem.zcontrol.early_ztest ? PQG_ZCOMP_ZCOMPLOC : PQG_ZCOMP); | 
					
						
							|  |  |  | 	//ERROR_LOG(VIDEO, "PerfQuery result: %d", g_perf_query->GetQueryResult(bpmem.zcontrol.early_ztest ? PQ_ZCOMP_OUTPUT_ZCOMPLOC : PQ_ZCOMP_OUTPUT));
 | 
					
						
							| 
									
										
										
										
											2010-09-30 15:24:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-16 10:12:57 +00:00
										 |  |  | 	// run through vertex groups again to set alpha
 | 
					
						
							| 
									
										
										
										
											2010-10-23 19:55:19 +00:00
										 |  |  | 	if (useDstAlpha && !dualSourcePossible) | 
					
						
							| 
									
										
										
										
											2009-04-03 14:35:49 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2013-02-13 13:12:19 +01:00
										 |  |  | 		ProgramShaderCache::SetShader(DSTALPHA_ALPHA_PASS,g_nativeVertexFmt->m_components); | 
					
						
							| 
									
										
										
										
											2012-12-24 11:09:52 -06:00
										 |  |  | 		if (!g_ActiveConfig.backend_info.bSupportsGLSLUBO) | 
					
						
							| 
									
										
										
										
											2012-12-28 14:24:12 +01:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2013-01-02 16:56:08 +01:00
										 |  |  | 			// Need to set these again, if we don't support UBO
 | 
					
						
							| 
									
										
										
										
											2012-12-28 14:24:12 +01:00
										 |  |  | 			VertexShaderManager::SetConstants(); | 
					
						
							| 
									
										
										
										
											2013-03-26 22:16:29 +01:00
										 |  |  | 			PixelShaderManager::SetConstants(g_nativeVertexFmt->m_components); | 
					
						
							| 
									
										
										
										
											2012-12-28 14:24:12 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-12-24 11:09:52 -06:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-08-10 18:57:37 +02:00
										 |  |  | 		// only update alpha
 | 
					
						
							|  |  |  | 		glColorMask(GL_FALSE, GL_FALSE, GL_FALSE, GL_TRUE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		glDisable(GL_BLEND); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-15 14:43:01 +01:00
										 |  |  | 		Draw(stride); | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-08-10 18:57:37 +02:00
										 |  |  | 		// restore color mask
 | 
					
						
							|  |  |  | 		g_renderer->SetColorMask(); | 
					
						
							| 
									
										
										
										
											2012-03-30 01:56:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-10 18:57:37 +02:00
										 |  |  | 		if (bpmem.blendmode.blendenable || bpmem.blendmode.subtract)  | 
					
						
							|  |  |  | 			glEnable(GL_BLEND); | 
					
						
							| 
									
										
										
										
											2010-06-16 10:12:57 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-12-05 14:15:36 +00:00
										 |  |  | 	GFX_DEBUGGER_PAUSE_AT(NEXT_FLUSH, true); | 
					
						
							| 
									
										
										
										
											2012-12-15 17:28:58 +01:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2010-06-16 10:12:57 +00:00
										 |  |  | #if defined(_DEBUG) || defined(DEBUGFAST)
 | 
					
						
							| 
									
										
										
										
											2009-09-13 08:21:35 +00:00
										 |  |  | 	if (g_ActiveConfig.iLog & CONF_SAVESHADERS)  | 
					
						
							| 
									
										
										
										
											2009-04-03 14:35:49 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 		// save the shaders
 | 
					
						
							| 
									
										
										
										
											2013-03-04 10:20:55 +01:00
										 |  |  | 		ProgramShaderCache::PCacheEntry prog = ProgramShaderCache::GetShaderProgram(); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 		char strfile[255]; | 
					
						
							| 
									
										
										
										
											2011-02-28 20:40:15 +00:00
										 |  |  | 		sprintf(strfile, "%sps%.3d.txt", File::GetUserPath(D_DUMPFRAMES_IDX).c_str(), g_ActiveConfig.iSaveTargetId); | 
					
						
							| 
									
										
										
										
											2013-02-28 19:33:39 -06:00
										 |  |  | 		std::ofstream fps; | 
					
						
							|  |  |  | 		OpenFStream(fps, strfile, std::ios_base::out); | 
					
						
							| 
									
										
										
										
											2013-03-04 10:20:55 +01:00
										 |  |  | 		fps << prog.shader.strpprog.c_str(); | 
					
						
							| 
									
										
										
										
											2011-02-28 20:40:15 +00:00
										 |  |  | 		sprintf(strfile, "%svs%.3d.txt", File::GetUserPath(D_DUMPFRAMES_IDX).c_str(), g_ActiveConfig.iSaveTargetId); | 
					
						
							| 
									
										
										
										
											2013-02-28 19:33:39 -06:00
										 |  |  | 		std::ofstream fvs; | 
					
						
							|  |  |  | 		OpenFStream(fvs, strfile, std::ios_base::out); | 
					
						
							| 
									
										
										
										
											2013-03-04 10:20:55 +01:00
										 |  |  | 		fvs << prog.shader.strvprog.c_str(); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-13 08:21:35 +00:00
										 |  |  | 	if (g_ActiveConfig.iLog & CONF_SAVETARGETS)  | 
					
						
							| 
									
										
										
										
											2009-04-03 14:35:49 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 		char str[128]; | 
					
						
							| 
									
										
										
										
											2011-02-28 20:40:15 +00:00
										 |  |  | 		sprintf(str, "%starg%.3d.tga", File::GetUserPath(D_DUMPFRAMES_IDX).c_str(), g_ActiveConfig.iSaveTargetId); | 
					
						
							| 
									
										
										
										
											2010-07-09 20:56:16 +00:00
										 |  |  | 		TargetRectangle tr; | 
					
						
							|  |  |  | 		tr.left = 0; | 
					
						
							|  |  |  | 		tr.right = Renderer::GetTargetWidth(); | 
					
						
							|  |  |  | 		tr.top = 0; | 
					
						
							|  |  |  | 		tr.bottom = Renderer::GetTargetHeight(); | 
					
						
							| 
									
										
										
										
											2010-11-18 02:21:26 +00:00
										 |  |  | 		g_renderer->SaveScreenshot(str, tr); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	g_Config.iSaveTargetId++; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-22 06:15:02 +02:00
										 |  |  | 	ClearEFBCache(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | 	GL_REPORT_ERRORD(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-10-03 00:41:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:25:12 +00:00
										 |  |  | }  // namespace
 |