/src/org/jitsi/impl/neomedia/jmfext/media/protocol/wasapi/DataSource.java

https://gitlab.com/bickelj/libjitsi · Java · 224 lines · 117 code · 23 blank · 84 comment · 12 complexity · 91dfa2c93cb118e50257f2f9cd31273b MD5 · raw file

  1. /*
  2. * Copyright @ 2015 Atlassian Pty Ltd
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. package org.jitsi.impl.neomedia.jmfext.media.protocol.wasapi;
  17. import java.io.*;
  18. import java.util.*;
  19. import javax.media.*;
  20. import javax.media.control.*;
  21. import javax.media.format.*;
  22. import org.jitsi.impl.neomedia.device.*;
  23. import org.jitsi.impl.neomedia.jmfext.media.protocol.*;
  24. import org.jitsi.util.*;
  25. /**
  26. * Implements <tt>CaptureDevice</tt> and <tt>DataSource</tt> using Windows Audio
  27. * Session API (WASAPI) and related Core Audio APIs such as Multimedia Device
  28. * (MMDevice) API.
  29. *
  30. * @author Lyubomir Marinov
  31. */
  32. public class DataSource
  33. extends AbstractPushBufferCaptureDevice
  34. {
  35. /**
  36. * The <tt>Logger</tt> used by the <tt>DataSource</tt> class and its
  37. * instances to log debugging information.
  38. */
  39. private static final Logger logger = Logger.getLogger(DataSource.class);
  40. /**
  41. * The indicator which determines whether the voice capture DMO is to be
  42. * used to perform echo cancellation and/or noise reduction.
  43. */
  44. final boolean aec;
  45. /**
  46. * The <tt>WASAPISystem</tt> which has contributed this
  47. * <tt>CaptureDevice</tt>/<tt>DataSource</tt>.
  48. */
  49. final WASAPISystem audioSystem;
  50. /**
  51. * Initializes a new <tt>DataSource</tt> instance.
  52. */
  53. public DataSource()
  54. {
  55. this(null);
  56. }
  57. /**
  58. * Initializes a new <tt>DataSource</tt> instance with a specific
  59. * <tt>MediaLocator</tt>.
  60. *
  61. * @param locator the <tt>MediaLocator</tt> to initialize the new instance
  62. * with
  63. */
  64. public DataSource(MediaLocator locator)
  65. {
  66. super(locator);
  67. audioSystem
  68. = (WASAPISystem)
  69. AudioSystem.getAudioSystem(AudioSystem.LOCATOR_PROTOCOL_WASAPI);
  70. aec = audioSystem.isDenoise() || audioSystem.isEchoCancel();
  71. }
  72. /**
  73. * {@inheritDoc}
  74. */
  75. @Override
  76. protected WASAPIStream createStream(
  77. int streamIndex,
  78. FormatControl formatControl)
  79. {
  80. return new WASAPIStream(this, formatControl);
  81. }
  82. /**
  83. * {@inheritDoc}
  84. */
  85. @Override
  86. protected void doConnect()
  87. throws IOException
  88. {
  89. super.doConnect();
  90. MediaLocator locator = getLocator();
  91. synchronized (getStreamSyncRoot())
  92. {
  93. for (Object stream : getStreams())
  94. ((WASAPIStream) stream).setLocator(locator);
  95. }
  96. }
  97. /**
  98. * {@inheritDoc}
  99. */
  100. @Override
  101. protected void doDisconnect()
  102. {
  103. try
  104. {
  105. synchronized (getStreamSyncRoot())
  106. {
  107. for (Object stream : getStreams())
  108. {
  109. try
  110. {
  111. ((WASAPIStream) stream).setLocator(null);
  112. }
  113. catch (IOException ioe)
  114. {
  115. logger.error(
  116. "Failed to disconnect "
  117. + stream.getClass().getName(),
  118. ioe);
  119. }
  120. }
  121. }
  122. }
  123. finally
  124. {
  125. super.doDisconnect();
  126. }
  127. }
  128. /**
  129. * Gets the <tt>Format</tt>s of media data supported by the audio endpoint
  130. * device associated with this instance.
  131. *
  132. * @return the <tt>Format</tt>s of media data supported by the audio
  133. * endpoint device associated with this instance
  134. */
  135. Format[] getIAudioClientSupportedFormats()
  136. {
  137. return getIAudioClientSupportedFormats(/* streamIndex */ 0);
  138. }
  139. /**
  140. * Gets the <tt>Format</tt>s of media data supported by the audio endpoint
  141. * device associated with this instance.
  142. *
  143. * @param streamIndex the index of the <tt>SourceStream</tt> within the list
  144. * of <tt>SourceStream</tt>s of this <tt>DataSource</tt> on behalf of which
  145. * the query is being made
  146. * @return the <tt>Format</tt>s of media data supported by the audio
  147. * endpoint device associated with this instance
  148. */
  149. private Format[] getIAudioClientSupportedFormats(int streamIndex)
  150. {
  151. Format[] superSupportedFormats = super.getSupportedFormats(streamIndex);
  152. /*
  153. * If the capture endpoint device reports to support no Format, then
  154. * acoustic echo cancellation (AEC) will surely not work.
  155. */
  156. if ((superSupportedFormats == null)
  157. || (superSupportedFormats.length == 0))
  158. return superSupportedFormats;
  159. // Return the NativelySupportedAudioFormat instances only.
  160. List<Format> supportedFormats
  161. = new ArrayList<Format>(superSupportedFormats.length);
  162. for (Format format : superSupportedFormats)
  163. {
  164. if ((format instanceof NativelySupportedAudioFormat)
  165. && !supportedFormats.contains(format))
  166. {
  167. supportedFormats.add(format);
  168. }
  169. }
  170. int supportedFormatCount = supportedFormats.size();
  171. return
  172. (supportedFormatCount == superSupportedFormats.length)
  173. ? superSupportedFormats
  174. : supportedFormats.toArray(new Format[supportedFormatCount]);
  175. }
  176. /**
  177. * {@inheritDoc}
  178. *
  179. * The <tt>Format</tt>s supported by this
  180. * <tt>CaptureDevice</tt>/<tt>DataSource</tt> are either the ones supported
  181. * by the capture endpoint device or the ones supported by the voice capture
  182. * DMO that implements the acoustic echo cancellation (AEC) feature
  183. * depending on whether the feature in question is disabled or enabled.
  184. */
  185. @Override
  186. protected Format[] getSupportedFormats(int streamIndex)
  187. {
  188. if (aec)
  189. {
  190. List<AudioFormat> aecSupportedFormats
  191. = audioSystem.getAECSupportedFormats();
  192. return
  193. aecSupportedFormats.toArray(
  194. new Format[aecSupportedFormats.size()]);
  195. }
  196. else
  197. {
  198. return getIAudioClientSupportedFormats(streamIndex);
  199. }
  200. }
  201. }