#1943 make sys free in texture console llmemorysavailphysicalmeminkb tell the truth (#1966)

* Also fix for crash when applying MoaP to PBR material
master
Dave Parks 2024-07-09 15:54:18 -05:00 committed by GitHub
parent 21f40280ec
commit e6e41e71b7
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
5 changed files with 73 additions and 70 deletions

View File

@ -51,13 +51,28 @@
//---------------------------------------------------------------------------- //----------------------------------------------------------------------------
//static //static
// most important memory metric for texture streaming
// On Windows, this should agree with resource monitor -> performance -> memory -> available
// On OS X, this should be activity monitor -> memory -> (physical memory - memory used)
// NOTE: this number MAY be less than the actual available memory on systems with more than MaxHeapSize64 GB of physical memory (default 16GB)
// In that case, should report min(available, sMaxHeapSizeInKB-sAllocateMemInKB)
U32Kilobytes LLMemory::sAvailPhysicalMemInKB(U32_MAX); U32Kilobytes LLMemory::sAvailPhysicalMemInKB(U32_MAX);
// Installed physical memory
U32Kilobytes LLMemory::sMaxPhysicalMemInKB(0); U32Kilobytes LLMemory::sMaxPhysicalMemInKB(0);
// Maximimum heap size according to the user's settings (default 16GB)
U32Kilobytes LLMemory::sMaxHeapSizeInKB(U32_MAX);
// Current memory usage
U32Kilobytes LLMemory::sAllocatedMemInKB(0);
U32Kilobytes LLMemory::sAllocatedPageSizeInKB(0);
static LLTrace::SampleStatHandle<F64Megabytes> sAllocatedMem("allocated_mem", "active memory in use by application"); static LLTrace::SampleStatHandle<F64Megabytes> sAllocatedMem("allocated_mem", "active memory in use by application");
static LLTrace::SampleStatHandle<F64Megabytes> sVirtualMem("virtual_mem", "virtual memory assigned to application"); static LLTrace::SampleStatHandle<F64Megabytes> sVirtualMem("virtual_mem", "virtual memory assigned to application");
U32Kilobytes LLMemory::sAllocatedMemInKB(0);
U32Kilobytes LLMemory::sAllocatedPageSizeInKB(0);
U32Kilobytes LLMemory::sMaxHeapSizeInKB(U32_MAX);
void ll_assert_aligned_func(uintptr_t ptr,U32 alignment) void ll_assert_aligned_func(uintptr_t ptr,U32 alignment)
{ {
@ -86,7 +101,14 @@ void LLMemory::initMaxHeapSizeGB(F32Gigabytes max_heap_size)
void LLMemory::updateMemoryInfo() void LLMemory::updateMemoryInfo()
{ {
LL_PROFILE_ZONE_SCOPED LL_PROFILE_ZONE_SCOPED
U32Kilobytes avail_phys;
sMaxPhysicalMemInKB = gSysMemory.getPhysicalMemoryKB();
U32Kilobytes avail_mem;
LLMemoryInfo::getAvailableMemoryKB(avail_mem);
sAvailPhysicalMemInKB = avail_mem;
#if LL_WINDOWS #if LL_WINDOWS
PROCESS_MEMORY_COUNTERS counters; PROCESS_MEMORY_COUNTERS counters;
@ -99,8 +121,6 @@ void LLMemory::updateMemoryInfo()
sAllocatedMemInKB = U32Kilobytes::convert(U64Bytes(counters.WorkingSetSize)); sAllocatedMemInKB = U32Kilobytes::convert(U64Bytes(counters.WorkingSetSize));
sAllocatedPageSizeInKB = U32Kilobytes::convert(U64Bytes(counters.PagefileUsage)); sAllocatedPageSizeInKB = U32Kilobytes::convert(U64Bytes(counters.PagefileUsage));
sample(sVirtualMem, sAllocatedPageSizeInKB); sample(sVirtualMem, sAllocatedPageSizeInKB);
U32Kilobytes avail_virtual;
LLMemoryInfo::getAvailableMemoryKB(avail_phys, avail_virtual) ;
#elif defined(LL_DARWIN) #elif defined(LL_DARWIN)
task_vm_info info; task_vm_info info;
@ -126,50 +146,20 @@ void LLMemory::updateMemoryInfo()
{ {
LL_WARNS() << "task_info failed" << LL_ENDL; LL_WARNS() << "task_info failed" << LL_ENDL;
} }
// Total installed and available physical memory are properties of the host, not just our process.
vm_statistics64_data_t vmstat;
mach_msg_type_number_t count = HOST_VM_INFO64_COUNT;
mach_port_t host = mach_host_self();
vm_size_t page_size;
host_page_size(host, &page_size);
kern_return_t result = host_statistics64(host, HOST_VM_INFO64, reinterpret_cast<host_info_t>(&vmstat), &count);
if (result == KERN_SUCCESS) {
// This is what Chrome reports as 'the "Physical Memory Free" value reported by the Memory Monitor in Instruments.'
// Note though that inactive pages are not included here and not yet free, but could become so under memory pressure.
avail_phys = U32Bytes(vmstat.free_count * page_size);
sMaxHeapSizeInKB = LLMemoryInfo::getHardwareMemSize();
}
else
{
LL_WARNS() << "task_info failed" << LL_ENDL;
}
#elif defined(LL_LINUX) #elif defined(LL_LINUX)
// Use sysinfo() to get the total physical memory. // Use sysinfo() to get the total physical memory.
struct sysinfo info; struct sysinfo info;
sysinfo(&info); sysinfo(&info);
sMaxHeapSizeInKB = U32Kilobytes::convert((U64Bytes)info.totalram); // Total RAM in system
avail_phys = U32Kilobytes::convert((U64Bytes)info.freeram); // Total Free RAM in system
sAllocatedMemInKB = U32Kilobytes::convert(U64Bytes(LLMemory::getCurrentRSS())); // represents the RAM allocated by this process only (in line with the windows implementation) sAllocatedMemInKB = U32Kilobytes::convert(U64Bytes(LLMemory::getCurrentRSS())); // represents the RAM allocated by this process only (in line with the windows implementation)
#else #else
//not valid for other systems for now. //not valid for other systems for now.
LL_WARNS() << "LLMemory::updateMemoryInfo() not implemented for this platform." << LL_ENDL; LL_WARNS() << "LLMemory::updateMemoryInfo() not implemented for this platform." << LL_ENDL;
sAllocatedMemInKB = U64Bytes(LLMemory::getCurrentRSS()); sAllocatedMemInKB = U64Bytes(LLMemory::getCurrentRSS());
sMaxPhysicalMemInKB = U64Bytes(U32_MAX);
sAvailPhysicalMemInKB = U64Bytes(U32_MAX);
#endif #endif
sample(sAllocatedMem, sAllocatedMemInKB); sample(sAllocatedMem, sAllocatedMemInKB);
// sMaxPhysicalMem - max this process can use = the lesser of (what we already have + what's available) or MaxHeap
sMaxPhysicalMemInKB = llmin(avail_phys + sAllocatedMemInKB, sMaxHeapSizeInKB);
if(sMaxPhysicalMemInKB > sAllocatedMemInKB) sAvailPhysicalMemInKB = llmin(sAvailPhysicalMemInKB, sMaxHeapSizeInKB - sAllocatedMemInKB);
{
sAvailPhysicalMemInKB = sMaxPhysicalMemInKB - sAllocatedMemInKB ;
}
else
{
sAvailPhysicalMemInKB = U32Kilobytes(0);
}
return ; return ;
} }
@ -206,10 +196,10 @@ void LLMemory::logMemoryInfo(bool update)
updateMemoryInfo() ; updateMemoryInfo() ;
} }
LL_INFOS() << "Current allocated physical memory(KB): " << sAllocatedMemInKB << LL_ENDL ; LL_INFOS() << llformat("Current allocated physical memory: %.2f MB", sAllocatedMemInKB / 1024.0) << LL_ENDL;
LL_INFOS() << "Current allocated page size (KB): " << sAllocatedPageSizeInKB << LL_ENDL ; LL_INFOS() << llformat("Current allocated page size: %.2f MB", sAllocatedPageSizeInKB / 1024.0) << LL_ENDL;
LL_INFOS() << "Current available physical memory(KB): " << sAvailPhysicalMemInKB << LL_ENDL ; LL_INFOS() << llformat("Current available physical memory: %.2f MB", sAvailPhysicalMemInKB / 1024.0) << LL_ENDL;
LL_INFOS() << "Current max usable memory(KB): " << sMaxPhysicalMemInKB << LL_ENDL ; LL_INFOS() << llformat("Current max usable memory: %.2f MB", sMaxPhysicalMemInKB / 1024.0) << LL_ENDL;
} }
//static //static

