LUCENE-1484: remove synchronization on SegmentReader.document by using ThreadLocal to maintain thread-private clones of FieldsReader

git-svn-id: https://svn.apache.org/repos/asf/lucene/java/trunk@727338 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Michael McCandless 2008-12-17 10:36:30 +00:00
parent 74e097f8eb
commit d3987d9ed4
3 changed files with 71 additions and 48 deletions

View File

@ -138,6 +138,9 @@ Optimizations
2. LUCENE-1443: Performance improvement for OpenBitSetDISI.inPlaceAnd() 2. LUCENE-1443: Performance improvement for OpenBitSetDISI.inPlaceAnd()
(Paul Elschot via yonik) (Paul Elschot via yonik)
3. LUCENE-1484: Remove synchronization of IndexReader.document() by
using CloseableThreadLocal internally. (Jason Rutherglen via Mike
McCandless).
Documentation Documentation

View File

@ -38,7 +38,7 @@ import java.util.zip.Inflater;
* *
* @version $Id$ * @version $Id$
*/ */
final class FieldsReader { final class FieldsReader implements Cloneable {
private final FieldInfos fieldInfos; private final FieldInfos fieldInfos;
// The main fieldStream, used only for cloning. // The main fieldStream, used only for cloning.
@ -48,6 +48,7 @@ final class FieldsReader {
// It should not be cloned outside of a synchronized context. // It should not be cloned outside of a synchronized context.
private final IndexInput fieldsStream; private final IndexInput fieldsStream;
private final IndexInput cloneableIndexStream;
private final IndexInput indexStream; private final IndexInput indexStream;
private int numTotalDocs; private int numTotalDocs;
private int size; private int size;
@ -60,6 +61,32 @@ final class FieldsReader {
private int docStoreOffset; private int docStoreOffset;
private CloseableThreadLocal fieldsStreamTL = new CloseableThreadLocal(); private CloseableThreadLocal fieldsStreamTL = new CloseableThreadLocal();
private boolean isOriginal = false;
/** Returns a cloned FieldsReader that shares open
* IndexInputs with the original one. It is the caller's
* job not to close the original FieldsReader until all
* clones are called (eg, currently SegmentReader manages
* this logic). */
public Object clone() {
ensureOpen();
return new FieldsReader(fieldInfos, numTotalDocs, size, format, formatSize, docStoreOffset, cloneableFieldsStream, cloneableIndexStream);
}
// Used only by clone
private FieldsReader(FieldInfos fieldInfos, int numTotalDocs, int size, int format, int formatSize,
int docStoreOffset, IndexInput cloneableFieldsStream, IndexInput cloneableIndexStream) {
this.fieldInfos = fieldInfos;
this.numTotalDocs = numTotalDocs;
this.size = size;
this.format = format;
this.formatSize = formatSize;
this.docStoreOffset = docStoreOffset;
this.cloneableFieldsStream = cloneableFieldsStream;
this.cloneableIndexStream = cloneableIndexStream;
fieldsStream = (IndexInput) cloneableFieldsStream.clone();
indexStream = (IndexInput) cloneableIndexStream.clone();
}
FieldsReader(Directory d, String segment, FieldInfos fn) throws IOException { FieldsReader(Directory d, String segment, FieldInfos fn) throws IOException {
this(d, segment, fn, BufferedIndexInput.BUFFER_SIZE, -1, 0); this(d, segment, fn, BufferedIndexInput.BUFFER_SIZE, -1, 0);
@ -71,17 +98,17 @@ final class FieldsReader {
FieldsReader(Directory d, String segment, FieldInfos fn, int readBufferSize, int docStoreOffset, int size) throws IOException { FieldsReader(Directory d, String segment, FieldInfos fn, int readBufferSize, int docStoreOffset, int size) throws IOException {
boolean success = false; boolean success = false;
isOriginal = true;
try { try {
fieldInfos = fn; fieldInfos = fn;
cloneableFieldsStream = d.openInput(segment + "." + IndexFileNames.FIELDS_EXTENSION, readBufferSize); cloneableFieldsStream = d.openInput(segment + "." + IndexFileNames.FIELDS_EXTENSION, readBufferSize);
indexStream = d.openInput(segment + "." + IndexFileNames.FIELDS_INDEX_EXTENSION, readBufferSize); cloneableIndexStream = d.openInput(segment + "." + IndexFileNames.FIELDS_INDEX_EXTENSION, readBufferSize);
// First version of fdx did not include a format // First version of fdx did not include a format
// header, but, the first int will always be 0 in that // header, but, the first int will always be 0 in that
// case // case
int firstInt = indexStream.readInt(); int firstInt = cloneableIndexStream.readInt();
if (firstInt == 0) if (firstInt == 0)
format = 0; format = 0;
else else
@ -101,7 +128,7 @@ final class FieldsReader {
fieldsStream = (IndexInput) cloneableFieldsStream.clone(); fieldsStream = (IndexInput) cloneableFieldsStream.clone();
final long indexSize = indexStream.length()-formatSize; final long indexSize = cloneableIndexStream.length()-formatSize;
if (docStoreOffset != -1) { if (docStoreOffset != -1) {
// We read only a slice out of this shared fields file // We read only a slice out of this shared fields file
@ -116,6 +143,7 @@ final class FieldsReader {
this.size = (int) (indexSize >> 3); this.size = (int) (indexSize >> 3);
} }
indexStream = (IndexInput) cloneableIndexStream.clone();
numTotalDocs = (int) (indexSize >> 3); numTotalDocs = (int) (indexSize >> 3);
success = true; success = true;
} finally { } finally {
@ -150,9 +178,14 @@ final class FieldsReader {
if (fieldsStream != null) { if (fieldsStream != null) {
fieldsStream.close(); fieldsStream.close();
} }
if (isOriginal) {
if (cloneableFieldsStream != null) { if (cloneableFieldsStream != null) {
cloneableFieldsStream.close(); cloneableFieldsStream.close();
} }
if (cloneableIndexStream != null) {
cloneableIndexStream.close();
}
}
if (indexStream != null) { if (indexStream != null) {
indexStream.close(); indexStream.close();
} }

View File

@ -47,7 +47,7 @@ class SegmentReader extends DirectoryIndexReader {
private int readBufferSize; private int readBufferSize;
FieldInfos fieldInfos; FieldInfos fieldInfos;
private FieldsReader fieldsReader; private FieldsReader fieldsReaderOrig = null;
TermInfosReader tis; TermInfosReader tis;
TermVectorsReader termVectorsReaderOrig = null; TermVectorsReader termVectorsReaderOrig = null;
@ -79,6 +79,16 @@ class SegmentReader extends DirectoryIndexReader {
// in case this is a re-opened reader // in case this is a re-opened reader
private SegmentReader referencedSegmentReader = null; private SegmentReader referencedSegmentReader = null;
/**
* Sets the initial value
*/
private class FieldsReaderLocal extends CloseableThreadLocal {
protected Object initialValue() {
return (FieldsReader) fieldsReaderOrig.clone();
}
}
CloseableThreadLocal fieldsReaderLocal = new FieldsReaderLocal();
private class Norm { private class Norm {
volatile int refCount; volatile int refCount;
boolean useSingleNormStream; boolean useSingleNormStream;
@ -354,12 +364,12 @@ class SegmentReader extends DirectoryIndexReader {
fieldsSegment = segment; fieldsSegment = segment;
if (doOpenStores) { if (doOpenStores) {
fieldsReader = new FieldsReader(storeDir, fieldsSegment, fieldInfos, readBufferSize, fieldsReaderOrig = new FieldsReader(storeDir, fieldsSegment, fieldInfos, readBufferSize,
si.getDocStoreOffset(), si.docCount); si.getDocStoreOffset(), si.docCount);
// Verify two sources of "maxDoc" agree: // Verify two sources of "maxDoc" agree:
if (si.getDocStoreOffset() == -1 && fieldsReader.size() != si.docCount) { if (si.getDocStoreOffset() == -1 && fieldsReaderOrig.size() != si.docCount) {
throw new CorruptIndexException("doc counts differ for segment " + si.name + ": fieldsReader shows " + fieldsReader.size() + " but segmentInfo shows " + si.docCount); throw new CorruptIndexException("doc counts differ for segment " + si.name + ": fieldsReader shows " + fieldsReaderOrig.size() + " but segmentInfo shows " + si.docCount);
} }
} }
@ -479,31 +489,9 @@ class SegmentReader extends DirectoryIndexReader {
clone.proxStream = proxStream; clone.proxStream = proxStream;
clone.termVectorsReaderOrig = termVectorsReaderOrig; clone.termVectorsReaderOrig = termVectorsReaderOrig;
if (fieldsReaderOrig != null) {
// we have to open a new FieldsReader, because it is not thread-safe clone.fieldsReaderOrig = (FieldsReader) fieldsReaderOrig.clone();
// and can thus not be shared among multiple SegmentReaders
// TODO: Change this in case FieldsReader becomes thread-safe in the future
final String fieldsSegment;
Directory storeDir = directory();
if (si.getDocStoreOffset() != -1) {
fieldsSegment = si.getDocStoreSegment();
if (storeCFSReader != null) {
storeDir = storeCFSReader;
} }
} else {
fieldsSegment = segment;
if (cfsReader != null) {
storeDir = cfsReader;
}
}
if (fieldsReader != null) {
clone.fieldsReader = new FieldsReader(storeDir, fieldsSegment, fieldInfos, readBufferSize,
si.getDocStoreOffset(), si.docCount);
}
if (!deletionsUpToDate) { if (!deletionsUpToDate) {
// load deleted docs // load deleted docs
@ -613,13 +601,14 @@ class SegmentReader extends DirectoryIndexReader {
} }
FieldsReader getFieldsReader() { FieldsReader getFieldsReader() {
return fieldsReader; return (FieldsReader) fieldsReaderLocal.get();
} }
protected void doClose() throws IOException { protected void doClose() throws IOException {
boolean hasReferencedReader = (referencedSegmentReader != null); boolean hasReferencedReader = (referencedSegmentReader != null);
termVectorsLocal.close(); termVectorsLocal.close();
fieldsReaderLocal.close();
if (hasReferencedReader) { if (hasReferencedReader) {
referencedSegmentReader.decRefReaderNotNorms(); referencedSegmentReader.decRefReaderNotNorms();
@ -637,11 +626,6 @@ class SegmentReader extends DirectoryIndexReader {
singleNormStream = null; singleNormStream = null;
} }
// re-opened SegmentReaders have their own instance of FieldsReader
if (fieldsReader != null) {
fieldsReader.close();
}
if (!hasReferencedReader) { if (!hasReferencedReader) {
// close everything, nothing is shared anymore with other readers // close everything, nothing is shared anymore with other readers
if (tis != null) { if (tis != null) {
@ -656,6 +640,9 @@ class SegmentReader extends DirectoryIndexReader {
if (termVectorsReaderOrig != null) if (termVectorsReaderOrig != null)
termVectorsReaderOrig.close(); termVectorsReaderOrig.close();
if (fieldsReaderOrig != null)
fieldsReaderOrig.close();
if (cfsReader != null) if (cfsReader != null)
cfsReader.close(); cfsReader.close();
@ -725,12 +712,12 @@ class SegmentReader extends DirectoryIndexReader {
* @throws CorruptIndexException if the index is corrupt * @throws CorruptIndexException if the index is corrupt
* @throws IOException if there is a low-level IO error * @throws IOException if there is a low-level IO error
*/ */
public synchronized Document document(int n, FieldSelector fieldSelector) throws CorruptIndexException, IOException { public Document document(int n, FieldSelector fieldSelector) throws CorruptIndexException, IOException {
ensureOpen(); ensureOpen();
if (isDeleted(n)) if (isDeleted(n))
throw new IllegalArgumentException throw new IllegalArgumentException
("attempt to access a deleted document"); ("attempt to access a deleted document");
return fieldsReader.doc(n, fieldSelector); return getFieldsReader().doc(n, fieldSelector);
} }
public synchronized boolean isDeleted(int n) { public synchronized boolean isDeleted(int n) {