Implement Java API for ConcurrentTaskLimiter class and compaction_thread_limiter field in ColumnFamilyOptions (#7347)

Summary:
as title

Pull Request resolved: https://github.com/facebook/rocksdb/pull/7347

Test Plan: unit tests included

Reviewed By: jay-zhuang

Differential Revision: D23592552

Pulled By: pdillinger

fbshipit-source-id: 1c3571b6f42bfd0cfd723ff49d01fbc02a1be45b
main
Tomasz Posluszny 4 years ago committed by Facebook GitHub Bot
parent 5c39d8df69
commit ec5add398c
  1. 3
      java/CMakeLists.txt
  2. 3
      java/Makefile
  3. 90
      java/rocksjni/concurrent_task_limiter.cc
  4. 29
      java/rocksjni/options.cc
  5. 18
      java/src/main/java/org/rocksdb/ColumnFamilyOptions.java
  6. 18
      java/src/main/java/org/rocksdb/ColumnFamilyOptionsInterface.java
  7. 38
      java/src/main/java/org/rocksdb/ConcurrentTaskLimiter.java
  8. 42
      java/src/main/java/org/rocksdb/ConcurrentTaskLimiterImpl.java
  9. 17
      java/src/main/java/org/rocksdb/Options.java
  10. 9
      java/src/test/java/org/rocksdb/ColumnFamilyOptionsTest.java
  11. 50
      java/src/test/java/org/rocksdb/ConcurrentTaskLimiterTest.java
  12. 9
      java/src/test/java/org/rocksdb/OptionsTest.java
  13. 1
      src.mk

@ -26,6 +26,7 @@ set(JNI_NATIVE_SOURCES
rocksjni/comparator.cc rocksjni/comparator.cc
rocksjni/comparatorjnicallback.cc rocksjni/comparatorjnicallback.cc
rocksjni/compression_options.cc rocksjni/compression_options.cc
rocksjni/concurrent_task_limiter.cc
rocksjni/config_options.cc rocksjni/config_options.cc
rocksjni/env.cc rocksjni/env.cc
rocksjni/env_options.cc rocksjni/env_options.cc
@ -153,6 +154,8 @@ set(JAVA_MAIN_CLASSES
src/main/java/org/rocksdb/InfoLogLevel.java src/main/java/org/rocksdb/InfoLogLevel.java
src/main/java/org/rocksdb/IngestExternalFileOptions.java src/main/java/org/rocksdb/IngestExternalFileOptions.java
src/main/java/org/rocksdb/LevelMetaData.java src/main/java/org/rocksdb/LevelMetaData.java
src/main/java/org/rocksdb/ConcurrentTaskLimiter.java
src/main/java/org/rocksdb/ConcurrentTaskLimiterImpl.java
src/main/java/org/rocksdb/LiveFileMetaData.java src/main/java/org/rocksdb/LiveFileMetaData.java
src/main/java/org/rocksdb/LogFile.java src/main/java/org/rocksdb/LogFile.java
src/main/java/org/rocksdb/Logger.java src/main/java/org/rocksdb/Logger.java

@ -36,6 +36,8 @@ NATIVE_JAVA_CLASSES = \
org.rocksdb.HashLinkedListMemTableConfig\ org.rocksdb.HashLinkedListMemTableConfig\
org.rocksdb.HashSkipListMemTableConfig\ org.rocksdb.HashSkipListMemTableConfig\
org.rocksdb.HdfsEnv\ org.rocksdb.HdfsEnv\
org.rocksdb.ConcurrentTaskLimiter\
org.rocksdb.ConcurrentTaskLimiterImpl\
org.rocksdb.Logger\ org.rocksdb.Logger\
org.rocksdb.LRUCache\ org.rocksdb.LRUCache\
org.rocksdb.MemoryUsageType\ org.rocksdb.MemoryUsageType\
@ -135,6 +137,7 @@ JAVA_TESTS = \
org.rocksdb.FlushTest\ org.rocksdb.FlushTest\
org.rocksdb.InfoLogLevelTest\ org.rocksdb.InfoLogLevelTest\
org.rocksdb.KeyMayExistTest\ org.rocksdb.KeyMayExistTest\
org.rocksdb.ConcurrentTaskLimiterTest\
org.rocksdb.LoggerTest\ org.rocksdb.LoggerTest\
org.rocksdb.LRUCacheTest\ org.rocksdb.LRUCacheTest\
org.rocksdb.MemoryUtilTest\ org.rocksdb.MemoryUtilTest\

@ -0,0 +1,90 @@
#include "rocksdb/concurrent_task_limiter.h"
#include <jni.h>
#include <memory>
#include <string>
#include "include/org_rocksdb_ConcurrentTaskLimiterImpl.h"
#include "rocksjni/portal.h"
/*
* Class: org_rocksdb_ConcurrentTaskLimiterImpl
* Method: newConcurrentTaskLimiterImpl0
* Signature: (Ljava/lang/String;I)J
*/
jlong Java_org_rocksdb_ConcurrentTaskLimiterImpl_newConcurrentTaskLimiterImpl0(
JNIEnv* env, jclass, jstring jname, jint limit) {
jboolean has_exception;
std::string name =
ROCKSDB_NAMESPACE::JniUtil::copyStdString(env, jname, &has_exception);
if (JNI_TRUE == has_exception) {
return 0;
}
auto* ptr = new std::shared_ptr<ROCKSDB_NAMESPACE::ConcurrentTaskLimiter>(
ROCKSDB_NAMESPACE::NewConcurrentTaskLimiter(name, limit));
return reinterpret_cast<jlong>(ptr);
}
/*
* Class: org_rocksdb_ConcurrentTaskLimiterImpl
* Method: name
* Signature: (J)Ljava/lang/String;
*/
jstring Java_org_rocksdb_ConcurrentTaskLimiterImpl_name(JNIEnv* env, jclass,
jlong handle) {
const auto& limiter = *reinterpret_cast<
std::shared_ptr<ROCKSDB_NAMESPACE::ConcurrentTaskLimiter>*>(handle);
return ROCKSDB_NAMESPACE::JniUtil::toJavaString(env, &limiter->GetName());
}
/*
* Class: org_rocksdb_ConcurrentTaskLimiterImpl
* Method: setMaxOutstandingTask
* Signature: (JI)V
*/
void Java_org_rocksdb_ConcurrentTaskLimiterImpl_setMaxOutstandingTask(
JNIEnv*, jclass, jlong handle, jint max_outstanding_task) {
const auto& limiter = *reinterpret_cast<
std::shared_ptr<ROCKSDB_NAMESPACE::ConcurrentTaskLimiter>*>(handle);
limiter->SetMaxOutstandingTask(static_cast<int32_t>(max_outstanding_task));
}
/*
* Class: org_rocksdb_ConcurrentTaskLimiterImpl
* Method: resetMaxOutstandingTask
* Signature: (J)V
*/
void Java_org_rocksdb_ConcurrentTaskLimiterImpl_resetMaxOutstandingTask(
JNIEnv*, jclass, jlong handle) {
const auto& limiter = *reinterpret_cast<
std::shared_ptr<ROCKSDB_NAMESPACE::ConcurrentTaskLimiter>*>(handle);
limiter->ResetMaxOutstandingTask();
}
/*
* Class: org_rocksdb_ConcurrentTaskLimiterImpl
* Method: outstandingTask
* Signature: (J)I
*/
jint Java_org_rocksdb_ConcurrentTaskLimiterImpl_outstandingTask(JNIEnv*, jclass,
jlong handle) {
const auto& limiter = *reinterpret_cast<
std::shared_ptr<ROCKSDB_NAMESPACE::ConcurrentTaskLimiter>*>(handle);
return static_cast<jint>(limiter->GetOutstandingTask());
}
/*
* Class: org_rocksdb_ConcurrentTaskLimiterImpl
* Method: disposeInternal
* Signature: (J)V
*/
void Java_org_rocksdb_ConcurrentTaskLimiterImpl_disposeInternal(JNIEnv*,
jobject,
jlong jhandle) {
auto* ptr = reinterpret_cast<
std::shared_ptr<ROCKSDB_NAMESPACE::ConcurrentTaskLimiter>*>(jhandle);
delete ptr; // delete std::shared_ptr
}

@ -1145,6 +1145,20 @@ void Java_org_rocksdb_Options_setSstPartitionerFactory(JNIEnv*, jobject,
options->sst_partitioner_factory = *factory; options->sst_partitioner_factory = *factory;
} }
/*
* Class: org_rocksdb_Options
* Method: setCompactionThreadLimiter
* Signature: (JJ)V
*/
void Java_org_rocksdb_Options_setCompactionThreadLimiter(
JNIEnv*, jclass, jlong jhandle, jlong jlimiter_handle) {
auto* options = reinterpret_cast<ROCKSDB_NAMESPACE::Options*>(jhandle);
auto* limiter = reinterpret_cast<
std::shared_ptr<ROCKSDB_NAMESPACE::ConcurrentTaskLimiter>*>(
jlimiter_handle);
options->compaction_thread_limiter = *limiter;
}
/* /*
* Class: org_rocksdb_Options * Class: org_rocksdb_Options
* Method: allowMmapReads * Method: allowMmapReads
@ -3649,6 +3663,21 @@ void Java_org_rocksdb_ColumnFamilyOptions_setSstPartitionerFactory(
options->sst_partitioner_factory.reset(factory); options->sst_partitioner_factory.reset(factory);
} }
/*
* Class: org_rocksdb_ColumnFamilyOptions
* Method: setCompactionThreadLimiter
* Signature: (JJ)V
*/
void Java_org_rocksdb_ColumnFamilyOptions_setCompactionThreadLimiter(
JNIEnv*, jclass, jlong jhandle, jlong jlimiter_handle) {
auto* options =
reinterpret_cast<ROCKSDB_NAMESPACE::ColumnFamilyOptions*>(jhandle);
auto* limiter = reinterpret_cast<
std::shared_ptr<ROCKSDB_NAMESPACE::ConcurrentTaskLimiter>*>(
jlimiter_handle);
options->compaction_thread_limiter = *limiter;
}
/* /*
* Method: tableFactoryName * Method: tableFactoryName
* Signature: (J)Ljava/lang/String * Signature: (J)Ljava/lang/String

@ -52,6 +52,7 @@ public class ColumnFamilyOptions extends RocksObject
this.compactionOptionsFIFO_ = other.compactionOptionsFIFO_; this.compactionOptionsFIFO_ = other.compactionOptionsFIFO_;
this.bottommostCompressionOptions_ = other.bottommostCompressionOptions_; this.bottommostCompressionOptions_ = other.bottommostCompressionOptions_;
this.compressionOptions_ = other.compressionOptions_; this.compressionOptions_ = other.compressionOptions_;
this.compactionThreadLimiter_ = other.compactionThreadLimiter_;
} }
/** /**
@ -851,6 +852,20 @@ public class ColumnFamilyOptions extends RocksObject
return this; return this;
} }
@Override
public ColumnFamilyOptions setCompactionThreadLimiter(
final ConcurrentTaskLimiter compactionThreadLimiter) {
setCompactionThreadLimiter(nativeHandle_, compactionThreadLimiter.nativeHandle_);
this.compactionThreadLimiter_ = compactionThreadLimiter;
return this;
}
@Override
public ConcurrentTaskLimiter compactionThreadLimiter() {
assert (isOwningHandle());
return this.compactionThreadLimiter_;
}
@Override @Override
public SstPartitionerFactory sstPartitionerFactory() { public SstPartitionerFactory sstPartitionerFactory() {
return sstPartitionerFactory_; return sstPartitionerFactory_;
@ -1018,6 +1033,8 @@ public class ColumnFamilyOptions extends RocksObject
final boolean forceConsistencyChecks); final boolean forceConsistencyChecks);
private native boolean forceConsistencyChecks(final long handle); private native boolean forceConsistencyChecks(final long handle);
private native void setSstPartitionerFactory(long nativeHandle_, long newFactoryHandle); private native void setSstPartitionerFactory(long nativeHandle_, long newFactoryHandle);
private static native void setCompactionThreadLimiter(
final long nativeHandle_, final long compactionThreadLimiterHandle);
// instance variables // instance variables
// NOTE: If you add new member variables, please update the copy constructor above! // NOTE: If you add new member variables, please update the copy constructor above!
@ -1032,4 +1049,5 @@ public class ColumnFamilyOptions extends RocksObject
private CompressionOptions bottommostCompressionOptions_; private CompressionOptions bottommostCompressionOptions_;
private CompressionOptions compressionOptions_; private CompressionOptions compressionOptions_;
private SstPartitionerFactory sstPartitionerFactory_; private SstPartitionerFactory sstPartitionerFactory_;
private ConcurrentTaskLimiter compactionThreadLimiter_;
} }

@ -454,6 +454,24 @@ public interface ColumnFamilyOptionsInterface<T extends ColumnFamilyOptionsInter
*/ */
SstPartitionerFactory sstPartitionerFactory(); SstPartitionerFactory sstPartitionerFactory();
/**
* Compaction concurrent thread limiter for the column family.
* If non-nullptr, use given concurrent thread limiter to control
* the max outstanding compaction tasks. Limiter can be shared with
* multiple column families across db instances.
*
* @param concurrentTaskLimiter The compaction thread limiter.
* @return the reference of the current options.
*/
T setCompactionThreadLimiter(ConcurrentTaskLimiter concurrentTaskLimiter);
/**
* Get compaction thread limiter
*
* @return Compaction thread limiter
*/
ConcurrentTaskLimiter compactionThreadLimiter();
/** /**
* Default memtable memory budget used with the following methods: * Default memtable memory budget used with the following methods:
* *

@ -0,0 +1,38 @@
package org.rocksdb;
public abstract class ConcurrentTaskLimiter extends RocksObject {
protected ConcurrentTaskLimiter(final long nativeHandle) {
super(nativeHandle);
}
/**
* Returns a name that identifies this concurrent task limiter.
*
* @return Concurrent task limiter name.
*/
public abstract String name();
/**
* Set max concurrent tasks.<br>
* limit = 0 means no new task allowed.<br>
* limit &lt; 0 means no limitation.
*
* @param maxOutstandinsTask max concurrent tasks.
* @return the reference to the current instance of ConcurrentTaskLimiter.
*/
public abstract ConcurrentTaskLimiter setMaxOutstandingTask(final int maxOutstandinsTask);
/**
* Reset to unlimited max concurrent task.
*
* @return the reference to the current instance of ConcurrentTaskLimiter.
*/
public abstract ConcurrentTaskLimiter resetMaxOutstandingTask();
/**
* Returns current outstanding task count.
*
* @return current outstanding task count.
*/
public abstract int outstandingTask();
}

@ -0,0 +1,42 @@
package org.rocksdb;
public class ConcurrentTaskLimiterImpl extends ConcurrentTaskLimiter {
public ConcurrentTaskLimiterImpl(final String name, final int maxOutstandingTask) {
super(newConcurrentTaskLimiterImpl0(name, maxOutstandingTask));
}
@Override
public String name() {
assert (isOwningHandle());
return name(nativeHandle_);
}
@Override
public ConcurrentTaskLimiter setMaxOutstandingTask(final int maxOutstandingTask) {
assert (isOwningHandle());
setMaxOutstandingTask(nativeHandle_, maxOutstandingTask);
return this;
}
@Override
public ConcurrentTaskLimiter resetMaxOutstandingTask() {
assert (isOwningHandle());
resetMaxOutstandingTask(nativeHandle_);
return this;
}
@Override
public int outstandingTask() {
assert (isOwningHandle());
return outstandingTask(nativeHandle_);
}
private static native long newConcurrentTaskLimiterImpl0(
final String name, final int maxOutstandingTask);
private static native String name(final long handle);
private static native void setMaxOutstandingTask(final long handle, final int limit);
private static native void resetMaxOutstandingTask(final long handle);
private static native int outstandingTask(final long handle);
@Override protected final native void disposeInternal(final long handle);
}

@ -91,6 +91,7 @@ public class Options extends RocksObject
this.compressionOptions_ = other.compressionOptions_; this.compressionOptions_ = other.compressionOptions_;
this.rowCache_ = other.rowCache_; this.rowCache_ = other.rowCache_;
this.writeBufferManager_ = other.writeBufferManager_; this.writeBufferManager_ = other.writeBufferManager_;
this.compactionThreadLimiter_ = other.compactionThreadLimiter_;
} }
@Override @Override
@ -1820,6 +1821,19 @@ public class Options extends RocksObject
return sstPartitionerFactory_; return sstPartitionerFactory_;
} }
@Override
public Options setCompactionThreadLimiter(final ConcurrentTaskLimiter compactionThreadLimiter) {
setCompactionThreadLimiter(nativeHandle_, compactionThreadLimiter.nativeHandle_);
this.compactionThreadLimiter_ = compactionThreadLimiter;
return this;
}
@Override
public ConcurrentTaskLimiter compactionThreadLimiter() {
assert (isOwningHandle());
return this.compactionThreadLimiter_;
}
private native static long newOptions(); private native static long newOptions();
private native static long newOptions(long dbOptHandle, private native static long newOptions(long dbOptHandle,
long cfOptHandle); long cfOptHandle);
@ -2191,6 +2205,8 @@ public class Options extends RocksObject
final boolean atomicFlush); final boolean atomicFlush);
private native boolean atomicFlush(final long handle); private native boolean atomicFlush(final long handle);
private native void setSstPartitionerFactory(long nativeHandle_, long newFactoryHandle); private native void setSstPartitionerFactory(long nativeHandle_, long newFactoryHandle);
private static native void setCompactionThreadLimiter(
final long nativeHandle_, final long newLimiterHandle);
// instance variables // instance variables
// NOTE: If you add new member variables, please update the copy constructor above! // NOTE: If you add new member variables, please update the copy constructor above!
@ -2210,4 +2226,5 @@ public class Options extends RocksObject
private WalFilter walFilter_; private WalFilter walFilter_;
private WriteBufferManager writeBufferManager_; private WriteBufferManager writeBufferManager_;
private SstPartitionerFactory sstPartitionerFactory_; private SstPartitionerFactory sstPartitionerFactory_;
private ConcurrentTaskLimiter compactionThreadLimiter_;
} }

