using UnityEngine; using UnityEngine.Rendering; namespace Unity.MLAgents.Sensors { /// /// A sensor that wraps a Camera object to generate visual observations for an agent. /// public class CameraSensor : ISensor, IBuiltInSensor, IDimensionPropertiesSensor { Camera m_Camera; int m_Width; int m_Height; bool m_Grayscale; string m_Name; //int[] m_Shape; private ObservationSpec m_ObservationSpec; SensorCompressionType m_CompressionType; static DimensionProperty[] s_DimensionProperties = new DimensionProperty[] { DimensionProperty.TranslationalEquivariance, DimensionProperty.TranslationalEquivariance, DimensionProperty.None }; /// /// The Camera used for rendering the sensor observations. /// public Camera Camera { get { return m_Camera; } set { m_Camera = value; } } /// /// The compression type used by the sensor. /// public SensorCompressionType CompressionType { get { return m_CompressionType; } set { m_CompressionType = value; } } /// /// Creates and returns the camera sensor. /// /// Camera object to capture images from. /// The width of the generated visual observation. /// The height of the generated visual observation. /// Whether to convert the generated image to grayscale or keep color. /// The name of the camera sensor. /// The compression to apply to the generated image. public CameraSensor( Camera camera, int width, int height, bool grayscale, string name, SensorCompressionType compression) { m_Camera = camera; m_Width = width; m_Height = height; m_Grayscale = grayscale; m_Name = name; m_ObservationSpec = ObservationSpec.FromShape(GenerateShape(width, height, grayscale)); m_CompressionType = compression; } /// /// Accessor for the name of the sensor. /// /// Sensor name. public string GetName() { return m_Name; } /// public ObservationSpec GetObservationSpec() { return m_ObservationSpec; } /// /// Accessor for the dimension properties of a camera sensor. A camera sensor /// Has translational equivariance along width and hight and no property along /// the channels dimension. /// /// public DimensionProperty[] GetDimensionProperties() { return s_DimensionProperties; } /// /// Generates a compressed image. This can be valuable in speeding-up training. /// /// Compressed image. public byte[] GetCompressedObservation() { using (TimerStack.Instance.Scoped("CameraSensor.GetCompressedObservation")) { var texture = ObservationToTexture(m_Camera, m_Width, m_Height); // TODO support more types here, e.g. JPG var compressed = texture.EncodeToPNG(); DestroyTexture(texture); return compressed; } } /// /// Writes out the generated, uncompressed image to the provided . /// /// Where the observation is written to. /// public int Write(ObservationWriter writer) { using (TimerStack.Instance.Scoped("CameraSensor.WriteToTensor")) { var texture = ObservationToTexture(m_Camera, m_Width, m_Height); var numWritten = writer.WriteTexture(texture, m_Grayscale); DestroyTexture(texture); return numWritten; } } /// public void Update() { } /// public void Reset() { } /// public SensorCompressionType GetCompressionType() { return m_CompressionType; } /// /// Renders a Camera instance to a 2D texture at the corresponding resolution. /// /// The 2D texture. /// Camera. /// Width of resulting 2D texture. /// Height of resulting 2D texture. /// Texture2D to render to. public static Texture2D ObservationToTexture(Camera obsCamera, int width, int height) { if (SystemInfo.graphicsDeviceType == GraphicsDeviceType.Null) { Debug.LogError("GraphicsDeviceType is Null. This will likely crash when trying to render."); } var texture2D = new Texture2D(width, height, TextureFormat.RGB24, false); var oldRec = obsCamera.rect; obsCamera.rect = new Rect(0f, 0f, 1f, 1f); var depth = 24; var format = RenderTextureFormat.Default; var readWrite = RenderTextureReadWrite.Default; var tempRt = RenderTexture.GetTemporary(width, height, depth, format, readWrite); var prevActiveRt = RenderTexture.active; var prevCameraRt = obsCamera.targetTexture; // render to offscreen texture (readonly from CPU side) RenderTexture.active = tempRt; obsCamera.targetTexture = tempRt; obsCamera.Render(); texture2D.ReadPixels(new Rect(0, 0, texture2D.width, texture2D.height), 0, 0); obsCamera.targetTexture = prevCameraRt; obsCamera.rect = oldRec; RenderTexture.active = prevActiveRt; RenderTexture.ReleaseTemporary(tempRt); return texture2D; } /// /// Computes the observation shape for a camera sensor based on the height, width /// and grayscale flag. /// /// Width of the image captures from the camera. /// Height of the image captures from the camera. /// Whether or not to convert the image to grayscale. /// The observation shape. internal static int[] GenerateShape(int width, int height, bool grayscale) { return new[] { height, width, grayscale ? 1 : 3 }; } static void DestroyTexture(Texture2D texture) { if (Application.isEditor) { // Edit Mode tests complain if we use Destroy() // TODO move to extension methods for UnityEngine.Object? Object.DestroyImmediate(texture); } else { Object.Destroy(texture); } } /// public BuiltInSensorType GetBuiltInSensorType() { return BuiltInSensorType.CameraSensor; } } }