summaryrefslogtreecommitdiff
path: root/src/core/tracer/recorder.cpp
diff options
context:
space:
mode:
authorGravatar Tony Wasserka2015-04-04 12:57:31 +0200
committerGravatar Tony Wasserka2015-07-13 22:27:20 +0200
commit902fa4da52737d43e04c2a028658ad9840811a89 (patch)
tree0a2346b4e08b5865c3369247f5720453b170e0c6 /src/core/tracer/recorder.cpp
parentGPU: Be robust against nullptr addresses; properly reset busy bits in the tri... (diff)
downloadyuzu-902fa4da52737d43e04c2a028658ad9840811a89.tar.gz
yuzu-902fa4da52737d43e04c2a028658ad9840811a89.tar.xz
yuzu-902fa4da52737d43e04c2a028658ad9840811a89.zip
Add CiTrace recording support.
This is exposed in the GUI as a new "CiTrace Recording" widget. Playback is implemented by a standalone 3DS homebrew application (which only runs reliably within Citra currently; on an actual 3DS it will often crash still).
Diffstat (limited to 'src/core/tracer/recorder.cpp')
-rw-r--r--src/core/tracer/recorder.cpp198
1 files changed, 198 insertions, 0 deletions
diff --git a/src/core/tracer/recorder.cpp b/src/core/tracer/recorder.cpp
new file mode 100644
index 000000000..73ebdd388
--- /dev/null
+++ b/src/core/tracer/recorder.cpp
@@ -0,0 +1,198 @@
1// Copyright 2015 Citra Emulator Project
2// Licensed under GPLv2 or any later version
3// Refer to the license.txt file included.
4
5#include <cstring>
6
7#include "common/assert.h"
8#include "common/file_util.h"
9#include "common/logging/log.h"
10
11#include "recorder.h"
12
13namespace CiTrace {
14
15Recorder::Recorder(u32* gpu_registers, u32 gpu_registers_size,
16 u32* lcd_registers, u32 lcd_registers_size,
17 u32* pica_registers, u32 pica_registers_size,
18 u32* vs_program_binary, u32 vs_program_binary_size,
19 u32* vs_swizzle_data, u32 vs_swizzle_data_size,
20 u32* vs_float_uniforms, u32 vs_float_uniforms_size,
21 u32* gs_program_binary, u32 gs_program_binary_size,
22 u32* gs_swizzle_data, u32 gs_swizzle_data_size,
23 u32* gs_float_uniforms, u32 gs_float_uniforms_size)
24 : gpu_registers(gpu_registers, gpu_registers + gpu_registers_size),
25 lcd_registers(lcd_registers, lcd_registers + lcd_registers_size),
26 pica_registers(pica_registers, pica_registers + pica_registers_size),
27 vs_program_binary(vs_program_binary, vs_program_binary + vs_program_binary_size),
28 vs_swizzle_data(vs_swizzle_data, vs_swizzle_data + vs_swizzle_data_size),
29 vs_float_uniforms(vs_float_uniforms, vs_float_uniforms + vs_float_uniforms_size),
30 gs_program_binary(gs_program_binary, gs_program_binary + gs_program_binary_size),
31 gs_swizzle_data(gs_swizzle_data, gs_swizzle_data + gs_swizzle_data_size),
32 gs_float_uniforms(gs_float_uniforms, gs_float_uniforms + gs_float_uniforms_size) {
33
34}
35
36void Recorder::Finish(const std::string& filename) {
37 // Setup CiTrace header
38 CTHeader header;
39 std::memcpy(header.magic, CTHeader::ExpectedMagicWord(), 4);
40 header.version = CTHeader::ExpectedVersion();
41 header.header_size = sizeof(CTHeader);
42
43 // Calculate file offsets
44 auto& initial = header.initial_state_offsets;
45
46 initial.gpu_registers_size = gpu_registers.size();
47 initial.lcd_registers_size = lcd_registers.size();
48 initial.pica_registers_size = pica_registers.size();
49 initial.vs_program_binary_size = vs_program_binary.size();
50 initial.vs_swizzle_data_size = vs_swizzle_data.size();
51 initial.vs_float_uniforms_size = vs_float_uniforms.size();
52 initial.gs_program_binary_size = gs_program_binary.size();
53 initial.gs_swizzle_data_size = gs_swizzle_data.size();
54 initial.gs_float_uniforms_size = gs_float_uniforms.size();
55 header.stream_size = stream.size();
56
57 initial.gpu_registers = sizeof(header);
58 initial.lcd_registers = initial.gpu_registers + initial.gpu_registers_size * sizeof(u32);
59 initial.pica_registers = initial.lcd_registers + initial.lcd_registers_size * sizeof(u32);;
60 initial.vs_program_binary = initial.pica_registers + initial.pica_registers_size * sizeof(u32);
61 initial.vs_swizzle_data = initial.vs_program_binary + initial.vs_program_binary_size * sizeof(u32);
62 initial.vs_float_uniforms = initial.vs_swizzle_data + initial.vs_swizzle_data_size * sizeof(u32);
63 initial.gs_program_binary = initial.vs_float_uniforms + initial.vs_float_uniforms_size * sizeof(u32);
64 initial.gs_swizzle_data = initial.gs_program_binary + initial.gs_program_binary_size * sizeof(u32);
65 initial.gs_float_uniforms = initial.gs_swizzle_data + initial.gs_swizzle_data_size * sizeof(u32);
66 header.stream_offset = initial.gs_float_uniforms + initial.gs_float_uniforms_size * sizeof(u32);
67
68 // Iterate through stream elements, update relevant stream element data
69 for (auto& stream_element : stream) {
70 switch (stream_element.data.type) {
71 case MemoryLoad:
72 {
73 auto& file_offset = memory_regions[stream_element.hash];
74 if (!stream_element.uses_existing_data) {
75 file_offset = header.stream_offset;
76 }
77 stream_element.data.memory_load.file_offset = file_offset;
78 break;
79 }
80
81 default:
82 // Other commands don't use any extra data
83 DEBUG_ASSERT(stream_element.extra_data.size() == 0);
84 break;
85 }
86 header.stream_offset += stream_element.extra_data.size();
87 }
88
89 try {
90 // Open file and write header
91 FileUtil::IOFile file(filename, "wb");
92 size_t written = file.WriteObject(header);
93 if (written != 1 || file.Tell() != initial.gpu_registers)
94 throw "Failed to write header";
95
96 // Write initial state
97 written = file.WriteArray(gpu_registers.data(), gpu_registers.size());
98 if (written != gpu_registers.size() || file.Tell() != initial.lcd_registers)
99 throw "Failed to write GPU registers";
100
101 written = file.WriteArray(lcd_registers.data(), lcd_registers.size());
102 if (written != lcd_registers.size() || file.Tell() != initial.pica_registers)
103 throw "Failed to write LCD registers";
104
105 written = file.WriteArray(pica_registers.data(), pica_registers.size());
106 if (written != pica_registers.size() || file.Tell() != initial.vs_program_binary)
107 throw "Failed to write Pica registers";
108
109 written = file.WriteArray(vs_program_binary.data(), vs_program_binary.size());
110 if (written != vs_program_binary.size() || file.Tell() != initial.vs_swizzle_data)
111 throw "Failed to write vertex shader program binary";
112
113 written = file.WriteArray(vs_swizzle_data.data(), vs_swizzle_data.size());
114 if (written != vs_swizzle_data.size() || file.Tell() != initial.vs_float_uniforms)
115 throw "Failed to write vertex shader swizzle data";
116
117 written = file.WriteArray(vs_float_uniforms.data(), vs_float_uniforms.size());
118 if (written != vs_float_uniforms.size() || file.Tell() != initial.gs_program_binary)
119 throw "Failed to write vertex shader float uniforms";
120
121 written = file.WriteArray(gs_program_binary.data(), gs_program_binary.size());
122 if (written != gs_program_binary.size() || file.Tell() != initial.gs_swizzle_data)
123 throw "Failed to write geomtry shader program binary";
124
125 written = file.WriteArray(gs_swizzle_data.data(), gs_swizzle_data.size());
126 if (written != gs_swizzle_data.size() || file.Tell() != initial.gs_float_uniforms)
127 throw "Failed to write geometry shader swizzle data";
128
129 written = file.WriteArray(gs_float_uniforms.data(), gs_float_uniforms.size());
130 if (written != gs_float_uniforms.size() || file.Tell() != initial.gs_float_uniforms + sizeof(u32) * initial.gs_float_uniforms_size)
131 throw "Failed to write geometry shader float uniforms";
132
133 // Iterate through stream elements, write "extra data"
134 for (const auto& stream_element : stream) {
135 if (stream_element.extra_data.size() == 0)
136 continue;
137
138 written = file.WriteBytes(stream_element.extra_data.data(), stream_element.extra_data.size());
139 if (written != stream_element.extra_data.size())
140 throw "Failed to write extra data";
141 }
142
143 if (file.Tell() != header.stream_offset)
144 throw "Unexpected end of extra data";
145
146 // Write actual stream elements
147 for (const auto& stream_element : stream) {
148 if (1 != file.WriteObject(stream_element.data))
149 throw "Failed to write stream element";
150 }
151 } catch(const char* str) {
152 LOG_ERROR(HW_GPU, "Writing CiTrace file failed: %s", str);
153 }
154}
155
156void Recorder::FrameFinished() {
157 stream.push_back( { FrameMarker } );
158}
159
160void Recorder::MemoryAccessed(const u8* data, u32 size, u32 physical_address) {
161 StreamElement element = { MemoryLoad };
162 element.data.memory_load.size = size;
163 element.data.memory_load.physical_address = physical_address;
164
165 // Compute hash over given memory region to check if the contents are already stored internally
166 boost::crc_32_type result;
167 result.process_bytes(data, size);
168 element.hash = result.checksum();
169
170 element.uses_existing_data = (memory_regions.find(element.hash) != memory_regions.end());
171 if (!element.uses_existing_data) {
172 element.extra_data.resize(size);
173 memcpy(element.extra_data.data(), data, size);
174 memory_regions.insert({element.hash, 0}); // file offset will be initialized in Finish()
175 }
176
177 stream.push_back(element);
178}
179
180template<typename T>
181void Recorder::RegisterWritten(u32 physical_address, T value) {
182 StreamElement element = { RegisterWrite };
183 element.data.register_write.size = (sizeof(T) == 1) ? CTRegisterWrite::SIZE_8
184 : (sizeof(T) == 2) ? CTRegisterWrite::SIZE_16
185 : (sizeof(T) == 4) ? CTRegisterWrite::SIZE_32
186 : CTRegisterWrite::SIZE_64;
187 element.data.register_write.physical_address = physical_address;
188 element.data.register_write.value = value;
189
190 stream.push_back(element);
191}
192
193template void Recorder::RegisterWritten(u32,u8);
194template void Recorder::RegisterWritten(u32,u16);
195template void Recorder::RegisterWritten(u32,u32);
196template void Recorder::RegisterWritten(u32,u64);
197
198}