Skip to content
Snippets Groups Projects
Commit 7161619d authored by andreykproductengine's avatar andreykproductengine
Browse files

MAINT-7978 - Fixed allocation crash in doRead(), workers should never crash

parent ece5e6b8
No related branches found
No related tags found
No related merge requests found
...@@ -387,22 +387,35 @@ bool LLTextureCacheRemoteWorker::doRead() ...@@ -387,22 +387,35 @@ bool LLTextureCacheRemoteWorker::doRead()
} }
// Allocate read buffer // Allocate read buffer
mReadData = (U8*)ALLOCATE_MEM(LLImageBase::getPrivatePool(), mDataSize); mReadData = (U8*)ALLOCATE_MEM(LLImageBase::getPrivatePool(), mDataSize);
S32 bytes_read = LLAPRFile::readEx(local_filename,
mReadData, mOffset, mDataSize, mCache->getLocalAPRFilePool()); if (mReadData)
if (bytes_read != mDataSize)
{ {
LL_WARNS() << "Error reading file from local cache: " << local_filename S32 bytes_read = LLAPRFile::readEx( local_filename,
<< " Bytes: " << mDataSize << " Offset: " << mOffset mReadData,
mOffset,
mDataSize,
mCache->getLocalAPRFilePool());
if (bytes_read != mDataSize)
{
LL_WARNS() << "Error reading file from local cache: " << local_filename
<< " Bytes: " << mDataSize << " Offset: " << mOffset
<< " / " << mDataSize << LL_ENDL; << " / " << mDataSize << LL_ENDL;
mDataSize = 0; mDataSize = 0;
FREE_MEM(LLImageBase::getPrivatePool(), mReadData); FREE_MEM(LLImageBase::getPrivatePool(), mReadData);
mReadData = NULL; mReadData = NULL;
}
else
{
mImageSize = local_size;
mImageLocal = TRUE;
}
} }
else else
{ {
//LL_INFOS() << "texture " << mID.asString() << " found in local_assets" << LL_ENDL; LL_WARNS() << "Error allocating memory for cache: " << local_filename
mImageSize = local_size; << " of size: " << mDataSize << LL_ENDL;
mImageLocal = TRUE; mDataSize = 0;
} }
// We're done... // We're done...
done = true; done = true;
...@@ -477,44 +490,55 @@ bool LLTextureCacheRemoteWorker::doRead() ...@@ -477,44 +490,55 @@ bool LLTextureCacheRemoteWorker::doRead()
// Reserve the whole data buffer first // Reserve the whole data buffer first
U8* data = (U8*)ALLOCATE_MEM(LLImageBase::getPrivatePool(), mDataSize); U8* data = (U8*)ALLOCATE_MEM(LLImageBase::getPrivatePool(), mDataSize);
if (data)
// Set the data file pointers taking the read offset into account. 2 cases:
if (mOffset < TEXTURE_CACHE_ENTRY_SIZE)
{ {
// Offset within the header record. That means we read something from the header cache. // Set the data file pointers taking the read offset into account. 2 cases:
// Note: most common case is (mOffset = 0), so this is the "normal" code path. if (mOffset < TEXTURE_CACHE_ENTRY_SIZE)
data_offset = TEXTURE_CACHE_ENTRY_SIZE - mOffset; // i.e. TEXTURE_CACHE_ENTRY_SIZE if mOffset nul (common case) {
file_offset = 0; // Offset within the header record. That means we read something from the header cache.
file_size = mDataSize - data_offset; // Note: most common case is (mOffset = 0), so this is the "normal" code path.
// Copy the raw data we've been holding from the header cache into the new sized buffer data_offset = TEXTURE_CACHE_ENTRY_SIZE - mOffset; // i.e. TEXTURE_CACHE_ENTRY_SIZE if mOffset nul (common case)
llassert_always(mReadData); file_offset = 0;
memcpy(data, mReadData, data_offset); file_size = mDataSize - data_offset;
FREE_MEM(LLImageBase::getPrivatePool(), mReadData); // Copy the raw data we've been holding from the header cache into the new sized buffer
mReadData = NULL; llassert_always(mReadData);
} memcpy(data, mReadData, data_offset);
else FREE_MEM(LLImageBase::getPrivatePool(), mReadData);
{ mReadData = NULL;
// Offset bigger than the header record. That means we haven't read anything yet. }
data_offset = 0; else
file_offset = mOffset - TEXTURE_CACHE_ENTRY_SIZE; {
file_size = mDataSize; // Offset bigger than the header record. That means we haven't read anything yet.
// No data from header cache to copy in that case, we skipped it all data_offset = 0;
} file_offset = mOffset - TEXTURE_CACHE_ENTRY_SIZE;
file_size = mDataSize;
// No data from header cache to copy in that case, we skipped it all
}
// Now use that buffer as the object read buffer // Now use that buffer as the object read buffer
llassert_always(mReadData == NULL); llassert_always(mReadData == NULL);
mReadData = data; mReadData = data;
// Read the data at last // Read the data at last
S32 bytes_read = LLAPRFile::readEx(filename, S32 bytes_read = LLAPRFile::readEx(filename,
mReadData + data_offset, mReadData + data_offset,
file_offset, file_size, file_offset, file_size,
mCache->getLocalAPRFilePool()); mCache->getLocalAPRFilePool());
if (bytes_read != file_size) if (bytes_read != file_size)
{
LL_WARNS() << "LLTextureCacheWorker: " << mID
<< " incorrect number of bytes read from body: " << bytes_read
<< " / " << file_size << LL_ENDL;
FREE_MEM(LLImageBase::getPrivatePool(), mReadData);
mReadData = NULL;
mDataSize = -1; // failed
done = true;
}
}
else
{ {
LL_WARNS() << "LLTextureCacheWorker: " << mID LL_WARNS() << "LLTextureCacheWorker: " << mID
<< " incorrect number of bytes read from body: " << bytes_read << " failed to allocate memory for reading: " << mDataSize << LL_ENDL;
<< " / " << file_size << LL_ENDL;
FREE_MEM(LLImageBase::getPrivatePool(), mReadData); FREE_MEM(LLImageBase::getPrivatePool(), mReadData);
mReadData = NULL; mReadData = NULL;
mDataSize = -1; // failed mDataSize = -1; // failed
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment