 c393cdf8e3
			
		
	
	
		c393cdf8e3
		
			
		
	
	
	
	
		
			
			* Keep track mapped buffers with fixed offsets * Started rewriting the memory manager * Initial support for MapPhysicalMemory and UnmapPhysicalMemory, other tweaks * MapPhysicalMemory/UnmapPhysicalMemory support, other tweaks * Rebased * Optimize the map/unmap physical memory svcs * Integrate shared font support * Fix address space reserve alignment * Some fixes related to gpu memory mapping * Some cleanup * Only try uploading const buffers that are really used * Check if memory region is contiguous * Rebased * Add missing count increment on IsRegionModified * Check for reads/writes outside of the address space, optimize translation with a tail call
		
			
				
	
	
		
			174 lines
		
	
	
		
			No EOL
		
	
	
		
			5.9 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			174 lines
		
	
	
		
			No EOL
		
	
	
		
			5.9 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using ChocolArm64.Memory;
 | |
| using Ryujinx.HLE.Loaders.Executables;
 | |
| using Ryujinx.HLE.OsHle;
 | |
| using Ryujinx.HLE.OsHle.Handles;
 | |
| using Ryujinx.HLE.OsHle.Utilities;
 | |
| using System.Collections.Generic;
 | |
| using System.IO;
 | |
| 
 | |
| namespace Ryujinx.HLE.Loaders
 | |
