diff --git a/indra/newview/lltextureinfo.cpp b/indra/newview/lltextureinfo.cpp
index 473d8ce709d4298d5302707bbbd80f013bebb87a..5f1e14e406c9d63101731d274d2f2bdef5f7d5a6 100644
--- a/indra/newview/lltextureinfo.cpp
+++ b/indra/newview/lltextureinfo.cpp
@@ -26,10 +26,19 @@
 
 #include "llviewerprecompiledheaders.h"
 
+#include "llagent.h"
+#include "llmeshrepository.h"
+#include "llsdutil.h"
 #include "lltextureinfo.h"
+#include "lltexturecache.h"
+#include "lltexturefetch.h"
 #include "lltexturestats.h"
-#include "llviewercontrol.h"
 #include "lltrace.h"
+#include "llviewercontrol.h"
+#include "llviewerregion.h"
+#include "llviewerstats.h"
+#include "llvocache.h"
+#include "llworld.h"
 
 static LLTrace::CountStatHandle<S32> sTextureDownloadsStarted("texture_downloads_started", "number of texture downloads initiated");
 static LLTrace::CountStatHandle<S32> sTextureDownloadsCompleted("texture_downloads_completed", "number of texture downloads completed");
@@ -175,6 +184,43 @@ void LLTextureInfo::setRequestCompleteTimeAndLog(const LLUUID& id, U64Microsecon
 			endTime << completeTime;
 			texture_data["end_time"] = endTime.str();
 			texture_data["averages"] = getAverages();
+
+			// Texture cache
+			LLSD texture_cache;
+			U32 cache_read = 0, cache_write = 0, res_wait = 0;
+			F64 cache_hit_rate = 0;
+			LLAppViewer::getTextureFetch()->getStateStats(&cache_read, &cache_write, &res_wait);
+			if (cache_read > 0 || cache_write > 0)
+			{
+				cache_hit_rate = cache_read / (cache_read + cache_write);
+			}
+			texture_cache["cache_read"] = LLSD::Integer(cache_read);
+			texture_cache["cache_write"] = LLSD::Integer(cache_write);
+			texture_cache["hit_rate"] = LLSD::Real(cache_hit_rate);
+			texture_cache["entries"] = LLSD::Integer(LLAppViewer::getTextureCache()->getEntries());
+			texture_cache["space_max"] = ll_sd_from_U64((U64)LLAppViewer::getTextureCache()->getMaxUsage().value()); // bytes
+			texture_cache["space_used"] = ll_sd_from_U64((U64)LLAppViewer::getTextureCache()->getUsage().value()); // bytes
+			texture_data["texture_cache"] = texture_cache;
+
+			// VO and mesh cache
+			LLSD object_cache;
+			object_cache["vo_entries_max"] = LLSD::Integer(LLVOCache::getInstance()->getCacheEntriesMax());
+			object_cache["vo_entries_curent"] = LLSD::Integer(LLVOCache::getInstance()->getCacheEntries());
+			object_cache["vo_active_entries"] = LLSD::Integer(LLWorld::getInstance()->getNumOfActiveCachedObjects());
+			U64 region_hit_count = gAgent.getRegion() != NULL ? gAgent.getRegion()->getRegionCacheHitCount() : 0;
+			U64 region_miss_count = gAgent.getRegion() != NULL ? gAgent.getRegion()->getRegionCacheMissCount() : 0;
+			F64 region_vocache_hit_rate = 0;
+			if (region_hit_count > 0 || region_miss_count > 0)
+			{
+				region_vocache_hit_rate = region_hit_count / (region_hit_count + region_miss_count);
+			}
+			object_cache["vo_region_hitcount"] = ll_sd_from_U64(region_hit_count);
+			object_cache["vo_region_misscount"] = ll_sd_from_U64(region_miss_count);
+			object_cache["vo_region_hitrate"] = LLSD::Real(region_vocache_hit_rate);
+			object_cache["mesh_reads"] = LLSD::Integer(LLMeshRepository::sCacheReads);
+			object_cache["mesh_writes"] = LLSD::Integer(LLMeshRepository::sCacheWrites);
+			texture_data["object_cache"] = object_cache;
+
 			send_texture_stats_to_sim(texture_data);
 			resetTextureStatistics();
 		}
diff --git a/indra/newview/lltexturestats.cpp b/indra/newview/lltexturestats.cpp
index 8ded148e178dfb7271c6d1d9a648fbb0895133fb..b55b4d9ca457e034c83d552175605ab32aebad02 100644
--- a/indra/newview/lltexturestats.cpp
+++ b/indra/newview/lltexturestats.cpp
@@ -30,6 +30,7 @@
 #include "llagent.h"
 #include "lltexturefetch.h" 
 #include "lltexturestats.h"