@ -643,4 +643,13 @@ public class ColumnFamilyOptionsTest {
} }
} }
@Test
public void compactionThreadLimiter() {
try (final ColumnFamilyOptions options = new ColumnFamilyOptions();
final ConcurrentTaskLimiter compactionThreadLimiter =
new ConcurrentTaskLimiterImpl("name", 3)) {
options.setCompactionThreadLimiter(compactionThreadLimiter);
assertThat(options.compactionThreadLimiter()).isEqualTo(compactionThreadLimiter);
}
}
} }

@ -0,0 +1,50 @@
package org.rocksdb;
import static org.junit.Assert.assertEquals;
import org.junit.After;
import org.junit.Before;
import org.junit.ClassRule;
import org.junit.Test;
public class ConcurrentTaskLimiterTest {
@ClassRule
public static final RocksNativeLibraryResource ROCKS_NATIVE_LIBRARY_RESOURCE =
new RocksNativeLibraryResource();
private static final String NAME = "name";
private ConcurrentTaskLimiter concurrentTaskLimiter;
@Before
public void beforeTest() {
concurrentTaskLimiter = new ConcurrentTaskLimiterImpl(NAME, 3);
}
@Test
public void name() {
assertEquals(NAME, concurrentTaskLimiter.name());
}
@Test
public void outstandingTask() {
assertEquals(0, concurrentTaskLimiter.outstandingTask());
}
@Test
public void setMaxOutstandingTask() {
assertEquals(concurrentTaskLimiter, concurrentTaskLimiter.setMaxOutstandingTask(4));
assertEquals(0, concurrentTaskLimiter.outstandingTask());
}
@Test
public void resetMaxOutstandingTask() {
assertEquals(concurrentTaskLimiter, concurrentTaskLimiter.resetMaxOutstandingTask());
assertEquals(0, concurrentTaskLimiter.outstandingTask());
}
@After
public void afterTest() {
concurrentTaskLimiter.close();
}
}

@ -1308,4 +1308,13 @@ public class OptionsTest {
} }
} }
@Test
public void compactionThreadLimiter() {
try (final Options options = new Options();
final ConcurrentTaskLimiter compactionThreadLimiter =
new ConcurrentTaskLimiterImpl("name", 3)) {
options.setCompactionThreadLimiter(compactionThreadLimiter);
assertThat(options.compactionThreadLimiter()).isEqualTo(compactionThreadLimiter);
}
}
} }

@ -531,6 +531,7 @@ JNI_NATIVE_SOURCES = \
java/rocksjni/comparator.cc \ java/rocksjni/comparator.cc \
java/rocksjni/comparatorjnicallback.cc \ java/rocksjni/comparatorjnicallback.cc \
java/rocksjni/compression_options.cc \ java/rocksjni/compression_options.cc \
java/rocksjni/concurrent_task_limiter.cc \
java/rocksjni/config_options.cc \ java/rocksjni/config_options.cc \
java/rocksjni/env.cc \ java/rocksjni/env.cc \
java/rocksjni/env_options.cc \ java/rocksjni/env_options.cc \

Loading…
Cancel
Save