parent
c5af85ecad
commit
0f2fdfe23a
@ -0,0 +1,66 @@ |
|||||||
|
// Copyright (c) 2016, Facebook, Inc. All rights reserved.
|
||||||
|
// This source code is licensed under the BSD-style license found in the
|
||||||
|
// LICENSE file in the root directory of this source tree. An additional grant
|
||||||
|
// of patent rights can be found in the PATENTS file in the same directory.
|
||||||
|
|
||||||
|
package org.rocksdb; |
||||||
|
|
||||||
|
import java.util.concurrent.atomic.AtomicBoolean; |
||||||
|
|
||||||
|
/** |
||||||
|
* Offers functionality for implementations of |
||||||
|
* {@link AbstractNativeReference} which have an immutable reference to the |
||||||
|
* underlying native C++ object |
||||||
|
*/ |
||||||
|
public abstract class AbstractImmutableNativeReference |
||||||
|
extends AbstractNativeReference { |
||||||
|
|
||||||
|
/** |
||||||
|
* A flag indicating whether the current {@code AbstractNativeReference} is |
||||||
|
* responsible to free the underlying C++ object |
||||||
|
*/ |
||||||
|
private final AtomicBoolean owningHandle_; |
||||||
|
|
||||||
|
protected AbstractImmutableNativeReference(final boolean owningHandle) { |
||||||
|
this.owningHandle_ = new AtomicBoolean(owningHandle); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public boolean isOwningHandle() { |
||||||
|
return owningHandle_.get(); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Releases this {@code AbstractNativeReference} from the responsibility of |
||||||
|
* freeing the underlying native C++ object |
||||||
|
* <p> |
||||||
|
* This will prevent the object from attempting to delete the underlying |
||||||
|
* native object in its finalizer. This must be used when another object |
||||||
|
* takes over ownership of the native object or both will attempt to delete |
||||||
|
* the underlying object when garbage collected. |
||||||
|
* <p> |
||||||
|
* When {@code disOwnNativeHandle()} is called, {@code dispose()} will |
||||||
|
* subsequently take no action. As a result, incorrect use of this function |
||||||
|
* may cause a memory leak. |
||||||
|
* </p> |
||||||
|
* |
||||||
|
* @see #dispose() |
||||||
|
*/ |
||||||
|
protected final void disOwnNativeHandle() { |
||||||
|
owningHandle_.set(false); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public final void dispose() { |
||||||
|
if (owningHandle_.compareAndSet(true, false)) { |
||||||
|
disposeInternal(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* The helper function of {@link AbstractImmutableNativeReference#dispose()} |
||||||
|
* which all subclasses of {@code AbstractImmutableNativeReference} must |
||||||
|
* implement to release their underlying native C++ objects. |
||||||
|
*/ |
||||||
|
protected abstract void disposeInternal(); |
||||||
|
} |
@ -0,0 +1,57 @@ |
|||||||
|
// Copyright (c) 2016, Facebook, Inc. All rights reserved.
|
||||||
|
// This source code is licensed under the BSD-style license found in the
|
||||||
|
// LICENSE file in the root directory of this source tree. An additional grant
|
||||||
|
// of patent rights can be found in the PATENTS file in the same directory.
|
||||||
|
|
||||||
|
package org.rocksdb; |
||||||
|
|
||||||
|
/** |
||||||
|
* AbstractNativeReference is the base-class of all RocksDB classes that have |
||||||
|
* a pointer to a native C++ {@code rocksdb} object. |
||||||
|
* <p> |
||||||
|
* AbstractNativeReference has the {@link AbstractNativeReference#dispose()} |
||||||
|
* method, which frees its associated C++ object.</p> |
||||||
|
* <p> |
||||||
|
* This function should be called manually, however, if required it will be |
||||||
|
* called automatically during the regular Java GC process via |
||||||
|
* {@link AbstractNativeReference#finalize()}.</p> |
||||||
|
* <p> |
||||||
|
* Note - Java can only see the long member variable (which is the C++ pointer |
||||||
|
* value to the native object), as such it does not know the real size of the |
||||||
|
* object and therefore may assign a low GC priority for it; So it is strongly |
||||||
|
* suggested that you manually dispose of objects when you are finished with |
||||||
|
* them.</p> |
||||||
|
*/ |
||||||
|
public abstract class AbstractNativeReference { |
||||||
|
|
||||||
|
/** |
||||||
|
* Returns true if we are responsible for freeing the underlying C++ object |
||||||
|
* |
||||||
|
* @return true if we are responsible to free the C++ object |
||||||
|
* @see #dispose() |
||||||
|
*/ |
||||||
|
protected abstract boolean isOwningHandle(); |
||||||
|
|
||||||
|
/** |
||||||
|
* Frees the underlying C++ object |
||||||
|
* <p> |
||||||
|
* It is strong recommended that the developer calls this after they |
||||||
|
* have finished using the object.</p> |
||||||
|
* <p> |
||||||
|
* Note, that once an instance of {@link AbstractNativeReference} has been |
||||||
|
* disposed, calling any of its functions will lead to undefined |
||||||
|
* behavior.</p> |
||||||
|
*/ |
||||||
|
public abstract void dispose(); |
||||||
|
|
||||||
|
/** |
||||||
|
* Simply calls {@link AbstractNativeReference#dispose()} to free |
||||||
|
* any underlying C++ object reference which has not yet been manually |
||||||
|
* released. |
||||||
|
*/ |
||||||
|
@Override |
||||||
|
protected void finalize() throws Throwable { |
||||||
|
dispose(); |
||||||
|
super.finalize(); |
||||||
|
} |
||||||
|
} |
@ -1,77 +0,0 @@ |
|||||||
package org.rocksdb; |
|
||||||
|
|
||||||
import java.util.concurrent.atomic.AtomicBoolean; |
|
||||||
|
|
||||||
public abstract class NativeReference { |
|
||||||
|
|
||||||
/** |
|
||||||
* A flag indicating whether the current {@code RocksObject} is responsible to |
|
||||||
* release the c++ object stored in its {@code nativeHandle_}. |
|
||||||
*/ |
|
||||||
private final AtomicBoolean owningHandle_; |
|
||||||
|
|
||||||
protected NativeReference(final boolean owningHandle) { |
|
||||||
this.owningHandle_ = new AtomicBoolean(owningHandle); |
|
||||||
} |
|
||||||
|
|
||||||
public boolean isOwningHandle() { |
|
||||||
return owningHandle_.get(); |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Revoke ownership of the native object. |
|
||||||
* <p> |
|
||||||
* This will prevent the object from attempting to delete the underlying |
|
||||||
* native object in its finalizer. This must be used when another object |
|
||||||
* takes over ownership of the native object or both will attempt to delete |
|
||||||
* the underlying object when garbage collected. |
|
||||||
* <p> |
|
||||||
* When {@code disOwnNativeHandle()} is called, {@code dispose()} will simply set |
|
||||||
* {@code nativeHandle_} to 0 without releasing its associated C++ resource. |
|
||||||
* As a result, incorrectly use this function may cause memory leak, and this |
|
||||||
* function call will not affect the return value of {@code isInitialized()}. |
|
||||||
* </p> |
|
||||||
* @see #dispose() |
|
||||||
*/ |
|
||||||
protected final void disOwnNativeHandle() { |
|
||||||
owningHandle_.set(false); |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Release the c++ object manually pointed by the native handle. |
|
||||||
* <p> |
|
||||||
* Note that {@code dispose()} will also be called during the GC process |
|
||||||
* if it was not called before its {@code RocksObject} went out-of-scope. |
|
||||||
* However, since Java may wrongly wrongly assume those objects are |
|
||||||
* small in that they seems to only hold a long variable. As a result, |
|
||||||
* they might have low priority in the GC process. To prevent this, |
|
||||||
* it is suggested to call {@code dispose()} manually. |
|
||||||
* </p> |
|
||||||
* <p> |
|
||||||
* Note that once an instance of {@code RocksObject} has been disposed, |
|
||||||
* calling its function will lead undefined behavior. |
|
||||||
* </p> |
|
||||||
*/ |
|
||||||
public final void dispose() { |
|
||||||
if (owningHandle_.compareAndSet(true, false)) { |
|
||||||
disposeInternal(); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* The helper function of {@code dispose()} which all subclasses of |
|
||||||
* {@code RocksObject} must implement to release their associated |
|
||||||
* C++ resource. |
|
||||||
*/ |
|
||||||
protected abstract void disposeInternal(); |
|
||||||
|
|
||||||
/** |
|
||||||
* Simply calls {@code dispose()} and release its c++ resource if it has not |
|
||||||
* yet released. |
|
||||||
*/ |
|
||||||
@Override |
|
||||||
protected void finalize() throws Throwable { |
|
||||||
dispose(); |
|
||||||
super.finalize(); |
|
||||||
} |
|
||||||
} |
|
@ -1,50 +1,69 @@ |
|||||||
|
// Copyright (c) 2016, Facebook, Inc. All rights reserved.
|
||||||
|
// This source code is licensed under the BSD-style license found in the
|
||||||
|
// LICENSE file in the root directory of this source tree. An additional grant
|
||||||
|
// of patent rights can be found in the PATENTS file in the same directory.
|
||||||
|
|
||||||
package org.rocksdb; |
package org.rocksdb; |
||||||
|
|
||||||
public abstract class RocksMutableObject /*extends NativeReference*/ { |
/** |
||||||
|
* RocksMutableObject is an implementation of {@link AbstractNativeReference} |
||||||
|
* whose reference to the underlying native C++ object can change. |
||||||
|
* |
||||||
|
* <p>The use of {@code RocksMutableObject} should be kept to a minimum, as it |
||||||
|
* has synchronization overheads and introduces complexity. Instead it is |
||||||
|
* recommended to use {@link RocksObject} where possible.</p> |
||||||
|
*/ |
||||||
|
public abstract class RocksMutableObject extends AbstractNativeReference { |
||||||
|
|
||||||
private long nativeHandle_; |
/** |
||||||
private boolean owningHandle_; |
* An mutable reference to the value of the C++ pointer pointing to some |
||||||
|
* underlying native RocksDB C++ object. |
||||||
|
*/ |
||||||
|
private long nativeHandle_; |
||||||
|
private boolean owningHandle_; |
||||||
|
|
||||||
protected RocksMutableObject() { |
protected RocksMutableObject() { |
||||||
} |
} |
||||||
|
|
||||||
protected RocksMutableObject(final long nativeHandle) { |
protected RocksMutableObject(final long nativeHandle) { |
||||||
this.nativeHandle_ = nativeHandle; |
this.nativeHandle_ = nativeHandle; |
||||||
this.owningHandle_ = true; |
this.owningHandle_ = true; |
||||||
} |
} |
||||||
|
|
||||||
public synchronized void setNativeHandle(final long nativeHandle, final boolean owningNativeHandle) { |
public synchronized void setNativeHandle(final long nativeHandle, |
||||||
this.nativeHandle_ = nativeHandle; |
final boolean owningNativeHandle) { |
||||||
this.owningHandle_ = owningNativeHandle; |
this.nativeHandle_ = nativeHandle; |
||||||
} |
this.owningHandle_ = owningNativeHandle; |
||||||
|
} |
||||||
|
|
||||||
//@Override
|
@Override |
||||||
protected synchronized boolean isOwningHandle() { |
protected synchronized boolean isOwningHandle() { |
||||||
return this.owningHandle_; |
return this.owningHandle_; |
||||||
} |
} |
||||||
|
|
||||||
protected synchronized long getNativeHandle() { |
/** |
||||||
assert(this.nativeHandle_ != 0); |
* Gets the value of the C++ pointer pointing to the underlying |
||||||
return this.nativeHandle_; |
* native C++ object |
||||||
} |
* |
||||||
|
* @return the pointer value for the native object |
||||||
public synchronized final void dispose() { |
*/ |
||||||
if(isOwningHandle()) { |
protected synchronized long getNativeHandle() { |
||||||
disposeInternal(); |
assert (this.nativeHandle_ != 0); |
||||||
this.owningHandle_ = false; |
return this.nativeHandle_; |
||||||
this.nativeHandle_ = 0; |
} |
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
@Override |
||||||
protected void finalize() throws Throwable { |
public synchronized final void dispose() { |
||||||
dispose(); |
if (isOwningHandle()) { |
||||||
super.finalize(); |
disposeInternal(); |
||||||
|
this.owningHandle_ = false; |
||||||
|
this.nativeHandle_ = 0; |
||||||
} |
} |
||||||
|
} |
||||||
|
|
||||||
protected void disposeInternal() { |
protected void disposeInternal() { |
||||||
disposeInternal(nativeHandle_); |
disposeInternal(nativeHandle_); |
||||||
} |
} |
||||||
|
|
||||||
protected abstract void disposeInternal(final long handle); |
protected abstract void disposeInternal(final long handle); |
||||||
} |
} |
||||||
|
Loading…
Reference in new issue