SOLR-593: getNewestSearcher() to fix inform deadlocks from getSearcher(), fix elevation component not decrementing it's reference

git-svn-id: https://svn.apache.org/repos/asf/lucene/solr/trunk@678418 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Yonik Seeley 2008-07-21 14:01:34 +00:00
parent 91170a2535
commit cb8eb4236a
2 changed files with 55 additions and 4 deletions

View File

@ -686,12 +686,21 @@ public final class SolrCore {
// get it (and it will increment the ref count at the same time) // get it (and it will increment the ref count at the same time)
private RefCounted<SolrIndexSearcher> _searcher; private RefCounted<SolrIndexSearcher> _searcher;
// All of the open searchers. Don't access this directly.
// protected by synchronizing on searcherLock.
private final LinkedList<RefCounted<SolrIndexSearcher>> _searchers = new LinkedList<RefCounted<SolrIndexSearcher>>();
final ExecutorService searcherExecutor = Executors.newSingleThreadExecutor(); final ExecutorService searcherExecutor = Executors.newSingleThreadExecutor();
private int onDeckSearchers; // number of searchers preparing private int onDeckSearchers; // number of searchers preparing
private Object searcherLock = new Object(); // the sync object for the searcher private Object searcherLock = new Object(); // the sync object for the searcher
private final int maxWarmingSearchers; // max number of on-deck searchers allowed private final int maxWarmingSearchers; // max number of on-deck searchers allowed
/**
* Return a registered {@link RefCounted}&lt;{@link SolrIndexSearcher}&gt; with
* the reference count incremented. It <b>must</b> be decremented when no longer needed.
* This method should not be called from SolrCoreAware.inform() since it can result
* in a deadlock if useColdSearcher==false.
*/
public RefCounted<SolrIndexSearcher> getSearcher() { public RefCounted<SolrIndexSearcher> getSearcher() {
try { try {
return getSearcher(false,true,null); return getSearcher(false,true,null);
@ -701,6 +710,28 @@ public final class SolrCore {
} }
} }
/**
* Return the newest {@link RefCounted}&lt;{@link SolrIndexSearcher}&gt; with
* the reference count incremented. It <b>must</b> be decremented when no longer needed.
* If no searcher is currently open, then if openNew==true a new searcher will be opened,
* or null is returned if openNew==false.
*/
public RefCounted<SolrIndexSearcher> getNewestSearcher(boolean openNew) {
synchronized (searcherLock) {
if (_searchers.isEmpty()) {
if (!openNew) return null;
// Not currently implemented since simply calling getSearcher during inform()
// can result in a deadlock. Right now, solr always opens a searcher first
// before calling inform() anyway, so this should never happen.
throw new UnsupportedOperationException();
}
RefCounted<SolrIndexSearcher> newest = _searchers.getLast();
newest.incref();
return newest;
}
}
/** /**
* Get a {@link SolrIndexSearcher} or start the process of creating a new one. * Get a {@link SolrIndexSearcher} or start the process of creating a new one.
* <p> * <p>
@ -810,6 +841,7 @@ public final class SolrCore {
RefCounted<SolrIndexSearcher> currSearcherHolder=null; RefCounted<SolrIndexSearcher> currSearcherHolder=null;
final RefCounted<SolrIndexSearcher> newSearchHolder=newHolder(newSearcher); final RefCounted<SolrIndexSearcher> newSearchHolder=newHolder(newSearcher);
if (returnSearcher) newSearchHolder.incref(); if (returnSearcher) newSearchHolder.incref();
// a signal to decrement onDeckSearchers if something goes wrong. // a signal to decrement onDeckSearchers if something goes wrong.
@ -820,6 +852,8 @@ public final class SolrCore {
boolean alreadyRegistered = false; boolean alreadyRegistered = false;
synchronized (searcherLock) { synchronized (searcherLock) {
_searchers.add(newSearchHolder);
if (_searcher == null) { if (_searcher == null) {
// if there isn't a current searcher then we may // if there isn't a current searcher then we may
// want to register this one before warming is complete instead of waiting. // want to register this one before warming is complete instead of waiting.
@ -965,6 +999,15 @@ public final class SolrCore {
RefCounted<SolrIndexSearcher> holder = new RefCounted<SolrIndexSearcher>(newSearcher) { RefCounted<SolrIndexSearcher> holder = new RefCounted<SolrIndexSearcher>(newSearcher) {
public void close() { public void close() {
try { try {
synchronized(searcherLock) {
// it's possible for someone to get a reference via the _searchers queue
// and increment the refcount while RefCounted.close() is being called.
// we check the refcount again to see if this has happened and abort the close.
// This relies on the RefCounted class allowing close() to be called every
// time the counter hits zero.
if (refcount.get() > 0) return;
_searchers.remove(this);
}
resource.close(); resource.close();
} catch (IOException e) { } catch (IOException e) {
log.severe("Error closing searcher:" + SolrException.toStr(e)); log.severe("Error closing searcher:" + SolrException.toStr(e));

View File

@ -61,7 +61,9 @@ import org.apache.solr.core.SolrCore;
import org.apache.solr.schema.FieldType; import org.apache.solr.schema.FieldType;
import org.apache.solr.schema.SchemaField; import org.apache.solr.schema.SchemaField;
import org.apache.solr.search.SortSpec; import org.apache.solr.search.SortSpec;
import org.apache.solr.search.SolrIndexSearcher;
import org.apache.solr.util.VersionedFile; import org.apache.solr.util.VersionedFile;
import org.apache.solr.util.RefCounted;
import org.apache.solr.util.plugin.SolrCoreAware; import org.apache.solr.util.plugin.SolrCoreAware;
import org.apache.solr.request.SolrQueryRequest; import org.apache.solr.request.SolrQueryRequest;
import org.w3c.dom.Node; import org.w3c.dom.Node;
@ -182,8 +184,14 @@ public class QueryElevationComponent extends SearchComponent implements SolrCore
} }
else { else {
// preload the first data // preload the first data
IndexReader reader = core.getSearcher().get().getReader(); RefCounted<SolrIndexSearcher> searchHolder = null;
try {
searchHolder = core.getNewestSearcher(true);
IndexReader reader = searchHolder.get().getReader();
getElevationMap( reader, core ); getElevationMap( reader, core );
} finally {
if (searchHolder != null) searchHolder.decref();
}
} }
} }
} }