| 
									
										
										
										
											2019-05-30 08:52:20 -04:00
										 |  |  | // Copyright 2019 yuzu Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-30 16:57:23 -04:00
										 |  |  | #include <atomic>
 | 
					
						
							|  |  |  | #include <mutex>
 | 
					
						
							| 
									
										
										
										
											2019-05-30 08:52:20 -04:00
										 |  |  | #include <optional>
 | 
					
						
							|  |  |  | #include <vector>
 | 
					
						
							|  |  |  | #include "common/common_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-30 16:57:23 -04:00
										 |  |  | namespace Core::Timing { | 
					
						
							|  |  |  | class CoreTiming; | 
					
						
							|  |  |  | struct EventType; | 
					
						
							|  |  |  | } // namespace Core::Timing
 | 
					
						
							| 
									
										
										
										
											2019-05-30 08:52:20 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Memory { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // A class that will effectively freeze memory values.
 | 
					
						
							|  |  |  | class Freezer { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     struct Entry { | 
					
						
							|  |  |  |         VAddr address; | 
					
						
							| 
									
										
										
										
											2019-05-30 16:57:23 -04:00
										 |  |  |         u32 width; | 
					
						
							| 
									
										
										
										
											2019-05-30 08:52:20 -04:00
										 |  |  |         u64 value; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-30 16:57:23 -04:00
										 |  |  |     explicit Freezer(Core::Timing::CoreTiming& core_timing); | 
					
						
							| 
									
										
										
										
											2019-05-30 08:52:20 -04:00
										 |  |  |     ~Freezer(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-30 16:57:23 -04:00
										 |  |  |     // Enables or disables the entire memory freezer.
 | 
					
						
							| 
									
										
										
										
											2019-05-30 08:52:20 -04:00
										 |  |  |     void SetActive(bool active); | 
					
						
							| 
									
										
										
										
											2019-05-30 16:57:23 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Returns whether or not the freezer is active.
 | 
					
						
							| 
									
										
										
										
											2019-05-30 08:52:20 -04:00
										 |  |  |     bool IsActive() const; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-30 16:57:23 -04:00
										 |  |  |     // Removes all entries from the freezer.
 | 
					
						
							| 
									
										
										
										
											2019-05-30 08:52:20 -04:00
										 |  |  |     void Clear(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-30 16:57:23 -04:00
										 |  |  |     // Freezes a value to its current memory address. The value the memory is kept at will be the
 | 
					
						
							|  |  |  |     // value that is read during this function. Width can be 1, 2, 4, or 8 (in bytes).
 | 
					
						
							|  |  |  |     u64 Freeze(VAddr address, u32 width); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Unfreezes the memory value at address. If the address isn't frozen, this is a no-op.
 | 
					
						
							| 
									
										
										
										
											2019-05-30 08:52:20 -04:00
										 |  |  |     void Unfreeze(VAddr address); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-30 16:57:23 -04:00
										 |  |  |     // Returns whether or not the address is frozen.
 | 
					
						
							|  |  |  |     bool IsFrozen(VAddr address) const; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Sets the value that address should be frozen to. This doesn't change the width set by using
 | 
					
						
							|  |  |  |     // Freeze(). If the value isn't frozen, this will not freeze it and is thus a no-op.
 | 
					
						
							| 
									
										
										
										
											2019-05-30 08:52:20 -04:00
										 |  |  |     void SetFrozenValue(VAddr address, u64 value); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-30 16:57:23 -04:00
										 |  |  |     // Returns the entry corresponding to the address if the address is frozen, otherwise
 | 
					
						
							|  |  |  |     // std::nullopt.
 | 
					
						
							|  |  |  |     std::optional<Entry> GetEntry(VAddr address) const; | 
					
						
							| 
									
										
										
										
											2019-05-30 08:52:20 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-30 16:57:23 -04:00
										 |  |  |     // Returns all the entries in the freezer, an empty vector means nothing is frozen.
 | 
					
						
							|  |  |  |     std::vector<Entry> GetEntries() const; | 
					
						
							| 
									
										
										
										
											2019-05-30 08:52:20 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     void FrameCallback(u64 userdata, s64 cycles_late); | 
					
						
							|  |  |  |     void FillEntryReads(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::atomic_bool active{false}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-30 16:57:23 -04:00
										 |  |  |     mutable std::mutex entries_mutex; | 
					
						
							| 
									
										
										
										
											2019-05-30 08:52:20 -04:00
										 |  |  |     std::vector<Entry> entries; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Core::Timing::EventType* event; | 
					
						
							|  |  |  |     Core::Timing::CoreTiming& core_timing; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Memory
 |