using System;
using System.Collections;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
namespace Unity.MLAgents.Actuators
{
///
/// ActionSegment{T} is a data structure that allows access to a segment of an underlying array
/// in order to avoid the copying and allocation of sub-arrays. The segment is defined by
/// the offset into the original array, and an length.
///
/// The type of object stored in the underlying
public readonly struct ActionSegment : IEnumerable, IEquatable>
where T : struct
{
///
/// The zero-based offset into the original array at which this segment starts.
///
public readonly int Offset;
///
/// The number of items this segment can access in the underlying array.
///
public readonly int Length;
///
/// An Empty segment which has an offset of 0, a Length of 0, and it's underlying array
/// is also empty.
///
public static ActionSegment Empty = new ActionSegment(System.Array.Empty(), 0, 0);
static void CheckParameters(IReadOnlyCollection actionArray, int offset, int length)
{
#if DEBUG
if (offset + length > actionArray.Count)
{
throw new ArgumentOutOfRangeException(nameof(offset),
$"Arguments offset: {offset} and length: {length} " +
$"are out of bounds of actionArray: {actionArray.Count}.");
}
#endif
}
///
/// Construct an with just an actionArray. The will
/// be set to 0 and the will be set to `actionArray.Length`.
///
/// The action array to use for the this segment.
public ActionSegment(T[] actionArray)
: this(actionArray ?? System.Array.Empty(), 0, actionArray?.Length ?? 0) { }
///
/// Construct an with an underlying array
/// and offset, and a length.
///
/// The underlying array which this segment has a view into
/// The zero-based offset into the underlying array.
/// The length of the segment.
public ActionSegment(T[] actionArray, int offset, int length)
{
#if DEBUG
CheckParameters(actionArray ?? System.Array.Empty(), offset, length);
#endif
Array = actionArray ?? System.Array.Empty();
Offset = offset;
Length = length;
}
///
/// Get the underlying of this segment.
///
public T[] Array { get; }
///
/// Allows access to the underlying array using array syntax.
///
/// The zero-based index of the segment.
/// Thrown when the index is less than 0 or
/// greater than or equal to
public T this[int index]
{
get
{
if (index < 0 || index > Length)
{
throw new IndexOutOfRangeException($"Index out of bounds, expected a number between 0 and {Length}");
}
return Array[Offset + index];
}
set
{
if (index < 0 || index > Length)
{
throw new IndexOutOfRangeException($"Index out of bounds, expected a number between 0 and {Length}");
}
Array[Offset + index] = value;
}
}
///
/// Sets the segment of the backing array to all zeros.
///
public void Clear()
{
System.Array.Clear(Array, Offset, Length);
}
///
/// Check if the segment is empty.
///
/// Whether or not the segment is empty.
public bool IsEmpty()
{
return Array == null || Array.Length == 0;
}
///
/// Returns an enumerator that iterates through the ActionSegment.
///
/// An IEnumerator object that can be used to iterate through the ActionSegment.
IEnumerator IEnumerable.GetEnumerator()
{
return new Enumerator(this);
}
///
/// Returns an enumerator that iterates through the ActionSegment.
///
/// An IEnumerator object that can be used to iterate through the ActionSegment.
public IEnumerator GetEnumerator()
{
return new Enumerator(this);
}
///
/// Indicates whether the current ActionSegment is equal to another ActionSegment.
///
/// An ActionSegment to compare with this ActionSegment.
/// true if the current ActionSegment is equal to the other parameter; otherwise, false.
public override bool Equals(object obj)
{
if (!(obj is ActionSegment))
{
return false;
}
return Equals((ActionSegment)obj);
}
///
/// Indicates whether the current ActionSegment is equal to another ActionSegment.
///
/// An ActionSegment to compare with this ActionSegment.
/// true if the current ActionSegment is equal to the other parameter; otherwise, false.
public bool Equals(ActionSegment other)
{
return Offset == other.Offset && Length == other.Length && Array.SequenceEqual(other.Array);
}
///
/// Computes the hash code of the ActionSegment.
///
/// A hash code for the current ActionSegment.
public override int GetHashCode()
{
unchecked
{
var hashCode = Offset;
hashCode = (hashCode * 397) ^ Length;
hashCode = (hashCode * 397) ^ (Array != null ? Array.GetHashCode() : 0);
return hashCode;
}
}
///
/// A private for the value type which follows its
/// rules of being a view into an underlying .
///
struct Enumerator : IEnumerator
{
readonly T[] m_Array;
readonly int m_Start;
readonly int m_End; // cache Offset + Count, since it's a little slow
int m_Current;
internal Enumerator(ActionSegment arraySegment)
{
Debug.Assert(arraySegment.Array != null);
Debug.Assert(arraySegment.Offset >= 0);
Debug.Assert(arraySegment.Length >= 0);
Debug.Assert(arraySegment.Offset + arraySegment.Length <= arraySegment.Array.Length);
m_Array = arraySegment.Array;
m_Start = arraySegment.Offset;
m_End = arraySegment.Offset + arraySegment.Length;
m_Current = arraySegment.Offset - 1;
}
public bool MoveNext()
{
if (m_Current < m_End)
{
m_Current++;
return m_Current < m_End;
}
return false;
}
public T Current
{
get
{
if (m_Current < m_Start)
throw new InvalidOperationException("Enumerator not started.");
if (m_Current >= m_End)
throw new InvalidOperationException("Enumerator has reached the end already.");
return m_Array[m_Current];
}
}
object IEnumerator.Current => Current;
void IEnumerator.Reset()
{
m_Current = m_Start - 1;
}
public void Dispose()
{
}
}
}
}