diff options
author | Rui Ueyama <ruiu@google.com> | 2014-07-25 19:46:31 +0000 |
---|---|---|
committer | Rui Ueyama <ruiu@google.com> | 2014-07-25 19:46:31 +0000 |
commit | a821d327297e3089d7d893814bb8bf5dc5b68499 (patch) | |
tree | e416238cc6c5e5d3789d4b0e0aa34980db94e528 /lld | |
parent | a57d91c2aede9cb8c5e43ab86cfaebe4e43f0161 (diff) | |
download | bcm5719-llvm-a821d327297e3089d7d893814bb8bf5dc5b68499.tar.gz bcm5719-llvm-a821d327297e3089d7d893814bb8bf5dc5b68499.zip |
Fix unsafe memory access
The following expression
m[i] = m[j]
where m is a DenseMap and i != j is not safe. m[j] returns a
reference, which would be invalidated when a rehashing occurs.
If rehashing occurs to make room for m[i], m[j] becomes
invalid, and that invalid reference would be used as the RHS
value of the expression.
llvm-svn: 213969
Diffstat (limited to 'lld')
-rw-r--r-- | lld/lib/Passes/LayoutPass.cpp | 13 |
1 files changed, 11 insertions, 2 deletions
diff --git a/lld/lib/Passes/LayoutPass.cpp b/lld/lib/Passes/LayoutPass.cpp index c0d8fa564ff..f1b2cf1cceb 100644 --- a/lld/lib/Passes/LayoutPass.cpp +++ b/lld/lib/Passes/LayoutPass.cpp @@ -347,7 +347,15 @@ void LayoutPass::buildFollowOnTable(MutableFile::DefinedAtomRange &range) { if (iter == _followOnRoots.end()) { // If the targetAtom is not a root of any chain, let's make the root of // the targetAtom to the root of the current chain. - _followOnRoots[targetAtom] = _followOnRoots[ai]; + + // The expression m[i] = m[j] where m is a DenseMap and i != j is not + // safe. m[j] returns a reference, which would be invalidated when a + // rehashing occurs. If rehashing occurs to make room for m[i], m[j] + // becomes invalid, and that invalid reference would be used as the RHS + // value of the expression. + // Copy the value to workaround. + const DefinedAtom *tmp = _followOnRoots[ai]; + _followOnRoots[targetAtom] = tmp; continue; } if (iter->second == targetAtom) { @@ -368,7 +376,8 @@ void LayoutPass::buildFollowOnTable(MutableFile::DefinedAtomRange &range) { if (currentAtomSize == 0) { const DefinedAtom *targetPrevAtom = findAtomFollowedBy(targetAtom); _followOnNexts[targetPrevAtom] = ai; - _followOnRoots[ai] = _followOnRoots[targetPrevAtom]; + const DefinedAtom *tmp = _followOnRoots[targetPrevAtom]; + _followOnRoots[ai] = tmp; continue; } if (!checkAllPrevAtomsZeroSize(targetAtom)) |