Search code examples
c#.nethashtablethread-sleepspinwait

Why there is a Thread.Sleep(1) in .NET internal Hashtable?


Recently I was reading implementation of .NET Hashtable and encountered piece of code that I don't understand. Part of the code is:

int num3 = 0;
int num4;
do
{
   num4 = this.version;
   bucket = bucketArray[index];
   if (++num3 % 8 == 0)
     Thread.Sleep(1);
}
while (this.isWriterInProgress || num4 != this.version);

The whole code is within public virtual object this[object key] of System.Collections.Hashtable (mscorlib Version=4.0.0.0).

The question is:

What is the reason of having Thread.Sleep(1) there?


Solution

  • Sleep(1) is a documented way in Windows to yield the processor and allow other threads to run. You can find this code in the Reference Source with comments:

       // Our memory model guarantee if we pick up the change in bucket from another processor,
       // we will see the 'isWriterProgress' flag to be true or 'version' is changed in the reader.
       //
       int spinCount = 0;
       do {
           // this is violate read, following memory accesses can not be moved ahead of it.
           currentversion = version;
           b = lbuckets[bucketNumber];
    
           // The contention between reader and writer shouldn't happen frequently.
           // But just in case this will burn CPU, yield the control of CPU if we spinned a few times.
           // 8 is just a random number I pick.
           if( (++spinCount) % 8 == 0 ) {
               Thread.Sleep(1);   // 1 means we are yeilding control to all threads, including low-priority ones.
           }
       } while ( isWriterInProgress || (currentversion != version) );
    

    The isWriterInProgress variable is a volatile bool. The author had some trouble with English "violate read" is "volatile read". The basic idea is try to avoid yielding, thread context switches are very expensive, with some hope that the writer gets done quickly. If that doesn't pan out then explicitly yield to avoid burning cpu. This would probably have been written with Spinlock today but Hashtable is very old. As are the assumptions about the memory model.