View File

@ -74,6 +74,8 @@ using namespace llsd;
# include <mach/mach_host.h> # include <mach/mach_host.h>
# include <mach/task.h> # include <mach/task.h>
# include <mach/task_info.h> # include <mach/task_info.h>
# include <sys/types.h>
# include <mach/mach_init.h>
#elif LL_LINUX #elif LL_LINUX
# include <errno.h> # include <errno.h>
# include <sys/utsname.h> # include <sys/utsname.h>
@ -85,6 +87,7 @@ const char MEMINFO_FILE[] = "/proc/meminfo";
#endif #endif
LLCPUInfo gSysCPU; LLCPUInfo gSysCPU;
LLMemoryInfo gSysMemory;
// Don't log memory info any more often than this. It also serves as our // Don't log memory info any more often than this. It also serves as our
// framerate sample size. // framerate sample size.
@ -797,33 +800,32 @@ U32Kilobytes LLMemoryInfo::getPhysicalMemoryKB() const
} }
//static //static
void LLMemoryInfo::getAvailableMemoryKB(U32Kilobytes& avail_physical_mem_kb, U32Kilobytes& avail_virtual_mem_kb) void LLMemoryInfo::getAvailableMemoryKB(U32Kilobytes& avail_mem_kb)
{ {
LL_PROFILE_ZONE_SCOPED_CATEGORY_MEMORY;
#if LL_WINDOWS #if LL_WINDOWS
// Sigh, this shouldn't be a static method, then we wouldn't have to // Sigh, this shouldn't be a static method, then we wouldn't have to
// reload this data separately from refresh() // reload this data separately from refresh()
LLSD statsMap(loadStatsMap()); LLSD statsMap(loadStatsMap());
avail_physical_mem_kb = (U32Kilobytes)statsMap["Avail Physical KB"].asInteger(); avail_mem_kb = (U32Kilobytes)statsMap["Avail Physical KB"].asInteger();
avail_virtual_mem_kb = (U32Kilobytes)statsMap["Avail Virtual KB"].asInteger();
#elif LL_DARWIN #elif LL_DARWIN
// mStatsMap is derived from vm_stat, look for (e.g.) "kb free": // use host_statistics64 to get memory info
// $ vm_stat vm_statistics64_data_t vmstat;
// Mach Virtual Memory Statistics: (page size of 4096 bytes) mach_msg_type_number_t count = HOST_VM_INFO64_COUNT;
// Pages free: 462078. mach_port_t host = mach_host_self();
// Pages active: 142010. vm_size_t page_size;
// Pages inactive: 220007. host_page_size(host, &page_size);
// Pages wired down: 159552. kern_return_t result = host_statistics64(host, HOST_VM_INFO64, reinterpret_cast<host_info_t>(&vmstat), &count);
// "Translation faults": 220825184. if (result == KERN_SUCCESS)
// Pages copy-on-write: 2104153. {
// Pages zero filled: 167034876. avail_mem_kb = U64Bytes((vmstat.free_count + vmstat.inactive_count) * page_size);
// Pages reactivated: 65153. }
// Pageins: 2097212. else
// Pageouts: 41759. {
// Object cache: 841598 hits of 7629869 lookups (11% hit rate) avail_mem_kb = (U32Kilobytes)-1;
avail_physical_mem_kb = (U32Kilobytes)-1 ; }
avail_virtual_mem_kb = (U32Kilobytes)-1 ;
#elif LL_LINUX #elif LL_LINUX
// mStatsMap is derived from MEMINFO_FILE: // mStatsMap is derived from MEMINFO_FILE:
@ -874,15 +876,14 @@ void LLMemoryInfo::getAvailableMemoryKB(U32Kilobytes& avail_physical_mem_kb, U32
// DirectMap4k: 434168 kB // DirectMap4k: 434168 kB
// DirectMap2M: 477184 kB // DirectMap2M: 477184 kB
// (could also run 'free', but easier to read a file than run a program) // (could also run 'free', but easier to read a file than run a program)
avail_physical_mem_kb = (U32Kilobytes)-1 ; LLSD statsMap(loadStatsMap());
avail_virtual_mem_kb = (U32Kilobytes)-1 ;
avail_mem_kb = (U32Kilobytes)statsMap["MemFree"].asInteger();
#else #else
//do not know how to collect available memory info for other systems. //do not know how to collect available memory info for other systems.
//leave it blank here for now. //leave it blank here for now.
avail_physical_mem_kb = (U32Kilobytes)-1 ; avail_mem_kb = (U32Kilobytes)-1 ;
avail_virtual_mem_kb = (U32Kilobytes)-1 ;
#endif #endif
} }
@ -977,7 +978,7 @@ LLSD LLMemoryInfo::loadStatsMap()
// specifically accepts PROCESS_MEMORY_COUNTERS*, and since this is a // specifically accepts PROCESS_MEMORY_COUNTERS*, and since this is a
// classic-C API, PROCESS_MEMORY_COUNTERS_EX isn't a subclass. Cast the // classic-C API, PROCESS_MEMORY_COUNTERS_EX isn't a subclass. Cast the
// pointer. // pointer.
GetProcessMemoryInfo(GetCurrentProcess(), PPROCESS_MEMORY_COUNTERS(&pmem), sizeof(pmem)); GetProcessMemoryInfo(GetCurrentProcess(), (PROCESS_MEMORY_COUNTERS*) &pmem, sizeof(pmem));
stats.add("Page Fault Count", pmem.PageFaultCount); stats.add("Page Fault Count", pmem.PageFaultCount);
stats.add("PeakWorkingSetSize KB", pmem.PeakWorkingSetSize/div); stats.add("PeakWorkingSetSize KB", pmem.PeakWorkingSetSize/div);

