aboutsummaryrefslogtreecommitdiffstats
path: root/src
diff options
context:
space:
mode:
authorJames Ahlborn <jtahlborn@yahoo.com>2012-09-24 01:42:23 +0000
committerJames Ahlborn <jtahlborn@yahoo.com>2012-09-24 01:42:23 +0000
commit74b23acf24e0b16f3d57654680930e55ddd697a0 (patch)
treeefb3e054cfb568e222aed59936730d2b07c48460 /src
parentfe78870364c29c07a8f4269b9c38ccd244665ed0 (diff)
downloadjackcess-74b23acf24e0b16f3d57654680930e55ddd697a0.tar.gz
jackcess-74b23acf24e0b16f3d57654680930e55ddd697a0.zip
Added the MemFileChannel to enable working with dbs completely in memory
git-svn-id: https://svn.code.sf.net/p/jackcess/code/jackcess/trunk@642 f203690c-595d-4dc9-a70b-905162fa7fd2
Diffstat (limited to 'src')
-rw-r--r--src/changes/changes.xml4
-rw-r--r--src/java/com/healthmarketscience/jackcess/Database.java7
-rw-r--r--src/java/com/healthmarketscience/jackcess/MemFileChannel.java477
3 files changed, 487 insertions, 1 deletions
diff --git a/src/changes/changes.xml b/src/changes/changes.xml
index afd3161..acccf8b 100644
--- a/src/changes/changes.xml
+++ b/src/changes/changes.xml
@@ -13,6 +13,10 @@
Added DatabaseBuilder in for more convenient and flexible Database
open/create.
</action>
+ <action dev="jahlborn" type="add">
+ Added the MemFileChannel (and associated support in DatabaseBuilder)
+ to enable working with Database files completely in memory.
+ </action>
</release>
<release version="1.2.8" date="2012-07-10">
<action dev="jahlborn" type="update" issue="3523179">
diff --git a/src/java/com/healthmarketscience/jackcess/Database.java b/src/java/com/healthmarketscience/jackcess/Database.java
index 1c6f347..5f3505a 100644
--- a/src/java/com/healthmarketscience/jackcess/Database.java
+++ b/src/java/com/healthmarketscience/jackcess/Database.java
@@ -279,6 +279,11 @@ public class Database
static final int SYSTEM_OBJECT_FLAGS =
SYSTEM_OBJECT_FLAG | ALT_SYSTEM_OBJECT_FLAG;
+ /** read-only channel access mode */
+ static final String RO_CHANNEL_MODE = "r";
+ /** read/write channel access mode */
+ static final String RW_CHANNEL_MODE = "rw";
+
/**
* Enum which indicates which version of Access created the database.
* @usage _general_class_
@@ -857,7 +862,7 @@ public class Database
static FileChannel openChannel(final File mdbFile, final boolean readOnly)
throws FileNotFoundException
{
- final String mode = (readOnly ? "r" : "rw");
+ final String mode = (readOnly ? RO_CHANNEL_MODE : RW_CHANNEL_MODE);
return new RandomAccessFile(mdbFile, mode).getChannel();
}
diff --git a/src/java/com/healthmarketscience/jackcess/MemFileChannel.java b/src/java/com/healthmarketscience/jackcess/MemFileChannel.java
new file mode 100644
index 0000000..221cb67
--- /dev/null
+++ b/src/java/com/healthmarketscience/jackcess/MemFileChannel.java
@@ -0,0 +1,477 @@
+/*
+Copyright (c) 2012 James Ahlborn
+
+This library is free software; you can redistribute it and/or
+modify it under the terms of the GNU Lesser General Public
+License as published by the Free Software Foundation; either
+version 2.1 of the License, or (at your option) any later version.
+
+This library is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+Lesser General Public License for more details.
+
+You should have received a copy of the GNU Lesser General Public
+License along with this library; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
+USA
+*/
+
+package com.healthmarketscience.jackcess;
+
+import java.io.File;
+import java.io.IOException;
+import java.io.InputStream;
+import java.io.RandomAccessFile;
+import java.nio.ByteBuffer;
+import java.nio.MappedByteBuffer;
+import java.nio.channels.Channels;
+import java.nio.channels.FileChannel;
+import java.nio.channels.FileLock;
+import java.nio.channels.NonWritableChannelException;
+import java.nio.channels.ReadableByteChannel;
+import java.nio.channels.WritableByteChannel;
+
+/**
+ * FileChannel implementation which maintains the entire "file" in memory.
+ * This enables working with a Database entirely in memory (for situations
+ * where disk usage may not be possible or desirable). Obviously, this
+ * requires enough jvm heap space to fit the file data. Use one of the
+ * {@code newChannel()} methods to construct an instance of this class.
+ * <p/>
+ * In order to use this class with a Database, you <i>must</i> use the {@link
+ * DatabaseBuilder} to open/create the Database instance, passing an instance
+ * of this class to the {@link DatabaseBuilder#setChannel} method.
+ * <p/>
+ * Implementation note: this class is optimized for use with {@link Database}.
+ * Therefore not all methods may be implemented and individual read/write
+ * operations are only supported within page boundaries.
+ *
+ * @author James Ahlborn
+ */
+public class MemFileChannel extends FileChannel
+{
+ private static final byte[][] EMPTY_DATA = new byte[0][];
+
+ // use largest possible Jet "page size" to ensure that reads/writes will
+ // always be within a single chunk
+ private static final int CHUNK_SIZE = 4096;
+ // this ensures that an "empty" mdb will fit in the initial chunk table
+ private static final int INIT_CHUNKS = 128;
+
+ /** current read/write position */
+ private long _position;
+ /** current amount of actual data in the file */
+ private long _size;
+ /** chunks containing the file data. the length of the chunk array is
+ always a power of 2 and the chunks are always CHUNK_SIZE. */
+ private byte[][] _data;
+
+ private MemFileChannel()
+ {
+ this(0L, 0L, EMPTY_DATA);
+ }
+
+ private MemFileChannel(long position, long size, byte[][] data) {
+ _position = position;
+ _size = size;
+ _data = data;
+ }
+
+ /**
+ * Creates a new read/write, empty MemFileChannel.
+ */
+ public static MemFileChannel newChannel() {
+ return new MemFileChannel();
+ }
+
+ /**
+ * Creates a new read/write MemFileChannel containing the contents of the
+ * given File. Note, modifications to the returned channel will <i>not</i>
+ * affect the original File source.
+ */
+ public static MemFileChannel newChannel(File file) throws IOException {
+ return newChannel(file, Database.RW_CHANNEL_MODE);
+ }
+
+ /**
+ * Creates a new MemFileChannel containing the contents of the
+ * given File with the given mode (for mode details see
+ * {@link RandomAccessFile#RandomAccessFile(File,String)}). Note,
+ * modifications to the returned channel will <i>not</i> affect the original
+ * File source.
+ */
+ public static MemFileChannel newChannel(File file, String mode)
+ throws IOException
+ {
+ FileChannel in = null;
+ try {
+ return newChannel(in = new RandomAccessFile(
+ file, Database.RO_CHANNEL_MODE).getChannel(),
+ mode);
+ } finally {
+ if(in != null) {
+ try {
+ in.close();
+ } catch(IOException e) {
+ // ignore close failure
+ }
+ }
+ }
+ }
+
+ /**
+ * Creates a new read/write MemFileChannel containing the contents of the
+ * given InputStream.
+ */
+ public static MemFileChannel newChannel(InputStream in) throws IOException {
+ return newChannel(in, Database.RW_CHANNEL_MODE);
+ }
+
+ /**
+ * Creates a new MemFileChannel containing the contents of the
+ * given InputStream with the given mode (for mode details see
+ * {@link RandomAccessFile#RandomAccessFile(File,String)}).
+ */
+ public static MemFileChannel newChannel(InputStream in, String mode)
+ throws IOException
+ {
+ return newChannel(Channels.newChannel(in), mode);
+ }
+
+ /**
+ * Creates a new read/write MemFileChannel containing the contents of the
+ * given ReadableByteChannel.
+ */
+ public static MemFileChannel newChannel(ReadableByteChannel in)
+ throws IOException
+ {
+ return newChannel(in, Database.RW_CHANNEL_MODE);
+ }
+
+ /**
+ * Creates a new MemFileChannel containing the contents of the
+ * given ReadableByteChannel with the given mode (for mode details see
+ * {@link RandomAccessFile#RandomAccessFile(File,String)}).
+ */
+ public static MemFileChannel newChannel(ReadableByteChannel in, String mode)
+ throws IOException
+ {
+ MemFileChannel channel = new MemFileChannel();
+ channel.transferFrom(in, 0L, Long.MAX_VALUE);
+ if(!mode.contains("w")) {
+ channel = new ReadOnlyChannel(channel);
+ }
+ return channel;
+ }
+
+ @Override
+ public int read(ByteBuffer dst) throws IOException {
+ int bytesRead = read(dst, _position);
+ if(bytesRead > 0) {
+ _position += bytesRead;
+ }
+ return bytesRead;
+ }
+
+ @Override
+ public int read(ByteBuffer dst, long position) throws IOException {
+ if(position >= _size) {
+ return -1;
+ }
+
+ // we assume reads will always be within a single chunk (due to how mdb
+ // files work)
+ byte[] chunk = _data[getChunkIndex(position)];
+ int chunkOffset = getChunkOffset(position);
+ int numBytes = dst.remaining();
+ dst.put(chunk, chunkOffset, numBytes);
+
+ return numBytes;
+ }
+
+ @Override
+ public int write(ByteBuffer src) throws IOException {
+ int bytesWritten = write(src, _position);
+ _position += bytesWritten;
+ return bytesWritten;
+ }
+
+ @Override
+ public int write(ByteBuffer src, long position) throws IOException {
+ int numBytes = src.remaining();
+ long newSize = position + numBytes;
+ ensureCapacity(newSize);
+
+ // we assume writes will always be within a single chunk (due to how mdb
+ // files work)
+ byte[] chunk = _data[getChunkIndex(position)];
+ int chunkOffset = getChunkOffset(position);
+ src.get(chunk, chunkOffset, numBytes);
+ if(newSize > _size) {
+ _size = newSize;
+ }
+
+ return numBytes;
+ }
+
+ @Override
+ public long position() throws IOException {
+ return _position;
+ }
+
+ @Override
+ public FileChannel position(long newPosition) throws IOException {
+ if(newPosition < 0L) {
+ throw new IllegalArgumentException("negative position");
+ }
+ _position = newPosition;
+ return this;
+ }
+
+ @Override
+ public long size() throws IOException {
+ return _size;
+ }
+
+ @Override
+ public FileChannel truncate(long newSize) throws IOException {
+ if(newSize < 0L) {
+ throw new IllegalArgumentException("negative size");
+ }
+ if(newSize < _size) {
+ // we'll optimize for memory over speed and aggressively free unused
+ // chunks
+ for(int i = getNumChunks(newSize); i < getNumChunks(_size); ++i) {
+ _data[i] = null;
+ }
+ _size = newSize;
+ }
+ _position = Math.min(newSize, _position);
+ return this;
+ }
+
+ @Override
+ public void force(boolean metaData) throws IOException {
+ // nothing to do
+ }
+
+ /**
+ * Convenience method for writing the entire contents of this channel to the
+ * given destination channel.
+ * @see #transferTo(long,long,WritableByteChannel)
+ */
+ public long transferTo(WritableByteChannel dst)
+ throws IOException
+ {
+ return transferTo(0L, _size, dst);
+ }
+
+ @Override
+ public long transferTo(long position, long count, WritableByteChannel dst)
+ throws IOException
+ {
+ if(position >= _size) {
+ return 0L;
+ }
+
+ count = Math.min(count, _size - position);
+
+ int chunkIndex = getChunkIndex(position);
+ int chunkOffset = getChunkOffset(position);
+
+ long numBytes = 0;
+ while(count > 0L) {
+
+ int chunkBytes = (int)Math.min(count, CHUNK_SIZE - chunkOffset);
+ ByteBuffer src = ByteBuffer.wrap(_data[chunkIndex], chunkOffset,
+ chunkBytes);
+
+ do {
+ int bytesWritten = dst.write(src);
+ if(bytesWritten == 0L) {
+ // dst full
+ return numBytes;
+ }
+ numBytes += bytesWritten;
+ count -= bytesWritten;
+ } while(src.hasRemaining());
+
+ ++chunkIndex;
+ chunkOffset = 0;
+ }
+
+ return numBytes;
+ }
+
+ /**
+ * Convenience method for writing the entire contents of this channel to the
+ * given destination stream.
+ * @see #transferTo(long,long,WritableByteChannel)
+ */
+ public long transferTo(OutputStream dst)
+ throws IOException
+ {
+ return transferTo(0L, _size, dst);
+ }
+
+ /**
+ * Convenience method for writing the selected portion of this channel to
+ * the given destination stream.
+ * @see #transferTo(long,long,WritableByteChannel)
+ */
+ public long transferTo(long position, long count, WritableByteChannel dst)
+ throws IOException
+ {
+ return transferTo(position, count, Channels.newChannel(dst));
+ }
+
+ @Override
+ public long transferFrom(ReadableByteChannel src,
+ long position, long count)
+ throws IOException
+ {
+ int chunkIndex = getChunkIndex(position);
+ int chunkOffset = getChunkOffset(position);
+
+ long numBytes = 0L;
+ while(count > 0L) {
+
+ ensureCapacity(position + numBytes + 1);
+
+ int chunkBytes = (int)Math.min(count, CHUNK_SIZE - chunkOffset);
+ ByteBuffer dst = ByteBuffer.wrap(_data[chunkIndex], chunkOffset,
+ chunkBytes);
+ do {
+ int bytesRead = src.read(dst);
+ if(bytesRead <= 0) {
+ // src empty
+ return numBytes;
+ }
+ numBytes += bytesRead;
+ count -= bytesRead;
+ _size = Math.max(_size, position + numBytes);
+ } while(dst.hasRemaining());
+
+ ++chunkIndex;
+ chunkOffset = 0;
+ }
+
+ return numBytes;
+ }
+
+ @Override
+ protected void implCloseChannel() throws IOException {
+ // release data
+ _data = EMPTY_DATA;
+ _size = _position = 0L;
+ }
+
+ private void ensureCapacity(long newSize)
+ {
+ if(newSize <= _size) {
+ // nothing to do
+ return;
+ }
+
+ int newNumChunks = getNumChunks(newSize);
+ int numChunks = getNumChunks(_size);
+
+ if(newNumChunks > _data.length) {
+
+ // need to extend chunk array (use powers of 2)
+ int newDataLen = Math.max(_data.length, INIT_CHUNKS);
+ while(newDataLen < newNumChunks) {
+ newDataLen <<= 1;
+ }
+
+ byte[][] newData = new byte[newDataLen][];
+
+ // copy existing chunks
+ System.arraycopy(_data, 0, newData, 0, numChunks);
+
+ _data = newData;
+ }
+
+ // allocate new chunks
+ for(int i = numChunks; i < newNumChunks; ++i) {
+ _data[i] = new byte[CHUNK_SIZE];
+ }
+ }
+
+ private static int getChunkIndex(long pos) {
+ return (int)(pos / CHUNK_SIZE);
+ }
+
+ private static int getChunkOffset(long pos) {
+ return (int)(pos % CHUNK_SIZE);
+ }
+
+ private static int getNumChunks(long size) {
+ return getChunkIndex(size + CHUNK_SIZE - 1);
+ }
+
+ @Override
+ public long write(ByteBuffer[] srcs, int offset, int length)
+ throws IOException
+ {
+ throw new UnsupportedOperationException();
+ }
+
+ @Override
+ public long read(ByteBuffer[] dsts, int offset, int length)
+ throws IOException
+ {
+ throw new UnsupportedOperationException();
+ }
+
+ @Override
+ public MappedByteBuffer map(MapMode mode, long position, long size)
+ throws IOException
+ {
+ throw new UnsupportedOperationException();
+ }
+
+ @Override
+ public FileLock lock(long position, long size, boolean shared)
+ throws IOException
+ {
+ throw new UnsupportedOperationException();
+ }
+
+ @Override
+ public FileLock tryLock(long position, long size, boolean shared)
+ throws IOException
+ {
+ throw new UnsupportedOperationException();
+ }
+
+ /**
+ * Subclass of MemFileChannel which is read-only.
+ */
+ private static final class ReadOnlyChannel extends MemFileChannel
+ {
+ private ReadOnlyChannel(MemFileChannel channel)
+ {
+ super(channel._position, channel._size, channel._data);
+ }
+
+ @Override
+ public int write(ByteBuffer src, long position) throws IOException {
+ throw new NonWritableChannelException();
+ }
+
+ @Override
+ public FileChannel truncate(long newSize) throws IOException {
+ throw new NonWritableChannelException();
+ }
+
+ @Override
+ public long transferFrom(ReadableByteChannel src,
+ long position, long count)
+ throws IOException
+ {
+ throw new NonWritableChannelException();
+ }
+ }
+}