VirtualBox

source: vbox/trunk/src/VBox/Devices/Audio/DrvHostOSSAudio.cpp@ 61362

Last change on this file since 61362 was 61332, checked in by vboxsync, 9 years ago

Audio: Update.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 33.6 KB
Line 
1/* $Id: DrvHostOSSAudio.cpp 61332 2016-05-31 13:23:20Z vboxsync $ */
2/** @file
3 * OSS (Open Sound System) host audio backend.
4 */
5
6/*
7 * Copyright (C) 2014-2016 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#include <errno.h>
19#include <fcntl.h>
20#include <sys/ioctl.h>
21#include <sys/mman.h>
22#include <sys/soundcard.h>
23#include <unistd.h>
24
25#include <iprt/alloc.h>
26#include <iprt/uuid.h> /* For PDMIBASE_2_PDMDRV. */
27
28#define LOG_GROUP LOG_GROUP_DRV_HOST_AUDIO
29#include <VBox/log.h>
30#include <VBox/vmm/pdmaudioifs.h>
31
32#include "DrvAudio.h"
33#include "AudioMixBuffer.h"
34
35#include "VBoxDD.h"
36
37/*********************************************************************************************************************************
38* Defines *
39*********************************************************************************************************************************/
40
41#if ((SOUND_VERSION > 360) && (defined(OSS_SYSINFO)))
42/* OSS > 3.6 has a new syscall available for querying a bit more detailed information
43 * about OSS' audio capabilities. This is handy for e.g. Solaris. */
44# define VBOX_WITH_OSS_SYSINFO 1
45#endif
46
47/** Makes DRVHOSTOSSAUDIO out of PDMIHOSTAUDIO. */
48#define PDMIHOSTAUDIO_2_DRVHOSTOSSAUDIO(pInterface) \
49 ( (PDRVHOSTOSSAUDIO)((uintptr_t)pInterface - RT_OFFSETOF(DRVHOSTOSSAUDIO, IHostAudio)) )
50
51/*********************************************************************************************************************************
52* Structures *
53*********************************************************************************************************************************/
54
55/**
56 * OSS host audio driver instance data.
57 * @implements PDMIAUDIOCONNECTOR
58 */
59typedef struct DRVHOSTOSSAUDIO
60{
61 /** Pointer to the driver instance structure. */
62 PPDMDRVINS pDrvIns;
63 /** Pointer to host audio interface. */
64 PDMIHOSTAUDIO IHostAudio;
65 /** Error count for not flooding the release log.
66 * UINT32_MAX for unlimited logging. */
67 uint32_t cLogErrors;
68} DRVHOSTOSSAUDIO, *PDRVHOSTOSSAUDIO;
69
70typedef struct OSSAUDIOSTREAMCFG
71{
72 PDMAUDIOFMT enmFormat;
73 PDMAUDIOENDIANNESS enmENDIANNESS;
74 uint16_t uFreq;
75 uint8_t cChannels;
76 uint16_t cFragments;
77 uint32_t cbFragmentSize;
78} OSSAUDIOSTREAMCFG, *POSSAUDIOSTREAMCFG;
79
80typedef struct OSSAUDIOSTREAMIN
81{
82 /** Note: Always must come first! */
83 PDMAUDIOSTREAM pStreamIn;
84 int hFile;
85 int cFragments;
86 int cbFragmentSize;
87 /** Own PCM buffer. */
88 void *pvBuf;
89 /** Size (in bytes) of own PCM buffer. */
90 size_t cbBuf;
91 int old_optr;
92} OSSAUDIOSTREAMIN, *POSSAUDIOSTREAMIN;
93
94typedef struct OSSAUDIOSTREAMOUT
95{
96 /** Note: Always must come first! */
97 PDMAUDIOSTREAM pStreamOut;
98 int hFile;
99 int cFragments;
100 int cbFragmentSize;
101#ifndef RT_OS_L4
102 /** Whether we use a memory mapped file instead of our
103 * own allocated PCM buffer below. */
104 bool fMemMapped;
105#endif
106 /** Own PCM buffer in case memory mapping is unavailable. */
107 void *pvBuf;
108 /** Size (in bytes) of own PCM buffer. */
109 size_t cbBuf;
110 int old_optr;
111} OSSAUDIOSTREAMOUT, *POSSAUDIOSTREAMOUT;
112
113typedef struct OSSAUDIOCFG
114{
115#ifndef RT_OS_L4
116 bool try_mmap;
117#endif
118 int nfrags;
119 int fragsize;
120 const char *devpath_out;
121 const char *devpath_in;
122 int debug;
123} OSSAUDIOCFG, *POSSAUDIOCFG;
124
125static OSSAUDIOCFG s_OSSConf =
126{
127#ifndef RT_OS_L4
128 false,
129#endif
130 4,
131 4096,
132 "/dev/dsp",
133 "/dev/dsp",
134 0
135};
136
137
138/* http://www.df.lth.se/~john_e/gems/gem002d.html */
139static uint32_t popcount(uint32_t u)
140{
141 u = ((u&0x55555555) + ((u>>1)&0x55555555));
142 u = ((u&0x33333333) + ((u>>2)&0x33333333));
143 u = ((u&0x0f0f0f0f) + ((u>>4)&0x0f0f0f0f));
144 u = ((u&0x00ff00ff) + ((u>>8)&0x00ff00ff));
145 u = ( u&0x0000ffff) + (u>>16);
146 return u;
147}
148
149static uint32_t lsbindex(uint32_t u)
150{
151 return popcount ((u&-u)-1);
152}
153
154static int ossAudioFmtToOSS(PDMAUDIOFMT fmt)
155{
156 switch (fmt)
157 {
158 case PDMAUDIOFMT_S8:
159 return AFMT_S8;
160
161 case PDMAUDIOFMT_U8:
162 return AFMT_U8;
163
164 case PDMAUDIOFMT_S16:
165 return AFMT_S16_LE;
166
167 case PDMAUDIOFMT_U16:
168 return AFMT_U16_LE;
169
170 default:
171 break;
172 }
173
174 AssertMsgFailed(("Format %ld not supported\n", fmt));
175 return AFMT_U8;
176}
177
178static int ossOSSToAudioFmt(int fmt, PDMAUDIOFMT *pFmt, PDMAUDIOENDIANNESS *pENDIANNESS)
179{
180 switch (fmt)
181 {
182 case AFMT_S8:
183 *pFmt = PDMAUDIOFMT_S8;
184 if (pENDIANNESS)
185 *pENDIANNESS = PDMAUDIOENDIANNESS_LITTLE;
186 break;
187
188 case AFMT_U8:
189 *pFmt = PDMAUDIOFMT_U8;
190 if (pENDIANNESS)
191 *pENDIANNESS = PDMAUDIOENDIANNESS_LITTLE;
192 break;
193
194 case AFMT_S16_LE:
195 *pFmt = PDMAUDIOFMT_S16;
196 if (pENDIANNESS)
197 *pENDIANNESS = PDMAUDIOENDIANNESS_LITTLE;
198 break;
199
200 case AFMT_U16_LE:
201 *pFmt = PDMAUDIOFMT_U16;
202 if (pENDIANNESS)
203 *pENDIANNESS = PDMAUDIOENDIANNESS_LITTLE;
204 break;
205
206 case AFMT_S16_BE:
207 *pFmt = PDMAUDIOFMT_S16;
208 if (pENDIANNESS)
209 *pENDIANNESS = PDMAUDIOENDIANNESS_BIG;
210 break;
211
212 case AFMT_U16_BE:
213 *pFmt = PDMAUDIOFMT_U16;
214 if (pENDIANNESS)
215 *pENDIANNESS = PDMAUDIOENDIANNESS_BIG;
216 break;
217
218 default:
219 AssertMsgFailed(("Format %ld not supported\n", fmt));
220 return VERR_NOT_SUPPORTED;
221 }
222
223 return VINF_SUCCESS;
224}
225
226static int ossStreamClose(int *phFile)
227{
228 if (!phFile || !*phFile)
229 return VINF_SUCCESS;
230
231 int rc;
232 if (close(*phFile))
233 {
234 LogRel(("OSS: Closing stream failed: %s\n", strerror(errno)));
235 rc = VERR_GENERAL_FAILURE; /** @todo */
236 }
237 else
238 {
239 *phFile = -1;
240 rc = VINF_SUCCESS;
241 }
242
243 return rc;
244}
245
246static int ossStreamOpen(const char *pszDev, int fOpen, POSSAUDIOSTREAMCFG pReq, POSSAUDIOSTREAMCFG pObt, int *phFile)
247{
248 AssertPtrReturn(pszDev, VERR_INVALID_POINTER);
249 AssertPtrReturn(pReq, VERR_INVALID_POINTER);
250 AssertPtrReturn(pObt, VERR_INVALID_POINTER);
251 AssertPtrReturn(phFile, VERR_INVALID_POINTER);
252
253 int rc;
254
255 int hFile = -1;
256 do
257 {
258 hFile = open(pszDev, fOpen);
259 if (hFile == -1)
260 {
261 LogRel(("OSS: Failed to open %s: %s (%d)\n", pszDev, strerror(errno), errno));
262 rc = RTErrConvertFromErrno(errno);
263 break;
264 }
265
266 int iFormat = ossAudioFmtToOSS(pReq->enmFormat);
267 if (ioctl(hFile, SNDCTL_DSP_SAMPLESIZE, &iFormat))
268 {
269 LogRel(("OSS: Failed to set audio format to %ld: %s (%d)\n", iFormat, strerror(errno), errno));
270 rc = RTErrConvertFromErrno(errno);
271 break;
272 }
273
274 int cChannels = pReq->cChannels;
275 if (ioctl(hFile, SNDCTL_DSP_CHANNELS, &cChannels))
276 {
277 LogRel(("OSS: Failed to set number of audio channels (%d): %s (%d)\n", pReq->cChannels, strerror(errno), errno));
278 rc = RTErrConvertFromErrno(errno);
279 break;
280 }
281
282 int freq = pReq->uFreq;
283 if (ioctl(hFile, SNDCTL_DSP_SPEED, &freq))
284 {
285 LogRel(("OSS: Failed to set audio frequency (%dHZ): %s (%d)\n", pReq->uFreq, strerror(errno), errno));
286 rc = RTErrConvertFromErrno(errno);
287 break;
288 }
289
290 /* Obsolete on Solaris (using O_NONBLOCK is sufficient). */
291#if !(defined(VBOX) && defined(RT_OS_SOLARIS))
292 if (ioctl(hFile, SNDCTL_DSP_NONBLOCK))
293 {
294 LogRel(("OSS: Failed to set non-blocking mode: %s (%d)\n", strerror(errno), errno));
295 rc = RTErrConvertFromErrno(errno);
296 break;
297 }
298#endif
299 int mmmmssss = (pReq->cFragments << 16) | lsbindex(pReq->cbFragmentSize);
300 if (ioctl(hFile, SNDCTL_DSP_SETFRAGMENT, &mmmmssss))
301 {
302 LogRel(("OSS: Failed to set %RU16 fragments to %RU32 bytes each: %s (%d)\n",
303 pReq->cFragments, pReq->cbFragmentSize, strerror(errno), errno));
304 rc = RTErrConvertFromErrno(errno);
305 break;
306 }
307
308 audio_buf_info abinfo;
309 if (ioctl(hFile, (fOpen & O_RDONLY) ? SNDCTL_DSP_GETISPACE : SNDCTL_DSP_GETOSPACE, &abinfo))
310 {
311 LogRel(("OSS: Failed to retrieve buffer length: %s (%d)\n", strerror(errno), errno));
312 rc = RTErrConvertFromErrno(errno);
313 break;
314 }
315
316 rc = ossOSSToAudioFmt(iFormat, &pObt->enmFormat, &pObt->enmENDIANNESS);
317 if (RT_SUCCESS(rc))
318 {
319 pObt->cChannels = cChannels;
320 pObt->uFreq = freq;
321 pObt->cFragments = abinfo.fragstotal;
322 pObt->cbFragmentSize = abinfo.fragsize;
323
324 *phFile = hFile;
325 }
326 }
327 while (0);
328
329 if (RT_FAILURE(rc))
330 ossStreamClose(&hFile);
331
332 LogFlowFuncLeaveRC(rc);
333 return rc;
334}
335
336static int ossControlStreamIn(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream,
337 PDMAUDIOSTREAMCMD enmStreamCmd)
338{
339 NOREF(pInterface);
340 NOREF(pStream);
341 NOREF(enmStreamCmd);
342
343 /** @todo Nothing to do here right now!? */
344
345 return VINF_SUCCESS;
346}
347
348static int ossControlStreamOut(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream,
349 PDMAUDIOSTREAMCMD enmStreamCmd)
350{
351 NOREF(pInterface);
352 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
353
354 POSSAUDIOSTREAMOUT pThisStream = (POSSAUDIOSTREAMOUT)pStream;
355
356#ifdef RT_OS_L4
357 return VINF_SUCCESS;
358#else
359 if (!pThisStream->fMemMapped)
360 return VINF_SUCCESS;
361#endif
362
363 int rc = VINF_SUCCESS;
364 int mask;
365 switch (enmStreamCmd)
366 {
367 case PDMAUDIOSTREAMCMD_ENABLE:
368 case PDMAUDIOSTREAMCMD_RESUME:
369 {
370 DrvAudioHlpClearBuf(&pStream->Props,
371 pThisStream->pvBuf, pThisStream->cbBuf, AudioMixBufSize(&pStream->MixBuf));
372
373 mask = PCM_ENABLE_OUTPUT;
374 if (ioctl(pThisStream->hFile, SNDCTL_DSP_SETTRIGGER, &mask) < 0)
375 {
376 LogRel(("OSS: Failed to enable output stream: %s\n", strerror(errno)));
377 rc = RTErrConvertFromErrno(errno);
378 }
379
380 break;
381 }
382
383 case PDMAUDIOSTREAMCMD_DISABLE:
384 case PDMAUDIOSTREAMCMD_PAUSE:
385 {
386 mask = 0;
387 if (ioctl(pThisStream->hFile, SNDCTL_DSP_SETTRIGGER, &mask) < 0)
388 {
389 LogRel(("OSS: Failed to disable output stream: %s\n", strerror(errno)));
390 rc = RTErrConvertFromErrno(errno);
391 }
392
393 break;
394 }
395
396 default:
397 AssertMsgFailed(("Invalid command %ld\n", enmStreamCmd));
398 rc = VERR_INVALID_PARAMETER;
399 break;
400 }
401
402 LogFlowFuncLeaveRC(rc);
403 return rc;
404}
405
406static DECLCALLBACK(int) drvHostOSSAudioInit(PPDMIHOSTAUDIO pInterface)
407{
408 NOREF(pInterface);
409
410 LogFlowFuncEnter();
411
412 return VINF_SUCCESS;
413}
414
415static DECLCALLBACK(int) drvHostOSSAudioStreamCapture(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream,
416 uint32_t *pcSamplesCaptured)
417{
418 NOREF(pInterface);
419 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
420
421 POSSAUDIOSTREAMIN pStrm = (POSSAUDIOSTREAMIN)pStream;
422
423 int rc = VINF_SUCCESS;
424 size_t cbToRead = RT_MIN(pStrm->cbBuf,
425 AudioMixBufFreeBytes(&pStream->MixBuf));
426
427 LogFlowFunc(("cbToRead=%zu\n", cbToRead));
428
429 uint32_t cWrittenTotal = 0;
430 uint32_t cbTemp;
431 ssize_t cbRead;
432 size_t offWrite = 0;
433
434 while (cbToRead)
435 {
436 cbTemp = RT_MIN(cbToRead, pStrm->cbBuf);
437 AssertBreakStmt(cbTemp, rc = VERR_NO_DATA);
438 cbRead = read(pStrm->hFile, (uint8_t *)pStrm->pvBuf + offWrite, cbTemp);
439
440 LogFlowFunc(("cbRead=%zi, cbTemp=%RU32, cbToRead=%zu\n", cbRead, cbTemp, cbToRead));
441
442 if (cbRead < 0)
443 {
444 switch (errno)
445 {
446 case 0:
447 {
448 LogFunc(("Failed to read %z frames\n", cbRead));
449 rc = VERR_ACCESS_DENIED;
450 break;
451 }
452
453 case EINTR:
454 case EAGAIN:
455 rc = VERR_NO_DATA;
456 break;
457
458 default:
459 LogFlowFunc(("Failed to read %zu input frames, rc=%Rrc\n", cbTemp, rc));
460 rc = VERR_GENERAL_FAILURE; /** @todo Fix this. */
461 break;
462 }
463
464 if (RT_FAILURE(rc))
465 break;
466 }
467 else if (cbRead)
468 {
469 uint32_t cWritten;
470 rc = AudioMixBufWriteCirc(&pStream->MixBuf, pStrm->pvBuf, cbRead, &cWritten);
471 if (RT_FAILURE(rc))
472 break;
473
474 uint32_t cbWritten = AUDIOMIXBUF_S2B(&pStream->MixBuf, cWritten);
475
476 Assert(cbToRead >= cbWritten);
477 cbToRead -= cbWritten;
478 offWrite += cbWritten;
479 cWrittenTotal += cWritten;
480 }
481 else /* No more data, try next round. */
482 break;
483 }
484
485 if (rc == VERR_NO_DATA)
486 rc = VINF_SUCCESS;
487
488 if (RT_SUCCESS(rc))
489 {
490 uint32_t cProcessed = 0;
491 if (cWrittenTotal)
492 rc = AudioMixBufMixToParent(&pStream->MixBuf, cWrittenTotal, &cProcessed);
493
494 if (pcSamplesCaptured)
495 *pcSamplesCaptured = cWrittenTotal;
496
497 LogFlowFunc(("cWrittenTotal=%RU32 (%RU32 processed), rc=%Rrc\n",
498 cWrittenTotal, cProcessed, rc));
499 }
500
501 LogFlowFuncLeaveRC(rc);
502 return rc;
503}
504
505static int ossDestroyStreamIn(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
506{
507 NOREF(pInterface);
508 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
509
510 POSSAUDIOSTREAMIN pStrm = (POSSAUDIOSTREAMIN)pStream;
511
512 LogFlowFuncEnter();
513
514 if (pStrm->pvBuf)
515 {
516 Assert(pStrm->cbBuf);
517
518 RTMemFree(pStrm->pvBuf);
519 pStrm->pvBuf = NULL;
520 }
521
522 pStrm->cbBuf = 0;
523
524 ossStreamClose(&pStrm->hFile);
525
526 return VINF_SUCCESS;
527}
528
529static int ossDestroyStreamOut(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
530{
531 NOREF(pInterface);
532 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
533
534 POSSAUDIOSTREAMOUT pStrm = (POSSAUDIOSTREAMOUT)pStream;
535
536 LogFlowFuncEnter();
537
538#ifndef RT_OS_L4
539 if (pStrm->fMemMapped)
540 {
541 if (pStrm->pvBuf)
542 {
543 Assert(pStrm->cbBuf);
544
545 int rc2 = munmap(pStrm->pvBuf, pStrm->cbBuf);
546 if (rc2 == 0)
547 {
548 pStrm->pvBuf = NULL;
549 pStrm->cbBuf = 0;
550
551 pStrm->fMemMapped = false;
552 }
553 else
554 LogRel(("OSS: Failed to memory unmap playback buffer on close: %s\n", strerror(errno)));
555 }
556 }
557 else
558 {
559#endif
560 if (pStrm->pvBuf)
561 {
562 Assert(pStrm->cbBuf);
563
564 RTMemFree(pStrm->pvBuf);
565 pStrm->pvBuf = NULL;
566 }
567
568 pStrm->cbBuf = 0;
569#ifndef RT_OS_L4
570 }
571#endif
572
573 ossStreamClose(&pStrm->hFile);
574
575 return VINF_SUCCESS;
576}
577
578static DECLCALLBACK(int) drvHostOSSAudioGetConfig(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDCFG pCfg)
579{
580 NOREF(pInterface);
581
582 pCfg->cbStreamIn = sizeof(OSSAUDIOSTREAMIN);
583 pCfg->cbStreamOut = sizeof(OSSAUDIOSTREAMOUT);
584
585 pCfg->cSources = 0;
586 pCfg->cSinks = 0;
587
588 int hFile = open("/dev/dsp", O_WRONLY | O_NONBLOCK, 0);
589 if (hFile == -1)
590 {
591 /* Try opening the mixing device instead. */
592 hFile = open("/dev/mixer", O_RDONLY | O_NONBLOCK, 0);
593 }
594
595 int ossVer = -1;
596
597#ifdef VBOX_WITH_OSS_SYSINFO
598 oss_sysinfo ossInfo;
599 RT_ZERO(ossInfo);
600#endif
601
602 if (hFile != -1)
603 {
604 int err = ioctl(hFile, OSS_GETVERSION, &ossVer);
605 if (err == 0)
606 {
607 LogRel2(("OSS: Using version: %d\n", ossVer));
608#ifdef VBOX_WITH_OSS_SYSINFO
609 err = ioctl(hFile, OSS_SYSINFO, &ossInfo);
610 if (err == 0)
611 {
612 LogRel2(("OSS: Number of DSPs: %d\n", ossInfo.numaudios));
613 LogRel2(("OSS: Number of mixers: %d\n", ossInfo.nummixers));
614
615 int cDev = ossInfo.nummixers;
616 if (!cDev)
617 cDev = ossInfo.numaudios;
618
619 pCfg->cSources = cDev;
620 pCfg->cSinks = cDev;
621
622 pCfg->cMaxStreamsIn = UINT32_MAX;
623 pCfg->cMaxStreamsOut = UINT32_MAX;
624 }
625 else
626 {
627#endif
628 /* Since we cannot query anything, assume that we have at least
629 * one input and one output if we found "/dev/dsp" or "/dev/mixer". */
630 pCfg->cSources = 1;
631 pCfg->cSinks = 1;
632
633 pCfg->cMaxStreamsIn = UINT32_MAX;
634 pCfg->cMaxStreamsOut = UINT32_MAX;
635#ifdef VBOX_WITH_OSS_SYSINFO
636 }
637#endif
638 }
639 else
640 LogRel(("OSS: Unable to determine installed version: %s (%d)\n", strerror(err), err));
641 }
642 else
643 LogRel(("OSS: No devices found, audio is not available\n"));
644
645 return VINF_SUCCESS;
646}
647
648static int ossCreateStreamIn(PPDMIHOSTAUDIO pInterface,
649 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg, uint32_t *pcSamples)
650{
651 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
652 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
653 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
654
655 PDRVHOSTOSSAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTOSSAUDIO(pInterface);
656 POSSAUDIOSTREAMIN pStrm = (POSSAUDIOSTREAMIN)pStream;
657
658 int rc;
659 int hFile = -1;
660
661 do
662 {
663 uint32_t cSamples;
664
665 OSSAUDIOSTREAMCFG reqStream, obtStream;
666 reqStream.enmFormat = pCfg->enmFormat;
667 reqStream.uFreq = pCfg->uHz;
668 reqStream.cChannels = pCfg->cChannels;
669 reqStream.cFragments = s_OSSConf.nfrags;
670 reqStream.cbFragmentSize = s_OSSConf.fragsize;
671
672 rc = ossStreamOpen(s_OSSConf.devpath_in, O_RDONLY | O_NONBLOCK, &reqStream, &obtStream, &hFile);
673 if (RT_SUCCESS(rc))
674 {
675 if (obtStream.cFragments * obtStream.cbFragmentSize & pStream->Props.uAlign)
676 LogRel(("OSS: Warning: Misaligned capturing buffer: Size = %zu, Alignment = %u\n",
677 obtStream.cFragments * obtStream.cbFragmentSize,
678 pStream->Props.uAlign + 1));
679
680 PDMAUDIOSTREAMCFG streamCfg;
681 streamCfg.enmFormat = obtStream.enmFormat;
682 streamCfg.uHz = obtStream.uFreq;
683 streamCfg.cChannels = pCfg->cChannels;
684 streamCfg.enmEndianness = obtStream.enmENDIANNESS;
685
686 rc = DrvAudioHlpStreamCfgToProps(&streamCfg, &pStream->Props);
687 if (RT_SUCCESS(rc))
688 {
689 cSamples = (obtStream.cFragments * obtStream.cbFragmentSize)
690 >> pStream->Props.cShift;
691 if (!cSamples)
692 rc = VERR_INVALID_PARAMETER;
693 }
694 }
695
696 if (RT_SUCCESS(rc))
697 {
698 size_t cbSample = (1 << pStream->Props.cShift);
699
700 size_t cbBuf = cSamples * cbSample;
701 void *pvBuf = RTMemAlloc(cbBuf);
702 if (!pvBuf)
703 {
704 LogRel(("OSS: Failed allocating capturing buffer with %RU32 samples (%zu bytes per sample)\n",
705 cSamples, cbSample));
706 rc = VERR_NO_MEMORY;
707 break;
708 }
709
710 pStrm->hFile = hFile;
711 pStrm->pvBuf = pvBuf;
712 pStrm->cbBuf = cbBuf;
713
714 if (pcSamples)
715 *pcSamples = cSamples;
716 }
717
718 } while (0);
719
720 if (RT_FAILURE(rc))
721 ossStreamClose(&hFile);
722
723 LogFlowFuncLeaveRC(rc);
724 return rc;
725}
726
727static int ossCreateStreamOut(PPDMIHOSTAUDIO pInterface,
728 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg,
729 uint32_t *pcSamples)
730{
731 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
732 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
733 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
734
735 PDRVHOSTOSSAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTOSSAUDIO(pInterface);
736 POSSAUDIOSTREAMOUT pStrm = (POSSAUDIOSTREAMOUT)pStream;
737
738 int rc;
739 int hFile = -1;
740
741 do
742 {
743 uint32_t cSamples;
744
745 OSSAUDIOSTREAMCFG reqStream, obtStream;
746 reqStream.enmFormat = pCfg->enmFormat;
747 reqStream.uFreq = pCfg->uHz;
748 reqStream.cChannels = pCfg->cChannels;
749 reqStream.cFragments = s_OSSConf.nfrags;
750 reqStream.cbFragmentSize = s_OSSConf.fragsize;
751
752 rc = ossStreamOpen(s_OSSConf.devpath_out, O_WRONLY | O_NONBLOCK, &reqStream, &obtStream, &hFile);
753 if (RT_SUCCESS(rc))
754 {
755 if (obtStream.cFragments * obtStream.cbFragmentSize & pStream->Props.uAlign)
756 LogRel(("OSS: Warning: Misaligned playback buffer: Size = %zu, Alignment = %u\n",
757 obtStream.cFragments * obtStream.cbFragmentSize,
758 pStream->Props.uAlign + 1));
759
760 PDMAUDIOSTREAMCFG streamCfg;
761 streamCfg.enmFormat = obtStream.enmFormat;
762 streamCfg.uHz = obtStream.uFreq;
763 streamCfg.cChannels = pCfg->cChannels;
764 streamCfg.enmEndianness = obtStream.enmENDIANNESS;
765
766 rc = DrvAudioHlpStreamCfgToProps(&streamCfg, &pStream->Props);
767 if (RT_SUCCESS(rc))
768 cSamples = (obtStream.cFragments * obtStream.cbFragmentSize)
769 >> pStream->Props.cShift;
770 }
771
772 if (RT_SUCCESS(rc))
773 {
774 pStrm->fMemMapped = false;
775
776 size_t cbSamples = cSamples << pStream->Props.cShift;
777 Assert(cbSamples);
778
779#ifndef RT_OS_L4
780 if (s_OSSConf.try_mmap)
781 {
782 pStrm->pvBuf = mmap(0, cbSamples, PROT_READ | PROT_WRITE, MAP_SHARED, hFile, 0);
783 if (pStrm->pvBuf == MAP_FAILED)
784 {
785 LogRel(("OSS: Failed to memory map %zu bytes of playback buffer: %s\n", cbSamples, strerror(errno)));
786 rc = RTErrConvertFromErrno(errno);
787 break;
788 }
789 else
790 {
791 int mask = 0;
792 if (ioctl(hFile, SNDCTL_DSP_SETTRIGGER, &mask) < 0)
793 {
794 LogRel(("OSS: Failed to retrieve initial trigger mask for playback buffer: %s\n", strerror(errno)));
795 rc = RTErrConvertFromErrno(errno);
796 /* Note: No break here, need to unmap file first! */
797 }
798 else
799 {
800 mask = PCM_ENABLE_OUTPUT;
801 if (ioctl (hFile, SNDCTL_DSP_SETTRIGGER, &mask) < 0)
802 {
803 LogRel(("OSS: Failed to retrieve PCM_ENABLE_OUTPUT mask: %s\n", strerror(errno)));
804 rc = RTErrConvertFromErrno(errno);
805 /* Note: No break here, need to unmap file first! */
806 }
807 else
808 pStrm->fMemMapped = true;
809 }
810
811 if (RT_FAILURE(rc))
812 {
813 int rc2 = munmap(pStrm->pvBuf, cbSamples);
814 if (rc2)
815 LogRel(("OSS: Failed to memory unmap playback buffer: %s\n", strerror(errno)));
816 break;
817 }
818 }
819 }
820#endif /* !RT_OS_L4 */
821
822 /* Memory mapping failed above? Try allocating an own buffer. */
823#ifndef RT_OS_L4
824 if (!pStrm->fMemMapped)
825 {
826#endif
827 void *pvBuf = RTMemAlloc(cbSamples);
828 if (!pvBuf)
829 {
830 LogRel(("OSS: Failed allocating playback buffer with %RU32 samples (%zu bytes)\n", cSamples, cbSamples));
831 rc = VERR_NO_MEMORY;
832 break;
833 }
834
835 pStrm->hFile = hFile;
836 pStrm->pvBuf = pvBuf;
837 pStrm->cbBuf = cbSamples;
838#ifndef RT_OS_L4
839 }
840#endif
841 if (pcSamples)
842 *pcSamples = cSamples;
843 }
844
845 } while (0);
846
847 if (RT_FAILURE(rc))
848 ossStreamClose(&hFile);
849
850 LogFlowFuncLeaveRC(rc);
851 return rc;
852}
853
854static DECLCALLBACK(bool) drvHostOSSAudioIsEnabled(PPDMIHOSTAUDIO pInterface, PDMAUDIODIR enmDir)
855{
856 NOREF(pInterface);
857 NOREF(enmDir);
858 return true; /* Always all enabled. */
859}
860
861static DECLCALLBACK(int) drvHostOSSAudioStreamPlay(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream,
862 uint32_t *pcSamplesPlayed)
863{
864 NOREF(pInterface);
865 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
866
867 POSSAUDIOSTREAMOUT pStrm = (POSSAUDIOSTREAMOUT)pStream;
868
869 int rc = VINF_SUCCESS;
870 uint32_t cbReadTotal = 0;
871 count_info cntinfo;
872
873 do
874 {
875 size_t cbBuf = AudioMixBufSizeBytes(&pStream->MixBuf);
876
877 uint32_t cLive = AudioMixBufAvail(&pStream->MixBuf);
878 uint32_t cToRead;
879
880#ifndef RT_OS_L4
881 if (pStrm->fMemMapped)
882 {
883 /* Get current playback pointer. */
884 int rc2 = ioctl(pStrm->hFile, SNDCTL_DSP_GETOPTR, &cntinfo);
885 if (!rc2)
886 {
887 LogRel(("OSS: Failed to retrieve current playback pointer: %s\n",
888 strerror(errno)));
889 rc = RTErrConvertFromErrno(errno);
890 break;
891 }
892
893 /* Nothing to play? */
894 if (cntinfo.ptr == pStrm->old_optr)
895 break;
896
897 int cbData;
898 if (cntinfo.ptr > pStrm->old_optr)
899 cbData = cntinfo.ptr - pStrm->old_optr;
900 else
901 cbData = cbBuf + cntinfo.ptr - pStrm->old_optr;
902 Assert(cbData);
903
904 cToRead = RT_MIN((uint32_t)AUDIOMIXBUF_B2S(&pStream->MixBuf, cbData),
905 cLive);
906 }
907 else
908 {
909#endif
910 audio_buf_info abinfo;
911 int rc2 = ioctl(pStrm->hFile, SNDCTL_DSP_GETOSPACE, &abinfo);
912 if (rc2 < 0)
913 {
914 LogRel(("OSS: Failed to retrieve current playback buffer: %s\n", strerror(errno)));
915 rc = RTErrConvertFromErrno(errno);
916 break;
917 }
918
919 if ((size_t)abinfo.bytes > cbBuf)
920 {
921 LogFlowFunc(("Warning: Invalid available size, size=%d, bufsize=%zu\n", abinfo.bytes, cbBuf));
922 abinfo.bytes = cbBuf;
923 /* Keep going. */
924 }
925
926 if (abinfo.bytes < 0)
927 {
928 LogFlowFunc(("Warning: Invalid available size, size=%d, bufsize=%zu\n", abinfo.bytes, cbBuf));
929 rc = VERR_INVALID_PARAMETER;
930 break;
931 }
932
933 cToRead = RT_MIN((uint32_t)AUDIOMIXBUF_B2S(&pStream->MixBuf, abinfo.bytes), cLive);
934 if (!cToRead)
935 break;
936#ifndef RT_OS_L4
937 }
938#endif
939 size_t cbToRead = AUDIOMIXBUF_S2B(&pStream->MixBuf, cToRead);
940 LogFlowFunc(("cbToRead=%zu\n", cbToRead));
941
942 uint32_t cRead, cbRead;
943 while (cbToRead)
944 {
945 rc = AudioMixBufReadCirc(&pStream->MixBuf, pStrm->pvBuf, cbToRead, &cRead);
946 if (RT_FAILURE(rc))
947 break;
948
949 cbRead = AUDIOMIXBUF_S2B(&pStream->MixBuf, cRead);
950 ssize_t cbWritten = write(pStrm->hFile, pStrm->pvBuf, cbRead);
951 if (cbWritten == -1)
952 {
953 LogRel(("OSS: Failed writing output data: %s\n", strerror(errno)));
954 rc = RTErrConvertFromErrno(errno);
955 break;
956 }
957
958 Assert(cbToRead >= cbRead);
959 cbToRead -= cbRead;
960 cbReadTotal += cbRead;
961 }
962
963#ifndef RT_OS_L4
964 /* Update read pointer. */
965 if (pStrm->fMemMapped)
966 pStrm->old_optr = cntinfo.ptr;
967#endif
968
969 } while(0);
970
971 if (RT_SUCCESS(rc))
972 {
973 uint32_t cReadTotal = AUDIOMIXBUF_B2S(&pStream->MixBuf, cbReadTotal);
974 if (cReadTotal)
975 AudioMixBufFinish(&pStream->MixBuf, cReadTotal);
976
977 if (pcSamplesPlayed)
978 *pcSamplesPlayed = cReadTotal;
979
980 LogFlowFunc(("cReadTotal=%RU32 (%RU32 bytes), rc=%Rrc\n", cReadTotal, cbReadTotal, rc));
981 }
982
983 LogFlowFuncLeaveRC(rc);
984 return rc;
985}
986
987static DECLCALLBACK(void) drvHostOSSAudioShutdown(PPDMIHOSTAUDIO pInterface)
988{
989 NOREF(pInterface);
990}
991
992static DECLCALLBACK(PDMAUDIOBACKENDSTS) drvHostOSSAudioGetStatus(PPDMIHOSTAUDIO pInterface, PDMAUDIODIR enmDir)
993{
994 AssertPtrReturn(pInterface, PDMAUDIOBACKENDSTS_UNKNOWN);
995
996 return PDMAUDIOBACKENDSTS_RUNNING;
997}
998
999static DECLCALLBACK(int) drvHostOSSAudioStreamCreate(PPDMIHOSTAUDIO pInterface,
1000 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg, uint32_t *pcSamples)
1001{
1002 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1003 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1004 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
1005
1006 int rc;
1007 if (pCfg->enmDir == PDMAUDIODIR_IN)
1008 rc = ossCreateStreamIn(pInterface, pStream, pCfg, pcSamples);
1009 else
1010 rc = ossCreateStreamOut(pInterface, pStream, pCfg, pcSamples);
1011
1012 LogFlowFunc(("%s: rc=%Rrc\n", pStream->szName, rc));
1013 return rc;
1014}
1015
1016static DECLCALLBACK(int) drvHostOSSAudioStreamDestroy(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
1017{
1018 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1019 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1020
1021 int rc;
1022 if (pStream->enmDir == PDMAUDIODIR_IN)
1023 rc = ossDestroyStreamIn(pInterface, pStream);
1024 else
1025 rc = ossDestroyStreamOut(pInterface, pStream);
1026
1027 return rc;
1028}
1029
1030static DECLCALLBACK(int) drvHostOSSAudioStreamControl(PPDMIHOSTAUDIO pInterface,
1031 PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd)
1032{
1033 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1034 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1035
1036 Assert(pStream->enmCtx == PDMAUDIOSTREAMCTX_HOST);
1037
1038 int rc;
1039 if (pStream->enmDir == PDMAUDIODIR_IN)
1040 rc = ossControlStreamIn(pInterface, pStream, enmStreamCmd);
1041 else
1042 rc = ossControlStreamOut(pInterface, pStream, enmStreamCmd);
1043
1044 return rc;
1045}
1046
1047static DECLCALLBACK(int) drvHostOSSAudioStreamIterate(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
1048{
1049 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1050 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1051
1052 LogFlowFuncEnter();
1053
1054 /* Nothing to do here for OSS. */
1055 return VINF_SUCCESS;
1056}
1057
1058static DECLCALLBACK(PDMAUDIOSTRMSTS) drvHostOSSAudioStreamGetStatus(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
1059{
1060 NOREF(pInterface);
1061 NOREF(pStream);
1062
1063 PDMAUDIOSTRMSTS strmSts = PDMAUDIOSTRMSTS_FLAG_INITIALIZED
1064 | PDMAUDIOSTRMSTS_FLAG_ENABLED;
1065
1066 strmSts |= pStream->enmDir == PDMAUDIODIR_IN
1067 ? PDMAUDIOSTRMSTS_FLAG_DATA_READABLE
1068 : PDMAUDIOSTRMSTS_FLAG_DATA_WRITABLE;
1069
1070 return strmSts;
1071}
1072/**
1073 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
1074 */
1075static DECLCALLBACK(void *) drvHostOSSAudioQueryInterface(PPDMIBASE pInterface, const char *pszIID)
1076{
1077 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
1078 PDRVHOSTOSSAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTOSSAUDIO);
1079 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
1080 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIHOSTAUDIO, &pThis->IHostAudio);
1081
1082 return NULL;
1083}
1084
1085/**
1086 * Constructs an OSS audio driver instance.
1087 *
1088 * @copydoc FNPDMDRVCONSTRUCT
1089 */
1090static DECLCALLBACK(int) drvHostOSSAudioConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
1091{
1092 PDRVHOSTOSSAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTOSSAUDIO);
1093 LogRel(("Audio: Initializing OSS driver\n"));
1094
1095 /*
1096 * Init the static parts.
1097 */
1098 pThis->pDrvIns = pDrvIns;
1099 /* IBase */
1100 pDrvIns->IBase.pfnQueryInterface = drvHostOSSAudioQueryInterface;
1101 /* IHostAudio */
1102 PDMAUDIO_IHOSTAUDIO_CALLBACKS(drvHostOSSAudio);
1103
1104 return VINF_SUCCESS;
1105}
1106
1107/**
1108 * Char driver registration record.
1109 */
1110const PDMDRVREG g_DrvHostOSSAudio =
1111{
1112 /* u32Version */
1113 PDM_DRVREG_VERSION,
1114 /* szName */
1115 "OSSAudio",
1116 /* szRCMod */
1117 "",
1118 /* szR0Mod */
1119 "",
1120 /* pszDescription */
1121 "OSS audio host driver",
1122 /* fFlags */
1123 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
1124 /* fClass. */
1125 PDM_DRVREG_CLASS_AUDIO,
1126 /* cMaxInstances */
1127 ~0U,
1128 /* cbInstance */
1129 sizeof(DRVHOSTOSSAUDIO),
1130 /* pfnConstruct */
1131 drvHostOSSAudioConstruct,
1132 /* pfnDestruct */
1133 NULL,
1134 /* pfnRelocate */
1135 NULL,
1136 /* pfnIOCtl */
1137 NULL,
1138 /* pfnPowerOn */
1139 NULL,
1140 /* pfnReset */
1141 NULL,
1142 /* pfnSuspend */
1143 NULL,
1144 /* pfnResume */
1145 NULL,
1146 /* pfnAttach */
1147 NULL,
1148 /* pfnDetach */
1149 NULL,
1150 /* pfnPowerOff */
1151 NULL,
1152 /* pfnSoftReset */
1153 NULL,
1154 /* u32EndVersion */
1155 PDM_DRVREG_VERSION
1156};
1157
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