| {
 | |
|     class Executable
 | |
|     {
 | |
|         private List<ElfDyn> Dynamic;
 | |
| 
 | |
|         private Dictionary<long, string> m_SymbolTable;
 | |
| 
 | |
|         public IReadOnlyDictionary<long, string> SymbolTable => m_SymbolTable;
 | |
| 
 | |
|         public string Name { get; private set; }
 | |
| 
 | |
|         public string FilePath { get; private set; }
 | |
| 
 | |
|         public long ImageBase { get; private set; }
 | |
|         public long ImageEnd  { get; private set; }
 | |
| 
 | |
|         private AMemory Memory;
 | |
| 
 | |
|         private KMemoryManager MemoryManager;
 | |
| 
 | |
|         public Executable(IExecutable Exe, KMemoryManager MemoryManager, AMemory Memory, long ImageBase)
 | |
|         {
 | |
|             Dynamic = new List<ElfDyn>();
 | |
| 
 | |
|             m_SymbolTable = new Dictionary<long, string>();
 | |
| 
 | |
|             FilePath = Exe.FilePath;
 | |
| 
 | |
|             if (FilePath != null)
 | |
|             {
 | |
|                 Name = Path.GetFileNameWithoutExtension(FilePath.Replace(Homebrew.TemporaryNroSuffix, ""));
 | |
|             }
 | |
| 
 | |
|             this.Memory        = Memory;
 | |
|             this.MemoryManager = MemoryManager;
 | |
|             this.ImageBase     = ImageBase;
 | |
|             this.ImageEnd      = ImageBase;
 | |
| 
 | |
|             long TextPosition = ImageBase + (uint)Exe.TextOffset;
 | |
|             long ROPosition   = ImageBase + (uint)Exe.ROOffset;
 | |
|             long DataPosition = ImageBase + (uint)Exe.DataOffset;
 | |
| 
 | |
|             long TextSize = (uint)IntUtils.AlignUp(Exe.Text.Length, KMemoryManager.PageSize);
 | |
|             long ROSize   = (uint)IntUtils.AlignUp(Exe.RO.Length,   KMemoryManager.PageSize);
 | |
|             long DataSize = (uint)IntUtils.AlignUp(Exe.Data.Length, KMemoryManager.PageSize);
 | |
| 
 | |
|             long DataAndBssSize = (uint)IntUtils.AlignUp(Exe.BssSize, KMemoryManager.PageSize) + DataSize;
 | |
| 
 | |
|             ImageEnd = DataPosition + DataAndBssSize;
 | |
| 
 | |
|             MemoryManager.HleMapProcessCode(TextPosition, TextSize + ROSize + DataAndBssSize);
 | |
| 
 | |
|             MemoryManager.SetProcessMemoryPermission(ROPosition,   ROSize,         MemoryPermission.Read);
 | |
|             MemoryManager.SetProcessMemoryPermission(DataPosition, DataAndBssSize, MemoryPermission.ReadAndWrite);
 | |
| 
 | |
|             Memory.WriteBytes(TextPosition, Exe.Text);
 | |
|             Memory.WriteBytes(ROPosition,   Exe.RO);
 | |
|             Memory.WriteBytes(DataPosition, Exe.Data);
 | |
| 
 | |
|             if (Exe.Mod0Offset == 0)
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             long Mod0Offset = ImageBase + Exe.Mod0Offset;
 | |
| 
 | |
|             int  Mod0Magic        = Memory.ReadInt32(Mod0Offset + 0x0);
 | |
|             long DynamicOffset    = Memory.ReadInt32(Mod0Offset + 0x4)  + Mod0Offset;
 | |
|             long BssStartOffset   = Memory.ReadInt32(Mod0Offset + 0x8)  + Mod0Offset;
 | |
|             long BssEndOffset     = Memory.ReadInt32(Mod0Offset + 0xc)  + Mod0Offset;
 | |
|             long EhHdrStartOffset = Memory.ReadInt32(Mod0Offset + 0x10) + Mod0Offset;
 | |
|             long EhHdrEndOffset   = Memory.ReadInt32(Mod0Offset + 0x14) + Mod0Offset;
 | |
|             long ModObjOffset     = Memory.ReadInt32(Mod0Offset + 0x18) + Mod0Offset;
 | |
| 
 | |
|             while (true)
 | |
|             {
 | |
|                 long TagVal = Memory.ReadInt64(DynamicOffset + 0);
 | |
|                 long Value  = Memory.ReadInt64(DynamicOffset + 8);
 | |
| 
 | |
|                 DynamicOffset += 0x10;
 | |
| 
 | |
|                 ElfDynTag Tag = (ElfDynTag)TagVal;
 | |
| 
 | |
|                 if (Tag == ElfDynTag.DT_NULL)
 | |
|                 {
 | |
|                     break;
 | |
|                 }
 | |
| 
 | |
|                 Dynamic.Add(new ElfDyn(Tag, Value));
 | |
|             }
 | |
| 
 | |
|             long StrTblAddr = ImageBase + GetFirstValue(ElfDynTag.DT_STRTAB);
 | |
|             long SymTblAddr = ImageBase + GetFirstValue(ElfDynTag.DT_SYMTAB);
 | |
| 
 | |
|             long SymEntSize = GetFirstValue(ElfDynTag.DT_SYMENT);
 | |
| 
 | |
|             while ((ulong)SymTblAddr < (ulong)StrTblAddr)
 | |
|             {
 | |
|                 ElfSym Sym = GetSymbol(SymTblAddr, StrTblAddr);
 | |
| 
 | |
|                 m_SymbolTable.TryAdd(Sym.Value, Sym.Name);
 | |
| 
 | |
|                 SymTblAddr += SymEntSize;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private ElfRel GetRelocation(long Position)
 | |
|         {
 | |
|             long Offset = Memory.ReadInt64(Position + 0);
 | |
|             long Info   = Memory.ReadInt64(Position + 8);
 | |
|             long Addend = Memory.ReadInt64(Position + 16);
 | |
| 
 | |
|             int RelType = (int)(Info >> 0);
 | |
|             int SymIdx  = (int)(Info >> 32);
 | |
| 
 | |
|             ElfSym Symbol = GetSymbol(SymIdx);
 | |
| 
 | |
|             return new ElfRel(Offset, Addend, Symbol, (ElfRelType)RelType);
 | |
|         }
 | |
| 
 | |
|         private ElfSym GetSymbol(int Index)
 | |
|         {
 | |
|             long StrTblAddr = ImageBase + GetFirstValue(ElfDynTag.DT_STRTAB);
 | |
|             long SymTblAddr = ImageBase + GetFirstValue(ElfDynTag.DT_SYMTAB);
 | |
| 
 | |
|             long SymEntSize = GetFirstValue(ElfDynTag.DT_SYMENT);
 | |
| 
 | |
|             long Position = SymTblAddr + Index * SymEntSize;
 | |
| 
 | |
|             return GetSymbol(Position, StrTblAddr);
 | |
|         }
 | |
| 
 | |
|         private ElfSym GetSymbol(long Position, long StrTblAddr)
 | |
|         {
 | |
|             int  NameIndex = Memory.ReadInt32(Position + 0);
 | |
|             int  Info      = Memory.ReadByte(Position + 4);
 | |
|             int  Other     = Memory.ReadByte(Position + 5);
 | |
|             int  SHIdx     = Memory.ReadInt16(Position + 6);
 | |
|             long Value     = Memory.ReadInt64(Position + 8);
 | |
|             long Size      = Memory.ReadInt64(Position + 16);
 | |
| 
 | |
|             string Name = string.Empty;
 | |
| 
 | |
|             for (int Chr; (Chr = Memory.ReadByte(StrTblAddr + NameIndex++)) != 0;)
 | |
|             {
 | |
|                 Name += (char)Chr;
 | |
|             }
 | |
| 
 | |
|             return new ElfSym(Name, Info, Other, SHIdx, Value, Size);
 | |
|         }
 | |
| 
 | |
|         private long GetFirstValue(ElfDynTag Tag)
 | |
|         {
 | |
|             foreach (ElfDyn Entry in Dynamic)
 | |
|             {
 | |
|                 if (Entry.Tag == Tag)
 | |
|                 {
 | |
|                     return Entry.Value;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return 0;
 | |
|         }
 | |
|     }
 | |
| } |