According to Arnold et al. [[JPL 06]], 14.10.2. Final Fields and Security:
... indeed you can use
final
fields to define immutable objects. There is a common misconception that shared access to immutable objects does not require any synchronization because the state of the object never changes. This is a misconception in general because it relies on the assumption that a thread will be guaranteed to see the initialized state of the immutable object, and that need not be the case. The problem is that, while the shared object is immutable, the reference used to access the shared object is itself shared and often mutable. Consequently, a correctly synchronized program must synchronize access to that shared reference, but often programs do not do this, because programmers do not recognize the need to do it. For example, suppose one thread creates aString
object and stores a reference to it in astatic
field. A second thread then uses that reference to access the string. There is no guarantee, based on what we've discussed so far, that the values written by the first thread when constructing the string will be seen by the second thread when it accesses the string.
It is unreasonable to assume that classes that use immutable objects are themselves immutable.
Noncompliant Code Example
This noncompliant code example publishes the helper
field prematurely through the getHelper()
method. Multiple threads may initialize the field, making class Foo
mutable.
class Foo { private Helper helper; public Helper getHelper() { return helper; } public void initialize(int num) { helper = new Helper(num); } } // Immutable Helper public class Helper { private final int n; public Helper(int n) { this.n = n; } // ... }
If the Helper
class is immutable, it cannot be changed after it is initialized and will always be properly constructed. However, this does not prevent a thread from accessing the helper
field of class Foo
such that it misses observing the most recent value set by some other thread.
Compliant Solution
This compliant solution synchronizes the methods of class Foo
to ensure that no thread sees a partially initialized helper
.
class Foo { private Helper helper; public synchronized Helper getHelper() { return helper; } public synchronized void initialize(int num) { helper = new Helper(num); } }
Risk Assessment
The assumption that classes containing immutable objects are immutable is misleading and can cause serious thread-safety issues.
Rule |
Severity |
Likelihood |
Remediation Cost |
Priority |
Level |
---|---|---|---|---|---|
CON28-J |
low |
probable |
medium |
P4 |
L2 |
Automated Detection
TODO
Related Vulnerabilities
Search for vulnerabilities resulting from the violation of this rule on the CERT website.
References
[[API 06]]
[[JPL 06]], 14.10.2. Final Fields and Security:
FIO36-J. Do not create multiple buffered wrappers on an InputStream 09. Input Output (FIO) 09. Input Output (FIO)