flightgear/3rdparty/iaxclient/lib/portmixer/px_mac/px_mac.c
2022-10-20 20:29:11 +08:00

410 lines
9.4 KiB
C
Executable File

/*
* PortMixer
* Mac OS 9 implementation
*
* Copyright (c) 2002
*
* Written by Dominic Mazzoni
*
* PortMixer is intended to work side-by-side with PortAudio,
* the Portable Real-Time Audio Library by Ross Bencina and
* Phil Burk.
*
* Permission is hereby granted, free of charge, to any person obtaining
* a copy of this software and associated documentation files
* (the "Software"), to deal in the Software without restriction,
* including without limitation the rights to use, copy, modify, merge,
* publish, distribute, sublicense, and/or sell copies of the Software,
* and to permit persons to whom the Software is furnished to do so,
* subject to the following conditions:
*
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
*
* Any person wishing to distribute modifications to the Software is
* requested to send the modifications to the original developer so that
* they can be incorporated into the canonical version.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR
* ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
* CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
* WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*
*/
#include <stdlib.h>
#include <string.h>
#include "portaudio.h"
#include "pa_host.h"
#include "portmixer.h"
#define PA_MAX_NUM_HOST_BUFFERS (16) /* Do not exceed!! */
typedef struct MultiBuffer
{
char *buffers[PA_MAX_NUM_HOST_BUFFERS];
int numBuffers;
int nextWrite;
int nextRead;
}
MultiBuffer;
/* Define structure to contain all Macintosh specific data. */
typedef struct PaHostSoundControl
{
UInt64 pahsc_EntryCount;
double pahsc_InverseMicrosPerHostBuffer; /* 1/Microseconds of real-time audio per user buffer. */
/* Use char instead of Boolean for atomic operation. */
volatile char pahsc_IsRecording; /* Recording in progress. Set by foreground. Cleared by background. */
volatile char pahsc_StopRecording; /* Signal sent to background. */
volatile char pahsc_IfInsideCallback;
/* Input */
SPB pahsc_InputParams;
SICompletionUPP pahsc_InputCompletionProc;
MultiBuffer pahsc_InputMultiBuffer;
int32 pahsc_BytesPerInputHostBuffer;
int32 pahsc_InputRefNum;
/* Output */
CmpSoundHeader pahsc_SoundHeaders[PA_MAX_NUM_HOST_BUFFERS];
int32 pahsc_BytesPerOutputHostBuffer;
SndChannelPtr pahsc_Channel;
SndCallBackUPP pahsc_OutputCompletionProc;
int32 pahsc_NumOutsQueued;
int32 pahsc_NumOutsPlayed;
PaTimestamp pahsc_NumFramesDone;
UInt64 pahsc_WhenFramesDoneIncremented;
/* Init Time -------------- */
int32 pahsc_NumHostBuffers;
int32 pahsc_FramesPerHostBuffer;
int32 pahsc_UserBuffersPerHostBuffer;
int32 pahsc_MinFramesPerHostBuffer; /* Can vary depending on virtual memory usage. */
}
PaHostSoundControl;
typedef struct PxSource
{
char name[256];
} PxSource;
typedef struct PxInfo
{
SPB *input;
int32 inputRefNum;
SndChannelPtr output;
int32 numSources;
PxSource *sources;
} PxInfo;
int Px_GetNumMixers( void *pa_stream )
{
return 0;
}
const char *Px_GetMixerName( void *pa_stream, int index )
{
return "Mac Sound Manager";
}
PxMixer *Px_OpenMixer( void *pa_stream, int index )
{
PxInfo *info;
internalPortAudioStream *past;
PaHostSoundControl *macInfo;
OSErr err;
int i, j;
Handle h;
unsigned char *data;
info = (PxInfo *)malloc(sizeof(PxInfo));
if (info == NULL)
return NULL;
past = (internalPortAudioStream *) pa_stream;
macInfo = (PaHostSoundControl *) past->past_DeviceData;
info->input = &macInfo->pahsc_InputParams;
info->inputRefNum = macInfo->pahsc_InputRefNum;
info->output = macInfo->pahsc_Channel;
info->numSources = 0;
info->sources = NULL;
err = SPBGetDeviceInfo (info->inputRefNum, siInputSourceNames, &h);
if (err)
return (PxMixer *)info;
HLock(h);
HNoPurge(h);
data = (unsigned char *)*h;
info->numSources = ((short *)data)[0];
if (info->numSources <= 0 || info->numSources > 50) {
HUnlock(h);
return (PxMixer *)info;
}
info->sources = (PxSource *)malloc(info->numSources * sizeof(PxSource));
data += 2;
for(i=0; i<info->numSources; i++) {
int len = *data++;
if (len > 63) {
info->numSources = 0;
free(info->sources);
info->sources = NULL;
HUnlock(h);
return (PxMixer *)info;
}
for(j=0; j<len; j++)
info->sources[i].name[j] = *data++;
info->sources[i].name[len] = 0;
}
HUnlock(h);
return (PxMixer *)info;
}
/*
Px_CloseMixer() closes a mixer opened using Px_OpenMixer and frees any
memory associated with it.
*/
void Px_CloseMixer(PxMixer *mixer)
{
PxInfo *info = (PxInfo *)mixer;
if (info->sources)
free(info->sources);
free(info);
}
/*
Master (output) volume
*/
PxVolume Px_GetMasterVolume( PxMixer *mixer )
{
PxInfo *info = (PxInfo *)mixer;
return 0.0;
}
void Px_SetMasterVolume( PxMixer *mixer, PxVolume volume )
{
PxInfo *info = (PxInfo *)mixer;
}
/*
PCM output volume
*/
int Px_SupportsPCMOutputVolume( PxMixer* mixer )
{
return 1 ;
}
PxVolume Px_GetPCMOutputVolume( PxMixer *mixer )
{
PxInfo *info = (PxInfo *)mixer;
OSErr err;
long packedVol;
SndCommand cmd;
cmd.cmd = getVolumeCmd;
cmd.param1 = 0;
cmd.param2 = (long)&packedVol;
err = SndDoImmediate(info->output, &cmd);
if (err)
return 0.0;
return ((packedVol & 0xFFFF) + ((packedVol & 0xFFFF0000) >> 16) / 2.0) / 256.0;
}
void Px_SetPCMOutputVolume( PxMixer *mixer, PxVolume volume )
{
PxInfo *info = (PxInfo *)mixer;
OSErr err;
long packedVol;
SndCommand cmd;
packedVol = (unsigned long)volume * 256.0;
packedVol += (packedVol << 16);
cmd.cmd = volumeCmd;
cmd.param1 = 0;
cmd.param2 = packedVol;
err = SndDoImmediate(info->output, &cmd);
}
/*
All output volumes
*/
int Px_GetNumOutputVolumes( PxMixer *mixer )
{
PxInfo *info = (PxInfo *)mixer;
return 0;
}
const char *Px_GetOutputVolumeName( PxMixer *mixer, int i )
{
PxInfo *info = (PxInfo *)mixer;
return NULL;
}
PxVolume Px_GetOutputVolume( PxMixer *mixer, int i )
{
PxInfo *info = (PxInfo *)mixer;
return NULL;
}
void Px_SetOutputVolume( PxMixer *mixer, int i, PxVolume volume )
{
PxInfo *info = (PxInfo *)mixer;
}
/*
Input sources
*/
int Px_GetNumInputSources( PxMixer *mixer )
{
PxInfo *info = (PxInfo *)mixer;
return info->numSources;
}
const char *Px_GetInputSourceName( PxMixer *mixer, int i)
{
PxInfo *info = (PxInfo *)mixer;
if (i >= 0 && i < info->numSources)
return info->sources[i].name;
else
return "";
}
int Px_GetCurrentInputSource( PxMixer *mixer )
{
PxInfo *info = (PxInfo *)mixer;
short selected;
OSErr err;
err = SPBGetDeviceInfo (info->inputRefNum, siInputSource, &selected);
if (err)
return 0;
return selected - 1;
}
void Px_SetCurrentInputSource( PxMixer *mixer, int i )
{
PxInfo *info = (PxInfo *)mixer;
short selected = i+1;
OSErr err;
err = SPBSetDeviceInfo (info->inputRefNum, siInputSource, &selected);
}
/*
Input volume
*/
PxVolume Px_GetInputVolume( PxMixer *mixer )
{
PxInfo *info = (PxInfo *)mixer;
Fixed fixedGain;
PxVolume vol;
OSErr err;
if (info->input) {
err = SPBGetDeviceInfo(info->inputRefNum, siInputGain, (Ptr)&fixedGain);
if (err)
return 0.0;
vol = (fixedGain / 65536.0) - 0.5;
return vol;
}
return 0.0;
}
void Px_SetInputVolume( PxMixer *mixer, PxVolume volume )
{
PxInfo *info = (PxInfo *)mixer;
Fixed fixedGain;
OSErr err;
if (info->input) {
fixedGain = (Fixed)((volume + 0.5) * 65536.0);
err = SPBSetDeviceInfo(info->inputRefNum, siInputGain, (Ptr)&fixedGain);
}
}
/*
Balance
*/
int Px_SupportsOutputBalance( PxMixer *mixer )
{
return 0;
}
PxBalance Px_GetOutputBalance( PxMixer *mixer )
{
return 0.0;
}
void Px_SetOutputBalance( PxMixer *mixer, PxBalance balance )
{
}
/*
Playthrough
*/
int Px_SupportsPlaythrough( PxMixer *mixer )
{
PxInfo *info = (PxInfo *)mixer;
return (info->input != NULL);
}
PxVolume Px_GetPlaythrough( PxMixer *mixer )
{
PxInfo *info = (PxInfo *)mixer;
OSErr err;
short level;
err = SPBGetDeviceInfo(info->inputRefNum, siPlayThruOnOff, (Ptr)&level);
if (err)
return 0.0;
if (level < 0)
level = 0;
if (level > 7)
level = 7;
return (PxVolume)(level / 7.0);
}
void Px_SetPlaythrough( PxMixer *mixer, PxVolume volume )
{
PxInfo *info = (PxInfo *)mixer;
OSErr err;
short level = (int)(volume * 7.0 + 0.5);
err = SPBSetDeviceInfo(info->inputRefNum, siPlayThruOnOff, (Ptr)&level);
}