|  | @@ -1,403 +0,0 @@
 | 
	
		
			
				|  |  | -//-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  | -// Copyright (c) 2012 GarageGames, LLC
 | 
	
		
			
				|  |  | -//
 | 
	
		
			
				|  |  | -// 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.
 | 
	
		
			
				|  |  | -//-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -#include "unit/test.h"
 | 
	
		
			
				|  |  | -#include "platform/threads/threadSafeDeque.h"
 | 
	
		
			
				|  |  | -#include "platform/threads/thread.h"
 | 
	
		
			
				|  |  | -#include "core/util/tVector.h"
 | 
	
		
			
				|  |  | -#include "console/console.h"
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -#ifndef TORQUE_SHIPPING
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -using namespace UnitTesting;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -#define TEST( x ) test( ( x ), "FAIL: " #x )
 | 
	
		
			
				|  |  | -#define XTEST( t, x ) t->test( ( x ), "FAIL: " #x )
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -// Test deque without concurrency.
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -CreateUnitTest( TestThreadSafeDequeSerial, "Platform/ThreadSafeDeque/Serial" )
 | 
	
		
			
				|  |  | -{
 | 
	
		
			
				|  |  | -   void test1()
 | 
	
		
			
				|  |  | -   {
 | 
	
		
			
				|  |  | -      ThreadSafeDeque< char > deque;
 | 
	
		
			
				|  |  | -      String str = "teststring";
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      for( U32 i = 0; i < str.length(); ++ i )
 | 
	
		
			
				|  |  | -         deque.pushBack( str[ i ] );
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      TEST( !deque.isEmpty() );
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      for( U32 i = 0; i < str.length(); ++ i )
 | 
	
		
			
				|  |  | -      {
 | 
	
		
			
				|  |  | -         char ch;
 | 
	
		
			
				|  |  | -         TEST( deque.tryPopFront( ch ) && ch == str[ i ] );
 | 
	
		
			
				|  |  | -      }
 | 
	
		
			
				|  |  | -   }
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -   void test2()
 | 
	
		
			
				|  |  | -   {
 | 
	
		
			
				|  |  | -      ThreadSafeDeque< char > deque;
 | 
	
		
			
				|  |  | -      String str = "teststring";
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      const char* p1 = str.c_str() + 4;
 | 
	
		
			
				|  |  | -      const char* p2 = p1 + 1;
 | 
	
		
			
				|  |  | -      while( *p2 )
 | 
	
		
			
				|  |  | -      {
 | 
	
		
			
				|  |  | -         deque.pushFront( *p1 );
 | 
	
		
			
				|  |  | -         deque.pushBack( *p2 );
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -         -- p1;
 | 
	
		
			
				|  |  | -         ++ p2;
 | 
	
		
			
				|  |  | -      }
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -#ifdef TORQUE_DEBUG
 | 
	
		
			
				|  |  | -      deque.dumpDebug();
 | 
	
		
			
				|  |  | -#endif
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      for( U32 i = 0; i < str.length(); ++ i )
 | 
	
		
			
				|  |  | -      {
 | 
	
		
			
				|  |  | -         char ch;
 | 
	
		
			
				|  |  | -         TEST( deque.tryPopFront( ch ) && ch == str[ i ] );
 | 
	
		
			
				|  |  | -      }
 | 
	
		
			
				|  |  | -   }
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -   void test3()
 | 
	
		
			
				|  |  | -   {
 | 
	
		
			
				|  |  | -      ThreadSafeDeque< char > deque;
 | 
	
		
			
				|  |  | -      String str = "teststring";
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      const char* p1 = str.c_str() + 4;
 | 
	
		
			
				|  |  | -      const char* p2 = p1 + 1;
 | 
	
		
			
				|  |  | -      while( *p2 )
 | 
	
		
			
				|  |  | -      {
 | 
	
		
			
				|  |  | -         deque.pushFront( *p1 );
 | 
	
		
			
				|  |  | -         deque.pushBack( *p2 );
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -         -- p1;
 | 
	
		
			
				|  |  | -         ++ p2;
 | 
	
		
			
				|  |  | -      }
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -#ifdef TORQUE_DEBUG
 | 
	
		
			
				|  |  | -      deque.dumpDebug();
 | 
	
		
			
				|  |  | -#endif
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      for( S32 i = ( str.length() - 1 ); i >= 0; -- i )
 | 
	
		
			
				|  |  | -      {
 | 
	
		
			
				|  |  | -         char ch;
 | 
	
		
			
				|  |  | -         TEST( deque.tryPopBack( ch ) && ch == str[ i ] );
 | 
	
		
			
				|  |  | -      }
 | 
	
		
			
				|  |  | -   }
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -   void test4()
 | 
	
		
			
				|  |  | -   {
 | 
	
		
			
				|  |  | -      ThreadSafeDeque< char > deque;
 | 
	
		
			
				|  |  | -      char ch;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      TEST( deque.isEmpty() );
 | 
	
		
			
				|  |  | -      
 | 
	
		
			
				|  |  | -      deque.pushFront( 'a' );
 | 
	
		
			
				|  |  | -      TEST( !deque.isEmpty() );
 | 
	
		
			
				|  |  | -      TEST( deque.tryPopFront( ch ) );
 | 
	
		
			
				|  |  | -      TEST( ch == 'a' );
 | 
	
		
			
				|  |  | -      
 | 
	
		
			
				|  |  | -      deque.pushBack( 'a' );
 | 
	
		
			
				|  |  | -      TEST( !deque.isEmpty() );
 | 
	
		
			
				|  |  | -      TEST( deque.tryPopFront( ch ) );
 | 
	
		
			
				|  |  | -      TEST( ch == 'a' );
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      deque.pushBack( 'a' );
 | 
	
		
			
				|  |  | -      TEST( !deque.isEmpty() );
 | 
	
		
			
				|  |  | -      TEST( deque.tryPopBack( ch ) );
 | 
	
		
			
				|  |  | -      TEST( ch == 'a' );
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      deque.pushFront( 'a' );
 | 
	
		
			
				|  |  | -      TEST( !deque.isEmpty() );
 | 
	
		
			
				|  |  | -      TEST( deque.tryPopBack( ch ) );
 | 
	
		
			
				|  |  | -      TEST( ch == 'a' );
 | 
	
		
			
				|  |  | -   }
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -   void run()
 | 
	
		
			
				|  |  | -   {
 | 
	
		
			
				|  |  | -      test1();
 | 
	
		
			
				|  |  | -      test2();
 | 
	
		
			
				|  |  | -      test3();
 | 
	
		
			
				|  |  | -      test4();
 | 
	
		
			
				|  |  | -   }
 | 
	
		
			
				|  |  | -};
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -// Test deque in a concurrent setting.
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -CreateUnitTest( TestThreadSafeDequeConcurrentSimple, "Platform/ThreadSafeDeque/ConcurrentSimple" )
 | 
	
		
			
				|  |  | -{
 | 
	
		
			
				|  |  | -public:
 | 
	
		
			
				|  |  | -   typedef TestThreadSafeDequeConcurrentSimple TestType;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -   enum
 | 
	
		
			
				|  |  | -   {
 | 
	
		
			
				|  |  | -      DEFAULT_NUM_VALUES = 100000,
 | 
	
		
			
				|  |  | -   };
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -   struct Value : public ThreadSafeRefCount< Value >
 | 
	
		
			
				|  |  | -   {
 | 
	
		
			
				|  |  | -      U32 mIndex;
 | 
	
		
			
				|  |  | -      U32 mTick;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      Value() {}
 | 
	
		
			
				|  |  | -      Value( U32 index, U32 tick )
 | 
	
		
			
				|  |  | -         : mIndex( index ), mTick( tick ) {}
 | 
	
		
			
				|  |  | -   };
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -   typedef ThreadSafeRef< Value > ValueRef;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -   struct Deque : public ThreadSafeDeque< ValueRef >
 | 
	
		
			
				|  |  | -   {
 | 
	
		
			
				|  |  | -      typedef ThreadSafeDeque<ValueRef> Parent;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      U32 mPushIndex;
 | 
	
		
			
				|  |  | -      U32 mPopIndex;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      Deque()
 | 
	
		
			
				|  |  | -         : mPushIndex( 0 ), mPopIndex( 0 ) {}
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      void pushBack( const ValueRef& value )
 | 
	
		
			
				|  |  | -      {
 | 
	
		
			
				|  |  | -         AssertFatal( value->mIndex == mPushIndex, "index out of line" );
 | 
	
		
			
				|  |  | -         mPushIndex ++;
 | 
	
		
			
				|  |  | -         Parent::pushBack( value );
 | 
	
		
			
				|  |  | -      }
 | 
	
		
			
				|  |  | -      bool tryPopFront( ValueRef& outValue )
 | 
	
		
			
				|  |  | -      {
 | 
	
		
			
				|  |  | -         if( Parent::tryPopFront( outValue ) )
 | 
	
		
			
				|  |  | -         {
 | 
	
		
			
				|  |  | -            AssertFatal( outValue->mIndex == mPopIndex, "index out of line" );
 | 
	
		
			
				|  |  | -            mPopIndex ++;
 | 
	
		
			
				|  |  | -            return true;
 | 
	
		
			
				|  |  | -         }
 | 
	
		
			
				|  |  | -         else
 | 
	
		
			
				|  |  | -            return false;
 | 
	
		
			
				|  |  | -      }
 | 
	
		
			
				|  |  | -   };
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -   Deque mDeque;
 | 
	
		
			
				|  |  | -   Vector< U32 > mValues;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -   struct ProducerThread : public Thread
 | 
	
		
			
				|  |  | -   {
 | 
	
		
			
				|  |  | -      ProducerThread( TestType* test )
 | 
	
		
			
				|  |  | -         : Thread( 0, test ) {}
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      virtual void run( void* arg )
 | 
	
		
			
				|  |  | -      {
 | 
	
		
			
				|  |  | -         _setName( "ProducerThread" );
 | 
	
		
			
				|  |  | -         Platform::outputDebugString( "Starting ProducerThread" );
 | 
	
		
			
				|  |  | -         
 | 
	
		
			
				|  |  | -         TestType* test = ( TestType* ) arg;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -         for( U32 i = 0; i < test->mValues.size(); ++ i )
 | 
	
		
			
				|  |  | -         {
 | 
	
		
			
				|  |  | -            U32 tick = Platform::getRealMilliseconds();
 | 
	
		
			
				|  |  | -            test->mValues[ i ] = tick;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -            ValueRef val = new Value( i, tick );
 | 
	
		
			
				|  |  | -            test->mDeque.pushBack( val );
 | 
	
		
			
				|  |  | -         }
 | 
	
		
			
				|  |  | -         Platform::outputDebugString( "Stopping ProducerThread" );
 | 
	
		
			
				|  |  | -      }
 | 
	
		
			
				|  |  | -   };
 | 
	
		
			
				|  |  | -   struct ConsumerThread : public Thread
 | 
	
		
			
				|  |  | -   {
 | 
	
		
			
				|  |  | -      ConsumerThread( TestType* test )
 | 
	
		
			
				|  |  | -         : Thread( 0, test ) {}
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      virtual void run( void* arg )
 | 
	
		
			
				|  |  | -      {
 | 
	
		
			
				|  |  | -         _setName( "ConsumerThread" );
 | 
	
		
			
				|  |  | -         Platform::outputDebugString( "Starting CosumerThread" );
 | 
	
		
			
				|  |  | -         TestType* t = ( TestType* ) arg;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -         for( U32 i = 0; i < t->mValues.size(); ++ i )
 | 
	
		
			
				|  |  | -         {
 | 
	
		
			
				|  |  | -            ValueRef value;
 | 
	
		
			
				|  |  | -            while( !t->mDeque.tryPopFront( value ) );
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -            XTEST( t, value->mIndex == i );
 | 
	
		
			
				|  |  | -            XTEST( t, t->mValues[ i ] == value->mTick );
 | 
	
		
			
				|  |  | -         }
 | 
	
		
			
				|  |  | -         Platform::outputDebugString( "Stopping ConsumerThread" );
 | 
	
		
			
				|  |  | -      }
 | 
	
		
			
				|  |  | -   };
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -   void run()
 | 
	
		
			
				|  |  | -   {
 | 
	
		
			
				|  |  | -      U32 numValues = Con::getIntVariable( "$testThreadSafeDeque::numValues", DEFAULT_NUM_VALUES );
 | 
	
		
			
				|  |  | -      mValues.setSize( numValues );
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      ProducerThread pThread( this );
 | 
	
		
			
				|  |  | -      ConsumerThread cThread( this );
 | 
	
		
			
				|  |  | -      
 | 
	
		
			
				|  |  | -      pThread.start();
 | 
	
		
			
				|  |  | -      cThread.start();
 | 
	
		
			
				|  |  | -      
 | 
	
		
			
				|  |  | -      pThread.join();
 | 
	
		
			
				|  |  | -      cThread.join();
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      mValues.clear();
 | 
	
		
			
				|  |  | -   }
 | 
	
		
			
				|  |  | -};
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -CreateUnitTest( TestThreadSafeDequeConcurrent, "Platform/ThreadSafeDeque/Concurrent" )
 | 
	
		
			
				|  |  | -{
 | 
	
		
			
				|  |  | -public:
 | 
	
		
			
				|  |  | -   typedef TestThreadSafeDequeConcurrent TestType;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -   enum
 | 
	
		
			
				|  |  | -   {
 | 
	
		
			
				|  |  | -      DEFAULT_NUM_VALUES = 100000,
 | 
	
		
			
				|  |  | -      DEFAULT_NUM_CONSUMERS = 10,
 | 
	
		
			
				|  |  | -      DEFAULT_NUM_PRODUCERS = 10
 | 
	
		
			
				|  |  | -   };
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -   struct Value : public ThreadSafeRefCount< Value >
 | 
	
		
			
				|  |  | -   {
 | 
	
		
			
				|  |  | -      U32 mIndex;
 | 
	
		
			
				|  |  | -      U32 mTick;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      Value() {}
 | 
	
		
			
				|  |  | -      Value( U32 index, U32 tick )
 | 
	
		
			
				|  |  | -         : mIndex( index ), mTick( tick ) {}
 | 
	
		
			
				|  |  | -   };
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -   typedef ThreadSafeRef< Value > ValueRef;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -   U32 mProducerIndex;
 | 
	
		
			
				|  |  | -   U32 mConsumerIndex;
 | 
	
		
			
				|  |  | -   ThreadSafeDeque< ValueRef > mDeque;
 | 
	
		
			
				|  |  | -   Vector< U32 > mValues;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -   struct ProducerThread : public Thread
 | 
	
		
			
				|  |  | -   {
 | 
	
		
			
				|  |  | -      ProducerThread( TestType* test )
 | 
	
		
			
				|  |  | -         : Thread( 0, test ) {}
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      virtual void run( void* arg )
 | 
	
		
			
				|  |  | -      {
 | 
	
		
			
				|  |  | -         _setName( "ProducerThread" );
 | 
	
		
			
				|  |  | -         Platform::outputDebugString( "Starting ProducerThread" );
 | 
	
		
			
				|  |  | -         TestType* test = ( TestType* ) arg;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -         while( 1 )
 | 
	
		
			
				|  |  | -         {
 | 
	
		
			
				|  |  | -            U32 index = test->mProducerIndex;
 | 
	
		
			
				|  |  | -            if( index == test->mValues.size() )
 | 
	
		
			
				|  |  | -               break;
 | 
	
		
			
				|  |  | -               
 | 
	
		
			
				|  |  | -            if( dCompareAndSwap( test->mProducerIndex, index, index + 1 ) )
 | 
	
		
			
				|  |  | -            {
 | 
	
		
			
				|  |  | -               U32 tick = Platform::getRealMilliseconds();
 | 
	
		
			
				|  |  | -               test->mValues[ index ] = tick;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -               ValueRef val = new Value( index, tick );
 | 
	
		
			
				|  |  | -               test->mDeque.pushBack( val );
 | 
	
		
			
				|  |  | -            }
 | 
	
		
			
				|  |  | -         }
 | 
	
		
			
				|  |  | -         Platform::outputDebugString( "Stopping ProducerThread" );
 | 
	
		
			
				|  |  | -      }
 | 
	
		
			
				|  |  | -   };
 | 
	
		
			
				|  |  | -   struct ConsumerThread : public Thread
 | 
	
		
			
				|  |  | -   {
 | 
	
		
			
				|  |  | -      ConsumerThread( TestType* test )
 | 
	
		
			
				|  |  | -         : Thread( 0, test ) {}
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      virtual void run( void* arg )
 | 
	
		
			
				|  |  | -      {
 | 
	
		
			
				|  |  | -         _setName( "ConsumerThread" );
 | 
	
		
			
				|  |  | -         Platform::outputDebugString( "Starting ConsumerThread" );
 | 
	
		
			
				|  |  | -         TestType* t = ( TestType* ) arg;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -         while( t->mConsumerIndex < t->mValues.size() )
 | 
	
		
			
				|  |  | -         {
 | 
	
		
			
				|  |  | -            ValueRef value;
 | 
	
		
			
				|  |  | -            if( t->mDeque.tryPopFront( value ) )
 | 
	
		
			
				|  |  | -            {
 | 
	
		
			
				|  |  | -               dFetchAndAdd( t->mConsumerIndex, 1 );
 | 
	
		
			
				|  |  | -               XTEST( t, t->mValues[ value->mIndex ] == value->mTick );
 | 
	
		
			
				|  |  | -               t->mValues[ value->mIndex ] = 0;
 | 
	
		
			
				|  |  | -            }
 | 
	
		
			
				|  |  | -         }
 | 
	
		
			
				|  |  | -         
 | 
	
		
			
				|  |  | -         Platform::outputDebugString( "Stopping ConsumerThread" );
 | 
	
		
			
				|  |  | -      }
 | 
	
		
			
				|  |  | -   };
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -   void run()
 | 
	
		
			
				|  |  | -   {
 | 
	
		
			
				|  |  | -      U32 numValues = Con::getIntVariable( "$testThreadSafeDeque::numValues", DEFAULT_NUM_VALUES );
 | 
	
		
			
				|  |  | -      U32 numConsumers = Con::getIntVariable( "$testThreadSafeDeque::numConsumers", DEFAULT_NUM_CONSUMERS );
 | 
	
		
			
				|  |  | -      U32 numProducers = Con::getIntVariable( "$testThreadSafeDeque::numProducers", DEFAULT_NUM_PRODUCERS );
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      mProducerIndex = 0;
 | 
	
		
			
				|  |  | -      mConsumerIndex = 0;
 | 
	
		
			
				|  |  | -      mValues.setSize( numValues );
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      U32 tick = Platform::getRealMilliseconds();
 | 
	
		
			
				|  |  | -      for( U32 i = 0; i < numValues; ++ i )
 | 
	
		
			
				|  |  | -         mValues[ i ] = tick;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      Vector< ProducerThread* > producers;
 | 
	
		
			
				|  |  | -      Vector< ConsumerThread* > consumers;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      producers.setSize( numProducers );
 | 
	
		
			
				|  |  | -      consumers.setSize( numConsumers );
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      for( U32 i = 0; i < numProducers; ++ i )
 | 
	
		
			
				|  |  | -      {
 | 
	
		
			
				|  |  | -         producers[ i ] = new ProducerThread( this );
 | 
	
		
			
				|  |  | -         producers[ i ]->start();
 | 
	
		
			
				|  |  | -      }
 | 
	
		
			
				|  |  | -      for( U32 i = 0; i < numConsumers; ++ i )
 | 
	
		
			
				|  |  | -      {
 | 
	
		
			
				|  |  | -         consumers[ i ] = new ConsumerThread( this );
 | 
	
		
			
				|  |  | -         consumers[ i ]->start();
 | 
	
		
			
				|  |  | -      }
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      for( U32 i = 0; i < numProducers; ++ i )
 | 
	
		
			
				|  |  | -      {
 | 
	
		
			
				|  |  | -         producers[ i ]->join();
 | 
	
		
			
				|  |  | -         delete producers[ i ];
 | 
	
		
			
				|  |  | -      }
 | 
	
		
			
				|  |  | -      for( U32 i = 0; i < numConsumers; ++ i )
 | 
	
		
			
				|  |  | -      {
 | 
	
		
			
				|  |  | -         consumers[ i ]->join();
 | 
	
		
			
				|  |  | -         delete consumers[ i ];
 | 
	
		
			
				|  |  | -      }
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      for( U32 i = 0; i < mValues.size(); ++ i )
 | 
	
		
			
				|  |  | -         TEST( mValues[ i ] == 0 );
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -      mValues.clear();
 | 
	
		
			
				|  |  | -   }
 | 
	
		
			
				|  |  | -};
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -#endif // !TORQUE_SHIPPING
 |