This document describes how memory is managed under J2ObjC translated code, and how programs behave when accessing shared memory.
One of J2ObjC's goals is to produce translated code that will integrate seamlessly into Objective-C's reference counting environment. This makes translated Java code easy to use from natively written Objective-C because there is no awkward transfer of ownership for objects being passed between Java and Objective-C environments.
Since Java uses garbage collection for memory management, Java code contains no explicit memory management of its objects. J2ObjC must therefore insert reference counting calls appropriately to ensure that objects are deallocated at the right time. We have settled on the following set of rules that we've found to be both performant and practical:
- All objects will live for at least the duration of the current autorelease pool.
- This general rule allows us to skip many retains and releases that would otherwise be necessary.
- Local variables are not retained.
- No reference counting calls on reads or writes of local variables.
- Fields are retained.
- Assignment of a field calls retain on the new value and autorelease on the old value.
- New objects are immediately autoreleased. (unless immediately assigned to a field)
A reference cycle exists when an object refers to itself either directly or indirectly through it's fields. Reference cycles can be cleaned up by Java's garbage collection, but will leak memory in Objective-C's reference counting environment. There is no automated way to prevent reference cycles from occurring; however, we do provide a Cycle Finder tool that automates detection of cycles. Here are some common ways to fix a reference cycle:
- Add a @Weak or @WeakOuter annotation to weaken one of the references.
- Add a
cleanup()method to one of the objects that sets some fields to null. Call
cleanup()before discarding the object.
- Redesign the code to avoid creating a reference cycle altogether.
In a multi-threaded program, some data can be shared by multiple threads. Java provides several tools to allow thread-safe access to shared data. This section describes J2ObjC's support for accessing shared data.
J2ObjC maps the
synchronized keyword directly to Objective-C
Java guarantees atomicity for loads and stores of all types except
JLS-17.7. With the
volatile fields (described below) J2ObjC provides no special treatment to ensure
atomic loads and stores. This implies the following:
- Since all iOS platforms are 32 or 64-bit, loads and stores of primitive types except
doubleare atomic on 32-bit devices, and all are atomic on 64-bit systems.
- Loads and stores of object types are not atomic in J2ObjC.
- Atomically updating reference counts is too costly.
- An object field can be made atomic by declaring it
volatile. (see below)
volatile fields, Java provides both atomicity and sequencially consistent ordering
(JLS-188.8.131.52), which can
be used for synchronization. J2ObjC provides the same guarantees as Java for all
J2ObjC uses the following mechanisms for
- Primitive types are mapped to c11 atomic types.
- Object fields are protected with pthread mutex locks. (cannot use spin locks due to priority inversion)
- Mutual exclusion is necessary to prevent race conditions with the reference counting.
- The implementation is very similar to Objective-C atomic properties.
Java provides a number of atomic types in the java.util.concurrent.atomic package. These are all fully supported in J2ObjC with custom implementations.
Java guarantees that a thread sees initialized values for an object's final fields without requiring any synchronization when sharing the object. (JSL-17.5) However, since J2ObjC does not support atomic access of non-volatile object types (see above), there is no safe way to share an object without synchronization. Therefore, no additional constraints are placed on the J2ObjC user by omitting the necessary memory fences for final fields.