/* Copyright 2013-2015 MongoDB Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
using System;
namespace MongoDB.Bson.IO
{
///
/// Represents a source of chunks optimized for input buffers.
///
public sealed class InputBufferChunkSource : IBsonChunkSource
{
// constants
const int DefaultMaxChunkSize = 1 * 1024 * 1024;
const int DefaultMaxUnpooledChunkSize = 4 * 1024;
const int DefaultMinChunkSize = 16 * 1024;
// fields
private readonly IBsonChunkSource _baseSource;
private bool _disposed;
private readonly int _maxChunkSize;
private readonly int _maxUnpooledChunkSize;
private readonly int _minChunkSize;
// constructors
///
/// Initializes a new instance of the class.
///
/// The chunk source.
/// The maximum size of an unpooled chunk.
/// The minimum size of a chunk.
/// The maximum size of a chunk.
public InputBufferChunkSource(
IBsonChunkSource baseSource,
int maxUnpooledChunkSize = DefaultMaxUnpooledChunkSize,
int minChunkSize = DefaultMinChunkSize,
int maxChunkSize = DefaultMaxChunkSize)
{
if (baseSource == null)
{
throw new ArgumentNullException("baseSource");
}
if (maxUnpooledChunkSize < 0)
{
throw new ArgumentOutOfRangeException("maxUnpooledChunkSize");
}
if (minChunkSize <= 0)
{
throw new ArgumentOutOfRangeException("minChunkSize");
}
if (maxChunkSize <= 0)
{
throw new ArgumentOutOfRangeException("maxChunkSize");
}
if (!PowerOf2.IsPowerOf2(minChunkSize))
{
throw new ArgumentException("minChunkSize is not a power of 2.", "minChunkSize");
}
if (!PowerOf2.IsPowerOf2(maxChunkSize))
{
throw new ArgumentException("maxChunkSize is not a power of 2.", "maxChunkSize");
}
if (maxChunkSize < minChunkSize)
{
throw new ArgumentException("maxChunkSize is less than minChunkSize", "maxChunkSize");
}
_baseSource = baseSource;
_maxUnpooledChunkSize = maxUnpooledChunkSize;
_minChunkSize = minChunkSize;
_maxChunkSize = maxChunkSize;
}
// properties
///
/// Gets the base source.
///
///
/// The base source.
///
public IBsonChunkSource BaseSource
{
get
{
ThrowIfDisposed();
return _baseSource;
}
}
///
/// Gets the maximum size of a chunk.
///
///
/// The maximum size of a chunk.
///
public int MaxChunkSize
{
get { return _maxChunkSize; }
}
///
/// Gets the minimum size of a chunk.
///
///
/// The minimum size of a chunk.
///
public int MinChunkSize
{
get { return _minChunkSize; }
}
///
/// Gets the maximum size of an unpooled chunk.
///
///
/// The maximum size of an unpooled chunk.
///
public int MaxUnpooledChunkSize
{
get { return _maxUnpooledChunkSize; }
}
// methods
///
public void Dispose()
{
_disposed = true;
}
///
public IBsonChunk GetChunk(int requestedSize)
{
if (requestedSize <= 0)
{
throw new ArgumentOutOfRangeException("requestedSize");
}
ThrowIfDisposed();
if (requestedSize <= _maxUnpooledChunkSize)
{
return new ByteArrayChunk(requestedSize);
}
var powerOf2Size = PowerOf2.RoundUpToPowerOf2(requestedSize);
if (powerOf2Size - requestedSize > _minChunkSize)
{
powerOf2Size = powerOf2Size / 2;
}
var chunkSize = Math.Max(Math.Min(powerOf2Size, _maxChunkSize), _minChunkSize);
return _baseSource.GetChunk(chunkSize);
}
private void ThrowIfDisposed()
{
if (_disposed)
{
throw new ObjectDisposedException(GetType().Name);
}
}
}
}