using System;
using Unity.Simulation;
using UnityEngine;
using UnityEngine.Perception.GroundTruth.DataModel;
#pragma warning disable 649
namespace UnityEngine.Perception.GroundTruth
{
///
/// Global manager for frame scheduling and output capture for simulations.
/// Data capture follows the schema defined in *TODO: Expose schema publicly*
///
public class DatasetCapture : MonoBehaviour
{
public static DatasetCapture Instance { get; protected set; }
public ConsumerEndpoint activeConsumer;
SimulationState m_SimulationState;
void Awake()
{
if (Instance != null && Instance != this)
{
Destroy(this);
Debug.LogError($"The simulation started with more than one instance of DatasetCapture, destroying this one");
}
else
{
Instance = this;
}
}
internal SimulationState simulationState
{
get { return m_SimulationState ?? (m_SimulationState = CreateSimulationData()); }
private set => m_SimulationState = value;
}
///
/// The json metadata schema version the DatasetCapture's output conforms to.
///
public static string SchemaVersion => "0.0.1";
///
/// Called when the simulation ends. The simulation ends on playmode exit, application exit, or when is called.
///
public event Action SimulationEnding;
public SensorHandle RegisterSensor(SensorDefinition sensor)
{
return simulationState.AddSensor(sensor, sensor.simulationDeltaTime);
}
public void RegisterMetric(MetricDefinition metricDefinition)
{
simulationState.RegisterMetric(metricDefinition);
}
public void RegisterAnnotationDefinition(AnnotationDefinition definition)
{
simulationState.RegisterAnnotationDefinition(definition);
}
///
/// Starts a new sequence in the capture.
///
public void StartNewSequence() => simulationState.StartNewSequence();
internal bool IsValid(string id) => simulationState.Contains(id);
SimulationState CreateSimulationData()
{
return new SimulationState();
}
[RuntimeInitializeOnLoadMethod]
void OnInitializeOnLoad()
{
Manager.Instance.ShutdownNotification += ResetSimulation;
}
///
/// Stop the current simulation and start a new one. All pending data is written to disk before returning.
///
public void ResetSimulation()
{
//this order ensures that exceptions thrown by End() do not prevent the state from being reset
SimulationEnding?.Invoke();
var oldSimulationState = simulationState;
simulationState = CreateSimulationData();
oldSimulationState.End();
}
}
///
/// Capture trigger modes for sensors.
///
public enum CaptureTriggerMode
{
///
/// Captures happen automatically based on a start frame and frame delta time.
///
Scheduled,
///
/// Captures should be triggered manually through calling the manual capture method of the sensor using this trigger mode.
///
Manual
}
public enum FutureType
{
Sensor,
Annotation,
Metric
}
public interface IAsyncFuture where T : SimulationState.IPendingId
{
T GetId();
FutureType GetFutureType();
bool IsPending();
}
public struct AsyncSensorFuture : IAsyncFuture
{
public AsyncSensorFuture(SimulationState.SPendingSensorId id, SimulationState simulationState)
{
m_Id = id;
m_SimulationState = simulationState;
}
SimulationState.SPendingSensorId m_Id;
SimulationState m_SimulationState;
public SimulationState.SPendingSensorId GetId()
{
return m_Id;
}
public FutureType GetFutureType()
{
return FutureType.Sensor;
}
public bool IsPending()
{
return m_SimulationState.IsPending(this);
}
public void Report(Sensor sensor)
{
m_SimulationState.ReportAsyncResult(this, sensor);
}
}
public struct AsyncAnnotationFuture : IAsyncFuture
{
public AsyncAnnotationFuture(SimulationState.SPendingCaptureId id, SimulationState simulationState)
{
m_Id = id;
m_SimulationState = simulationState;
}
SimulationState.SPendingCaptureId m_Id;
SimulationState m_SimulationState;
public SimulationState.SPendingCaptureId GetId()
{
return m_Id;
}
public FutureType GetFutureType()
{
return FutureType.Annotation;
}
public bool IsPending()
{
return m_SimulationState.IsPending(this);
}
public void Report(Annotation annotation)
{
m_SimulationState.ReportAsyncResult(this, annotation);
}
}
public struct AsyncMetricFuture : IAsyncFuture
{
public AsyncMetricFuture(SimulationState.SPendingCaptureId id, SimulationState simulationState)
{
m_Id = id;
m_SimulationState = simulationState;
}
SimulationState.SPendingCaptureId m_Id;
SimulationState m_SimulationState;
public SimulationState.SPendingCaptureId GetId()
{
return m_Id;
}
public FutureType GetFutureType()
{
return FutureType.Metric;
}
public bool IsPending()
{
return m_SimulationState.IsPending(this);
}
public void Report(Metric metric)
{
m_SimulationState.ReportAsyncResult(this, metric);
}
}
///
/// A handle to a sensor managed by the . It can be used to check whether the sensor
/// is expected to capture this frame and report captures, annotations, and metrics regarding the sensor.
///
public struct SensorHandle : IDisposable, IEquatable
{
public string Id { get; internal set; }
internal SensorHandle(string id)
{
Id = id ?? string.Empty;
}
///
/// Whether the sensor is currently enabled. When disabled, the DatasetCapture will no longer schedule frames for running captures on this sensor.
///
public bool Enabled
{
get => DatasetCapture.Instance.simulationState.IsEnabled(this);
set
{
CheckValid();
DatasetCapture.Instance.simulationState.SetEnabled(this, value);
}
}
public void ReportAnnotation(AnnotationDefinition definition, Annotation annotation)
{
if (!ShouldCaptureThisFrame)
throw new InvalidOperationException("Annotation reported on SensorHandle in frame when its ShouldCaptureThisFrame is false.");
if (!definition.IsValid())
throw new ArgumentException("The given annotationDefinition is invalid", nameof(definition));
DatasetCapture.Instance.simulationState.ReportAnnotation(this, definition, annotation);
}
///
/// Creates an async annotation for reporting the values for an annotation during a future frame.
///
/// The AnnotationDefinition of this annotation.
/// Returns a handle to the , which can be used to report annotation data during a subsequent frame.
/// Thrown if this method is called during a frame where is false.
/// Thrown if the given AnnotationDefinition is invalid.
public AsyncAnnotationFuture ReportAnnotationAsync(AnnotationDefinition annotationDefinition)
{
if (!ShouldCaptureThisFrame)
throw new InvalidOperationException("Annotation reported on SensorHandle in frame when its ShouldCaptureThisFrame is false.");
if (!annotationDefinition.IsValid())
throw new ArgumentException("The given annotationDefinition is invalid", nameof(annotationDefinition));
return DatasetCapture.Instance.simulationState.ReportAnnotationAsync(annotationDefinition, this);
}
public AsyncSensorFuture ReportSensorAsync(SensorDefinition sensorDefinition)
{
if (!ShouldCaptureThisFrame)
throw new InvalidOperationException("Annotation reported on SensorHandle in frame when its ShouldCaptureThisFrame is false.");
if (!sensorDefinition.IsValid())
throw new ArgumentException("The given annotationDefinition is invalid", nameof(sensorDefinition));
return DatasetCapture.Instance.simulationState.ReportSensorAsync(sensorDefinition);
}
public void ReportSensor(SensorDefinition definition, Sensor sensor)
{
if (!ShouldCaptureThisFrame)
throw new InvalidOperationException("Annotation reported on SensorHandle in frame when its ShouldCaptureThisFrame is false.");
if (!definition.IsValid())
throw new ArgumentException("The given annotationDefinition is invalid", nameof(definition));
DatasetCapture.Instance.simulationState.ReportSensor(definition, sensor);
}
///
/// Whether the sensor should capture this frame. Sensors are expected to call this method each frame to determine whether
/// they should capture during the frame. Captures should only be reported when this is true.
///
public bool ShouldCaptureThisFrame => DatasetCapture.Instance.simulationState.ShouldCaptureThisFrame(this);
///
/// Requests a capture from this sensor on the next rendered frame. Can only be used with manual capture mode ().
///
public void RequestCapture()
{
DatasetCapture.Instance.simulationState.SetNextCaptureTimeToNowForSensor(this);
}
#if false
public MetricHandle ReportMetric(MetricDefinition definition, Metric metric)
{
if (metric == null)
throw new ArgumentNullException(nameof(metric));
if (!ShouldCaptureThisFrame)
throw new InvalidOperationException($"Sensor-based metrics may only be reported when SensorHandle.ShouldCaptureThisFrame is true");
return DatasetCapture.Instance.simulationState.ReportMetric(this, definition, metric, default);
}
#endif
///
/// Start an async metric for reporting metric values for this frame in a subsequent frame.
///
/// The of the metric
/// Thrown if is false
/// An which should be used to report the metric values, potentially in a later frame
public AsyncMetricFuture ReportMetricAsync(MetricDefinition metricDefinition)
{
if (!ShouldCaptureThisFrame)
throw new InvalidOperationException($"Sensor-based metrics may only be reported when SensorHandle.ShouldCaptureThisFrame is true");
if (!metricDefinition.IsValid())
throw new ArgumentException("The passed in metric definition is invalid", nameof(metricDefinition));
return DatasetCapture.Instance.simulationState.CreateAsyncMetric(metricDefinition, this);
}
///
/// Dispose this SensorHandle.
///
public void Dispose()
{
this.Enabled = false;
}
///
/// Returns whether this SensorHandle is valid in the current simulation. Nil SensorHandles are never valid.
///
public bool IsValid => DatasetCapture.Instance.IsValid(this.Id);
///
/// Returns true if this SensorHandle was default-instantiated.
///
public bool IsNil => this == default;
void CheckValid()
{
if (!DatasetCapture.Instance.IsValid(this.Id))
throw new InvalidOperationException("SensorHandle has been disposed or its simulation has ended");
}
///
public bool Equals(SensorHandle other)
{
switch (Id)
{
case null when other.Id == null:
return true;
case null:
return false;
default:
return Id.Equals(other.Id);
}
}
///
public override bool Equals(object obj)
{
return obj is SensorHandle other && Equals(other);
}
///
public override int GetHashCode()
{
return Id.GetHashCode();
}
///
/// Compares two instances for equality.
///
/// The first SensorHandle.
/// The second SensorHandle.
/// Returns true if the two SensorHandles refer to the same sensor.
public static bool operator==(SensorHandle left, SensorHandle right)
{
return left.Equals(right);
}
///
/// Compares two instances for inequality.
///
/// The first SensorHandle.
/// The second SensorHandle.
/// Returns false if the two SensorHandles refer to the same sensor.
public static bool operator!=(SensorHandle left, SensorHandle right)
{
return !left.Equals(right);
}
}
///
/// A handle to an annotation. Can be used to report metrics on the annotation.
///
public readonly struct AnnotationHandle : IEquatable
{
readonly AnnotationDefinition m_Definition;
///
/// The ID of the annotation which will be used in the json metadata.
///
public string Id => m_Definition.id;
///
/// The SensorHandle on which the annotation was reported
///
public readonly SensorHandle SensorHandle;
internal AnnotationHandle(SensorHandle sensorHandle, AnnotationDefinition definition, int sequence, int step)
{
m_Definition = definition;
SensorHandle = sensorHandle;
}
///
/// Returns true if the annotation is nil (created using default instantiation).
///
public bool IsNil => Id == string.Empty;
///
public bool Equals(AnnotationHandle other)
{
return SensorHandle.Equals(other.SensorHandle) && m_Definition.Equals(other.m_Definition);
}
///
public override bool Equals(object obj)
{
return obj is AnnotationHandle other && Equals(other);
}
///
public override int GetHashCode()
{
var hash = (Id != null ? StringComparer.InvariantCulture.GetHashCode(Id) : 0);
return hash;
}
}
}