VirtualBox

source: vbox/trunk/src/VBox/Devices/Audio/DrvHostAudioOss.cpp@ 88574

Last change on this file since 88574 was 88534, checked in by vboxsync, 4 years ago

Audio: Merged the cbStreamOut and cbStreamIn fields in PDMAUDIOBACKENDCFG (into cbStream). Added a fFlags member to PDMAUDIOBACKENDCFG, currently must-be-zero. bugref:9890

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 32.2 KB
Line 
1/* $Id: DrvHostAudioOss.cpp 88534 2021-04-15 12:16:56Z vboxsync $ */
2/** @file
3 * Host audio driver - OSS (Open Sound System).
4 */
5
6/*
7 * Copyright (C) 2014-2020 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#include <errno.h>
23#include <fcntl.h>
24#include <sys/ioctl.h>
25#include <sys/mman.h>
26#include <sys/soundcard.h>
27#include <unistd.h>
28
29#include <iprt/alloc.h>
30#include <iprt/uuid.h> /* For PDMIBASE_2_PDMDRV. */
31
32#define LOG_GROUP LOG_GROUP_DRV_HOST_AUDIO
33#include <VBox/log.h>
34#include <VBox/vmm/pdmaudioifs.h>
35#include <VBox/vmm/pdmaudioinline.h>
36
37#include "VBoxDD.h"
38
39
40/*********************************************************************************************************************************
41* Defines *
42*********************************************************************************************************************************/
43#if ((SOUND_VERSION > 360) && (defined(OSS_SYSINFO)))
44/* OSS > 3.6 has a new syscall available for querying a bit more detailed information
45 * about OSS' audio capabilities. This is handy for e.g. Solaris. */
46# define VBOX_WITH_AUDIO_OSS_SYSINFO 1
47#endif
48
49
50/*********************************************************************************************************************************
51* Structures *
52*********************************************************************************************************************************/
53/**
54 * OSS host audio driver instance data.
55 * @implements PDMIAUDIOCONNECTOR
56 */
57typedef struct DRVHOSTOSSAUDIO
58{
59 /** Pointer to the driver instance structure. */
60 PPDMDRVINS pDrvIns;
61 /** Pointer to host audio interface. */
62 PDMIHOSTAUDIO IHostAudio;
63 /** Error count for not flooding the release log.
64 * UINT32_MAX for unlimited logging. */
65 uint32_t cLogErrors;
66} DRVHOSTOSSAUDIO;
67/** Pointer to the instance data for an OSS host audio driver. */
68typedef DRVHOSTOSSAUDIO *PDRVHOSTOSSAUDIO;
69
70/**
71 * OSS audio stream configuration.
72 */
73typedef struct OSSAUDIOSTREAMCFG
74{
75 PDMAUDIOPCMPROPS Props;
76 uint16_t cFragments;
77 /** The log2 of cbFragment. */
78 uint16_t cbFragmentLog2;
79 uint32_t cbFragment;
80} OSSAUDIOSTREAMCFG;
81/** Pointer to an OSS audio stream configuration. */
82typedef OSSAUDIOSTREAMCFG *POSSAUDIOSTREAMCFG;
83
84/**
85 * OSS audio stream.
86 */
87typedef struct OSSAUDIOSTREAM
88{
89 /** The file descriptor. */
90 int hFile;
91 /** Buffer alignment. */
92 uint8_t uAlign;
93 /** Set if we're draining the stream (output only). */
94 bool fDraining;
95 /** Internal stream byte offset. */
96 uint64_t offInternal;
97 /** The stream's acquired configuration. */
98 PDMAUDIOSTREAMCFG Cfg;
99 /** The acquired OSS configuration. */
100 OSSAUDIOSTREAMCFG OssCfg;
101 /** Handle to the thread draining output streams. */
102 RTTHREAD hThreadDrain;
103
104} OSSAUDIOSTREAM;
105/** Pointer to an OSS audio stream. */
106typedef OSSAUDIOSTREAM *POSSAUDIOSTREAM;
107
108
109/*********************************************************************************************************************************
110* Global Variables *
111*********************************************************************************************************************************/
112/** The path to the output OSS device. */
113static char g_szPathOutputDev[] = "/dev/dsp";
114/** The path to the input OSS device. */
115static char g_szPathInputDev[] = "/dev/dsp";
116
117
118
119static int ossOSSToAudioProps(PPDMAUDIOPCMPROPS pProps, int fmt, int cChannels, int uHz)
120{
121 switch (fmt)
122 {
123 case AFMT_S8:
124 PDMAudioPropsInit(pProps, 1 /*8-bit*/, true /*signed*/, cChannels, uHz);
125 break;
126
127 case AFMT_U8:
128 PDMAudioPropsInit(pProps, 1 /*8-bit*/, false /*signed*/, cChannels, uHz);
129 break;
130
131 case AFMT_S16_LE:
132 PDMAudioPropsInitEx(pProps, 2 /*16-bit*/, true /*signed*/, cChannels, uHz, true /*fLittleEndian*/, false /*fRaw*/);
133 break;
134
135 case AFMT_U16_LE:
136 PDMAudioPropsInitEx(pProps, 2 /*16-bit*/, false /*signed*/, cChannels, uHz, true /*fLittleEndian*/, false /*fRaw*/);
137 break;
138
139 case AFMT_S16_BE:
140 PDMAudioPropsInitEx(pProps, 2 /*16-bit*/, true /*signed*/, cChannels, uHz, false /*fLittleEndian*/, false /*fRaw*/);
141 break;
142
143 case AFMT_U16_BE:
144 PDMAudioPropsInitEx(pProps, 2 /*16-bit*/, false /*signed*/, cChannels, uHz, false /*fLittleEndian*/, false /*fRaw*/);
145 break;
146
147 default:
148 AssertMsgFailedReturn(("Format %d not supported\n", fmt), VERR_NOT_SUPPORTED);
149 }
150
151 return VINF_SUCCESS;
152}
153
154
155static int ossStreamClose(int *phFile)
156{
157 if (!phFile || !*phFile || *phFile == -1)
158 return VINF_SUCCESS;
159
160 int rc;
161 if (close(*phFile))
162 {
163 rc = RTErrConvertFromErrno(errno);
164 LogRel(("OSS: Closing stream failed: %s / %Rrc\n", strerror(errno), rc));
165 }
166 else
167 {
168 *phFile = -1;
169 rc = VINF_SUCCESS;
170 }
171
172 return rc;
173}
174
175
176/**
177 * @interface_method_impl{PDMIHOSTAUDIO,pfnGetConfig}
178 */
179static DECLCALLBACK(int) drvHostOssAudioHA_GetConfig(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDCFG pBackendCfg)
180{
181 RT_NOREF(pInterface);
182
183 /*
184 * Fill in the config structure.
185 */
186 RTStrCopy(pBackendCfg->szName, sizeof(pBackendCfg->szName), "OSS");
187 pBackendCfg->cbStream = sizeof(OSSAUDIOSTREAM);
188 pBackendCfg->fFlags = 0;
189 pBackendCfg->cMaxStreamsIn = 0;
190 pBackendCfg->cMaxStreamsOut = 0;
191
192 int hFile = open("/dev/dsp", O_WRONLY | O_NONBLOCK, 0);
193 if (hFile == -1)
194 {
195 /* Try opening the mixing device instead. */
196 hFile = open("/dev/mixer", O_RDONLY | O_NONBLOCK, 0);
197 }
198 if (hFile != -1)
199 {
200 int ossVer = -1;
201 int err = ioctl(hFile, OSS_GETVERSION, &ossVer);
202 if (err == 0)
203 {
204 LogRel2(("OSS: Using version: %d\n", ossVer));
205#ifdef VBOX_WITH_AUDIO_OSS_SYSINFO
206 oss_sysinfo ossInfo;
207 RT_ZERO(ossInfo);
208 err = ioctl(hFile, OSS_SYSINFO, &ossInfo);
209 if (err == 0)
210 {
211 LogRel2(("OSS: Number of DSPs: %d\n", ossInfo.numaudios));
212 LogRel2(("OSS: Number of mixers: %d\n", ossInfo.nummixers));
213
214 int cDev = ossInfo.nummixers;
215 if (!cDev)
216 cDev = ossInfo.numaudios;
217
218 pBackendCfg->cMaxStreamsIn = UINT32_MAX;
219 pBackendCfg->cMaxStreamsOut = UINT32_MAX;
220 }
221 else
222#endif
223 {
224 /* Since we cannot query anything, assume that we have at least
225 * one input and one output if we found "/dev/dsp" or "/dev/mixer". */
226
227 pBackendCfg->cMaxStreamsIn = UINT32_MAX;
228 pBackendCfg->cMaxStreamsOut = UINT32_MAX;
229 }
230 }
231 else
232 LogRel(("OSS: Unable to determine installed version: %s (%d)\n", strerror(err), err));
233 close(hFile);
234 }
235 else
236 LogRel(("OSS: No devices found, audio is not available\n"));
237
238 return VINF_SUCCESS;
239}
240
241
242
243
244/**
245 * @interface_method_impl{PDMIHOSTAUDIO,pfnGetStatus}
246 */
247static DECLCALLBACK(PDMAUDIOBACKENDSTS) drvHostOssAudioHA_GetStatus(PPDMIHOSTAUDIO pInterface, PDMAUDIODIR enmDir)
248{
249 AssertPtrReturn(pInterface, PDMAUDIOBACKENDSTS_UNKNOWN);
250 RT_NOREF(enmDir);
251
252 return PDMAUDIOBACKENDSTS_RUNNING;
253}
254
255
256static int ossStreamConfigure(int hFile, bool fInput, POSSAUDIOSTREAMCFG pOSSReq, POSSAUDIOSTREAMCFG pOSSAcq)
257{
258 /*
259 * Format.
260 */
261 int iFormat;
262 switch (PDMAudioPropsSampleSize(&pOSSReq->Props))
263 {
264 case 1:
265 iFormat = pOSSReq->Props.fSigned ? AFMT_S8 : AFMT_U8;
266 break;
267
268 case 2:
269 if (PDMAudioPropsIsLittleEndian(&pOSSReq->Props))
270 iFormat = pOSSReq->Props.fSigned ? AFMT_S16_LE : AFMT_U16_LE;
271 else
272 iFormat = pOSSReq->Props.fSigned ? AFMT_S16_BE : AFMT_U16_BE;
273 break;
274
275 default:
276 LogRel2(("OSS: Unsupported sample size: %u\n", PDMAudioPropsSampleSize(&pOSSReq->Props)));
277 return VERR_AUDIO_STREAM_COULD_NOT_CREATE;
278 }
279 AssertLogRelMsgReturn(ioctl(hFile, SNDCTL_DSP_SAMPLESIZE, &iFormat) >= 0,
280 ("OSS: Failed to set audio format to %d: %s (%d)\n", iFormat, strerror(errno), errno),
281 RTErrConvertFromErrno(errno));
282
283 /*
284 * Channel count.
285 */
286 int cChannels = PDMAudioPropsChannels(&pOSSReq->Props);
287 AssertLogRelMsgReturn(ioctl(hFile, SNDCTL_DSP_CHANNELS, &cChannels) >= 0,
288 ("OSS: Failed to set number of audio channels (%RU8): %s (%d)\n",
289 PDMAudioPropsChannels(&pOSSReq->Props), strerror(errno), errno),
290 RTErrConvertFromErrno(errno));
291
292 /*
293 * Frequency.
294 */
295 int iFrequenc = pOSSReq->Props.uHz;
296 AssertLogRelMsgReturn(ioctl(hFile, SNDCTL_DSP_SPEED, &iFrequenc) >= 0,
297 ("OSS: Failed to set audio frequency to %d Hz: %s (%d)\n", pOSSReq->Props.uHz, strerror(errno), errno),
298 RTErrConvertFromErrno(errno));
299
300
301 /*
302 * Set obsolete non-blocking call for input streams.
303 */
304 if (fInput)
305 {
306#if !(defined(VBOX) && defined(RT_OS_SOLARIS)) /* Obsolete on Solaris (using O_NONBLOCK is sufficient). */
307 AssertLogRelMsgReturn(ioctl(hFile, SNDCTL_DSP_NONBLOCK, NULL) >= 0,
308 ("OSS: Failed to set non-blocking mode: %s (%d)\n", strerror(errno), errno),
309 RTErrConvertFromErrno(errno));
310#endif
311 }
312
313 /*
314 * Set fragment size and count.
315 */
316 LogRel2(("OSS: Requested %RU16 %s fragments, %RU32 bytes each\n",
317 pOSSReq->cFragments, fInput ? "input" : "output", pOSSReq->cbFragment));
318
319 int mmmmssss = (pOSSReq->cFragments << 16) | pOSSReq->cbFragmentLog2;
320 AssertLogRelMsgReturn(ioctl(hFile, SNDCTL_DSP_SETFRAGMENT, &mmmmssss) >= 0,
321 ("OSS: Failed to set %RU16 fragments to %RU32 bytes each: %s (%d)\n",
322 pOSSReq->cFragments, pOSSReq->cbFragment, strerror(errno), errno),
323 RTErrConvertFromErrno(errno));
324
325 /*
326 * Get parameters and popuplate pOSSAcq.
327 */
328 audio_buf_info BufInfo = { 0, 0, 0, 0 };
329 AssertLogRelMsgReturn(ioctl(hFile, fInput ? SNDCTL_DSP_GETISPACE : SNDCTL_DSP_GETOSPACE, &BufInfo) >= 0,
330 ("OSS: Failed to retrieve %s buffer length: %s (%d)\n",
331 fInput ? "input" : "output", strerror(errno), errno),
332 RTErrConvertFromErrno(errno));
333
334 int rc = ossOSSToAudioProps(&pOSSAcq->Props, iFormat, cChannels, iFrequenc);
335 if (RT_SUCCESS(rc))
336 {
337 pOSSAcq->cFragments = BufInfo.fragstotal;
338 pOSSAcq->cbFragment = BufInfo.fragsize;
339 pOSSAcq->cbFragmentLog2 = ASMBitFirstSetU32(BufInfo.fragsize) - 1;
340 Assert(RT_BIT_32(pOSSAcq->cbFragmentLog2) == pOSSAcq->cbFragment);
341
342 LogRel2(("OSS: Got %RU16 %s fragments, %RU32 bytes each\n",
343 pOSSAcq->cFragments, fInput ? "input" : "output", pOSSAcq->cbFragment));
344 }
345
346 return rc;
347}
348
349
350/**
351 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamCreate}
352 */
353static DECLCALLBACK(int) drvHostOssAudioHA_StreamCreate(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream,
354 PPDMAUDIOSTREAMCFG pCfgReq, PPDMAUDIOSTREAMCFG pCfgAcq)
355{
356 AssertPtr(pInterface); RT_NOREF(pInterface);
357 POSSAUDIOSTREAM pStreamOSS = (POSSAUDIOSTREAM)pStream;
358 AssertPtrReturn(pStreamOSS, VERR_INVALID_POINTER);
359 AssertPtrReturn(pCfgReq, VERR_INVALID_POINTER);
360 AssertPtrReturn(pCfgAcq, VERR_INVALID_POINTER);
361
362 pStreamOSS->hThreadDrain = NIL_RTTHREAD;
363
364 /*
365 * Open the device
366 */
367 int rc;
368 if (pCfgReq->enmDir == PDMAUDIODIR_IN)
369 pStreamOSS->hFile = open(g_szPathInputDev, O_RDONLY | O_NONBLOCK);
370 else
371 pStreamOSS->hFile = open(g_szPathOutputDev, O_WRONLY);
372 if (pStreamOSS->hFile >= 0)
373 {
374 /*
375 * Configure it.
376 */
377 OSSAUDIOSTREAMCFG ReqOssCfg;
378 RT_ZERO(ReqOssCfg);
379
380 memcpy(&ReqOssCfg.Props, &pCfgReq->Props, sizeof(PDMAUDIOPCMPROPS));
381 ReqOssCfg.cbFragmentLog2 = 12;
382 ReqOssCfg.cbFragment = RT_BIT_32(ReqOssCfg.cbFragmentLog2);
383 uint32_t const cbBuffer = PDMAudioPropsFramesToBytes(&pCfgReq->Props, pCfgReq->Backend.cFramesBufferSize);
384 ReqOssCfg.cFragments = cbBuffer >> ReqOssCfg.cbFragmentLog2;
385 AssertLogRelStmt(cbBuffer < ((uint32_t)0x7ffe << ReqOssCfg.cbFragmentLog2), ReqOssCfg.cFragments = 0x7ffe);
386
387 rc = ossStreamConfigure(pStreamOSS->hFile, pCfgReq->enmDir == PDMAUDIODIR_IN, &ReqOssCfg, &pStreamOSS->OssCfg);
388 if (RT_SUCCESS(rc))
389 {
390 pStreamOSS->uAlign = 0; /** @todo r=bird: Where did the correct assignment of this go? */
391
392 /*
393 * Complete the stream structure and fill in the pCfgAcq bits.
394 */
395 if ((pStreamOSS->OssCfg.cFragments * pStreamOSS->OssCfg.cbFragment) & pStreamOSS->uAlign)
396 LogRel(("OSS: Warning: Misaligned playback buffer: Size = %zu, Alignment = %u\n",
397 pStreamOSS->OssCfg.cFragments * pStreamOSS->OssCfg.cbFragment, pStreamOSS->uAlign + 1));
398
399 memcpy(&pCfgAcq->Props, &pStreamOSS->OssCfg.Props, sizeof(PDMAUDIOPCMPROPS));
400 pCfgAcq->Backend.cFramesPeriod = PDMAudioPropsBytesToFrames(&pCfgAcq->Props, pStreamOSS->OssCfg.cbFragment);
401 pCfgAcq->Backend.cFramesBufferSize = pCfgAcq->Backend.cFramesPeriod * pStreamOSS->OssCfg.cFragments;
402 if (pCfgReq->enmDir == PDMAUDIODIR_OUT)
403 pCfgAcq->Backend.cFramesPreBuffering = (uint64_t)pCfgReq->Backend.cFramesPreBuffering
404 * pCfgAcq->Backend.cFramesBufferSize
405 / RT_MAX(pCfgReq->Backend.cFramesBufferSize, 1);
406 else
407 pCfgAcq->Backend.cFramesPreBuffering = 0; /** @todo is this sane? */
408
409 /*
410 * Copy the stream config and we're done!
411 */
412 PDMAudioStrmCfgCopy(&pStreamOSS->Cfg, pCfgAcq);
413 return VINF_SUCCESS;
414 }
415 ossStreamClose(&pStreamOSS->hFile);
416 }
417 else
418 {
419 rc = RTErrConvertFromErrno(errno);
420 LogRel(("OSS: Failed to open '%s': %s (%d) / %Rrc\n",
421 pCfgReq->enmDir == PDMAUDIODIR_IN ? g_szPathInputDev : g_szPathOutputDev, strerror(errno), errno, rc));
422 }
423 return rc;
424}
425
426
427/**
428 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamDestroy}
429 */
430static DECLCALLBACK(int) drvHostOssAudioHA_StreamDestroy(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream)
431{
432 RT_NOREF(pInterface);
433 POSSAUDIOSTREAM pStreamOSS = (POSSAUDIOSTREAM)pStream;
434 AssertPtrReturn(pStreamOSS, VERR_INVALID_POINTER);
435
436 ossStreamClose(&pStreamOSS->hFile);
437
438 if (pStreamOSS->hThreadDrain != NIL_RTTHREAD)
439 {
440 int rc = RTThreadWait(pStreamOSS->hThreadDrain, 1, NULL);
441 AssertRC(rc);
442 pStreamOSS->hThreadDrain = NIL_RTTHREAD;
443 }
444
445 return VINF_SUCCESS;
446}
447
448
449/**
450 * @ interface_method_impl{PDMIHOSTAUDIO,pfnStreamEnable}
451 */
452static DECLCALLBACK(int) drvHostOssAudioHA_StreamEnable(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream)
453{
454 RT_NOREF(pInterface);
455 POSSAUDIOSTREAM pStreamOSS = (POSSAUDIOSTREAM)pStream;
456
457 pStreamOSS->fDraining = false;
458
459 int rc;
460 if (pStreamOSS->Cfg.enmDir == PDMAUDIODIR_IN)
461 rc = VINF_SUCCESS; /** @todo apparently nothing to do here? */
462 else
463 {
464 int fMask = PCM_ENABLE_OUTPUT;
465 if (ioctl(pStreamOSS->hFile, SNDCTL_DSP_SETTRIGGER, &fMask) >= 0)
466 rc = VINF_SUCCESS;
467 else
468 {
469 LogRel(("OSS: Failed to enable output stream: %s (%d)\n", strerror(errno), errno));
470 rc = RTErrConvertFromErrno(errno);
471 }
472 }
473
474 LogFlowFunc(("returns %Rrc for '%s'\n", rc, pStreamOSS->Cfg.szName));
475 return rc;
476}
477
478
479/**
480 * @ interface_method_impl{PDMIHOSTAUDIO,pfnStreamDisable}
481 */
482static DECLCALLBACK(int) drvHostOssAudioHA_StreamDisable(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream)
483{
484 RT_NOREF(pInterface);
485 POSSAUDIOSTREAM pStreamOSS = (POSSAUDIOSTREAM)pStream;
486
487 int rc;
488 if (pStreamOSS->Cfg.enmDir == PDMAUDIODIR_IN)
489 rc = VINF_SUCCESS; /** @todo apparently nothing to do here? */
490 else if (pStreamOSS->fDraining)
491 {
492 LogFlow(("Ignoring stream disable because we're draining...\n"));
493 rc = VINF_SUCCESS; /** @todo apparently nothing to do here? */
494 }
495 else
496 {
497 /** @todo Official documentation says this isn't the right way to stop playback.
498 * It may work in some implementations but fail in all others... Suggest
499 * using SNDCTL_DSP_RESET / SNDCTL_DSP_HALT. */
500 int fMask = 0;
501 if (ioctl(pStreamOSS->hFile, SNDCTL_DSP_SETTRIGGER, &fMask) >= 0)
502 rc = VINF_SUCCESS;
503 else
504 {
505 LogRel(("OSS: Failed to enable output stream: %s (%d)\n", strerror(errno), errno));
506 rc = RTErrConvertFromErrno(errno);
507 }
508 }
509 LogFlowFunc(("returns %Rrc for '%s'\n", rc, pStreamOSS->Cfg.szName));
510 return rc;
511}
512
513
514/**
515 * @ interface_method_impl{PDMIHOSTAUDIO,pfnStreamPause}
516 */
517static DECLCALLBACK(int) drvHostOssAudioHA_StreamPause(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream)
518{
519 return drvHostOssAudioHA_StreamDisable(pInterface, pStream);
520}
521
522
523/**
524 * @ interface_method_impl{PDMIHOSTAUDIO,pfnStreamResume}
525 */
526static DECLCALLBACK(int) drvHostOssAudioHA_StreamResume(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream)
527{
528 return drvHostOssAudioHA_StreamEnable(pInterface, pStream);
529}
530
531
532/**
533 * @callback_method_impl{FNRTTHREAD,
534 * Thread for calling SNDCTL_DSP_SYNC (blocking) on an output stream.}
535 */
536static DECLCALLBACK(int) drvHostOssAudioDrainThread(RTTHREAD ThreadSelf, void *pvUser)
537{
538 RT_NOREF(ThreadSelf);
539 POSSAUDIOSTREAM pStreamOSS = (POSSAUDIOSTREAM)pvUser;
540 int rc;
541
542 /* Make it blocking (for Linux). */
543 int fOrgFlags = fcntl(pStreamOSS->hFile, F_GETFL, 0);
544 LogFunc(("F_GETFL -> %#x\n", fOrgFlags));
545 Assert(fOrgFlags != -1);
546 if (fOrgFlags != -1)
547 {
548 rc = fcntl(pStreamOSS->hFile, F_SETFL, fOrgFlags & ~O_NONBLOCK);
549 AssertStmt(rc != -1, fOrgFlags = -1);
550 }
551
552 /* Drain it. */
553 LogFunc(("Calling SNDCTL_DSP_SYNC now...\n"));
554 rc = ioctl(pStreamOSS->hFile, SNDCTL_DSP_SYNC, NULL);
555 LogFunc(("SNDCTL_DSP_SYNC returned %d / errno=%d\n", rc, errno)); RT_NOREF(rc);
556
557 /* Re-enable non-blocking mode and disable it. */
558 if (fOrgFlags != -1)
559 {
560 rc = fcntl(pStreamOSS->hFile, F_SETFL, fOrgFlags);
561 Assert(rc != -1);
562
563 int fMask = 0;
564 rc = ioctl(pStreamOSS->hFile, SNDCTL_DSP_SETTRIGGER, &fMask);
565 Assert(rc >= 0);
566
567 pStreamOSS->fDraining = false;
568 LogFunc(("Restored non-block mode and cleared the trigger mask\n"));
569 }
570
571 return VINF_SUCCESS;
572}
573
574
575/**
576 * @ interface_method_impl{PDMIHOSTAUDIO,pfnStreamDisable}
577 */
578static DECLCALLBACK(int) drvHostOssAudioHA_StreamDrain(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream)
579{
580 PDRVHOSTOSSAUDIO pThis = RT_FROM_MEMBER(pInterface, DRVHOSTOSSAUDIO, IHostAudio);
581 POSSAUDIOSTREAM pStreamOSS = (POSSAUDIOSTREAM)pStream;
582 AssertReturn(pStreamOSS->Cfg.enmDir == PDMAUDIODIR_OUT, VERR_WRONG_ORDER);
583
584 pStreamOSS->fDraining = true;
585
586 /*
587 * Because the SNDCTL_DSP_SYNC call is blocking on real OSS,
588 * we kick off a thread to deal with it as we're probably on EMT
589 * and cannot block for extended periods.
590 */
591 if (pStreamOSS->hThreadDrain != NIL_RTTHREAD)
592 {
593 int rc = RTThreadWait(pStreamOSS->hThreadDrain, 0, NULL);
594 if (RT_SUCCESS(rc))
595 {
596 pStreamOSS->hThreadDrain = NIL_RTTHREAD;
597 LogFunc(("Cleaned up stale thread handle.\n"));
598 }
599 else
600 {
601 LogFunc(("Drain thread already running (%Rrc).\n", rc));
602 AssertMsg(rc == VERR_TIMEOUT, ("%Rrc\n", rc));
603 return rc == VERR_TIMEOUT ? VINF_SUCCESS : rc;
604 }
605 }
606
607 int rc = RTThreadCreateF(&pStreamOSS->hThreadDrain, drvHostOssAudioDrainThread, pStreamOSS, 0,
608 RTTHREADTYPE_IO, RTTHREADFLAGS_WAITABLE, "ossdrai%u", pThis->pDrvIns->iInstance);
609 LogFunc(("Started drain thread: %Rrc\n", rc));
610 AssertRCReturn(rc, rc);
611
612 return VINF_SUCCESS;
613}
614
615
616
617/**
618 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamControl}
619 */
620static DECLCALLBACK(int) drvHostOssAudioHA_StreamControl(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream,
621 PDMAUDIOSTREAMCMD enmStreamCmd)
622{
623 /** @todo r=bird: I'd like to get rid of this pfnStreamControl method,
624 * replacing it with individual StreamXxxx methods. That would save us
625 * potentally huge switches and more easily see which drivers implement
626 * which operations (grep for pfnStreamXxxx). */
627 switch (enmStreamCmd)
628 {
629 case PDMAUDIOSTREAMCMD_ENABLE:
630 return drvHostOssAudioHA_StreamEnable(pInterface, pStream);
631 case PDMAUDIOSTREAMCMD_DISABLE:
632 return drvHostOssAudioHA_StreamDisable(pInterface, pStream);
633 case PDMAUDIOSTREAMCMD_PAUSE:
634 return drvHostOssAudioHA_StreamPause(pInterface, pStream);
635 case PDMAUDIOSTREAMCMD_RESUME:
636 return drvHostOssAudioHA_StreamResume(pInterface, pStream);
637 case PDMAUDIOSTREAMCMD_DRAIN:
638 return drvHostOssAudioHA_StreamDrain(pInterface, pStream);
639 /** @todo the drain call for OSS is SNDCTL_DSP_SYNC, however in the non-ALSA
640 * implementation of OSS it is probably blocking. Also, it comes with
641 * caveats about clicks and silence... */
642 case PDMAUDIOSTREAMCMD_END:
643 case PDMAUDIOSTREAMCMD_32BIT_HACK:
644 case PDMAUDIOSTREAMCMD_INVALID:
645 /* no default*/
646 break;
647 }
648 return VERR_NOT_SUPPORTED;
649}
650
651
652/**
653 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamGetReadable}
654 */
655static DECLCALLBACK(uint32_t) drvHostOssAudioHA_StreamGetReadable(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream)
656{
657 RT_NOREF(pInterface, pStream);
658 Log4Func(("returns UINT32_MAX\n"));
659 return UINT32_MAX;
660}
661
662
663/**
664 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamGetWritable}
665 */
666static DECLCALLBACK(uint32_t) drvHostOssAudioHA_StreamGetWritable(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream)
667{
668 RT_NOREF(pInterface);
669 POSSAUDIOSTREAM pStreamOSS = (POSSAUDIOSTREAM)pStream;
670 AssertPtr(pStreamOSS);
671
672 /*
673 * Note! This logic was found in StreamPlay and corrected a little.
674 *
675 * The logic here must match what StreamPlay does.
676 */
677 audio_buf_info BufInfo = { 0, 0, 0, 0 };
678 int rc2 = ioctl(pStreamOSS->hFile, SNDCTL_DSP_GETOSPACE, &BufInfo);
679 AssertMsgReturn(rc2 >= 0, ("SNDCTL_DSP_GETOSPACE failed: %s (%d)\n", strerror(errno), errno), 0);
680
681#if 0 /** @todo we could return BufInfo.bytes here iff StreamPlay didn't use the fragmented approach */
682 /** @todo r=bird: WTF do we make a fuss over BufInfo.bytes for when we don't
683 * even use it?!? */
684 AssertLogRelMsgReturn(BufInfo.bytes >= 0, ("OSS: Warning: Invalid available size: %d\n", BufInfo.bytes), VERR_INTERNAL_ERROR_3);
685 if ((unsigned)BufInfo.bytes > cbBuf)
686 {
687 LogRel2(("OSS: Warning: Too big output size (%d > %RU32), limiting to %RU32\n", BufInfo.bytes, cbBuf, cbBuf));
688 BufInfo.bytes = cbBuf;
689 /* Keep going. */
690 }
691#endif
692
693 uint32_t cbRet = (uint32_t)(BufInfo.fragments * BufInfo.fragsize);
694 Log4Func(("returns %#x (%u)\n", cbRet, cbRet));
695 return cbRet;
696}
697
698
699/**
700 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamGetStatus}
701 */
702static DECLCALLBACK(PDMAUDIOSTREAMSTS) drvHostOssAudioHA_StreamGetStatus(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream)
703{
704 RT_NOREF(pInterface, pStream);
705 return PDMAUDIOSTREAMSTS_FLAGS_INITIALIZED | PDMAUDIOSTREAMSTS_FLAGS_ENABLED;
706}
707
708
709/**
710 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamPlay}
711 */
712static DECLCALLBACK(int) drvHostOssAudioHA_StreamPlay(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream,
713 const void *pvBuf, uint32_t cbBuf, uint32_t *pcbWritten)
714{
715 RT_NOREF(pInterface);
716 POSSAUDIOSTREAM pStreamOSS = (POSSAUDIOSTREAM)pStream;
717 AssertPtrReturn(pStreamOSS, VERR_INVALID_POINTER);
718
719 /*
720 * Figure out now much to write.
721 */
722 audio_buf_info BufInfo;
723 int rc2 = ioctl(pStreamOSS->hFile, SNDCTL_DSP_GETOSPACE, &BufInfo);
724 AssertLogRelMsgReturn(rc2 >= 0, ("OSS: Failed to retrieve current playback buffer: %s (%d)\n", strerror(errno), errno),
725 RTErrConvertFromErrno(errno));
726
727#if 0 /** @todo r=bird: WTF do we make a fuss over BufInfo.bytes for when we don't even use it?!? */
728 AssertLogRelMsgReturn(BufInfo.bytes >= 0, ("OSS: Warning: Invalid available size: %d\n", BufInfo.bytes), VERR_INTERNAL_ERROR_3);
729 if ((unsigned)BufInfo.bytes > cbBuf)
730 {
731 LogRel2(("OSS: Warning: Too big output size (%d > %RU32), limiting to %RU32\n", BufInfo.bytes, cbBuf, cbBuf));
732 BufInfo.bytes = cbBuf;
733 /* Keep going. */
734 }
735#endif
736 uint32_t cbToWrite = (uint32_t)(BufInfo.fragments * BufInfo.fragsize);
737 cbToWrite = RT_MIN(cbToWrite, cbBuf);
738 Log3Func(("@%#RX64 cbBuf=%#x BufInfo: fragments=%#x fragstotal=%#x fragsize=%#x bytes=%#x %s cbToWrite=%#x\n",
739 pStreamOSS->offInternal, cbBuf, BufInfo.fragments, BufInfo.fragstotal, BufInfo.fragsize, BufInfo.bytes,
740 pStreamOSS->Cfg.szName, cbToWrite));
741
742 /*
743 * Write.
744 */
745 uint8_t const *pbBuf = (uint8_t const *)pvBuf;
746 uint32_t cbChunk = cbToWrite;
747 uint32_t offChunk = 0;
748 while (cbChunk > 0)
749 {
750 ssize_t cbWritten = write(pStreamOSS->hFile, &pbBuf[offChunk], RT_MIN(cbChunk, pStreamOSS->OssCfg.cbFragment));
751 if (cbWritten > 0)
752 {
753 AssertLogRelMsg(!(cbWritten & pStreamOSS->uAlign),
754 ("OSS: Misaligned write (written %#zx, alignment %#x)\n", cbWritten, pStreamOSS->uAlign));
755
756 Assert((uint32_t)cbWritten <= cbChunk);
757 offChunk += (uint32_t)cbWritten;
758 cbChunk -= (uint32_t)cbWritten;
759 pStreamOSS->offInternal += cbWritten;
760 }
761 else if (cbWritten == 0)
762 {
763 LogFunc(("@%#RX64 write(%#x) returned zeroed (previously wrote %#x bytes)!\n",
764 pStreamOSS->offInternal, RT_MIN(cbChunk, pStreamOSS->OssCfg.cbFragment), cbWritten));
765 break;
766 }
767 else
768 {
769 LogRel(("OSS: Failed writing output data: %s (%d)\n", strerror(errno), errno));
770 return RTErrConvertFromErrno(errno);
771 }
772 }
773
774 *pcbWritten = offChunk;
775 return VINF_SUCCESS;
776}
777
778
779/**
780 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamCapture}
781 */
782static DECLCALLBACK(int) drvHostOssAudioHA_StreamCapture(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream,
783 void *pvBuf, uint32_t cbBuf, uint32_t *pcbRead)
784{
785 RT_NOREF(pInterface);
786 POSSAUDIOSTREAM pStreamOSS = (POSSAUDIOSTREAM)pStream;
787 AssertPtrReturn(pStreamOSS, VERR_INVALID_POINTER);
788 Log3Func(("@%#RX64 cbBuf=%#x %s\n", pStreamOSS->offInternal, cbBuf, pStreamOSS->Cfg.szName));
789
790 size_t cbToRead = cbBuf;
791 uint8_t * const pbDst = (uint8_t *)pvBuf;
792 size_t offWrite = 0;
793 while (cbToRead > 0)
794 {
795 ssize_t cbRead = read(pStreamOSS->hFile, &pbDst[offWrite], cbToRead);
796 if (cbRead)
797 {
798 LogFlowFunc(("cbRead=%zi, offWrite=%zu cbToRead=%zu\n", cbRead, offWrite, cbToRead));
799 Assert((ssize_t)cbToRead >= cbRead);
800 cbToRead -= cbRead;
801 offWrite += cbRead;
802 pStreamOSS->offInternal += cbRead;
803 }
804 else
805 {
806 LogFunc(("cbRead=%zi, offWrite=%zu cbToRead=%zu errno=%d\n", cbRead, offWrite, cbToRead, errno));
807
808 /* Don't complain about errors if we've retrieved some audio data already. */
809 if (cbRead < 0 && offWrite == 0 && errno != EINTR && errno != EAGAIN)
810 {
811 AssertStmt(errno != 0, errno = EACCES);
812 int rc = RTErrConvertFromErrno(errno);
813 LogFunc(("Failed to read %zu input frames, errno=%d rc=%Rrc\n", cbToRead, errno, rc));
814 return rc;
815 }
816 break;
817 }
818 }
819
820 *pcbRead = offWrite;
821 return VINF_SUCCESS;
822}
823
824
825
826/**
827 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
828 */
829static DECLCALLBACK(void *) drvHostOSSAudioQueryInterface(PPDMIBASE pInterface, const char *pszIID)
830{
831 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
832 PDRVHOSTOSSAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTOSSAUDIO);
833
834 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
835 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIHOSTAUDIO, &pThis->IHostAudio);
836
837 return NULL;
838}
839
840/**
841 * Constructs an OSS audio driver instance.
842 *
843 * @copydoc FNPDMDRVCONSTRUCT
844 */
845static DECLCALLBACK(int) drvHostOSSAudioConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
846{
847 RT_NOREF(pCfg, fFlags);
848 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
849 PDRVHOSTOSSAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTOSSAUDIO);
850 LogRel(("Audio: Initializing OSS driver\n"));
851
852 /*
853 * Init the static parts.
854 */
855 pThis->pDrvIns = pDrvIns;
856 /* IBase */
857 pDrvIns->IBase.pfnQueryInterface = drvHostOSSAudioQueryInterface;
858 /* IHostAudio */
859 pThis->IHostAudio.pfnGetConfig = drvHostOssAudioHA_GetConfig;
860 pThis->IHostAudio.pfnGetDevices = NULL;
861 pThis->IHostAudio.pfnGetStatus = drvHostOssAudioHA_GetStatus;
862 pThis->IHostAudio.pfnStreamCreate = drvHostOssAudioHA_StreamCreate;
863 pThis->IHostAudio.pfnStreamDestroy = drvHostOssAudioHA_StreamDestroy;
864 pThis->IHostAudio.pfnStreamControl = drvHostOssAudioHA_StreamControl;
865 pThis->IHostAudio.pfnStreamGetReadable = drvHostOssAudioHA_StreamGetReadable;
866 pThis->IHostAudio.pfnStreamGetWritable = drvHostOssAudioHA_StreamGetWritable;
867 pThis->IHostAudio.pfnStreamGetStatus = drvHostOssAudioHA_StreamGetStatus;
868 pThis->IHostAudio.pfnStreamPlay = drvHostOssAudioHA_StreamPlay;
869 pThis->IHostAudio.pfnStreamCapture = drvHostOssAudioHA_StreamCapture;
870 pThis->IHostAudio.pfnStreamGetPending = NULL;
871
872 return VINF_SUCCESS;
873}
874
875/**
876 * Char driver registration record.
877 */
878const PDMDRVREG g_DrvHostOSSAudio =
879{
880 /* u32Version */
881 PDM_DRVREG_VERSION,
882 /* szName */
883 "OSSAudio",
884 /* szRCMod */
885 "",
886 /* szR0Mod */
887 "",
888 /* pszDescription */
889 "OSS audio host driver",
890 /* fFlags */
891 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
892 /* fClass. */
893 PDM_DRVREG_CLASS_AUDIO,
894 /* cMaxInstances */
895 ~0U,
896 /* cbInstance */
897 sizeof(DRVHOSTOSSAUDIO),
898 /* pfnConstruct */
899 drvHostOSSAudioConstruct,
900 /* pfnDestruct */
901 NULL,
902 /* pfnRelocate */
903 NULL,
904 /* pfnIOCtl */
905 NULL,
906 /* pfnPowerOn */
907 NULL,
908 /* pfnReset */
909 NULL,
910 /* pfnSuspend */
911 NULL,
912 /* pfnResume */
913 NULL,
914 /* pfnAttach */
915 NULL,
916 /* pfnDetach */
917 NULL,
918 /* pfnPowerOff */
919 NULL,
920 /* pfnSoftReset */
921 NULL,
922 /* u32EndVersion */
923 PDM_DRVREG_VERSION
924};
925
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette