276 lines
6.7 KiB
C#
276 lines
6.7 KiB
C#
|
//
|
||
|
// Author(s):
|
||
|
// Marek Habersack <mhabersack@novell.com>
|
||
|
//
|
||
|
// (C) 2009-2010 Novell, Inc (http://novell.com)
|
||
|
//
|
||
|
//
|
||
|
// Permission is hereby granted, free of charge, to any person obtaining
|
||
|
// a copy of this software and associated documentation files (the
|
||
|
// "Software"), to deal in the Software without restriction, including
|
||
|
// without limitation the rights to use, copy, modify, merge, publish,
|
||
|
// distribute, sublicense, and/or sell copies of the Software, and to
|
||
|
// permit persons to whom the Software is furnished to do so, subject to
|
||
|
// the following conditions:
|
||
|
//
|
||
|
// The above copyright notice and this permission notice shall be
|
||
|
// included in all copies or substantial portions of the Software.
|
||
|
//
|
||
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||
|
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||
|
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||
|
//
|
||
|
using System;
|
||
|
using System.Collections;
|
||
|
using System.Collections.Generic;
|
||
|
using System.IO;
|
||
|
using System.Text;
|
||
|
using System.Threading;
|
||
|
using System.Xml;
|
||
|
|
||
|
namespace System.Web.Caching
|
||
|
{
|
||
|
sealed partial class CacheItemPriorityQueue
|
||
|
{
|
||
|
const int INITIAL_HEAP_SIZE = 32;
|
||
|
const int HEAP_RESIZE_THRESHOLD = 8192;
|
||
|
|
||
|
CacheItem[] heap;
|
||
|
int heapSize = 0;
|
||
|
int heapCount = 0;
|
||
|
|
||
|
// See comment for the cacheLock field at top of System.Web.Caching/Cache.cs
|
||
|
ReaderWriterLockSlim queueLock;
|
||
|
|
||
|
public int Count {
|
||
|
get { return heapCount; }
|
||
|
}
|
||
|
|
||
|
public int Size {
|
||
|
get { return heapSize; }
|
||
|
}
|
||
|
|
||
|
public CacheItem[] Heap {
|
||
|
get { return heap; }
|
||
|
}
|
||
|
|
||
|
public CacheItemPriorityQueue ()
|
||
|
{
|
||
|
queueLock = new ReaderWriterLockSlim ();
|
||
|
InitDebugMode ();
|
||
|
}
|
||
|
|
||
|
void ResizeHeap (int newSize)
|
||
|
{
|
||
|
CacheItem[] oldHeap = heap;
|
||
|
Array.Resize <CacheItem> (ref heap, newSize);
|
||
|
heapSize = newSize;
|
||
|
|
||
|
// TODO: The code helps the GC in case the array is pinned. In such instance clearing
|
||
|
// the old array will release references to the CacheItems stored in there. If the
|
||
|
// array is not pinned, otoh, this is a waste of time.
|
||
|
// Currently we don't know if the array is pinned or not so it's safer to always clear it.
|
||
|
// However when we have more precise stack scanning the code should be
|
||
|
// revisited.
|
||
|
if (oldHeap != null) {
|
||
|
((IList)oldHeap).Clear ();
|
||
|
oldHeap = null;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
CacheItem[] GetHeapWithGrow ()
|
||
|
{
|
||
|
if (heap == null) {
|
||
|
heap = new CacheItem [INITIAL_HEAP_SIZE];
|
||
|
heapSize = INITIAL_HEAP_SIZE;
|
||
|
heapCount = 0;
|
||
|
return heap;
|
||
|
}
|
||
|
|
||
|
if (heapCount >= heapSize)
|
||
|
ResizeHeap (heapSize <<= 1);
|
||
|
|
||
|
return heap;
|
||
|
}
|
||
|
|
||
|
CacheItem[] GetHeapWithShrink ()
|
||
|
{
|
||
|
if (heap == null)
|
||
|
return null;
|
||
|
|
||
|
if (heapSize > HEAP_RESIZE_THRESHOLD) {
|
||
|
int halfTheSize = heapSize >> 1;
|
||
|
|
||
|
if (heapCount < halfTheSize)
|
||
|
ResizeHeap (halfTheSize + (heapCount / 3));
|
||
|
}
|
||
|
|
||
|
return heap;
|
||
|
}
|
||
|
|
||
|
public void Enqueue (CacheItem item)
|
||
|
{
|
||
|
if (item == null)
|
||
|
return;
|
||
|
|
||
|
CacheItem[] heap;
|
||
|
|
||
|
try {
|
||
|
queueLock.EnterWriteLock ();
|
||
|
heap = GetHeapWithGrow ();
|
||
|
heap [heapCount] = item;
|
||
|
if (heapCount == 0)
|
||
|
item.PriorityQueueIndex = 0;
|
||
|
BubbleUp (heap, heapCount++);
|
||
|
AddSequenceEntry (item, EDSequenceEntryType.Enqueue);
|
||
|
} finally {
|
||
|
// See comment at the top of the file, above queueLock declaration
|
||
|
queueLock.ExitWriteLock ();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public CacheItem Dequeue ()
|
||
|
{
|
||
|
CacheItem ret = null;
|
||
|
CacheItem[] heap;
|
||
|
int index;
|
||
|
|
||
|
try {
|
||
|
queueLock.EnterWriteLock ();
|
||
|
heap = GetHeapWithShrink ();
|
||
|
if (heap == null || heapCount == 0)
|
||
|
return null;
|
||
|
|
||
|
ret = heap [0];
|
||
|
index = --heapCount;
|
||
|
heap [0] = heap [index];
|
||
|
heap [index] = null;
|
||
|
|
||
|
if (heapCount > 0)
|
||
|
BubbleDown (heap, 0);
|
||
|
|
||
|
AddSequenceEntry (ret, EDSequenceEntryType.Dequeue);
|
||
|
return ret;
|
||
|
} finally {
|
||
|
// See comment at the top of the file, above queueLock declaration
|
||
|
queueLock.ExitWriteLock ();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public bool Update (CacheItem item)
|
||
|
{
|
||
|
if (item == null || item.PriorityQueueIndex <= 0 || item.PriorityQueueIndex >= heapCount - 1)
|
||
|
return false;
|
||
|
|
||
|
try {
|
||
|
queueLock.EnterWriteLock ();
|
||
|
CacheItem stored = heap [item.PriorityQueueIndex];
|
||
|
if (stored == null ||
|
||
|
String.Compare (stored.Key, item.Key, StringComparison.Ordinal) != 0
|
||
|
#if DEBUG
|
||
|
|| stored.Guid != item.Guid
|
||
|
#endif
|
||
|
)
|
||
|
return false;
|
||
|
|
||
|
int oldIndex = item.PriorityQueueIndex;
|
||
|
int index = BubbleUp (heap, oldIndex);
|
||
|
if (index > -1 && index >= oldIndex)
|
||
|
BubbleDown (heap, index);
|
||
|
|
||
|
AddSequenceEntry (item, EDSequenceEntryType.Update);
|
||
|
} finally {
|
||
|
queueLock.ExitWriteLock ();
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
public CacheItem Peek ()
|
||
|
{
|
||
|
CacheItem ret;
|
||
|
|
||
|
try {
|
||
|
queueLock.EnterReadLock ();
|
||
|
if (heap == null || heapCount == 0)
|
||
|
return null;
|
||
|
|
||
|
ret = heap [0];
|
||
|
AddSequenceEntry (ret, EDSequenceEntryType.Peek);
|
||
|
|
||
|
return ret;
|
||
|
} finally {
|
||
|
// See comment at the top of the file, above queueLock declaration
|
||
|
queueLock.ExitReadLock ();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int BubbleDown (CacheItem[] heap, int startIndex)
|
||
|
{
|
||
|
int index = startIndex;
|
||
|
int left = startIndex + 1;
|
||
|
int right = startIndex + 2;
|
||
|
CacheItem item = heap [index], tmpItem;
|
||
|
|
||
|
int selected = (right < heapCount && heap [right].ExpiresAt < heap [left].ExpiresAt) ? 2 : 1;
|
||
|
|
||
|
do {
|
||
|
selected = index;
|
||
|
left = (index << 1) + 1;
|
||
|
right = left + 1;
|
||
|
if (heapCount > left && heap [index].ExpiresAt > heap [left].ExpiresAt)
|
||
|
index = left;
|
||
|
if (heapCount > right && heap [index].ExpiresAt > heap [right].ExpiresAt)
|
||
|
index = right;
|
||
|
if (index == selected)
|
||
|
break;
|
||
|
tmpItem = heap [index];
|
||
|
heap [index] = heap [selected];
|
||
|
heap [index].PriorityQueueIndex = index;
|
||
|
heap [selected] = tmpItem;
|
||
|
tmpItem.PriorityQueueIndex = selected;
|
||
|
} while (true);
|
||
|
|
||
|
item.PriorityQueueIndex = index;
|
||
|
return index;
|
||
|
}
|
||
|
|
||
|
int BubbleUp (CacheItem[] heap, int startIndex)
|
||
|
{
|
||
|
int index, parentIndex;
|
||
|
CacheItem parent, item;
|
||
|
|
||
|
if (heapCount <= 1)
|
||
|
return -1;
|
||
|
|
||
|
int maxIndex = heapCount - 1;
|
||
|
if (startIndex < 0 || startIndex > maxIndex)
|
||
|
return -1;
|
||
|
|
||
|
index = startIndex;
|
||
|
parentIndex = (index - 1) >> 1;
|
||
|
|
||
|
item = heap [index];
|
||
|
while (index > 0) {
|
||
|
parent = heap [parentIndex];
|
||
|
if (heap [index].ExpiresAt >= parent.ExpiresAt)
|
||
|
break;
|
||
|
|
||
|
heap [index] = parent;
|
||
|
parent.PriorityQueueIndex = index;
|
||
|
index = parentIndex;
|
||
|
parentIndex = (index - 1) >> 1;
|
||
|
}
|
||
|
|
||
|
heap [index] = item;
|
||
|
item.PriorityQueueIndex = index;
|
||
|
|
||
|
return index;
|
||
|
}
|
||
|
}
|
||
|
}
|