Code:
/ 4.0 / 4.0 / DEVDIV_TFS / Dev10 / Releases / RTMRel / ndp / clr / src / ManagedLibraries / Remoting / Channels / CORE / ByteBufferPool.cs / 1305376 / ByteBufferPool.cs
// ==++== // // Copyright (c) Microsoft Corporation. All rights reserved. // // ==--== //========================================================================== // File: ByteBufferPool.cs // // Summary: Stream used for reading from a socket by remoting channels. // //========================================================================= using System; using System.Threading; namespace System.IO { internal interface IByteBufferPool { byte[] GetBuffer(); void ReturnBuffer(byte[] buffer); } // This isn't actually a buffer pool. It always creates a new byte buffer. internal class ByteBufferAllocator : IByteBufferPool { private int _bufferSize; public ByteBufferAllocator(int bufferSize) { _bufferSize = bufferSize; } public byte[] GetBuffer() { return new byte[_bufferSize]; } public void ReturnBuffer(byte[] buffer) { } } // ByteBufferAllocator internal class ByteBufferPool : IByteBufferPool { private byte[][] _bufferPool = null; private int _current; // -1 for none private int _last; private int _max; // maximum number of buffers to pool private int _bufferSize; private Object _controlCookie = "cookie object"; public ByteBufferPool(int maxBuffers, int bufferSize) { _max = maxBuffers; _bufferPool = new byte[_max][]; _bufferSize = bufferSize; _current = -1; _last = -1; } // ByteBufferPool public byte[] GetBuffer() { Object cookie = null; try { // If a ThreadAbortException gets thrown after the exchange, // but before the result is assigned to cookie, then the // control cookie is lost forever. However, the buffer pool // will still function normally and return everybody a new // buffer each time (that isn't very likely to happen, // so we don't really care). cookie = Interlocked.Exchange(ref _controlCookie, null); if (cookie != null) { // we have the control cookie, so take a buffer if (_current == -1) { _controlCookie = cookie; // no pooled buffers available return new byte[_bufferSize]; } else { // grab next available buffer byte[] buffer = _bufferPool[_current]; _bufferPool[_current] = null; // update "current" index if (_current == _last) { // this is the last free buffer _current = -1; } else { _current = (_current + 1) % _max; } _controlCookie = cookie; return buffer; } } else { // we don't have the control cookie, so just create a new buffer since // there will probably be a lot of contention anyway. return new byte[_bufferSize]; } } catch (ThreadAbortException) { if (cookie != null) { // This should be rare, so just reset // everything to the initial state. _current = -1; _last = -1; // restore cookie _controlCookie = cookie; } throw; } } // GetBuffer public void ReturnBuffer(byte[] buffer) { if (buffer == null) throw new ArgumentNullException("buffer"); // The purpose of the buffer pool is to try to reduce the // amount of garbage generated, so it doesn't matter if // the buffer gets tossed out. Since we don't want to // take the perf hit of taking a lock, we only return // the buffer if we can grab the control cookie. Object cookie = null; try { // If a ThreadAbortException gets thrown after the exchange, // but before the result is assigned to cookie, then the // control cookie is lost forever. However, the buffer pool // will still function normally and return everybody a new // buffer each time (that isn't very likely to happen, // so we don't really care). cookie = Interlocked.Exchange(ref _controlCookie, null); if (cookie != null) { if (_current == -1) { _bufferPool[0] = buffer; _current = 0; _last = 0; } else { int newLast = (_last + 1) % _max; if (newLast != _current) { // the pool isn't full so store this buffer _last = newLast; _bufferPool[_last] = buffer; } } _controlCookie = cookie; } } catch (ThreadAbortException) { if (cookie != null) { // This should be rare, so just reset // everything to the initial state. _current = -1; _last = -1; // restore cookie _controlCookie = cookie; } throw; } } // ReturnBuffer } // ByteBufferPool } // namespace System.IO // File provided for Reference Use Only by Microsoft Corporation (c) 2007. // ==++== // // Copyright (c) Microsoft Corporation. All rights reserved. // // ==--== //========================================================================== // File: ByteBufferPool.cs // // Summary: Stream used for reading from a socket by remoting channels. // //========================================================================= using System; using System.Threading; namespace System.IO { internal interface IByteBufferPool { byte[] GetBuffer(); void ReturnBuffer(byte[] buffer); } // This isn't actually a buffer pool. It always creates a new byte buffer. internal class ByteBufferAllocator : IByteBufferPool { private int _bufferSize; public ByteBufferAllocator(int bufferSize) { _bufferSize = bufferSize; } public byte[] GetBuffer() { return new byte[_bufferSize]; } public void ReturnBuffer(byte[] buffer) { } } // ByteBufferAllocator internal class ByteBufferPool : IByteBufferPool { private byte[][] _bufferPool = null; private int _current; // -1 for none private int _last; private int _max; // maximum number of buffers to pool private int _bufferSize; private Object _controlCookie = "cookie object"; public ByteBufferPool(int maxBuffers, int bufferSize) { _max = maxBuffers; _bufferPool = new byte[_max][]; _bufferSize = bufferSize; _current = -1; _last = -1; } // ByteBufferPool public byte[] GetBuffer() { Object cookie = null; try { // If a ThreadAbortException gets thrown after the exchange, // but before the result is assigned to cookie, then the // control cookie is lost forever. However, the buffer pool // will still function normally and return everybody a new // buffer each time (that isn't very likely to happen, // so we don't really care). cookie = Interlocked.Exchange(ref _controlCookie, null); if (cookie != null) { // we have the control cookie, so take a buffer if (_current == -1) { _controlCookie = cookie; // no pooled buffers available return new byte[_bufferSize]; } else { // grab next available buffer byte[] buffer = _bufferPool[_current]; _bufferPool[_current] = null; // update "current" index if (_current == _last) { // this is the last free buffer _current = -1; } else { _current = (_current + 1) % _max; } _controlCookie = cookie; return buffer; } } else { // we don't have the control cookie, so just create a new buffer since // there will probably be a lot of contention anyway. return new byte[_bufferSize]; } } catch (ThreadAbortException) { if (cookie != null) { // This should be rare, so just reset // everything to the initial state. _current = -1; _last = -1; // restore cookie _controlCookie = cookie; } throw; } } // GetBuffer public void ReturnBuffer(byte[] buffer) { if (buffer == null) throw new ArgumentNullException("buffer"); // The purpose of the buffer pool is to try to reduce the // amount of garbage generated, so it doesn't matter if // the buffer gets tossed out. Since we don't want to // take the perf hit of taking a lock, we only return // the buffer if we can grab the control cookie. Object cookie = null; try { // If a ThreadAbortException gets thrown after the exchange, // but before the result is assigned to cookie, then the // control cookie is lost forever. However, the buffer pool // will still function normally and return everybody a new // buffer each time (that isn't very likely to happen, // so we don't really care). cookie = Interlocked.Exchange(ref _controlCookie, null); if (cookie != null) { if (_current == -1) { _bufferPool[0] = buffer; _current = 0; _last = 0; } else { int newLast = (_last + 1) % _max; if (newLast != _current) { // the pool isn't full so store this buffer _last = newLast; _bufferPool[_last] = buffer; } } _controlCookie = cookie; } } catch (ThreadAbortException) { if (cookie != null) { // This should be rare, so just reset // everything to the initial state. _current = -1; _last = -1; // restore cookie _controlCookie = cookie; } throw; } } // ReturnBuffer } // ByteBufferPool } // namespace System.IO // File provided for Reference Use Only by Microsoft Corporation (c) 2007.
Link Menu
This book is available now!
Buy at Amazon US or
Buy at Amazon UK
- ExpanderAutomationPeer.cs
- SafeNativeMethods.cs
- BinaryParser.cs
- SQLBinaryStorage.cs
- TargetControlTypeCache.cs
- EventSetter.cs
- SoapEnvelopeProcessingElement.cs
- BamlCollectionHolder.cs
- uribuilder.cs
- ProfileGroupSettingsCollection.cs
- ElementHostAutomationPeer.cs
- LOSFormatter.cs
- FontFamilyConverter.cs
- ReadOnlyAttribute.cs
- StickyNoteAnnotations.cs
- OdbcException.cs
- TogglePatternIdentifiers.cs
- FixedDocument.cs
- LeafCellTreeNode.cs
- DataGridViewColumn.cs
- LoginName.cs
- CountdownEvent.cs
- DebugView.cs
- FacetValues.cs
- WinInetCache.cs
- TextAnchor.cs
- SpellerHighlightLayer.cs
- storagemappingitemcollection.viewdictionary.cs
- ScriptResourceHandler.cs
- TextEndOfSegment.cs
- CompModSwitches.cs
- Switch.cs
- LeaseManager.cs
- WebPartEditVerb.cs
- AssemblyAttributes.cs
- SqlBuilder.cs
- BamlCollectionHolder.cs
- MobileListItemCollection.cs
- Int16AnimationUsingKeyFrames.cs
- AttachedAnnotation.cs
- Geometry.cs
- DataObjectPastingEventArgs.cs
- UInt64Converter.cs
- ListViewItemMouseHoverEvent.cs
- MainMenu.cs
- VisualTarget.cs
- SspiSecurityTokenParameters.cs
- ImageListUtils.cs
- ThreadStartException.cs
- Route.cs
- GridViewColumnCollection.cs
- ProfileParameter.cs
- ZipIOExtraFieldZip64Element.cs
- ConstructorBuilder.cs
- RadioButtonFlatAdapter.cs
- TimeoutException.cs
- XmlSchemaException.cs
- MissingManifestResourceException.cs
- LayoutTableCell.cs
- InputBindingCollection.cs
- NullNotAllowedCollection.cs
- AsyncMethodInvoker.cs
- DefaultBinder.cs
- BitmapImage.cs
- DragDropHelper.cs
- UrlMappingsModule.cs
- ResolveDuplexAsyncResult.cs
- TranslateTransform.cs
- UIElementHelper.cs
- TimeSpanMinutesOrInfiniteConverter.cs
- OdbcCommand.cs
- DataBindingList.cs
- WSFederationHttpBindingCollectionElement.cs
- basecomparevalidator.cs
- PersonalizationStateInfo.cs
- MouseGestureConverter.cs
- PointLight.cs
- PermissionListSet.cs
- WebReferencesBuildProvider.cs
- RectangleF.cs
- SupportsEventValidationAttribute.cs
- SqlInfoMessageEvent.cs
- ConfigXmlWhitespace.cs
- NumericExpr.cs
- ProfileManager.cs
- precedingquery.cs
- CodeDelegateInvokeExpression.cs
- ColorTranslator.cs
- WebPartTransformerAttribute.cs
- DataTableMapping.cs
- ListViewItem.cs
- DataGridItem.cs
- DescendantOverDescendantQuery.cs
- WebPartEditorCancelVerb.cs
- Transform3D.cs
- PostBackOptions.cs
- WindowsListViewGroupHelper.cs
- MarkupCompilePass1.cs
- EnumType.cs
- EtwTrace.cs