LUCENE-431: RAMInputStream and RAMOutputStream subclass IndexInput and IndexOutput directly now

git-svn-id: https://svn.apache.org/repos/asf/lucene/java/trunk@529749 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Michael Busch 2007-04-17 20:08:41 +00:00
parent 2333dd080b
commit 3eaeba7fb7
4 changed files with 161 additions and 58 deletions

View File

@ -130,6 +130,10 @@ Optimizations
of SegmentTermPositions instead of SegmentTermDocs without additional costs. of SegmentTermPositions instead of SegmentTermDocs without additional costs.
(Michael Busch) (Michael Busch)
2. LUCENE-431: RAMInputStream and RAMOutputStream extend IndexInput and
IndexOutput directly now. This avoids further buffering and thus avoids
unneccessary array copies. (Michael Busch)
Documentation: Documentation:
1. LUCENE 791 && INFRA-1173: Infrastructure moved the Wiki to http://wiki.apache.org/lucene-java/ Updated the links in the docs and wherever else I found references. (Grant Ingersoll, Joe Schaefer) 1. LUCENE 791 && INFRA-1173: Infrastructure moved the Wiki to http://wiki.apache.org/lucene-java/ Updated the links in the docs and wherever else I found references. (Grant Ingersoll, Joe Schaefer)

View File

@ -1,5 +1,7 @@
package org.apache.lucene.store; package org.apache.lucene.store;
import java.io.IOException;
/** /**
* Licensed to the Apache Software Foundation (ASF) under one or more * Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with * contributor license agreements. See the NOTICE file distributed with
@ -23,42 +25,86 @@ package org.apache.lucene.store;
* @version $Id$ * @version $Id$
*/ */
class RAMInputStream extends BufferedIndexInput implements Cloneable { class RAMInputStream extends IndexInput implements Cloneable {
static final int BUFFER_SIZE = BufferedIndexOutput.BUFFER_SIZE;
private RAMFile file; private RAMFile file;
private long pointer = 0;
private long length; private long length;
private byte[] currentBuffer;
private int currentBufferIndex;
private int bufferPosition;
private long bufferStart;
private int bufferLength;
public RAMInputStream(RAMFile f) { public RAMInputStream(RAMFile f) {
file = f; file = f;
length = file.length; length = file.length;
}
public void readInternal(byte[] dest, int destOffset, int len) { // make sure that we switch to the
int remainder = len; // first needed buffer lazily
long start = pointer; currentBufferIndex = -1;
while (remainder != 0) { currentBuffer = null;
int bufferNumber = (int)(start/BUFFER_SIZE);
int bufferOffset = (int)(start%BUFFER_SIZE);
int bytesInBuffer = BUFFER_SIZE - bufferOffset;
int bytesToCopy = bytesInBuffer >= remainder ? remainder : bytesInBuffer;
byte[] buffer = (byte[])file.buffers.get(bufferNumber);
System.arraycopy(buffer, bufferOffset, dest, destOffset, bytesToCopy);
destOffset += bytesToCopy;
start += bytesToCopy;
remainder -= bytesToCopy;
}
pointer += len;
} }
public void close() { public void close() {
} // nothing to do here
public void seekInternal(long pos) {
pointer = pos;
} }
public long length() { public long length() {
return length; return length;
} }
public byte readByte() throws IOException {
if (bufferPosition >= bufferLength) {
currentBufferIndex++;
switchCurrentBuffer();
}
return currentBuffer[bufferPosition++];
}
public void readBytes(byte[] b, int offset, int len) throws IOException {
while (len > 0) {
if (bufferPosition >= bufferLength) {
currentBufferIndex++;
switchCurrentBuffer();
}
int remainInBuffer = bufferLength - bufferPosition;
int bytesToCopy = len < remainInBuffer ? len : remainInBuffer;
System.arraycopy(currentBuffer, bufferPosition, b, offset, bytesToCopy);
offset += bytesToCopy;
len -= bytesToCopy;
bufferPosition += bytesToCopy;
}
}
private final void switchCurrentBuffer() throws IOException {
if (currentBufferIndex >= file.buffers.size()) {
// end of file reached, no more buffers left
throw new IOException("Read past EOF");
} else {
currentBuffer = (byte[]) file.buffers.get(currentBufferIndex);
bufferPosition = 0;
bufferStart = BUFFER_SIZE * currentBufferIndex;
bufferLength = (int) (length - bufferStart);
if (bufferLength > BUFFER_SIZE) {
bufferLength = BUFFER_SIZE;
}
}
}
public long getFilePointer() {
return currentBufferIndex < 0 ? 0 : bufferStart + bufferPosition;
}
public void seek(long pos) throws IOException {
long bufferStart = currentBufferIndex * BUFFER_SIZE;
if (pos < bufferStart || pos >= bufferStart + BUFFER_SIZE) {
currentBufferIndex = (int) (pos / BUFFER_SIZE);
switchCurrentBuffer();
}
bufferPosition = (int) (pos % BUFFER_SIZE);
}
} }

View File

