mirror of https://github.com/apache/activemq.git
Updated fix for this issue allows for enabling non-forced metadata updates to the file channel via FileChannel#force(false) enable this by defining "org.apache.activemq.kahaDB.files.skipMetadataUpdate=true"
This commit is contained in:
parent
efb988655c
commit
ef619b6a9b
|
@ -17,11 +17,11 @@
|
||||||
package org.apache.activemq.store.kahadb.disk.journal;
|
package org.apache.activemq.store.kahadb.disk.journal;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.RandomAccessFile;
|
|
||||||
import java.util.zip.Adler32;
|
import java.util.zip.Adler32;
|
||||||
import java.util.zip.Checksum;
|
import java.util.zip.Checksum;
|
||||||
import org.apache.activemq.util.ByteSequence;
|
|
||||||
import org.apache.activemq.store.kahadb.disk.util.DataByteArrayOutputStream;
|
import org.apache.activemq.store.kahadb.disk.util.DataByteArrayOutputStream;
|
||||||
|
import org.apache.activemq.util.ByteSequence;
|
||||||
import org.apache.activemq.util.RecoverableRandomAccessFile;
|
import org.apache.activemq.util.RecoverableRandomAccessFile;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -49,6 +49,7 @@ class CallerBufferingDataFileAppender extends DataFileAppender {
|
||||||
append(write);
|
append(write);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public void append(Journal.WriteCommand write) throws IOException {
|
public void append(Journal.WriteCommand write) throws IOException {
|
||||||
super.append(write);
|
super.append(write);
|
||||||
forceToDisk |= appendToBuffer(write, buff);
|
forceToDisk |= appendToBuffer(write, buff);
|
||||||
|
@ -130,9 +131,9 @@ class CallerBufferingDataFileAppender extends DataFileAppender {
|
||||||
buff.skip(5+Journal.BATCH_CONTROL_RECORD_MAGIC.length);
|
buff.skip(5+Journal.BATCH_CONTROL_RECORD_MAGIC.length);
|
||||||
buff.writeInt(sequence.getLength()-Journal.BATCH_CONTROL_RECORD_SIZE);
|
buff.writeInt(sequence.getLength()-Journal.BATCH_CONTROL_RECORD_SIZE);
|
||||||
if( journal.isChecksum() ) {
|
if( journal.isChecksum() ) {
|
||||||
Checksum checksum = new Adler32();
|
Checksum checksum = new Adler32();
|
||||||
checksum.update(sequence.getData(), sequence.getOffset()+Journal.BATCH_CONTROL_RECORD_SIZE, sequence.getLength()-Journal.BATCH_CONTROL_RECORD_SIZE);
|
checksum.update(sequence.getData(), sequence.getOffset()+Journal.BATCH_CONTROL_RECORD_SIZE, sequence.getLength()-Journal.BATCH_CONTROL_RECORD_SIZE);
|
||||||
buff.writeLong(checksum.getValue());
|
buff.writeLong(checksum.getValue());
|
||||||
}
|
}
|
||||||
|
|
||||||
// Now do the 1 big write.
|
// Now do the 1 big write.
|
||||||
|
@ -151,11 +152,11 @@ class CallerBufferingDataFileAppender extends DataFileAppender {
|
||||||
file.write(sequence.getData(), sequence.getOffset(), sequence.getLength());
|
file.write(sequence.getData(), sequence.getOffset(), sequence.getLength());
|
||||||
ReplicationTarget replicationTarget = journal.getReplicationTarget();
|
ReplicationTarget replicationTarget = journal.getReplicationTarget();
|
||||||
if( replicationTarget!=null ) {
|
if( replicationTarget!=null ) {
|
||||||
replicationTarget.replicate(wb.writes.getHead().location, sequence, forceToDisk);
|
replicationTarget.replicate(wb.writes.getHead().location, sequence, forceToDisk);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (forceToDisk) {
|
if (forceToDisk) {
|
||||||
file.getFD().sync();
|
file.sync();
|
||||||
}
|
}
|
||||||
|
|
||||||
Journal.WriteCommand lastWrite = wb.writes.getTail();
|
Journal.WriteCommand lastWrite = wb.writes.getTail();
|
||||||
|
|
|
@ -17,7 +17,6 @@
|
||||||
package org.apache.activemq.store.kahadb.disk.journal;
|
package org.apache.activemq.store.kahadb.disk.journal;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.RandomAccessFile;
|
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
import org.apache.activemq.util.ByteSequence;
|
import org.apache.activemq.util.ByteSequence;
|
||||||
|
@ -70,7 +69,7 @@ final class DataFileAccessor {
|
||||||
throw new IOException("Invalid location: " + location);
|
throw new IOException("Invalid location: " + location);
|
||||||
}
|
}
|
||||||
|
|
||||||
Journal.WriteCommand asyncWrite = (Journal.WriteCommand)inflightWrites.get(new Journal.WriteKey(location));
|
Journal.WriteCommand asyncWrite = inflightWrites.get(new Journal.WriteKey(location));
|
||||||
if (asyncWrite != null) {
|
if (asyncWrite != null) {
|
||||||
return asyncWrite.data;
|
return asyncWrite.data;
|
||||||
}
|
}
|
||||||
|
@ -105,7 +104,7 @@ final class DataFileAccessor {
|
||||||
}
|
}
|
||||||
|
|
||||||
public void readLocationDetails(Location location) throws IOException {
|
public void readLocationDetails(Location location) throws IOException {
|
||||||
Journal.WriteCommand asyncWrite = (Journal.WriteCommand)inflightWrites.get(new Journal.WriteKey(location));
|
Journal.WriteCommand asyncWrite = inflightWrites.get(new Journal.WriteKey(location));
|
||||||
if (asyncWrite != null) {
|
if (asyncWrite != null) {
|
||||||
location.setSize(asyncWrite.location.getSize());
|
location.setSize(asyncWrite.location.getSize());
|
||||||
location.setType(asyncWrite.location.getType());
|
location.setType(asyncWrite.location.getType());
|
||||||
|
@ -155,9 +154,7 @@ final class DataFileAccessor {
|
||||||
int size = Math.min(data.getLength(), location.getSize());
|
int size = Math.min(data.getLength(), location.getSize());
|
||||||
file.write(data.getData(), data.getOffset(), size);
|
file.write(data.getData(), data.getOffset(), size);
|
||||||
if (sync) {
|
if (sync) {
|
||||||
file.getFD().sync();
|
file.sync();
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -18,16 +18,15 @@ package org.apache.activemq.store.kahadb.disk.journal;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InterruptedIOException;
|
import java.io.InterruptedIOException;
|
||||||
import java.io.RandomAccessFile;
|
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.concurrent.CountDownLatch;
|
import java.util.concurrent.CountDownLatch;
|
||||||
import java.util.concurrent.atomic.AtomicReference;
|
import java.util.concurrent.atomic.AtomicReference;
|
||||||
import java.util.zip.Adler32;
|
import java.util.zip.Adler32;
|
||||||
import java.util.zip.Checksum;
|
import java.util.zip.Checksum;
|
||||||
|
|
||||||
import org.apache.activemq.util.ByteSequence;
|
|
||||||
import org.apache.activemq.store.kahadb.disk.util.DataByteArrayOutputStream;
|
import org.apache.activemq.store.kahadb.disk.util.DataByteArrayOutputStream;
|
||||||
import org.apache.activemq.store.kahadb.disk.util.LinkedNodeList;
|
import org.apache.activemq.store.kahadb.disk.util.LinkedNodeList;
|
||||||
|
import org.apache.activemq.util.ByteSequence;
|
||||||
import org.apache.activemq.util.RecoverableRandomAccessFile;
|
import org.apache.activemq.util.RecoverableRandomAccessFile;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
@ -67,10 +66,12 @@ class DataFileAppender implements FileAppender {
|
||||||
hash = (int)(file ^ offset);
|
hash = (int)(file ^ offset);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public int hashCode() {
|
public int hashCode() {
|
||||||
return hash;
|
return hash;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public boolean equals(Object obj) {
|
public boolean equals(Object obj) {
|
||||||
if (obj instanceof WriteKey) {
|
if (obj instanceof WriteKey) {
|
||||||
WriteKey di = (WriteKey)obj;
|
WriteKey di = (WriteKey)obj;
|
||||||
|
@ -132,6 +133,7 @@ class DataFileAppender implements FileAppender {
|
||||||
this.syncOnComplete = this.journal.isEnableAsyncDiskSync();
|
this.syncOnComplete = this.journal.isEnableAsyncDiskSync();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public Location storeItem(ByteSequence data, byte type, boolean sync) throws IOException {
|
public Location storeItem(ByteSequence data, byte type, boolean sync) throws IOException {
|
||||||
|
|
||||||
// Write the packet our internal buffer.
|
// Write the packet our internal buffer.
|
||||||
|
@ -160,6 +162,7 @@ class DataFileAppender implements FileAppender {
|
||||||
return location;
|
return location;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public Location storeItem(ByteSequence data, byte type, Runnable onComplete) throws IOException {
|
public Location storeItem(ByteSequence data, byte type, Runnable onComplete) throws IOException {
|
||||||
// Write the packet our internal buffer.
|
// Write the packet our internal buffer.
|
||||||
int size = data.getLength() + Journal.RECORD_HEAD_SPACE;
|
int size = data.getLength() + Journal.RECORD_HEAD_SPACE;
|
||||||
|
@ -185,6 +188,7 @@ class DataFileAppender implements FileAppender {
|
||||||
if (!running) {
|
if (!running) {
|
||||||
running = true;
|
running = true;
|
||||||
thread = new Thread() {
|
thread = new Thread() {
|
||||||
|
@Override
|
||||||
public void run() {
|
public void run() {
|
||||||
processQueue();
|
processQueue();
|
||||||
}
|
}
|
||||||
|
@ -246,6 +250,7 @@ class DataFileAppender implements FileAppender {
|
||||||
return new WriteBatch(file, file.getLength(), write);
|
return new WriteBatch(file, file.getLength(), write);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public void close() throws IOException {
|
public void close() throws IOException {
|
||||||
synchronized (enqueueMutex) {
|
synchronized (enqueueMutex) {
|
||||||
if (!shutdown) {
|
if (!shutdown) {
|
||||||
|
@ -365,7 +370,7 @@ class DataFileAppender implements FileAppender {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (forceToDisk) {
|
if (forceToDisk) {
|
||||||
file.getFD().sync();
|
file.sync();
|
||||||
}
|
}
|
||||||
|
|
||||||
Journal.WriteCommand lastWrite = wb.writes.getTail();
|
Journal.WriteCommand lastWrite = wb.writes.getTail();
|
||||||
|
|
|
@ -42,9 +42,15 @@ import java.util.concurrent.atomic.AtomicLong;
|
||||||
import java.util.zip.Adler32;
|
import java.util.zip.Adler32;
|
||||||
import java.util.zip.Checksum;
|
import java.util.zip.Checksum;
|
||||||
|
|
||||||
import org.apache.activemq.util.*;
|
|
||||||
import org.apache.activemq.store.kahadb.disk.util.Sequence;
|
import org.apache.activemq.store.kahadb.disk.util.Sequence;
|
||||||
import org.apache.activemq.store.kahadb.disk.util.SequenceSet;
|
import org.apache.activemq.store.kahadb.disk.util.SequenceSet;
|
||||||
|
import org.apache.activemq.util.DataByteArrayOutputStream;
|
||||||
|
import org.apache.activemq.util.IOExceptionSupport;
|
||||||
|
import org.apache.activemq.util.IOHelper;
|
||||||
|
import org.apache.activemq.util.IntrospectionSupport;
|
||||||
|
import org.apache.activemq.util.LFUCache;
|
||||||
|
import org.apache.activemq.util.LRUCache;
|
||||||
|
import org.apache.activemq.util.RecoverableRandomAccessFile;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
@ -75,7 +81,7 @@ public class PageFile {
|
||||||
private static final Logger LOG = LoggerFactory.getLogger(PageFile.class);
|
private static final Logger LOG = LoggerFactory.getLogger(PageFile.class);
|
||||||
|
|
||||||
// A PageFile will use a couple of files in this directory
|
// A PageFile will use a couple of files in this directory
|
||||||
private File directory;
|
private final File directory;
|
||||||
// And the file names in that directory will be based on this name.
|
// And the file names in that directory will be based on this name.
|
||||||
private final String name;
|
private final String name;
|
||||||
|
|
||||||
|
@ -97,7 +103,7 @@ public class PageFile {
|
||||||
// The number of pages in the current recovery buffer
|
// The number of pages in the current recovery buffer
|
||||||
private int recoveryPageCount;
|
private int recoveryPageCount;
|
||||||
|
|
||||||
private AtomicBoolean loaded = new AtomicBoolean();
|
private final AtomicBoolean loaded = new AtomicBoolean();
|
||||||
// The number of pages we are aiming to write every time we
|
// The number of pages we are aiming to write every time we
|
||||||
// write to disk.
|
// write to disk.
|
||||||
int writeBatchSize = DEFAULT_WRITE_BATCH_SIZE;
|
int writeBatchSize = DEFAULT_WRITE_BATCH_SIZE;
|
||||||
|
@ -118,23 +124,23 @@ public class PageFile {
|
||||||
private boolean enabledWriteThread = false;
|
private boolean enabledWriteThread = false;
|
||||||
|
|
||||||
// These are used if enableAsyncWrites==true
|
// These are used if enableAsyncWrites==true
|
||||||
private AtomicBoolean stopWriter = new AtomicBoolean();
|
private final AtomicBoolean stopWriter = new AtomicBoolean();
|
||||||
private Thread writerThread;
|
private Thread writerThread;
|
||||||
private CountDownLatch checkpointLatch;
|
private CountDownLatch checkpointLatch;
|
||||||
|
|
||||||
// Keeps track of writes that are being written to disk.
|
// Keeps track of writes that are being written to disk.
|
||||||
private TreeMap<Long, PageWrite> writes = new TreeMap<Long, PageWrite>();
|
private final TreeMap<Long, PageWrite> writes = new TreeMap<Long, PageWrite>();
|
||||||
|
|
||||||
// Keeps track of free pages.
|
// Keeps track of free pages.
|
||||||
private final AtomicLong nextFreePageId = new AtomicLong();
|
private final AtomicLong nextFreePageId = new AtomicLong();
|
||||||
private SequenceSet freeList = new SequenceSet();
|
private SequenceSet freeList = new SequenceSet();
|
||||||
|
|
||||||
private AtomicLong nextTxid = new AtomicLong();
|
private final AtomicLong nextTxid = new AtomicLong();
|
||||||
|
|
||||||
// Persistent settings stored in the page file.
|
// Persistent settings stored in the page file.
|
||||||
private MetaData metaData;
|
private MetaData metaData;
|
||||||
|
|
||||||
private ArrayList<File> tmpFilesForRemoval = new ArrayList<File>();
|
private final ArrayList<File> tmpFilesForRemoval = new ArrayList<File>();
|
||||||
|
|
||||||
private boolean useLFRUEviction = false;
|
private boolean useLFRUEviction = false;
|
||||||
private float LFUEvictionFactor = 0.2f;
|
private float LFUEvictionFactor = 0.2f;
|
||||||
|
@ -521,6 +527,7 @@ public class PageFile {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
public String toString() {
|
public String toString() {
|
||||||
return "Page File: " + getMainPageFile();
|
return "Page File: " + getMainPageFile();
|
||||||
}
|
}
|
||||||
|
@ -610,10 +617,10 @@ public class PageFile {
|
||||||
// So we don't loose it.. write it 2 times...
|
// So we don't loose it.. write it 2 times...
|
||||||
writeFile.seek(0);
|
writeFile.seek(0);
|
||||||
writeFile.write(d);
|
writeFile.write(d);
|
||||||
writeFile.getFD().sync();
|
writeFile.sync();
|
||||||
writeFile.seek(PAGE_FILE_HEADER_SIZE / 2);
|
writeFile.seek(PAGE_FILE_HEADER_SIZE / 2);
|
||||||
writeFile.write(d);
|
writeFile.write(d);
|
||||||
writeFile.getFD().sync();
|
writeFile.sync();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void storeFreeList() throws IOException {
|
private void storeFreeList() throws IOException {
|
||||||
|
@ -880,14 +887,17 @@ public class PageFile {
|
||||||
private <T> void write(Page<T> page, byte[] data) throws IOException {
|
private <T> void write(Page<T> page, byte[] data) throws IOException {
|
||||||
final PageWrite write = new PageWrite(page, data);
|
final PageWrite write = new PageWrite(page, data);
|
||||||
Entry<Long, PageWrite> entry = new Entry<Long, PageWrite>() {
|
Entry<Long, PageWrite> entry = new Entry<Long, PageWrite>() {
|
||||||
|
@Override
|
||||||
public Long getKey() {
|
public Long getKey() {
|
||||||
return write.getPage().getPageId();
|
return write.getPage().getPageId();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public PageWrite getValue() {
|
public PageWrite getValue() {
|
||||||
return write;
|
return write;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public PageWrite setValue(PageWrite value) {
|
public PageWrite setValue(PageWrite value) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
@ -1081,9 +1091,9 @@ public class PageFile {
|
||||||
if (enableDiskSyncs) {
|
if (enableDiskSyncs) {
|
||||||
// Sync to make sure recovery buffer writes land on disk..
|
// Sync to make sure recovery buffer writes land on disk..
|
||||||
if (enableRecoveryFile) {
|
if (enableRecoveryFile) {
|
||||||
recoveryFile.getFD().sync();
|
writeFile.sync();
|
||||||
}
|
}
|
||||||
writeFile.getFD().sync();
|
writeFile.sync();
|
||||||
}
|
}
|
||||||
} finally {
|
} finally {
|
||||||
synchronized (writes) {
|
synchronized (writes) {
|
||||||
|
@ -1185,7 +1195,7 @@ public class PageFile {
|
||||||
}
|
}
|
||||||
|
|
||||||
// And sync it to disk
|
// And sync it to disk
|
||||||
writeFile.getFD().sync();
|
writeFile.sync();
|
||||||
return nextTxId;
|
return nextTxId;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -27,12 +27,15 @@ import java.util.Arrays;
|
||||||
*/
|
*/
|
||||||
public class DiskBenchmark {
|
public class DiskBenchmark {
|
||||||
|
|
||||||
|
private static final boolean SKIP_METADATA_UPDATE =
|
||||||
|
Boolean.getBoolean("org.apache.activemq.file.skipMetadataUpdate");
|
||||||
|
|
||||||
boolean verbose;
|
boolean verbose;
|
||||||
// reads and writes work with 4k of data at a time.
|
// reads and writes work with 4k of data at a time.
|
||||||
int bs=1024*4;
|
int bs = 1024 * 4;
|
||||||
// Work with 100 meg file.
|
// Work with 100 meg file.
|
||||||
long size=1024*1024*500;
|
long size = 1024 * 1024 * 500;
|
||||||
long sampleInterval = 10*1000;
|
long sampleInterval = 10 * 1000;
|
||||||
|
|
||||||
public static void main(String[] args) {
|
public static void main(String[] args) {
|
||||||
|
|
||||||
|
@ -83,63 +86,53 @@ public class DiskBenchmark {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String toString() {
|
public String toString() {
|
||||||
return
|
return "Writes: \n" + " " + writes + " writes of size " + size + " written in " + (writeDuration / 1000.0) + " seconds.\n" + " " + getWriteRate()
|
||||||
"Writes: \n" +
|
+ " writes/second.\n" + " " + getWriteSizeRate() + " megs/second.\n" + "\n" + "Sync Writes: \n" + " " + syncWrites + " writes of size "
|
||||||
" "+writes+" writes of size "+size+" written in "+(writeDuration/1000.0)+" seconds.\n"+
|
+ size + " written in " + (syncWriteDuration / 1000.0) + " seconds.\n" + " " + getSyncWriteRate() + " writes/second.\n" + " "
|
||||||
" "+getWriteRate()+" writes/second.\n"+
|
+ getSyncWriteSizeRate() + " megs/second.\n" + "\n" + "Reads: \n" + " " + reads + " reads of size " + size + " read in "
|
||||||
" "+getWriteSizeRate()+" megs/second.\n"+
|
+ (readDuration / 1000.0) + " seconds.\n" + " " + getReadRate() + " writes/second.\n" + " " + getReadSizeRate() + " megs/second.\n" + "\n"
|
||||||
"\n"+
|
+ "";
|
||||||
"Sync Writes: \n" +
|
|
||||||
" "+syncWrites+" writes of size "+size+" written in "+(syncWriteDuration/1000.0)+" seconds.\n"+
|
|
||||||
" "+getSyncWriteRate()+" writes/second.\n"+
|
|
||||||
" "+getSyncWriteSizeRate()+" megs/second.\n"+
|
|
||||||
"\n"+
|
|
||||||
"Reads: \n" +
|
|
||||||
" "+reads+" reads of size "+size+" read in "+(readDuration/1000.0)+" seconds.\n"+
|
|
||||||
" "+getReadRate()+" writes/second.\n"+
|
|
||||||
" "+getReadSizeRate()+" megs/second.\n"+
|
|
||||||
"\n"+
|
|
||||||
"";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private float getWriteSizeRate() {
|
private float getWriteSizeRate() {
|
||||||
float rc = writes;
|
float rc = writes;
|
||||||
rc *= size;
|
rc *= size;
|
||||||
rc /= (1024*1024); // put it in megs
|
rc /= (1024 * 1024); // put it in megs
|
||||||
rc /= (writeDuration/1000.0); // get rate.
|
rc /= (writeDuration / 1000.0); // get rate.
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
private float getWriteRate() {
|
private float getWriteRate() {
|
||||||
float rc = writes;
|
float rc = writes;
|
||||||
rc /= (writeDuration/1000.0); // get rate.
|
rc /= (writeDuration / 1000.0); // get rate.
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
private float getSyncWriteSizeRate() {
|
private float getSyncWriteSizeRate() {
|
||||||
float rc = syncWrites;
|
float rc = syncWrites;
|
||||||
rc *= size;
|
rc *= size;
|
||||||
rc /= (1024*1024); // put it in megs
|
rc /= (1024 * 1024); // put it in megs
|
||||||
rc /= (syncWriteDuration/1000.0); // get rate.
|
rc /= (syncWriteDuration / 1000.0); // get rate.
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
private float getSyncWriteRate() {
|
private float getSyncWriteRate() {
|
||||||
float rc = syncWrites;
|
float rc = syncWrites;
|
||||||
rc /= (syncWriteDuration/1000.0); // get rate.
|
rc /= (syncWriteDuration / 1000.0); // get rate.
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
private float getReadSizeRate() {
|
private float getReadSizeRate() {
|
||||||
float rc = reads;
|
float rc = reads;
|
||||||
rc *= size;
|
rc *= size;
|
||||||
rc /= (1024*1024); // put it in megs
|
rc /= (1024 * 1024); // put it in megs
|
||||||
rc /= (readDuration/1000.0); // get rate.
|
rc /= (readDuration / 1000.0); // get rate.
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
private float getReadRate() {
|
private float getReadRate() {
|
||||||
float rc = reads;
|
float rc = reads;
|
||||||
rc /= (readDuration/1000.0); // get rate.
|
rc /= (readDuration / 1000.0); // get rate.
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -200,14 +193,13 @@ public class DiskBenchmark {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public Report benchmark(File file) throws IOException {
|
public Report benchmark(File file) throws IOException {
|
||||||
Report rc = new Report();
|
Report rc = new Report();
|
||||||
|
|
||||||
// Initialize the block we will be writing to disk.
|
// Initialize the block we will be writing to disk.
|
||||||
byte []data = new byte[bs];
|
byte[] data = new byte[bs];
|
||||||
for (int i = 0; i < data.length; i++) {
|
for (int i = 0; i < data.length; i++) {
|
||||||
data[i] = (byte)('a'+(i%26));
|
data[i] = (byte) ('a' + (i % 26));
|
||||||
}
|
}
|
||||||
|
|
||||||
rc.size = data.length;
|
rc.size = data.length;
|
||||||
|
@ -217,47 +209,47 @@ public class DiskBenchmark {
|
||||||
// Figure out how many writes we can do in the sample interval.
|
// Figure out how many writes we can do in the sample interval.
|
||||||
long start = System.currentTimeMillis();
|
long start = System.currentTimeMillis();
|
||||||
long now = System.currentTimeMillis();
|
long now = System.currentTimeMillis();
|
||||||
int ioCount=0;
|
int ioCount = 0;
|
||||||
while( true ) {
|
while (true) {
|
||||||
if( (now-start)>sampleInterval ) {
|
if ((now - start) > sampleInterval) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
raf.seek(0);
|
raf.seek(0);
|
||||||
for( long i=0; i+data.length < size; i+=data.length) {
|
for (long i = 0; i + data.length < size; i += data.length) {
|
||||||
raf.write(data);
|
raf.write(data);
|
||||||
ioCount++;
|
ioCount++;
|
||||||
now = System.currentTimeMillis();
|
now = System.currentTimeMillis();
|
||||||
if( (now-start)>sampleInterval ) {
|
if ((now - start) > sampleInterval) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Sync to disk so that the we actually write the data to disk.. otherwise
|
// Sync to disk so that the we actually write the data to disk..
|
||||||
// OS buffering might not really do the write.
|
// otherwise OS buffering might not really do the write.
|
||||||
raf.getFD().sync();
|
raf.getChannel().force(!SKIP_METADATA_UPDATE);
|
||||||
}
|
}
|
||||||
raf.getFD().sync();
|
raf.getChannel().force(!SKIP_METADATA_UPDATE);
|
||||||
raf.close();
|
raf.close();
|
||||||
now = System.currentTimeMillis();
|
now = System.currentTimeMillis();
|
||||||
|
|
||||||
rc.size = data.length;
|
rc.size = data.length;
|
||||||
rc.writes = ioCount;
|
rc.writes = ioCount;
|
||||||
rc.writeDuration = (now-start);
|
rc.writeDuration = (now - start);
|
||||||
|
|
||||||
raf = new RandomAccessFile(file, "rw");
|
raf = new RandomAccessFile(file, "rw");
|
||||||
start = System.currentTimeMillis();
|
start = System.currentTimeMillis();
|
||||||
now = System.currentTimeMillis();
|
now = System.currentTimeMillis();
|
||||||
ioCount=0;
|
ioCount = 0;
|
||||||
while( true ) {
|
while (true) {
|
||||||
if( (now-start)>sampleInterval ) {
|
if ((now - start) > sampleInterval) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
for( long i=0; i+data.length < size; i+=data.length) {
|
for (long i = 0; i + data.length < size; i += data.length) {
|
||||||
raf.seek(i);
|
raf.seek(i);
|
||||||
raf.write(data);
|
raf.write(data);
|
||||||
raf.getFD().sync();
|
raf.getChannel().force(false);
|
||||||
ioCount++;
|
ioCount++;
|
||||||
now = System.currentTimeMillis();
|
now = System.currentTimeMillis();
|
||||||
if( (now-start)>sampleInterval ) {
|
if ((now - start) > sampleInterval) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -265,23 +257,23 @@ public class DiskBenchmark {
|
||||||
raf.close();
|
raf.close();
|
||||||
now = System.currentTimeMillis();
|
now = System.currentTimeMillis();
|
||||||
rc.syncWrites = ioCount;
|
rc.syncWrites = ioCount;
|
||||||
rc.syncWriteDuration = (now-start);
|
rc.syncWriteDuration = (now - start);
|
||||||
|
|
||||||
raf = new RandomAccessFile(file, "rw");
|
raf = new RandomAccessFile(file, "rw");
|
||||||
start = System.currentTimeMillis();
|
start = System.currentTimeMillis();
|
||||||
now = System.currentTimeMillis();
|
now = System.currentTimeMillis();
|
||||||
ioCount=0;
|
ioCount = 0;
|
||||||
while( true ) {
|
while (true) {
|
||||||
if( (now-start)>sampleInterval ) {
|
if ((now - start) > sampleInterval) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
raf.seek(0);
|
raf.seek(0);
|
||||||
for( long i=0; i+data.length < size; i+=data.length) {
|
for (long i = 0; i + data.length < size; i += data.length) {
|
||||||
raf.seek(i);
|
raf.seek(i);
|
||||||
raf.readFully(data);
|
raf.readFully(data);
|
||||||
ioCount++;
|
ioCount++;
|
||||||
now = System.currentTimeMillis();
|
now = System.currentTimeMillis();
|
||||||
if( (now-start)>sampleInterval ) {
|
if ((now - start) > sampleInterval) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -289,48 +281,39 @@ public class DiskBenchmark {
|
||||||
raf.close();
|
raf.close();
|
||||||
|
|
||||||
rc.reads = ioCount;
|
rc.reads = ioCount;
|
||||||
rc.readDuration = (now-start);
|
rc.readDuration = (now - start);
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public boolean isVerbose() {
|
public boolean isVerbose() {
|
||||||
return verbose;
|
return verbose;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public void setVerbose(boolean verbose) {
|
public void setVerbose(boolean verbose) {
|
||||||
this.verbose = verbose;
|
this.verbose = verbose;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public int getBs() {
|
public int getBs() {
|
||||||
return bs;
|
return bs;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public void setBs(int bs) {
|
public void setBs(int bs) {
|
||||||
this.bs = bs;
|
this.bs = bs;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public long getSize() {
|
public long getSize() {
|
||||||
return size;
|
return size;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public void setSize(long size) {
|
public void setSize(long size) {
|
||||||
this.size = size;
|
this.size = size;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public long getSampleInterval() {
|
public long getSampleInterval() {
|
||||||
return sampleInterval;
|
return sampleInterval;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public void setSampleInterval(long sampleInterval) {
|
public void setSampleInterval(long sampleInterval) {
|
||||||
this.sampleInterval = sampleInterval;
|
this.sampleInterval = sampleInterval;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,10 +16,18 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.activemq.util;
|
package org.apache.activemq.util;
|
||||||
|
|
||||||
import java.io.*;
|
import java.io.File;
|
||||||
|
import java.io.FileDescriptor;
|
||||||
|
import java.io.FileNotFoundException;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.RandomAccessFile;
|
||||||
|
import java.nio.channels.FileChannel;
|
||||||
|
|
||||||
public class RecoverableRandomAccessFile implements java.io.DataOutput, java.io.DataInput, java.io.Closeable {
|
public class RecoverableRandomAccessFile implements java.io.DataOutput, java.io.DataInput, java.io.Closeable {
|
||||||
|
|
||||||
|
private static final boolean SKIP_METADATA_UPDATE =
|
||||||
|
Boolean.getBoolean("org.apache.activemq.kahaDB.files.skipMetadataUpdate");
|
||||||
|
|
||||||
RandomAccessFile raf;
|
RandomAccessFile raf;
|
||||||
File file;
|
File file;
|
||||||
String mode;
|
String mode;
|
||||||
|
@ -389,6 +397,24 @@ public class RecoverableRandomAccessFile implements java.io.DataOutput, java.io.
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void sync() throws IOException {
|
||||||
|
try {
|
||||||
|
getRaf().getChannel().force(!SKIP_METADATA_UPDATE);;
|
||||||
|
} catch (IOException ioe) {
|
||||||
|
handleException();
|
||||||
|
throw ioe;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public FileChannel getChannel() throws IOException {
|
||||||
|
try {
|
||||||
|
return getRaf().getChannel();
|
||||||
|
} catch (IOException ioe) {
|
||||||
|
handleException();
|
||||||
|
throw ioe;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public int read(byte[] b, int off, int len) throws IOException {
|
public int read(byte[] b, int off, int len) throws IOException {
|
||||||
try {
|
try {
|
||||||
return getRaf().read(b, off, len);
|
return getRaf().read(b, off, len);
|
||||||
|
|
Loading…
Reference in New Issue