source-engine/game/client/econ/tool_items/custom_texture_cache.cpp

1112 lines
31 KiB
C++
Raw Normal View History

2020-04-22 12:56:21 -04:00
//========= Copyright Valve Corporation, All rights reserved. ============//
#include "cbase.h"
#include "custom_texture_cache.h"
#include "materialsystem/imaterialproxy.h"
#include "materialsystem/imaterialvar.h"
#include "materialsystem/itexture.h"
#include "pixelwriter.h"
#include "checksum_md5.h"
#include "imageutils.h"
#include "toolframework_client.h"
#include "econ_gcmessages.h"
#include "econ_item_inventory.h"
#include "VGuiMatSurface/IMatSystemSurface.h"
#include "bitmap/bitmap.h"
using namespace CustomTextureSystem;
ITexture *CustomTextureSystem::g_pPreviewCustomTexture = NULL;
CEconItemView *CustomTextureSystem::g_pPreviewEconItem = NULL;
bool CustomTextureSystem::g_pPreviewCustomTextureDirty = true;
const char CustomTextureSystem::k_rchCustomTextureFilterPreviewImageName[] = "__CustomTextureFilterPreview";
const char CustomTextureSystem::k_rchCustomTextureFilterPreviewTextureName[] = "vgui/__CustomTextureFilterPreview";
//-----------------------------------------------------------------------------
static ISteamRemoteStorage *GetISteamRemoteStorage()
{
return steamapicontext?steamapicontext->SteamRemoteStorage():NULL;
// return Steam3Client().SteamRemoteStorage();
}
static void CalcMD5Ascii( char *szDigestAscii, const void *data, int dataSz )
{
MD5Context_t context;
unsigned char digest[ MD5_DIGEST_LENGTH ];
MD5Init( &context );
MD5Update( &context, (const unsigned char *)data, dataSz );
MD5Final( digest, &context );
Q_binarytohex( digest, MD5_DIGEST_LENGTH, szDigestAscii, MD5_DIGEST_LENGTH*2+1 );
}
static bool BReadSteamRemoteFileToBuffer( CUtlBuffer &outBuffer, const char *pchRemoteFilename )
{
outBuffer.Purge();
ISteamRemoteStorage *pRemoteStorage = GetISteamRemoteStorage();
if ( !pRemoteStorage )
return false;
if ( !pRemoteStorage->FileExists( pchRemoteFilename ))
return false;
int nFileSize = pRemoteStorage->GetFileSize( pchRemoteFilename );
if ( nFileSize <= 0 )
return false;
// Allocate space
outBuffer.SeekPut( CUtlBuffer::SEEK_HEAD, nFileSize );
int nSizeRead = pRemoteStorage->FileRead( pchRemoteFilename, outBuffer.Base(), nFileSize );
return ( nSizeRead == nFileSize );
}
//-----------------------------------------------------------------------------
// Local cache of custom images. This cache contains files from the cloud, and
// and also a few virtual textures that are used during autioning / tweaking
/// Name of cloud-backed config file remembering the custom images that the user
/// has uploaded to the cloud.
static const char k_szCustomTextureRecentListFilename[] = "stamped_items_mru.txt";
/// Track a single entry
struct SCustomImageCacheEntry : private ITextureRegenerator
{
/// If this has been assigned a cloud ID, what is it?
UGCHandle_t m_hCloudID;
/// If this is one of our files, then we know the MD5.
/// This is empty for other people's files
char m_szDigestAscii[ MD5_DIGEST_LENGTH*2 + 4];
//
// Bookkeeping for steam downloads of UGC.
//
/// -1 = failure, 0 = not started, 1 = in progress, 2 = finished OK and image should be in memory
int m_nStatus;
/// Handle to the active download, or k_uAPICallInvalid if not active
SteamAPICall_t m_hDownloadApiCall;
/// Procedural texture object. We hold a reference.
ITexture *m_pTexture;
/// Procedurally-created material. We hold a reference.
IMaterial *m_pMaterial;
/// GUI texture handle. (It's bound to the material.)
int m_iVguiHandle;
/// The raw image
Bitmap_t m_image;
/// Doubly-linked list. We keep it in MRU order so we know what to eject from the cache
SCustomImageCacheEntry *m_pPrev;
SCustomImageCacheEntry *m_pNext;
SCustomImageCacheEntry()
: m_hCloudID(0)
, m_pTexture(NULL)
, m_nStatus(0)
, m_hDownloadApiCall(k_uAPICallInvalid)
, m_pPrev(NULL)
, m_pNext(NULL)
, m_pMaterial(NULL)
, m_iVguiHandle(0)
{
m_szDigestAscii[0] = '\0';
}
virtual ~SCustomImageCacheEntry()
{
Clear();
}
// Release texture / VGUI resources. This doesn't free the image we have
// loaded or stop any async actions that were in progress. (Use Clear())
void ReleaseResources()
{
if ( m_pTexture )
{
ITexture *tex = m_pTexture;
m_pTexture = NULL; // clear pointer first, to prevent infinite recursion
tex->SetTextureRegenerator( NULL );
tex->Release();
}
if ( m_pMaterial )
{
m_pMaterial->Release();
m_pMaterial = NULL;
}
if ( m_iVguiHandle != 0 )
{
g_pMatSystemSurface->DestroyTextureID( m_iVguiHandle );
m_iVguiHandle = 0;
}
}
/// Inherited from ITextureRegenerator
///
/// Gets called when our ITextureRegenerator interface gets detached from the texture.
/// We should be the only ones doing this --- so that means we had better have already
/// cleared our texture at this point!
virtual void Release()
{
Assert( m_pTexture == NULL );
}
/// Inherited from ITextureRegenerator
///
/// The main interface function that actually supplies the texture bits
virtual void RegenerateTextureBits( ITexture *pTexture, IVTFTexture *pVTFTexture, Rect_t *pRect )
{
Assert( pVTFTexture->FrameCount() == 1 );
Assert( pVTFTexture->FaceCount() == 1 );
Assert( pTexture == m_pTexture );
Assert( !pTexture->IsMipmapped() );
int nWidth, nHeight, nDepth;
pVTFTexture->ComputeMipLevelDimensions( 0, &nWidth, &nHeight, &nDepth );
Assert( nDepth == 1 );
Assert( nWidth == m_image.Width() && nHeight == m_image.Height() );
CPixelWriter pixelWriter;
pixelWriter.SetPixelMemory( pVTFTexture->Format(),
pVTFTexture->ImageData( 0, 0, 0 ), pVTFTexture->RowSizeInBytes( 0 ) );
// !SPEED! 'Tis probably DEATHLY slow...
for ( int y = 0; y < nHeight; ++y )
{
pixelWriter.Seek( 0, y );
for ( int x = 0; x < nWidth; ++x )
{
Color c = m_image.GetColor( x, y );
pixelWriter.WritePixel( c.r(), c.g(), c.b(), c.a() );
}
}
}
void Clear()
{
ReleaseResources();
m_image.Clear();
m_szDigestAscii[0] = '\0';
m_nStatus = 0;
m_hCloudID = 0;
// !KLUDGE! How can I clean this up properly if something is
// in progress?
m_hDownloadApiCall = k_uAPICallInvalid;
}
/// Poll the entry and update bookeeping if we're busy.
void Poll()
{
// We must know our cloud ID
if ( m_hCloudID == 0 )
{
Assert( m_hCloudID != 0 );
return;
}
// If texture already exists, then we are definitely done!
if ( m_pTexture )
{
Assert( m_nStatus == 2 );
return;
}
// Check if we have not yet initiated anything
if ( m_nStatus == 0 )
{
// We'll need to download it.
// Start by assuming failure.
m_nStatus = -1;
// Start download
ISteamRemoteStorage *pRemoteStorage = GetISteamRemoteStorage();
if ( pRemoteStorage )
{
m_hDownloadApiCall = pRemoteStorage->UGCDownload( m_hCloudID, 0 );
if ( m_hDownloadApiCall != k_uAPICallInvalid )
{
// Mark download as in progress
Msg( "Started download of cloud file %08X%08X\n", (uint32)(m_hCloudID>>32), (uint32)m_hCloudID );
m_nStatus = 1;
}
}
}
// If we're in progress, poll the result
if ( m_nStatus == 1 )
{
PollDownload();
}
// If result has completed, then fetch the texture
Assert( m_pTexture == NULL );
if ( m_nStatus == 2 && m_image.IsValid() )
{
// Generate the logical texture name
char rchTextureName[MAX_PATH];
GenerateLocalTextureName( rchTextureName );
ITexture *pTexture = NULL;
if ( g_pMaterialSystem->IsTextureLoaded( rchTextureName ) )
{
pTexture = g_pMaterialSystem->FindTexture( rchTextureName, TEXTURE_GROUP_VGUI );
pTexture->AddRef();
Assert( pTexture );
}
else
{
pTexture = g_pMaterialSystem->CreateProceduralTexture(
rchTextureName,
TEXTURE_GROUP_VGUI,
k_nCustomImageSize, k_nCustomImageSize,
IMAGE_FORMAT_RGBA8888,
TEXTUREFLAGS_CLAMPS | TEXTUREFLAGS_CLAMPT | TEXTUREFLAGS_NOMIP | TEXTUREFLAGS_NOLOD
);
Assert( pTexture );
}
pTexture->SetTextureRegenerator( this ); // note carefully order of operations here. See Release()
m_pTexture = pTexture;
// Upload the data now
m_pTexture->Download();
}
else
{
Assert( m_nStatus < 2 );
Assert( !m_image.IsValid() );
}
}
void PollDownload()
{
Assert( m_nStatus == 1 );
// Sanity check we have everything we need
ISteamRemoteStorage *pRemoteStorage = GetISteamRemoteStorage();
if ( m_hDownloadApiCall == k_uAPICallInvalid || !steamapicontext || !steamapicontext->SteamUtils() || !pRemoteStorage )
{
// ???
Assert( m_hDownloadApiCall != k_uAPICallInvalid );
Assert( steamapicontext && steamapicontext->SteamUtils() );
Assert( pRemoteStorage );
m_nStatus = -1;
return;
}
// Poll progress
bool bFailed;
RemoteStorageDownloadUGCResult_t result;
if ( !steamapicontext->SteamUtils()->GetAPICallResult(m_hDownloadApiCall,
&result, sizeof(result), RemoteStorageDownloadUGCResult_t::k_iCallback, &bFailed) )
{
// Still busy.
return;
}
// Make sure we got back the file we were expecting
Assert( result.m_hFile == m_hCloudID );
// Clear status, mark success
m_hDownloadApiCall = k_uAPICallInvalid;
// Completed. Did we succeed?
if ( bFailed )
{
Warning( "Download of custom image file from UFS (UGC=%08X%08X) failed.\n", (uint32)(m_hCloudID >> 32), (uint32)(m_hCloudID) );
m_nStatus = -1;
return;
}
// Fetch file details
AppId_t nAppID;
char *pchName;
int32 nFileSizeInBytes = -1;
CSteamID steamIDOwner;
if ( !pRemoteStorage->GetUGCDetails( m_hCloudID, &nAppID, &pchName, &nFileSizeInBytes, &steamIDOwner )
|| nFileSizeInBytes <= 0 || nFileSizeInBytes >= k_nMaxCustomImageFileSize )
{
Warning( "GetUGCDetails failed? (UGC=%08X%08X nFileSizeInBytes=%d).\n", (uint32)(m_hCloudID >> 32), (uint32)(m_hCloudID), nFileSizeInBytes );
m_nStatus = -1;
return;
}
// Load the file data
CUtlBuffer fileData;
fileData.SeekPut( CUtlBuffer::SEEK_HEAD, nFileSizeInBytes );
// Read in the data. Phil says this is supposed to be basically a memcpy
// or some other fast, local operation.
if ( pRemoteStorage->UGCRead( m_hCloudID, fileData.Base( ), nFileSizeInBytes, 0, k_EUGCRead_ContinueReadingUntilFinished ) != nFileSizeInBytes )
{
Warning( "UGCRead failed? (UGC=%08X%08X).\n", (uint32)(m_hCloudID >> 32), (uint32)(m_hCloudID) );
m_nStatus = -1;
return;
}
// Parse the PNG file data
if ( ImgUtl_LoadPNGBitmapFromBuffer( fileData, m_image ) != CE_SUCCESS )
{
Warning( "Corrupt PNG file, UGC=%08X%08X.\n", (uint32)(m_hCloudID >> 32), (uint32)(m_hCloudID) );
m_nStatus = -1;
return;
}
// We have the raw data
m_nStatus = 2;
}
int GetGuiHandle()
{
// Should never be called on entries without a cloud ID
if ( m_hCloudID == 0 )
{
Assert( m_hCloudID != 0 );
return 0;
}
// Already have one?
if ( m_iVguiHandle != 0 )
{
return m_iVguiHandle;
}
// Process texture downloading, etc
Poll();
// If we don't have a texture yet, or don't know our logical name, then we cannot draw
if ( m_pTexture == NULL )
{
return 0;
}
// Make a material, if we don't already have one
if ( m_pMaterial == NULL )
{
// Generate the material name
char rchImageName[MAX_PATH], rchMaterialName[MAX_PATH];
GenerateLocalImageNameBase( rchImageName );
Q_snprintf( rchMaterialName, MAX_PATH, "vgui/%s.mtl", rchImageName );
// Does it already exist?
if ( g_pMaterialSystem->IsMaterialLoaded( rchMaterialName ) )
{
m_pMaterial = g_pMaterialSystem->FindMaterial( rchMaterialName, TEXTURE_GROUP_VGUI );
Assert( m_pMaterial );
}
else
{
// Fetch the texture name
char rchTextureName[MAX_PATH];
GenerateLocalTextureName( rchTextureName );
// Create dummy material KV data
KeyValues *pVMTKeyValues = new KeyValues( "UnlitGeneric" );
pVMTKeyValues->SetString( "$basetexture", rchTextureName );
pVMTKeyValues->SetInt( "$vertexcolor", 1 );
pVMTKeyValues->SetInt( "$vertexalpha", 1 );
pVMTKeyValues->SetInt( "$translucent", 1 );
// Create the material
m_pMaterial = g_pMaterialSystem->CreateMaterial(
rchMaterialName,
pVMTKeyValues
);
}
// Bind the material to a new VGUI texture object
m_iVguiHandle = g_pMatSystemSurface->CreateNewTextureID();
g_pMatSystemSurface->DrawSetTextureMaterial( m_iVguiHandle, m_pMaterial );
}
return m_iVguiHandle;
}
// Generate logical image name, with no leading materials or vgui directories
// nor a file extension.
void GenerateLocalImageNameBase( char *result ) const
{
Assert( m_hCloudID != 0 );
// Generate the local filenames. !KLUDGE! I'm not sure the platform-safe way
// to print a 64-bit int, so I'll just print both halves myself
Q_snprintf( result, 64, "cloud_custom_images/%08X%08X", (uint32)(m_hCloudID >> 32), (uint32)(m_hCloudID) );
}
/// Logical texture name, including "vgui" but not "materials"
void GenerateLocalTextureName( char *result ) const
{
char rchImageName[MAX_PATH];
GenerateLocalImageNameBase( rchImageName );
Q_snprintf( result, MAX_PATH, "vgui/%s.vtf", rchImageName );
}
/// Full local filename, including leading "materials" directory
void GenerateLocalFilename( char *result ) const
{
char szLocalTextureName[MAX_PATH];
GenerateLocalTextureName( szLocalTextureName );
Q_snprintf( result, MAX_PATH, "materials/%s", szLocalTextureName );
}
};
/// Head of linked list of entries, in MRU order
static SCustomImageCacheEntry *mruCustomImageEntry = NULL;
/// Map of entries, indexed by cloud ID
typedef CUtlMap<UGCHandle_t, SCustomImageCacheEntry *, int> tCustomTextureInfoMap;
static tCustomTextureInfoMap g_mapCustomTextureInfoByCloudId( DefLessFunc(UGCHandle_t) );
// Remove from linked list, without deleting. The item must already be in the list
static void CustomTextureCache_Remove(SCustomImageCacheEntry *pEntry)
{
Assert( pEntry );
// List had better not be empty. Commence paranoia.
Assert( mruCustomImageEntry );
Assert( !mruCustomImageEntry->m_pPrev );
SCustomImageCacheEntry *p = pEntry->m_pPrev;
SCustomImageCacheEntry *n = pEntry->m_pNext;
// Detach from next, if we're not last
if ( n != NULL )
{
Assert( n->m_pPrev == pEntry);
n->m_pPrev = p;
}
// At the head?
if ( !p )
{
Assert( mruCustomImageEntry == pEntry );
mruCustomImageEntry = n;
}
else
{
// Detach from previous
Assert( p->m_pNext == pEntry );
p->m_pNext = n;
}
// Clear pointers
pEntry->m_pPrev = pEntry->m_pNext = NULL;
}
// Insert the item at the head (MRU) slot. The item shouldn't
// already be in the list
static void CustomTextureCache_InsertAtHead(SCustomImageCacheEntry *pEntry)
{
Assert( pEntry );
Assert( !pEntry->m_pNext );
Assert( !pEntry->m_pPrev );
// Edge case of inserting into empty list
if ( mruCustomImageEntry )
{
Assert( !mruCustomImageEntry->m_pPrev );
mruCustomImageEntry->m_pPrev = pEntry;
}
else
{
// Inserting into an empty list
}
// Do the head insertion.
pEntry->m_pNext = mruCustomImageEntry;
mruCustomImageEntry = pEntry;
}
// Reorder list, setting item at the head (MRU) slot. The item must already
// be in the list somewhere.
static void CustomTextureCache_SetMRU(SCustomImageCacheEntry *pEntry)
{
// Note: even if we are already at the head, go through the motions, anyway,
// to exercise all of the sanity checking code.
CustomTextureCache_Remove(pEntry);
CustomTextureCache_InsertAtHead(pEntry);
}
static SCustomImageCacheEntry *CustomTextureCache_NewEntry()
{
SCustomImageCacheEntry *pEntry = new SCustomImageCacheEntry;
// Go ahead and put us at the head
CustomTextureCache_InsertAtHead(pEntry);
// Return the new entry
return pEntry;
}
static SCustomImageCacheEntry *CustomTextureCache_FindOrAddByCloudId( UGCHandle_t ugcHandle )
{
// Locate the bookeeping entry, if one exists
int idx = g_mapCustomTextureInfoByCloudId.Find( ugcHandle );
SCustomImageCacheEntry *pEntry;
if ( g_mapCustomTextureInfoByCloudId.IsValidIndex( idx ) )
{
pEntry = g_mapCustomTextureInfoByCloudId[idx];
// We're accessing it, so move it to the head, the MRU slot
CustomTextureCache_SetMRU(pEntry);
}
else
{
// Grab a new entry
pEntry = CustomTextureCache_NewEntry();
// Assign the cloud ID
pEntry->m_hCloudID = ugcHandle;
// Add it to the map by cloud ID
idx = g_mapCustomTextureInfoByCloudId.Insert( ugcHandle );
g_mapCustomTextureInfoByCloudId[idx] = pEntry;
}
// Return the entry
return pEntry;
}
// Locate an entry by hash create a new entry if one doesn't already exist
static SCustomImageCacheEntry *CustomTextureCache_FindOrAddByDigest( const char *szDigestAscii )
{
Assert( strlen(szDigestAscii) == MD5_DIGEST_LENGTH*2 );
// Brute-force linear search. This should never be called in time-critical
// situations
SCustomImageCacheEntry *pEntry = mruCustomImageEntry;
while ( pEntry )
{
// Match?
if ( !Q_stricmp(pEntry->m_szDigestAscii, szDigestAscii) )
{
// Found. Se at MRU and return it.
CustomTextureCache_SetMRU(pEntry);
return pEntry;
}
// Keep looking
pEntry = pEntry->m_pNext;
}
// Not found. Make a new entry
pEntry = CustomTextureCache_NewEntry();
V_strcpy_safe(pEntry->m_szDigestAscii, szDigestAscii);
return pEntry;
}
//-----------------------------------------------------------------------------
int GetCustomTextureGuiHandle( uint64 hCloudId )
{
// Find or create the entry
SCustomImageCacheEntry *pEntry = CustomTextureCache_FindOrAddByCloudId( hCloudId );
// Poll entry and return GUI handle if it's finally ready
return pEntry->GetGuiHandle();
}
//-----------------------------------------------------------------------------
class CCustomTextureOnItemProxy : public IMaterialProxy
{
public:
CCustomTextureOnItemProxy();
virtual ~CCustomTextureOnItemProxy();
virtual bool Init( IMaterial* pMaterial, KeyValues *pKeyValues );
virtual void OnBind( void *pC_BaseEntity );
virtual void Release();
virtual IMaterial *GetMaterial();
protected:
virtual void OnBindInternal( CEconItemView *pScriptItem );
private:
IMaterialVar *m_pBaseTextureVar;
ITexture *m_pOriginalTexture;
};
EXPOSE_INTERFACE( CCustomTextureOnItemProxy, IMaterialProxy, "CustomSteamImageOnModel" IMATERIAL_PROXY_INTERFACE_VERSION );
CCustomTextureOnItemProxy::CCustomTextureOnItemProxy()
: m_pBaseTextureVar( NULL )
, m_pOriginalTexture( NULL )
{
}
CCustomTextureOnItemProxy::~CCustomTextureOnItemProxy()
{
}
bool CCustomTextureOnItemProxy::Init( IMaterial *pMaterial, KeyValues *pKeyValues )
{
Release();
bool found = false;
m_pBaseTextureVar = pMaterial->FindVar( "$basetexture", &found );
if ( !found )
{
return false;
}
// No! Don't do this until, because the material/texture might
// not have been cached. If we call this, it causes the material
// to try to get cached, but instead of loading the texture
// synchronously, it just goes into a queue, and we get the error
// texture instead. We'll just defer it until later when we know
// for sure that everything is ready to go.
//m_pOriginalTexture = m_pBaseTextureVar->GetTextureValue();
//if ( m_pOriginalTexture )
//{
// m_pOriginalTexture->AddRef();
//}
return true;
}
void CCustomTextureOnItemProxy::OnBind( void *pC_BaseEntity )
{
if ( pC_BaseEntity )
{
CEconItemView *pScriptItem = NULL;
IClientRenderable *pRend = (IClientRenderable *)pC_BaseEntity;
C_BaseEntity *pEntity = pRend->GetIClientUnknown()->GetBaseEntity();
if ( pEntity )
{
CEconEntity *pItem = dynamic_cast< CEconEntity* >( pEntity );
if ( pItem )
{
pScriptItem = pItem->GetAttributeContainer()->GetItem();
}
}
else
{
// Proxy data can be a script created item itself, if we're in a vgui CModelPanel
pScriptItem = dynamic_cast< CEconItemView* >( pRend );
}
if ( pScriptItem )
{
OnBindInternal( pScriptItem );
}
}
}
void CCustomTextureOnItemProxy::Release()
{
if ( m_pOriginalTexture )
{
m_pOriginalTexture->Release();
m_pOriginalTexture = NULL;
}
}
IMaterial *CCustomTextureOnItemProxy::GetMaterial()
{
return m_pBaseTextureVar->GetOwningMaterial();
}
void CCustomTextureOnItemProxy::OnBindInternal( CEconItemView *pScriptItem )
{
if ( !m_pBaseTextureVar || !m_pBaseTextureVar->IsTexture() )
{
return;
}
// Snag the original texture object the first time.
// And make sure we're 100% ready to go.
if ( m_pOriginalTexture == NULL )
{
m_pOriginalTexture = m_pBaseTextureVar->GetTextureValue();
if ( m_pOriginalTexture == NULL )
{
return;
}
if ( m_pOriginalTexture->IsError() )
{
m_pOriginalTexture = NULL;
return;
}
// Success! Let's hang on to this guy
m_pOriginalTexture->AddRef();
}
ITexture *texture = m_pOriginalTexture;
// Fetch the UGC handle from the item
UGCHandle_t ugcHandle = pScriptItem->GetCustomUserTextureID();
// Are we in a preview window?
if ( pScriptItem == g_pPreviewEconItem ) // !KLUDGE!
{
Assert( g_pPreviewCustomTexture );
if ( g_pPreviewCustomTexture )
{
texture = g_pPreviewCustomTexture;
// Re-fetch the bits if necessary
if ( g_pPreviewCustomTextureDirty )
{
g_pPreviewCustomTexture->Download();
Assert( !g_pPreviewCustomTextureDirty );
}
}
}
else if (ugcHandle != 0)
{
SCustomImageCacheEntry *pEntry = CustomTextureCache_FindOrAddByCloudId(ugcHandle);
pEntry->Poll();
texture = pEntry->m_pTexture; // might be NULL if texture isn't ready yet
}
if ( texture )
{
m_pBaseTextureVar->SetTextureValue( texture );
}
if ( ToolsEnabled() )
{
ToolFramework_RecordMaterialParams( GetMaterial() );
}
}
//-----------------------------------------------------------------------------
// The custom texture cache needs to init/shutdown and get some frame ticking
//-----------------------------------------------------------------------------
class CCustomTextureToolCache : public CBaseGameSystemPerFrame
{
public:
CCustomTextureToolCache() {}
virtual ~CCustomTextureToolCache() {}
//
// CAutoGameSystemPerFrame overrides
//
virtual char const *Name()
{
return "CCustomTextureToolCache";
}
virtual bool Init()
{
return true;
}
virtual void Shutdown()
{
// Destroy all the cache entries
SCustomImageCacheEntry *pEntry = mruCustomImageEntry;
mruCustomImageEntry = NULL;
while ( pEntry != NULL )
{
SCustomImageCacheEntry *pNext = pEntry->m_pNext;
delete pEntry;
pEntry = pNext;
}
}
// At level shutdown, release all of our GPU resources.
// We'll still hang on to the bitmap data, since it isn't
// that large and is in regular virtual memory which is easily
// swapped out if stale. But the video RAM we want to be more
// agressive at cleaning out.
virtual void LevelShutdownPreEntity()
{
// Destroy all the cache entries
for ( SCustomImageCacheEntry *pEntry = mruCustomImageEntry ; pEntry ; pEntry = pEntry->m_pNext )
{
pEntry->ReleaseResources();
}
}
// CAutoGameSystemPerFrame defines different stuff depending on which DLL we're building
#ifdef CLIENT_DLL
// Do our frame-time processing after rendering
virtual void PostRender()
{
// !FIXME! Here's where we should scan the list and eject
// entries that haven't been used recently to limit the
// hardware resources we're using.
}
#else
// This file shouldn't be compiled outside of client.dll. Right?
#error "Say what?"
#endif
};
static CCustomTextureToolCache s_CustomTextureToolCache;
IGameSystem *CustomTextureToolCacheGameSystem()
{
return &s_CustomTextureToolCache;
}
CApplyCustomTextureJob::CApplyCustomTextureJob( itemid_t nToolItemID, itemid_t nSubjectItemID, const void *pPNGData, int nPNGDataBytes )
: GCSDK::CGCClientJob( GCClientSystem()->GetGCClient() )
, m_nToolItemID( nToolItemID )
, m_nSubjectItemID( nSubjectItemID )
, m_hCloudID( 0 )
{
m_chRemoteStorageName[0] = '\0';
m_bufPNGData.Put( pPNGData, nPNGDataBytes );
}
bool CApplyCustomTextureJob::BYieldingRunGCJob()
{
YieldingRunJob();
CleanUp();
return true;
}
void CApplyCustomTextureJob::CleanUp()
{
// If we had a cloud file, delete it from the logical
// cloud filespace. We are using the cloud system really just
// to get the file into the UGC system and get a handle to it.
// But once it's up there, it really isn't this user. They will
// fetch it by UGC handle just like any other user. They paid
// for this action, and we don't want it taking up any of their
// quota.
ISteamRemoteStorage *pRemoteStorage = GetISteamRemoteStorage();
if ( pRemoteStorage && m_chRemoteStorageName[0] != '\0' )
{
pRemoteStorage->FileDelete( m_chRemoteStorageName );
}
}
EResult CApplyCustomTextureJob::YieldingRunJob()
{
EResult result = YieldingFindFileIncacheOrUploadFileToCDN();
if ( result != k_EResultOK )
{
return result;
}
Assert( m_hCloudID != 0 );
result = YieldingApplyTool();
if ( result != k_EResultOK )
{
return result;
}
// OK!
return k_EResultOK;
}
EResult CApplyCustomTextureJob::YieldingFindFileIncacheOrUploadFileToCDN()
{
int nFileSize = m_bufPNGData.TellPut();
Assert( nFileSize <= k_nMaxCustomImageFileSize ); // what the heck is out image converter doing?!
// Generate the hash
char szDigestAscii[ MD5_DIGEST_LENGTH*2 + 4];
CalcMD5Ascii( szDigestAscii, m_bufPNGData.Base(), nFileSize );
// Find or create an existing cache entry
SCustomImageCacheEntry *pSelectedCacheEntry = CustomTextureCache_FindOrAddByDigest( szDigestAscii );
KeyValuesAD pkvMruFile( "StampedItems" );
{
// Load up list of images recently used and uploaded
CUtlBuffer listFileData;
listFileData.SetBufferType( true, true );
if ( BReadSteamRemoteFileToBuffer( listFileData, k_szCustomTextureRecentListFilename ) )
{
if ( !pkvMruFile->LoadFromBuffer( k_szCustomTextureRecentListFilename, listFileData ) )
{
pkvMruFile->Clear();
}
}
}
KeyValues *pkvMruUploadedImages = pkvMruFile->FindKey( "Uploaded", true );
// !FIXME! Check for duplicates!
// Make sure we are ready
Assert( pSelectedCacheEntry != NULL );
Assert( pSelectedCacheEntry->m_hCloudID == 0 );
Assert( strlen(pSelectedCacheEntry->m_szDigestAscii) == MD5_DIGEST_LENGTH*2 );
Assert( m_bufPNGData.TellPut() > 0 );
// Generate filename in the cloud file space. Each user has their own
// namespace, and Phil requested that we keep the filenames simple
// and easily optimizeable by string table. (I.e. don't use the
// hash or something else)
//
// We *could* just always use the same filename, and each file would
// be its own "version." But that doesn't seem to be the proper
// spirit of the cloud system. So I'll just use a simple integer name,
// based on how many images they have uploaded. It isn't critical what
// this logical filename is, because once the GC gets a message to tag the file,
// that UGC ID should always refer to that version of the file and can never
// be changed or deleted, even if we reuse the filename.
int iFileIndex = 1;
KeyValues *pKey;
for ( pKey = pkvMruUploadedImages->GetFirstTrueSubKey() ; pKey ; pKey = pKey->GetNextTrueSubKey() )
{
int index = atoi(pKey->GetName());
iFileIndex = MAX( iFileIndex, index+1 );
}
Q_snprintf( m_chRemoteStorageName, sizeof( m_chRemoteStorageName ), "my_custom_images/%d.png", iFileIndex );
// Write the local copy of the file
Msg( "Saving %s to cloud....\n", m_chRemoteStorageName );
ISteamRemoteStorage *pRemoteStorage = GetISteamRemoteStorage();
if ( !pRemoteStorage || !pRemoteStorage->FileWrite( m_chRemoteStorageName, m_bufPNGData.Base(), m_bufPNGData.TellPut() ) )
{
Warning( "Failed to save local copy of custom image %s\n", m_chRemoteStorageName);
return k_EResultFail;
}
// Share it. This initiates the upload to cloud
Msg( "Starting upload of %s to UFS....\n", m_chRemoteStorageName );
SteamAPICall_t hFileShareApiCall = pRemoteStorage->FileShare( m_chRemoteStorageName );
if ( hFileShareApiCall == k_uAPICallInvalid )
{
return k_EResultFail;
}
bool bFailed;
RemoteStorageFileShareResult_t shareResult;
while ( !steamapicontext->SteamUtils()->GetAPICallResult(hFileShareApiCall,
&shareResult, sizeof(shareResult), RemoteStorageFileShareResult_t::k_iCallback, &bFailed) )
{
BYield();
}
if ( bFailed || shareResult.m_eResult != k_EResultOK )
{
Warning( "Custom texture uploaded to cloud FAILED\n" );
return k_EResultFail;
}
Msg( "Custom texture uploaded to cloud completed OK, assigned UGC ID %08X%08X\n", (uint32)(shareResult.m_hFile >> 32), (uint32)(shareResult.m_hFile) );
// Remember the handle to the cloud file
m_hCloudID = pSelectedCacheEntry->m_hCloudID = shareResult.m_hFile;
// Update the MRU list
pKey = pkvMruUploadedImages->GetFirstTrueSubKey();
while ( pKey )
{
int index = atoi(pKey->GetName());
int mruValue = pKey->GetInt( "mru", 0 );
const char *entryDigsetAscii = pKey->GetString("md5", "");
UGCHandle_t ugcID = pKey->GetUint64( "ugcid", 0);
if ( index <= 0 || mruValue <= 0 || strlen(entryDigsetAscii) != MD5_DIGEST_LENGTH*2 || ugcID == 0 )
{
// Bah! Bogus data!
Assert(false);
continue;
}
// Is this the one they selected?
if ( ugcID == pSelectedCacheEntry->m_hCloudID )
{
// This *can* happen if the list file gets lost and they reuse an image. It means we are wasting
// some of their cloud quota, but should be rare, and it's harmless.
Assert( !Q_stricmp(entryDigsetAscii, pSelectedCacheEntry->m_szDigestAscii) );
break;
}
pKey = pKey->GetNextTrueSubKey();
}
// Found it?
int oldIndex = 0x7fffffff;
if ( pKey )
{
// Renumber them in MRU order
oldIndex = pKey->GetInt( "mru", 1 );
}
else
{
// Create a new key
pKey = pkvMruUploadedImages->CreateNewKey();
// Remember hash and cloud file location in subkeys
pKey->SetString( "md5", pSelectedCacheEntry->m_szDigestAscii );
pKey->SetUint64( "ugcid", pSelectedCacheEntry->m_hCloudID );
//pKey->SetString( "remoteStorageName", m_chSelectedRemoteStorageNameBase );
}
for ( KeyValues *p = pkvMruUploadedImages->GetFirstTrueSubKey() ; p ; p = p->GetNextTrueSubKey() )
{
if ( p != pKey )
{
int mruValue = p->GetInt( "mru", 0 );
Assert( mruValue > 0 );
if (mruValue < oldIndex)
{
p->SetInt( "mru", mruValue+1 );
}
}
}
pKey->SetInt( "mru", 1);
// Re-save the cloud-backed MRU list file
Msg( "Saving MRU list file %s\n", k_szCustomTextureRecentListFilename );
if ( pRemoteStorage )
{
CUtlBuffer listFileData;
listFileData.SetBufferType( true, true );
pkvMruFile->RecursiveSaveToFile( listFileData, 0 );
pRemoteStorage->FileWrite( k_szCustomTextureRecentListFilename, listFileData.Base(), listFileData.TellPut() );
}
return k_EResultOK;
}
EResult CApplyCustomTextureJob::YieldingApplyTool()
{
Msg( "Sending tool request to GC.\n" );
// At this point, we need to know the cloud ID and hash of the image we are applying
Assert( m_hCloudID != 0 );
// Send the message to the GC
GCSDK::CGCMsg< MsgGCCustomizeItemTexture_t > msg( k_EMsgGCCustomizeItemTexture );
msg.Body().m_unToolItemID = m_nToolItemID;
msg.Body().m_unSubjectItemID = m_nSubjectItemID;
msg.Body().m_unImageUGCHandle = m_hCloudID;
GCSDK::CGCMsg<MsgGCStandardResponse_t> msgReply;
if ( !BYldSendMessageAndGetReply( msg, 10, &msgReply, k_EMsgGCCustomizeItemTextureResponse ) )
{
Warning( "Customize texture tool failed: Did not get reply from GC\n" );
return k_EResultTimeout;
}
// OK!
InventoryManager()->ShowItemsPickedUp( true );
return k_EResultOK;
};