forked from Alepha/Alepha
		
	Deal with last-item-in-the-queue bug.
This commit is contained in:
		| @ -102,12 +102,21 @@ namespace Alepha::Hydrogen::Atomic  ::detail::  Dropbox_m | |||||||
| 				drops.producer.releaseBox(); | 				drops.producer.releaseBox(); | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
|  | 			void | ||||||
|  | 			recycle() | ||||||
|  | 			{ | ||||||
|  | 				auto &filled= drops.consumer.acquireBox(); | ||||||
|  | 				filled.notification= nullptr; | ||||||
|  | 				filled.items.clear(); | ||||||
|  | 				drops.consumer.releaseBox(); | ||||||
|  | 			} | ||||||
|  |  | ||||||
| 		public: | 		public: | ||||||
| 			explicit | 			explicit | ||||||
| 			Dropbox( const size_t lim ) | 			Dropbox( const size_t lim ) | ||||||
| 				: boxLimit( lim ) | 				: boxLimit( lim ) | ||||||
| 			{ | 			{ | ||||||
| 				drops.consumer.releaseBox(); | 				recycle(); | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			Item | 			Item | ||||||
| @ -118,14 +127,24 @@ namespace Alepha::Hydrogen::Atomic  ::detail::  Dropbox_m | |||||||
| 				{ | 				{ | ||||||
| 					[[unlikely]]; | 					[[unlikely]]; | ||||||
| 					auto notification= std::move( filled.notification ); | 					auto notification= std::move( filled.notification ); | ||||||
| 					filled.notification= nullptr; | 					recycle(); | ||||||
| 					drops.consumer.releaseBox(); |  | ||||||
| 					std::rethrow_exception( std::move( notification ) ); | 					std::rethrow_exception( std::move( notification ) ); | ||||||
| 				} | 				} | ||||||
|  | 				assert( not filled.items.empty() ); | ||||||
| 				auto &rv= *filled.pos++; | 				auto &rv= *filled.pos++; | ||||||
|  | 				// If it was the last item and we're going to rotate/reccyle, | ||||||
|  | 				// be sure that we don't reference the now-stale value. | ||||||
|  | 				// This is a special case that will only come up at the end of each queue | ||||||
|  | 				// cycle. | ||||||
|  | 				// | ||||||
|  | 				// I need to research if NRV applies in this circumstance -- if so, then the | ||||||
|  | 				// `rv` variable can be an instance instead of a reference. | ||||||
| 				if( filled.pos == end( filled.items ) and filled.notification == nullptr ) | 				if( filled.pos == end( filled.items ) and filled.notification == nullptr ) | ||||||
| 				{ | 				{ | ||||||
| 					drops.consumer.releaseBox(); | 					[[unlikely]]; | ||||||
|  | 					auto rvTemp= std::move( rv ); | ||||||
|  | 					recycle(); | ||||||
|  | 					return rvTemp; | ||||||
| 				} | 				} | ||||||
| 				return std::move( rv ); | 				return std::move( rv ); | ||||||
| 			} | 			} | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user