Skip to content
GitLab
Explore
Sign in
Register
Primary navigation
Search or go to…
Project
Alchemy Viewer
Manage
Activity
Members
Labels
Plan
Issues
Issue boards
Milestones
Iterations
Wiki
Code
Merge requests
Repository
Branches
Commits
Tags
Repository graph
Compare revisions
Locked files
Deploy
Releases
Package registry
Operate
Terraform modules
Help
Help
Support
GitLab documentation
Compare GitLab plans
Community forum
Contribute to GitLab
Provide feedback
Terms and privacy
Keyboard shortcuts
?
Snippets
Groups
Projects
Silent mode is enabled
All outbound communications are blocked.
Learn more
.
Show more breadcrumbs
Alchemy Viewer
Alchemy Viewer
Commits
7161619d
Commit
7161619d
authored
7 years ago
by
andreykproductengine
Browse files
Options
Downloads
Patches
Plain Diff
MAINT-7978 - Fixed allocation crash in doRead(), workers should never crash
parent
ece5e6b8
No related branches found
Branches containing commit
No related tags found
Tags containing commit
No related merge requests found
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
indra/newview/lltexturecache.cpp
+68
-44
68 additions, 44 deletions
indra/newview/lltexturecache.cpp
with
68 additions
and
44 deletions
indra/newview/lltexturecache.cpp
+
68
−
44
View file @
7161619d
...
@@ -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_
INFO
S() << "
texture " << mID.asString() << " found in local_assets" << LL_ENDL;
LL_
WARN
S
()
<<
"
Error allocating memory for cache: "
<<
local_filename
mImageSize
=
local_size
;
<<
" of size: "
<<
mDataSize
<<
LL_ENDL
;
m
ImageLocal
=
TRUE
;
m
DataSize
=
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
...
...
This diff is collapsed.
Click to expand it.
Preview
0%
Loading
Try again
or
attach a new file
.
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Save comment
Cancel
Please
register
or
sign in
to comment