@ -25,9 +25,17 @@ import java.io.IOException;
* @version $Id$ * @version $Id$
*/ */
public class RAMOutputStream extends BufferedIndexOutput { public class RAMOutputStream extends IndexOutput {
static final int BUFFER_SIZE = BufferedIndexOutput.BUFFER_SIZE;
private RAMFile file; private RAMFile file;
private long pointer = 0;
private byte[] currentBuffer;
private int currentBufferIndex;
private int bufferPosition;
private long bufferStart;
private int bufferLength;
/** Construct an empty output buffer. */ /** Construct an empty output buffer. */
public RAMOutputStream() { public RAMOutputStream() {
@ -36,6 +44,11 @@ public class RAMOutputStream extends BufferedIndexOutput {
RAMOutputStream(RAMFile f) { RAMOutputStream(RAMFile f) {
file = f; file = f;
// make sure that we switch to the
// first needed buffer lazily
currentBufferIndex = -1;
currentBuffer = null;
} }
/** Copy the current contents of this buffer to the named output. */ /** Copy the current contents of this buffer to the named output. */
@ -66,41 +79,74 @@ public class RAMOutputStream extends BufferedIndexOutput {
file.setLength(0); file.setLength(0);
} }
public void flushBuffer(byte[] src, int offset, int len) throws IOException {
byte[] buffer;
int bufferPos = 0;
while (bufferPos != len) {
int bufferNumber = (int)(pointer/BUFFER_SIZE);
int bufferOffset = (int)(pointer%BUFFER_SIZE);
int bytesInBuffer = BUFFER_SIZE - bufferOffset;
int remainInSrcBuffer = len - bufferPos;
int bytesToCopy = bytesInBuffer >= remainInSrcBuffer ? remainInSrcBuffer : bytesInBuffer;
if (bufferNumber == file.buffers.size())
buffer = file.addBuffer(BUFFER_SIZE);
else
buffer = (byte[]) file.buffers.get(bufferNumber);
System.arraycopy(src, offset + bufferPos, buffer, bufferOffset, bytesToCopy);
bufferPos += bytesToCopy;
pointer += bytesToCopy;
}
if (pointer > file.length)
file.setLength(pointer);
file.setLastModified(System.currentTimeMillis());
}
public void close() throws IOException { public void close() throws IOException {
super.close(); flush();
} }
public void seek(long pos) throws IOException { public void seek(long pos) throws IOException {
super.seek(pos); // set the file length in case we seek back
pointer = pos; // and flush() has not been called yet
setFileLength();
if (pos < bufferStart || pos >= bufferStart + bufferLength) {
currentBufferIndex = (int) (pos / BUFFER_SIZE);
switchCurrentBuffer();
} }
bufferPosition = (int) (pos % BUFFER_SIZE);
}
public long length() { public long length() {
return file.length; return file.length;
} }
public void writeByte(byte b) throws IOException {
if (bufferPosition == bufferLength) {
currentBufferIndex++;
switchCurrentBuffer();
}
currentBuffer[bufferPosition++] = b;
}
public void writeBytes(byte[] b, int offset, int len) throws IOException {
while (len > 0) {
if (bufferPosition == bufferLength) {
currentBufferIndex++;
switchCurrentBuffer();
}
int remainInBuffer = currentBuffer.length - bufferPosition;
int bytesToCopy = len < remainInBuffer ? len : remainInBuffer;
System.arraycopy(b, offset, currentBuffer, bufferPosition, bytesToCopy);
offset += bytesToCopy;
len -= bytesToCopy;
bufferPosition += bytesToCopy;
}
}
private final void switchCurrentBuffer() throws IOException {
if (currentBufferIndex == file.buffers.size()) {
currentBuffer = file.addBuffer(BUFFER_SIZE);
} else {
currentBuffer = (byte[]) file.buffers.get(currentBufferIndex);
}
bufferPosition = 0;
bufferStart = BUFFER_SIZE * currentBufferIndex;
bufferLength = currentBuffer.length;
}
private void setFileLength() {
long pointer = bufferStart + bufferPosition;
if (pointer > file.length) {
file.setLength(pointer);
}
}
public void flush() throws IOException {
file.setLastModified(System.currentTimeMillis());
setFileLength();
}
public long getFilePointer() {
return currentBufferIndex < 0 ? 0 : bufferStart + bufferPosition;
}
} }

View File

@ -31,6 +31,8 @@ public class MockRAMOutputStream extends RAMOutputStream {
private MockRAMDirectory dir; private MockRAMDirectory dir;
private boolean first=true; private boolean first=true;
byte[] singleByte = new byte[1];
/** Construct an empty output buffer. */ /** Construct an empty output buffer. */
public MockRAMOutputStream(MockRAMDirectory dir, RAMFile f) { public MockRAMOutputStream(MockRAMDirectory dir, RAMFile f) {
super(f); super(f);
@ -48,7 +50,12 @@ public class MockRAMOutputStream extends RAMOutputStream {
} }
} }
public void flushBuffer(byte[] src, int offset, int len) throws IOException { public void writeByte(byte b) throws IOException {
singleByte[0] = b;
writeBytes(singleByte, 0, 1);
}
public void writeBytes(byte[] b, int offset, int len) throws IOException {
long freeSpace = dir.maxSize - dir.sizeInBytes(); long freeSpace = dir.maxSize - dir.sizeInBytes();
long realUsage = 0; long realUsage = 0;
@ -63,14 +70,14 @@ public class MockRAMOutputStream extends RAMOutputStream {
if (dir.maxSize != 0 && freeSpace <= len) { if (dir.maxSize != 0 && freeSpace <= len) {
if (freeSpace > 0 && freeSpace < len) { if (freeSpace > 0 && freeSpace < len) {
realUsage += freeSpace; realUsage += freeSpace;
super.flushBuffer(src, offset, (int) freeSpace); super.writeBytes(b, offset, (int) freeSpace);
} }
if (realUsage > dir.maxUsedSize) { if (realUsage > dir.maxUsedSize) {
dir.maxUsedSize = realUsage; dir.maxUsedSize = realUsage;
} }
throw new IOException("fake disk full at " + dir.getRecomputedActualSizeInBytes() + " bytes"); throw new IOException("fake disk full at " + dir.getRecomputedActualSizeInBytes() + " bytes");
} else { } else {
super.flushBuffer(src, offset, len); super.writeBytes(b, offset, len);
} }
if (first) { if (first) {