 4c2ab880ef
			
		
	
	
		4c2ab880ef
		
			
		
	
	
	
	
		
			
			* Ryujinx.Audio: Remove BOM from files * misc: Relicense Ryujinx.Audio under the terms of the MIT license With the approvals of all the Ryujinx.Audio contributors, this commit changes Ryujinx.Audio license from LGPLv3 to MIT.
		
			
				
	
	
		
			296 lines
		
	
	
	
		
			9.6 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			296 lines
		
	
	
	
		
			9.6 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using Ryujinx.Audio.Common;
 | |
| using Ryujinx.Audio.Integration;
 | |
| using Ryujinx.Common.Logging;
 | |
| using Ryujinx.Memory;
 | |
| using System;
 | |
| using System.Diagnostics;
 | |
| using System.Linq;
 | |
| using System.Threading;
 | |
| 
 | |
| namespace Ryujinx.Audio.Output
 | |
| {
 | |
|     /// <summary>
 | |
|     /// The audio output manager.
 | |
|     /// </summary>
 | |
|     public class AudioOutputManager : IDisposable
 | |
|     {
 | |
|         private object _lock = new object();
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Lock used for session allocation.
 | |
|         /// </summary>
 | |
|         private object _sessionLock = new object();
 | |
| 
 | |
|         /// <summary>
 | |
|         /// The session ids allocation table.
 | |
|         /// </summary>
 | |
|         private int[] _sessionIds;
 | |
| 
 | |
|         /// <summary>
 | |
|         /// The device driver.
 | |
|         /// </summary>
 | |
|         private IHardwareDeviceDriver _deviceDriver;
 | |
| 
 | |
|         /// <summary>
 | |
|         /// The events linked to each session.
 | |
|         /// </summary>
 | |
|         private IWritableEvent[] _sessionsBufferEvents;
 | |
| 
 | |
|         /// <summary>
 | |
|         /// The <see cref="AudioOutputSystem"/> session instances.
 | |
|         /// </summary>
 | |
|         private AudioOutputSystem[] _sessions;
 | |
| 
 | |
|         /// <summary>
 | |
|         /// The count of active sessions.
 | |
|         /// </summary>
 | |
|         private int _activeSessionCount;
 | |
| 
 | |
|         /// <summary>
 | |
|         /// The dispose state.
 | |
|         /// </summary>
 | |
|         private int _disposeState;
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Create a new <see cref="AudioOutputManager"/>.
 | |
|         /// </summary>
 | |
|         public AudioOutputManager()
 | |
|         {
 | |
|             _sessionIds = new int[Constants.AudioOutSessionCountMax];
 | |
|             _sessions = new AudioOutputSystem[Constants.AudioOutSessionCountMax];
 | |
|             _activeSessionCount = 0;
 | |
| 
 | |
|             for (int i = 0; i < _sessionIds.Length; i++)
 | |
|             {
 | |
|                 _sessionIds[i] = i;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Initialize the <see cref="AudioOutputManager"/>.
 | |
|         /// </summary>
 | |
|         /// <param name="deviceDriver">The device driver.</param>
 | |
|         /// <param name="sessionRegisterEvents">The events associated to each session.</param>
 | |
|         public void Initialize(IHardwareDeviceDriver deviceDriver, IWritableEvent[] sessionRegisterEvents)
 | |
|         {
 | |
|             _deviceDriver = deviceDriver;
 | |
|             _sessionsBufferEvents = sessionRegisterEvents;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Acquire a new session id.
 | |
|         /// </summary>
 | |
|         /// <returns>A new session id.</returns>
 | |
|         private int AcquireSessionId()
 | |
|         {
 | |
|             lock (_sessionLock)
 | |
|             {
 | |
|                 int index = _activeSessionCount;
 | |
| 
 | |
|                 Debug.Assert(index < _sessionIds.Length);
 | |
| 
 | |
|                 int sessionId = _sessionIds[index];
 | |
| 
 | |
|                 _sessionIds[index] = -1;
 | |
| 
 | |
|                 _activeSessionCount++;
 | |
| 
 | |
|                 Logger.Info?.Print(LogClass.AudioRenderer, $"Registered new output ({sessionId})");
 | |
| 
 | |
|                 return sessionId;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Release a given <paramref name="sessionId"/>.
 | |
|         /// </summary>
 | |
|         /// <param name="sessionId">The session id to release.</param>
 | |
|         private void ReleaseSessionId(int sessionId)
 | |
|         {
 | |
|             lock (_sessionLock)
 | |
|             {
 | |
|                 Debug.Assert(_activeSessionCount > 0);
 | |
| 
 | |
|                 int newIndex = --_activeSessionCount;
 | |
| 
 | |
|                 _sessionIds[newIndex] = sessionId;
 | |
|             }
 | |
| 
 | |
|             Logger.Info?.Print(LogClass.AudioRenderer, $"Unregistered output ({sessionId})");
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Used to update audio output system.
 | |
|         /// </summary>
 | |
|         public void Update()
 | |
|         {
 | |
|             lock (_sessionLock)
 | |
|             {
 | |
|                 foreach (AudioOutputSystem output in _sessions)
 | |
|                 {
 | |
|                     output?.Update();
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Register a new <see cref="AudioOutputSystem"/>.
 | |
|         /// </summary>
 | |
|         /// <param name="output">The <see cref="AudioOutputSystem"/> to register.</param>
 | |
|         private void Register(AudioOutputSystem output)
 | |
|         {
 | |
|             lock (_sessionLock)
 | |
|             {
 | |
|                 _sessions[output.GetSessionId()] = output;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Unregister a new <see cref="AudioOutputSystem"/>.
 | |
|         /// </summary>
 | |
|         /// <param name="output">The <see cref="AudioOutputSystem"/> to unregister.</param>
 | |
|         internal void Unregister(AudioOutputSystem output)
 | |
|         {
 | |
|             lock (_sessionLock)
 | |
|             {
 | |
|                 int sessionId = output.GetSessionId();
 | |
| 
 | |
|                 _sessions[output.GetSessionId()] = null;
 | |
| 
 | |
|                 ReleaseSessionId(sessionId);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Get the list of all audio outputs name.
 | |
|         /// </summary>
 | |
|         /// <returns>The list of all audio outputs name</returns>
 | |
|         public string[] ListAudioOuts()
 | |
|         {
 | |
|             return new string[] { Constants.DefaultDeviceOutputName };
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Open a new <see cref="AudioOutputSystem"/>.
 | |
|         /// </summary>
 | |
|         /// <param name="outputDeviceName">The output device name selected by the <see cref="AudioOutputSystem"/></param>
 | |
|         /// <param name="outputConfiguration">The output audio configuration selected by the <see cref="AudioOutputSystem"/></param>
 | |
|         /// <param name="obj">The new <see cref="AudioOutputSystem"/></param>
 | |
|         /// <param name="memoryManager">The memory manager that will be used for all guest memory operations</param>
 | |
|         /// <param name="inputDeviceName">The input device name wanted by the user</param>
 | |
|         /// <param name="sampleFormat">The sample format to use</param>
 | |
|         /// <param name="parameter">The user configuration</param>
 | |
|         /// <param name="appletResourceUserId">The applet resource user id of the application</param>
 | |
|         /// <param name="processHandle">The process handle of the application</param>
 | |
|         /// <returns>A <see cref="ResultCode"/> reporting an error or a success</returns>
 | |
|         public ResultCode OpenAudioOut(out string outputDeviceName,
 | |
|                                        out AudioOutputConfiguration outputConfiguration,
 | |
|                                        out AudioOutputSystem obj,
 | |
|                                        IVirtualMemoryManager memoryManager,
 | |
|                                        string inputDeviceName,
 | |
|                                        SampleFormat sampleFormat,
 | |
|                                        ref AudioInputConfiguration parameter,
 | |
|                                        ulong appletResourceUserId,
 | |
|                                        uint processHandle,
 | |
|                                        float volume)
 | |
|         {
 | |
|             int sessionId = AcquireSessionId();
 | |
| 
 | |
|             _sessionsBufferEvents[sessionId].Clear();
 | |
| 
 | |
|             IHardwareDeviceSession deviceSession = _deviceDriver.OpenDeviceSession(IHardwareDeviceDriver.Direction.Output, memoryManager, sampleFormat, parameter.SampleRate, parameter.ChannelCount, volume);
 | |
| 
 | |
|             AudioOutputSystem audioOut = new AudioOutputSystem(this, _lock, deviceSession, _sessionsBufferEvents[sessionId]);
 | |
| 
 | |
|             ResultCode result = audioOut.Initialize(inputDeviceName, sampleFormat, ref parameter, sessionId);
 | |
| 
 | |
|             if (result == ResultCode.Success)
 | |
|             {
 | |
|                 outputDeviceName = audioOut.DeviceName;
 | |
|                 outputConfiguration = new AudioOutputConfiguration
 | |
|                 {
 | |
|                     ChannelCount  = audioOut.ChannelCount,
 | |
|                     SampleFormat  = audioOut.SampleFormat,
 | |
|                     SampleRate    = audioOut.SampleRate,
 | |
|                     AudioOutState = audioOut.GetState(),
 | |
|                 };
 | |
| 
 | |
|                 obj = audioOut;
 | |
| 
 | |
|                 Register(audioOut);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 ReleaseSessionId(sessionId);
 | |
| 
 | |
|                 obj = null;
 | |
|                 outputDeviceName = null;
 | |
|                 outputConfiguration = default;
 | |
|             }
 | |
| 
 | |
|             return result;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Sets the volume for all output devices.
 | |
|         /// </summary>
 | |
|         /// <param name="volume">The volume to set.</param>
 | |
|         public void SetVolume(float volume)
 | |
|         {
 | |
|             if (_sessions != null)
 | |
|             {
 | |
|                 foreach (AudioOutputSystem session in _sessions)
 | |
|                 {
 | |
|                     session?.SetVolume(volume);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Gets the volume for all output devices.
 | |
|         /// </summary>
 | |
|         /// <returns>A float indicating the volume level.</returns>
 | |
|         public float GetVolume()
 | |
|         {
 | |
|             if (_sessions != null)
 | |
|             {
 | |
|                 foreach (AudioOutputSystem session in _sessions)
 | |
|                 {
 | |
|                     if (session != null)
 | |
|                     {
 | |
|                         return session.GetVolume();
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return 0.0f;
 | |
|         }
 | |
| 
 | |
|         public void Dispose()
 | |
|         {
 | |
|             if (Interlocked.CompareExchange(ref _disposeState, 1, 0) == 0)
 | |
|             {
 | |
|                 Dispose(true);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         protected virtual void Dispose(bool disposing)
 | |
|         {
 | |
|             if (disposing)
 | |
|             {
 | |
|                 // Clone the sessions array to dispose them outside the lock.
 | |
|                 AudioOutputSystem[] sessions;
 | |
| 
 | |
|                 lock (_sessionLock)
 | |
|                 {
 | |
|                     sessions = _sessions.ToArray();
 | |
|                 }
 | |
| 
 | |
|                 foreach (AudioOutputSystem output in sessions)
 | |
|                 {
 | |
|                     output?.Dispose();
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 |