+#include "llversioninfo.h"
 #include "llviewerregion.h"
 #include "llcorehttputil.h"
 
@@ -45,6 +46,8 @@ void send_texture_stats_to_sim(const LLSD &texture_stats)
 	LLUUID agent_id = gAgent.getID();
 	texture_stats_report["agent_id"] = agent_id;
 	texture_stats_report["region_id"] = gAgent.getRegion()->getRegionID();
+	texture_stats_report["viewer_channel"] = LLVersionInfo::getChannel();
+	texture_stats_report["viewer_version"] = LLVersionInfo::getVersion();
 	texture_stats_report["stats_data"] = texture_stats;
 
 	std::string texture_cap_url = gAgent.getRegion()->getCapability("TextureStats");
diff --git a/indra/newview/llviewerregion.cpp b/indra/newview/llviewerregion.cpp
index eb37613c955e64f2aa31186e89ea3aa8df4fc38b..3fd2af87defa00346d28fc8c4f33a5e623a30f58 100644
--- a/indra/newview/llviewerregion.cpp
+++ b/indra/newview/llviewerregion.cpp
@@ -532,7 +532,9 @@ LLViewerRegion::LLViewerRegion(const U64 &handle,
 	mDead(FALSE),
 	mLastVisitedEntry(NULL),
 	mInvisibilityCheckHistory(-1),
-	mPaused(FALSE)
+	mPaused(FALSE),
+	mRegionCacheHitCount(0),
+	mRegionCacheMissCount(0)
 {
 	mWidth = region_width_meters;
 	mImpl->mOriginGlobal = from_region_handle(handle); 
@@ -2440,6 +2442,7 @@ LLVOCacheEntry* LLViewerRegion::getCacheEntry(U32 local_id, bool valid)
 
 void LLViewerRegion::addCacheMiss(U32 id, LLViewerRegion::eCacheMissType miss_type)
 {
+	mRegionCacheMissCount++;
 #if 0
 	mCacheMissList.insert(CacheMissItem(id, miss_type));
 #else
@@ -2491,6 +2494,7 @@ bool LLViewerRegion::probeCache(U32 local_id, U32 crc, U32 flags, U8 &cache_miss
 		if (entry->getCRC() == crc)
 		{
 			// Record a hit
+			mRegionCacheHitCount++;
 			entry->recordHit();
 		cache_miss_type = CACHE_MISS_TYPE_NONE;
 			entry->setUpdateFlags(flags);
diff --git a/indra/newview/llviewerregion.h b/indra/newview/llviewerregion.h
index 61ce5b454d84be9faf61a35b97f774ec184ef670..69fb9c4d4e41d04bf83cac38e7492cc07ecbe8d5 100644
--- a/indra/newview/llviewerregion.h
+++ b/indra/newview/llviewerregion.h
@@ -338,6 +338,8 @@ class LLViewerRegion: public LLCapabilityProvider // implements this interface
 	LLVOCacheEntry* getCacheEntryForOctree(U32 local_id);
 	LLVOCacheEntry* getCacheEntry(U32 local_id, bool valid = true);
 	bool probeCache(U32 local_id, U32 crc, U32 flags, U8 &cache_miss_type);
+	U64 getRegionCacheHitCount() { return mRegionCacheHitCount; }
+	U64 getRegionCacheMissCount() { return mRegionCacheMissCount; }
 	void requestCacheMisses();
 	void addCacheMissFull(const U32 local_id);
 	//update object cache if the object receives a full-update or terse update
@@ -534,7 +536,9 @@ class LLViewerRegion: public LLCapabilityProvider // implements this interface
 		typedef std::list<CacheMissItem> cache_miss_list_t;
 	};
 	CacheMissItem::cache_miss_list_t   mCacheMissList;
-	
+	U64 mRegionCacheHitCount;
+	U64 mRegionCacheMissCount;
+
 	caps_received_signal_t mCapabilitiesReceivedSignal;		
 	caps_received_signal_t mSimulatorFeaturesReceivedSignal;		
 
diff --git a/indra/newview/llvocache.h b/indra/newview/llvocache.h
index 230a81fb7632774077e727f3ee5ab4f0b3c20e1f..7d450c5231c9fb7661735ea0893bdc8b1b3628d5 100644
--- a/indra/newview/llvocache.h
+++ b/indra/newview/llvocache.h
@@ -268,6 +268,9 @@ class LLVOCache : public LLSingleton<LLVOCache>
 
 	void setReadOnly(bool read_only) {mReadOnly = read_only;} 
 
+	U32 getCacheEntries() { return mNumEntries; }
+	U32 getCacheEntriesMax() { return mCacheSize; }
+
 private:
 	void setDirNames(ELLPath location);	
 	// determine the cache filename for the region from the region handle