View File

@ -134,8 +134,8 @@ public:
static U32Kilobytes getHardwareMemSize(); // Because some Mac linkers won't let us reference extern gSysMemory from a different lib. static U32Kilobytes getHardwareMemSize(); // Because some Mac linkers won't let us reference extern gSysMemory from a different lib.
#endif #endif
//get the available memory infomation in KiloBytes. //get the available memory in KiloBytes.
static void getAvailableMemoryKB(U32Kilobytes& avail_physical_mem_kb, U32Kilobytes& avail_virtual_mem_kb); static void getAvailableMemoryKB(U32Kilobytes& avail_mem_kb);
// Retrieve a map of memory statistics. The keys of the map are platform- // Retrieve a map of memory statistics. The keys of the map are platform-
// dependent. The values are in kilobytes to try to avoid integer overflow. // dependent. The values are in kilobytes to try to avoid integer overflow.
@ -169,6 +169,7 @@ bool LL_COMMON_API gunzip_file(const std::string& srcfile, const std::string& ds
// gzip srcfile into dstfile. Returns false on error. // gzip srcfile into dstfile. Returns false on error.
bool LL_COMMON_API gzip_file(const std::string& srcfile, const std::string& dstfile); bool LL_COMMON_API gzip_file(const std::string& srcfile, const std::string& dstfile);
extern LL_COMMON_API LLMemoryInfo gSysMemory;
extern LL_COMMON_API LLCPUInfo gSysCPU; extern LL_COMMON_API LLCPUInfo gSysCPU;
#endif // LL_LLSYS_H #endif // LL_LLSYS_H

View File

@ -347,7 +347,6 @@ LLFrameTimer gRestoreGLTimer;
bool gRestoreGL = false; bool gRestoreGL = false;
bool gUseWireframe = false; bool gUseWireframe = false;
LLMemoryInfo gSysMemory;
U64Bytes gMemoryAllocated(0); // updated in display_stats() in llviewerdisplay.cpp U64Bytes gMemoryAllocated(0); // updated in display_stats() in llviewerdisplay.cpp
std::string gLastVersionChannel; std::string gLastVersionChannel;

View File

@ -3489,7 +3489,19 @@ void LLViewerMediaTexture::setPlaying(bool playing)
for(std::list< LLFace* >::iterator iter = mMediaFaceList.begin(); iter!= mMediaFaceList.end(); ++iter) for(std::list< LLFace* >::iterator iter = mMediaFaceList.begin(); iter!= mMediaFaceList.end(); ++iter)
{ {
switchTexture(LLRender::DIFFUSE_MAP, *iter); LLFace* facep = *iter;
const LLTextureEntry* te = facep->getTextureEntry();
if (te->getGLTFMaterial())
{
// PBR material, switch emissive and basecolor
switchTexture(LLRender::EMISSIVE_MAP, *iter);
switchTexture(LLRender::BASECOLOR_MAP, *iter);
}
else
{
// blinn-phong material, switch diffuse map only
switchTexture(LLRender::DIFFUSE_MAP, *iter);
}
} }
} }
else //stop playing this media else //stop playing this media