/* File: AudioFileObject.cpp Abstract: AudioFileObject.h Version: 1.1 Disclaimer: IMPORTANT: This Apple software is supplied to you by Apple Inc. ("Apple") in consideration of your agreement to the following terms, and your use, installation, modification or redistribution of this Apple software constitutes acceptance of these terms. If you do not agree with these terms, please do not use, install, modify or redistribute this Apple software. In consideration of your agreement to abide by the following terms, and subject to these terms, Apple grants you a personal, non-exclusive license, under Apple's copyrights in this original Apple software (the "Apple Software"), to use, reproduce, modify and redistribute the Apple Software, with or without modifications, in source and/or binary forms; provided that if you redistribute the Apple Software in its entirety and without modifications, you must retain this notice and the following text and disclaimers in all such redistributions of the Apple Software. Neither the name, trademarks, service marks or logos of Apple Inc. may be used to endorse or promote products derived from the Apple Software without specific prior written permission from Apple. Except as expressly stated in this notice, no other rights or licenses, express or implied, are granted by Apple herein, including but not limited to any patent rights that may be infringed by your derivative works or by other works in which the Apple Software may be incorporated. The Apple Software is provided by Apple on an "AS IS" basis. APPLE MAKES NO WARRANTIES, EXPRESS OR IMPLIED, INCLUDING WITHOUT LIMITATION THE IMPLIED WARRANTIES OF NON-INFRINGEMENT, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE, REGARDING THE APPLE SOFTWARE OR ITS USE AND OPERATION ALONE OR IN COMBINATION WITH YOUR PRODUCTS. IN NO EVENT SHALL APPLE BE LIABLE FOR ANY SPECIAL, INDIRECT, INCIDENTAL OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) ARISING IN ANY WAY OUT OF THE USE, REPRODUCTION, MODIFICATION AND/OR DISTRIBUTION OF THE APPLE SOFTWARE, HOWEVER CAUSED AND WHETHER UNDER THEORY OF CONTRACT, TORT (INCLUDING NEGLIGENCE), STRICT LIABILITY OR OTHERWISE, EVEN IF APPLE HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. Copyright (C) 2014 Apple Inc. All Rights Reserved. */ #include "AudioFileObject.h" #include "CADebugMacros.h" #include #include #define kAudioFileNoCacheMask 0x20 const SInt64 kScanToEnd = 0x7fffFFFFffffFFFFLL; ////////////////////////////////////////////////////////////////////////////////////////////////////////// AudioFileObject::~AudioFileObject() { delete mDataSource; DeletePacketTable(); SetURL(NULL); } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::DoCreate( CFURLRef inFileRef, const AudioStreamBasicDescription *inFormat, UInt32 inFlags) { // common prologue if (!IsDataFormatValid(inFormat)) return kAudioFileUnsupportedDataFormatError; if (!IsDataFormatSupported(inFormat)) return kAudioFileUnsupportedDataFormatError; SetPermissions(kAudioFileReadWritePermission); SetAlignDataWithFillerChunks(!(inFlags & 2 /* kAudioFileFlags_DontPageAlignAudioData */ )); // call virtual method for particular format. return Create(inFileRef, inFormat); } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::Create( CFURLRef inFileRef, const AudioStreamBasicDescription *inFormat) { int fileD; OSStatus err = CreateDataFile (inFileRef, fileD); FailIf (err != noErr, Bail, "CreateDataFile failed"); SetURL (inFileRef); err = OpenFile(kAudioFileReadWritePermission, fileD); FailIf (err != noErr, Bail, "OpenFile failed"); err = SetDataFormat(inFormat); FailIf (err != noErr, Bail, "SetDataFormat failed"); mIsInitialized = false; Bail: return err; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::DoOpen( CFURLRef inFileRef, SInt8 inPermissions, int inFD) { OSStatus err = noErr; SetPermissions(inPermissions); err = Open(inFileRef, inPermissions, inFD); FailIf (err != noErr, Bail, "Open failed"); err = ValidateFormatAndData(); FailIf (err != noErr, Bail, "ValidateFormatAndData failed"); Bail: return err; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::Open( CFURLRef inFileRef, SInt8 inPermissions, int inFD) { if (!(inPermissions & kAudioFileReadPermission)) return kAudioFilePermissionsError; // file must have read permissions SetURL(inFileRef); OSStatus err = OpenFile(inPermissions, inFD); FailIf (err != noErr, Bail, "OpenFile failed"); err = OpenFromDataSource(); FailIf (err != noErr, Bail, "OpenFromDataSource failed"); Bail: return err; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::ValidateFormatAndData() { AudioStreamBasicDescription asbd = GetDataFormat(); if (!IsDataFormatValid(&asbd)) return kAudioFileInvalidFileError; if (asbd.mFormatID == kAudioFormatLinearPCM) { SInt64 maxPackets = GetNumBytes() / asbd.mBytesPerPacket; if (GetNumPackets() > maxPackets) return kAudioFileInvalidFileError; } return noErr; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::DoOpenWithCallbacks( void * inRefCon, AudioFile_ReadProc inReadFunc, AudioFile_WriteProc inWriteFunc, AudioFile_GetSizeProc inGetSizeFunc, AudioFile_SetSizeProc inSetSizeFunc) { SInt8 perms = (inSetSizeFunc || inWriteFunc) ? kAudioFileReadWritePermission : kAudioFileReadPermission; SetPermissions(perms); DataSource* dataSource = new Seekable_DataSource(inRefCon, inReadFunc, inWriteFunc, inGetSizeFunc, inSetSizeFunc); SetDataSource(dataSource); return OpenFromDataSource(); } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::OpenFromDataSource(void) { return noErr; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::DoInitializeWithCallbacks( void * inRefCon, AudioFile_ReadProc inReadFunc, AudioFile_WriteProc inWriteFunc, AudioFile_GetSizeProc inGetSizeFunc, AudioFile_SetSizeProc inSetSizeFunc, UInt32 inFileType, const AudioStreamBasicDescription *inFormat, UInt32 inFlags) { DataSource* dataSource = new Seekable_DataSource(inRefCon, inReadFunc, inWriteFunc, inGetSizeFunc, inSetSizeFunc); if (!dataSource->CanWrite()) return -54/*permErr*/; dataSource->SetSize(0); SetDataSource(dataSource); SetPermissions(kAudioFileReadWritePermission); SetAlignDataWithFillerChunks(!(inFlags & 2 /* kAudioFileFlags_DontPageAlignAudioData */ )); OSStatus err = SetDataFormat(inFormat); if (err) return err; return InitializeDataSource(inFormat, inFlags); } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::DoInitialize( CFURLRef inFileRef, const AudioStreamBasicDescription *inFormat, UInt32 inFlags) { SetURL (inFileRef); SetPermissions(kAudioFileReadWritePermission); SetAlignDataWithFillerChunks(!(inFlags & 2 /* kAudioFileFlags_DontPageAlignAudioData */ )); return Initialize(inFileRef, inFormat, inFlags); } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::Initialize( CFURLRef inFileRef, const AudioStreamBasicDescription *inFormat, UInt32 inFlags) { OSStatus err = noErr; UInt8 fPath[FILENAME_MAX]; if (!CFURLGetFileSystemRepresentation (inFileRef, true, fPath, FILENAME_MAX)) return kAudio_FileNotFoundError; #if TARGET_OS_WIN32 int filePerms = 0; int flags = O_TRUNC | O_RDWR | O_BINARY; #else mode_t filePerms = 0; int flags = O_TRUNC | O_RDWR; #endif int fileD = open((const char*)fPath, flags, filePerms); if (fileD < 0) return AudioFileTranslateErrno(errno); err = OpenFile(kAudioFileReadWritePermission, fileD); FailIf (err != noErr, Bail, "OpenFile failed"); // don't need to do this as open has an option to truncate the file // GetDataSource()->SetSize(0); err = SetDataFormat(inFormat); FailIf (err != noErr, Bail, "SetDataFormat failed"); InitializeDataSource(inFormat, inFlags); Bail: return err; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::InitializeDataSource(const AudioStreamBasicDescription *inFormat, UInt32 /*inFlags*/) { return noErr; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::DoClose() { OSStatus err = UpdateSizeIfNeeded(); if (err) return err; return Close(); } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::Close() { try { delete mDataSource; mDataSource = 0; } catch (OSStatus err) { return err; } catch (...) { return kAudioFileUnspecifiedError; } return noErr; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::Optimize() { // default is that nothing needs to be done. This happens to be true for Raw, SD2 and NeXT/Sun types. SetIsOptimized(true); return noErr; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::DoOptimize() { if (!CanWrite()) return kAudioFilePermissionsError; OSStatus err = UpdateSizeIfNeeded(); if (err) return err; if (IsOptimized()) return noErr; err = Optimize(); return err; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::UpdateNumBytes(SInt64 inNumBytes) { OSStatus err = noErr; if (inNumBytes != GetNumBytes()) { SetNumBytes(inNumBytes); // #warning " this will not work for vbr formats" SetNumPackets(GetNumBytes() / mDataFormat.mBytesPerPacket); SizeChanged(); } return err; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::UpdateNumPackets(SInt64 inNumPackets) { OSStatus err = noErr; if (inNumPackets != GetNumPackets()) { // sync current state. SetNeedsSizeUpdate(true); UpdateSizeIfNeeded(); SetNumPackets(inNumPackets); // #warning " this will not work for vbr formats" SetNumBytes(GetNumPackets() * mDataFormat.mBytesPerFrame); SizeChanged(); } return err; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::PacketToFrame(SInt64 inPacket, SInt64& outFirstFrameInPacket) { if (mDataFormat.mFramesPerPacket == 0) { OSStatus err = ScanForPackets(inPacket+1); // the packet count must be one greater than the packet index if (err) return err; SInt64 packetTableSize = GetPacketTableSize(); if (mPacketTable && inPacket >= packetTableSize) return kAudioFileEndOfFileError; CompressedPacketTable* packetTable = GetPacketTable(); if (!packetTable) return kAudioFileInvalidPacketOffsetError; if (inPacket < 0 || inPacket >= packetTableSize) return kAudioFileInvalidPacketOffsetError; outFirstFrameInPacket = (*packetTable)[(size_t)inPacket].mFrameOffset; } else { outFirstFrameInPacket = inPacket * mDataFormat.mFramesPerPacket; } return noErr; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::FrameToPacket(SInt64 inFrame, SInt64& outPacket, UInt32& outFrameOffsetInPacket) { if (mDataFormat.mFramesPerPacket == 0) { CompressedPacketTable* packetTable = GetPacketTable(); if (!packetTable) return kAudioFileInvalidPacketOffsetError; // search packet table AudioStreamPacketDescriptionExtended pext; memset(&pext, 0, sizeof(pext)); pext.mFrameOffset = inFrame; CompressedPacketTable::iterator iter = std::lower_bound(packetTable->begin(), packetTable->end(), pext); if (iter == packetTable->end()) return kAudioFileInvalidPacketOffsetError; if (iter > packetTable->begin()) --iter; outPacket = iter - packetTable->begin(); outFrameOffsetInPacket = (UInt32)(inFrame - iter->mFrameOffset); } else { outPacket = inFrame / mDataFormat.mFramesPerPacket; outFrameOffsetInPacket = (UInt32)(inFrame % mDataFormat.mFramesPerPacket); } return noErr; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::PacketToByte(AudioBytePacketTranslation* abpt) { if (abpt->mPacket < 0) return kAudioFileInvalidPacketOffsetError; if (mDataFormat.mBytesPerPacket == 0) { CompressedPacketTable* packetTable = GetPacketTable(); if (!packetTable) return kAudioFileInvalidPacketOffsetError; if (abpt->mPacket < GetPacketTableSize()) { abpt->mByte = (*packetTable)[(int)abpt->mPacket].mStartOffset; abpt->mFlags = 0; } else { SInt64 numPackets = packetTable->size(); if (numPackets < 8) return 'more' /*kAudioFileStreamError_DataUnavailable*/ ; const AudioStreamPacketDescriptionExtended lastPacket = (*packetTable)[numPackets - 1]; SInt64 bytesReadSoFar = lastPacket.mStartOffset + lastPacket.mDataByteSize; double averageBytesPerPacket = (double)(bytesReadSoFar - GetDataOffset()) / (double)numPackets; abpt->mByte = (SInt64)floor((double)abpt->mPacket * averageBytesPerPacket); abpt->mFlags = kBytePacketTranslationFlag_IsEstimate; } } else { abpt->mByte = abpt->mPacket * mDataFormat.mBytesPerPacket; abpt->mFlags = 0; } return noErr; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ inline bool byte_less_than (const AudioStreamPacketDescriptionExtended& a, const AudioStreamPacketDescriptionExtended& b) { return a.mStartOffset < b.mStartOffset; } OSStatus AudioFileObject::ByteToPacket(AudioBytePacketTranslation* abpt) { if (abpt->mByte < 0) return kAudioFileInvalidPacketOffsetError; if (mDataFormat.mBytesPerPacket == 0) { CompressedPacketTable* packetTable = GetPacketTable(); if (!packetTable) return kAudioFileInvalidPacketOffsetError; // search packet table AudioStreamPacketDescriptionExtended pext; memset(&pext, 0, sizeof(pext)); pext.mStartOffset = abpt->mByte; CompressedPacketTable::iterator iter = std::lower_bound(packetTable->begin(), packetTable->end(), pext, byte_less_than); if (iter == packetTable->end()) { SInt64 numPackets = packetTable->size(); if (numPackets < 8) return 'more' /*kAudioFileStreamError_DataUnavailable*/ ; const AudioStreamPacketDescriptionExtended lastPacket = (*packetTable)[numPackets - 1]; SInt64 bytesReadSoFar = lastPacket.mStartOffset + lastPacket.mDataByteSize; double averageBytesPerPacket = (double)(bytesReadSoFar - GetDataOffset()) / (double)numPackets; double fpacket = (double)abpt->mByte / averageBytesPerPacket; abpt->mPacket = (SInt64)floor(fpacket); abpt->mByteOffsetInPacket = (UInt32)floor((fpacket - (double)abpt->mPacket) * averageBytesPerPacket); abpt->mFlags = kBytePacketTranslationFlag_IsEstimate; } else { if (iter > packetTable->begin()) --iter; abpt->mPacket = iter - packetTable->begin(); abpt->mByteOffsetInPacket = (UInt32)(abpt->mByte - iter->mStartOffset); abpt->mFlags = 0; } } else { abpt->mPacket = abpt->mByte / mDataFormat.mBytesPerPacket; abpt->mByteOffsetInPacket = (UInt32)(abpt->mByte % mDataFormat.mBytesPerPacket); abpt->mFlags = 0; } return noErr; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::ReadBytes( Boolean inUseCache, SInt64 inStartingByte, UInt32 *ioNumBytes, void *outBuffer) { OSStatus err = noErr; UInt16 mode = SEEK_SET; SInt64 fileOffset = mDataOffset + inStartingByte; bool readingPastEnd = false; FailWithAction((ioNumBytes == NULL) || (outBuffer == NULL), err = kAudio_ParamError, Bail, "invalid num bytes parameter"); //printf("inStartingByte %lld GetNumBytes %lld\n", inStartingByte, GetNumBytes()); if (inStartingByte >= GetNumBytes()) { *ioNumBytes = 0; return kAudioFileEndOfFileError; } if ((fileOffset + *ioNumBytes) > (GetNumBytes() + mDataOffset)) { *ioNumBytes = (UInt32)(GetNumBytes() + mDataOffset - fileOffset); readingPastEnd = true; } //printf("fileOffset %lld mDataOffset %lld readingPastEnd %d\n", fileOffset, mDataOffset, readingPastEnd); if (!inUseCache) mode |= kAudioFileNoCacheMask; err = GetDataSource()->ReadBytes(mode, fileOffset, *ioNumBytes, outBuffer, ioNumBytes); if (readingPastEnd && err == noErr) err = kAudioFileEndOfFileError; Bail: return err; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::WriteBytes( Boolean inUseCache, SInt64 inStartingByte, UInt32 *ioNumBytes, const void *inBuffer) { OSStatus err = noErr; UInt16 mode = SEEK_SET; bool extendingTheAudioData; if (!CanWrite()) return kAudioFilePermissionsError; FailWithAction((ioNumBytes == NULL) || (inBuffer == NULL), err = kAudioFileUnspecifiedError, Bail, "invalid parameters"); // Do not try to write to a postion greater than 32 bits for some file types // see if starting byte + ioNumBytes is greater than 32 bits // if so, see if file type supports this and bail if not err = IsValidFilePosition(inStartingByte + *ioNumBytes); FailIf(err != noErr, Bail, "invalid file position"); extendingTheAudioData = inStartingByte + *ioNumBytes > GetNumBytes(); // if file is not optimized, then do not write data that would overwrite chunks following the sound data chunk FailWithAction( extendingTheAudioData && !IsOptimized(), err = kAudioFileNotOptimizedError, Bail, "Can't write more data until the file is optimized"); if (!inUseCache) mode |= kAudioFileNoCacheMask; err = GetDataSource()->WriteBytes(mode, mDataOffset + inStartingByte, *ioNumBytes, inBuffer, ioNumBytes); FailIf(err != noErr, Bail, "couldn't write new data"); if (extendingTheAudioData) { SInt64 nuEOF; // Get the total bytes of audio data SInt64 nuByteTotal; err = GetDataSource()->GetSize(nuEOF); FailIf(err != noErr, Bail, "GetSize failed"); nuByteTotal = nuEOF - mDataOffset; err = UpdateNumBytes(nuByteTotal); } Bail: return err; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::ReadPackets( Boolean inUseCache, UInt32 *outNumBytes, AudioStreamPacketDescription *outPacketDescriptions, SInt64 inStartingPacket, UInt32 *ioNumPackets, void *outBuffer) { // This only works with CBR. To suppport VBR you must override. OSStatus err = noErr; FailWithAction(outBuffer == NULL, err = kAudio_ParamError, Bail, "NULL buffer"); FailWithAction((ioNumPackets == NULL) || (*ioNumPackets < 1), err = kAudio_ParamError, Bail, "invalid num packets parameter"); { UInt32 byteCount = *ioNumPackets * mDataFormat.mBytesPerPacket; SInt64 startingByte = inStartingPacket * mDataFormat.mBytesPerPacket; err = ReadBytes (inUseCache, startingByte, &byteCount, outBuffer); if ((err == noErr) || (err == kAudioFileEndOfFileError)) { if (byteCount != (*ioNumPackets * mDataFormat.mBytesPerPacket)) { *ioNumPackets = byteCount / mDataFormat.mBytesPerPacket; byteCount = *ioNumPackets * mDataFormat.mBytesPerPacket; } if (outNumBytes) *outNumBytes = byteCount; if (err == kAudioFileEndOfFileError) err = noErr; } } Bail: return err; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::ReadPacketData( Boolean inUseCache, UInt32 *ioNumBytes, AudioStreamPacketDescription *outPacketDescriptions, SInt64 inStartingPacket, UInt32 *ioNumPackets, void *outBuffer) { OSStatus err = noErr; FailWithAction(ioNumPackets == NULL || *ioNumPackets < 1, err = kAudio_ParamError, Bail, "invalid ioNumPackets parameter"); FailWithAction(ioNumBytes == NULL || *ioNumBytes < 1, err = kAudio_ParamError, Bail, "invalid ioNumBytes parameter"); if (mDataFormat.mBytesPerPacket) { // CBR FailWithAction(outBuffer == NULL, err = kAudio_ParamError, Bail, "NULL buffer"); UInt32 maxPackets = *ioNumBytes / mDataFormat.mBytesPerPacket; if (*ioNumPackets > maxPackets) *ioNumPackets = maxPackets; if (outBuffer) { UInt32 byteCount = *ioNumPackets * mDataFormat.mBytesPerPacket; SInt64 startingByte = inStartingPacket * mDataFormat.mBytesPerPacket; err = ReadBytes (inUseCache, startingByte, &byteCount, outBuffer); if (err == noErr || err == kAudioFileEndOfFileError) { if (byteCount != (*ioNumPackets * mDataFormat.mBytesPerPacket)) { *ioNumPackets = byteCount / mDataFormat.mBytesPerPacket; byteCount = *ioNumPackets * mDataFormat.mBytesPerPacket; } *ioNumBytes = byteCount; } } } else { FailWithAction(outPacketDescriptions == NULL, err = kAudio_ParamError, Bail, "invalid outPacketDescriptions parameter"); err = ReadPacketDataVBR(inUseCache, ioNumBytes, outPacketDescriptions, inStartingPacket, ioNumPackets, outBuffer); } Bail: return err; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::ReadPacketDataVBR( Boolean inUseCache, UInt32 *ioNumBytes, AudioStreamPacketDescription *outPacketDescriptions, SInt64 inStartingPacket, UInt32 *ioNumPackets, void *outBuffer) { OSStatus err = ScanForPackets(inStartingPacket+1); // need to scan packets up to start if (err && err != kAudioFileEndOfFileError) return err; SInt64 dataOffset = GetDataOffset(); CompressedPacketTable* packetTable = GetPacketTable(); if (!packetTable) return kAudioFileInvalidFileError; SInt64 packetTableSize = GetPacketTableSize(); if (inStartingPacket >= packetTableSize) { *ioNumBytes = 0; *ioNumPackets = 0; return kAudioFileEndOfFileError; } if (inStartingPacket + *ioNumPackets <= packetTableSize) { err = ReadPacketDataVBR_InTable(inUseCache, ioNumBytes, outPacketDescriptions, inStartingPacket, ioNumPackets, outBuffer); } else { AudioStreamPacketDescription firstPacket = (*packetTable)[inStartingPacket]; SInt64 firstPacketOffset = firstPacket.mStartOffset; UInt32 bytesRead = *ioNumBytes; SInt64 fileSize = 0; GetDataSource()->GetSize(fileSize); SInt64 remainingBytesInFile = fileSize - firstPacketOffset - dataOffset; if (bytesRead > remainingBytesInFile) bytesRead = (UInt32)remainingBytesInFile; err = ReadBytes (inUseCache, firstPacketOffset, &bytesRead, outBuffer); if (err && err != kAudioFileEndOfFileError) { *ioNumBytes = 0; *ioNumPackets = 0; return err; } Buffer_DataSource bufSrc(bytesRead, outBuffer, dataOffset + firstPacketOffset); OSStatus scanErr = ScanForPackets(kScanToEnd, &bufSrc, false); if (scanErr && scanErr != kAudioFileEndOfFileError) return scanErr; packetTableSize = packetTable->size(); UInt32 numPacketsRead = 0; UInt32 endOfData = 0; SInt64 packetNumber = inStartingPacket; for (; numPacketsRead < *ioNumPackets && packetNumber < packetTableSize; ++numPacketsRead, ++packetNumber) { AudioStreamPacketDescription curPacket = (*packetTable)[numPacketsRead + inStartingPacket]; SInt64 curPacketOffset = curPacket.mStartOffset - firstPacketOffset; SInt64 endOfPacket = curPacketOffset + curPacket.mDataByteSize; if (endOfPacket > bytesRead) break; endOfData = (UInt32)endOfPacket; outPacketDescriptions[numPacketsRead] = curPacket; outPacketDescriptions[numPacketsRead].mStartOffset = curPacketOffset; } *ioNumBytes = endOfData; *ioNumPackets = numPacketsRead; } return err; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::HowManyPacketsCanBeReadIntoBuffer(UInt32* ioNumBytes, SInt64 inStartingPacket, UInt32 *ioNumPackets) { CompressedPacketTable* packetTable = GetPacketTable(); SInt64 packetTableSize = GetPacketTableSize(); if (inStartingPacket + *ioNumPackets > (SInt64)packetTableSize) { *ioNumPackets = (UInt32)(packetTableSize - inStartingPacket); } AudioStreamPacketDescription firstPacket = (*packetTable)[inStartingPacket]; if (*ioNumBytes < firstPacket.mDataByteSize) { *ioNumBytes = 0; *ioNumPackets = 0; return kAudio_ParamError; } SInt64 lastPacketIndex = inStartingPacket + *ioNumPackets - 1; if (lastPacketIndex >= packetTableSize) lastPacketIndex = packetTableSize - 1; AudioStreamPacketDescription lastPacket = (*packetTable)[lastPacketIndex]; SInt64 readBytes = lastPacket.mStartOffset + lastPacket.mDataByteSize - firstPacket.mStartOffset; if (readBytes <= *ioNumBytes) { *ioNumBytes = (UInt32)readBytes; return noErr; } SInt64 lowBound = inStartingPacket; SInt64 highBound = lastPacketIndex + 1; SInt64 okIndex = lowBound; while (highBound >= lowBound) { SInt64 tryBound = (lowBound + highBound) >> 1; if (tryBound > lastPacketIndex) break; AudioStreamPacketDescription tryPacket = (*packetTable)[tryBound]; SInt64 readBytes = tryPacket.mStartOffset + tryPacket.mDataByteSize - firstPacket.mStartOffset; if (readBytes > (SInt64)*ioNumBytes) { highBound = tryBound - 1; } else if (readBytes < (SInt64)*ioNumBytes) { okIndex = tryBound; lowBound = tryBound + 1; } else { okIndex = tryBound; break; } } SInt64 numPackets = okIndex - inStartingPacket + 1; if (numPackets > *ioNumPackets) { numPackets = *ioNumPackets; okIndex = inStartingPacket + numPackets - 1; } AudioStreamPacketDescription packet = (*packetTable)[okIndex]; *ioNumBytes = (UInt32)(packet.mStartOffset + packet.mDataByteSize - firstPacket.mStartOffset); *ioNumPackets = (UInt32)numPackets; return noErr; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::ReadPacketDataVBR_InTable( Boolean inUseCache, UInt32 *ioNumBytes, AudioStreamPacketDescription *outPacketDescriptions, SInt64 inStartingPacket, UInt32 *ioNumPackets, void *outBuffer) { CompressedPacketTable* packetTable = GetPacketTable(); if (!packetTable) return kAudioFileInvalidFileError; OSStatus err = HowManyPacketsCanBeReadIntoBuffer(ioNumBytes, inStartingPacket, ioNumPackets); if (err) return err; AudioStreamPacketDescription firstPacket = (*packetTable)[inStartingPacket]; SInt64 firstPacketOffset = firstPacket.mStartOffset; UInt32 bytesRead = *ioNumBytes; if (outBuffer) { err = ReadBytes (inUseCache, firstPacketOffset, &bytesRead, outBuffer); if (err && err != kAudioFileEndOfFileError) { *ioNumBytes = 0; *ioNumPackets = 0; return err; } *ioNumBytes = bytesRead; } // fill out packet descriptions if (outPacketDescriptions) { for (UInt32 i = 0; i < *ioNumPackets; i++) { AudioStreamPacketDescription curPacket = (*packetTable)[i + inStartingPacket]; outPacketDescriptions[i] = curPacket; outPacketDescriptions[i].mStartOffset = curPacket.mStartOffset - firstPacketOffset; } } return err; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::WritePackets( Boolean inUseCache, UInt32 inNumBytes, const AudioStreamPacketDescription *inPacketDescriptions, SInt64 inStartingPacket, UInt32 *ioNumPackets, const void *inBuffer) { // This only works with CBR. To suppport VBR you must override. OSStatus err = noErr; FailWithAction(inStartingPacket > GetNumPackets(), err = kAudioFileInvalidPacketOffsetError, Bail, "write past end"); FailWithAction((ioNumPackets == NULL) || (inBuffer == NULL), err = kAudioFileUnspecifiedError, Bail, "invalid parameter"); { UInt32 byteCount = *ioNumPackets * mDataFormat.mBytesPerPacket; SInt64 startingByte = inStartingPacket * mDataFormat.mBytesPerPacket; err = WriteBytes(inUseCache, startingByte, &byteCount, inBuffer); FailIf (err != noErr, Bail, "Write Bytes Failed"); if (byteCount != (*ioNumPackets * mDataFormat.mBytesPerPacket)) *ioNumPackets = byteCount / mDataFormat.mBytesPerPacket; } Bail: return (err); } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::GetBitRate( UInt32 *outBitRate) { if (!outBitRate) return kAudioFileUnspecifiedError; UInt32 bytesPerPacket = GetDataFormat().mBytesPerPacket; UInt32 framesPerPacket = GetDataFormat().mFramesPerPacket; Float64 sampleRate = GetDataFormat().mSampleRate; const Float64 bitsPerByte = 8.; if (bytesPerPacket && framesPerPacket) { *outBitRate = (UInt32)(bitsPerByte * (Float64)bytesPerPacket * sampleRate / (Float64)framesPerPacket); } else { SInt64 numPackets = GetNumPackets(); SInt64 numBytes = GetNumBytes(); SInt64 numFrames = 0; if (framesPerPacket) { numFrames = numPackets * framesPerPacket; } else { // count frames CompressedPacketTable* packetTable = GetPacketTable(); if (packetTable) { if (packetTable->size() != numPackets) { return kAudioFileInvalidFileError; } #if !TARGET_OS_WIN32 for (ssize_t i = 0; i < numPackets; i++) #else for (int i = 0; i < numPackets; i++) #endif { numFrames += (*packetTable)[i].mVariableFramesInPacket; } } else { return kAudioFileUnsupportedPropertyError; } } if (numFrames == 0 || (sampleRate == 0.)) { *outBitRate = 0; return noErr; } Float64 duration = (Float64)numFrames / sampleRate; *outBitRate = (UInt32)(bitsPerByte * (Float64)numBytes / duration); } return noErr; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::GetMagicCookieDataSize( UInt32 *outDataSize, UInt32 *isWritable) { if (outDataSize) *outDataSize = 0; if (isWritable) *isWritable = 0; return kAudioFileUnsupportedPropertyError; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::GetMagicCookieData( UInt32 *ioDataSize, void *ioPropertyData) { *ioDataSize = 0; return kAudioFileUnsupportedPropertyError; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::SetMagicCookieData( UInt32 /*inDataSize*/, const void *inPropertyData) { return kAudioFileInvalidChunkError; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::GetMarkerListSize( UInt32 *outDataSize, UInt32 *isWritable) { if (outDataSize) *outDataSize = 0; if (isWritable) *isWritable = 0; return kAudioFileUnsupportedPropertyError; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::GetMarkerList( UInt32 *ioDataSize, AudioFileMarkerList* /*ioPropertyData*/) { *ioDataSize = 0; return kAudioFileUnsupportedPropertyError; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::SetMarkerList( UInt32 /*inDataSize*/, const AudioFileMarkerList* /*inPropertyData*/) { return kAudioFileUnsupportedPropertyError; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::GetRegionListSize( UInt32 *outDataSize, UInt32 *isWritable) { if (outDataSize) *outDataSize = 0; if (isWritable) *isWritable = 0; return kAudioFileUnsupportedPropertyError; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::GetRegionList( UInt32 *ioDataSize, AudioFileRegionList *ioPropertyData) { *ioDataSize = 0; return kAudioFileUnsupportedPropertyError; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::SetRegionList( UInt32 /*inDataSize*/, const AudioFileRegionList* /*inPropertyData*/) { return kAudioFileUnsupportedPropertyError; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::GetChannelLayoutSize( UInt32 *outDataSize, UInt32 *isWritable) { if (outDataSize) *outDataSize = 0; if (isWritable) *isWritable = 0; return kAudioFileUnsupportedPropertyError; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::GetChannelLayout( UInt32 *ioDataSize, AudioChannelLayout* /*ioPropertyData*/) { *ioDataSize = 0; return kAudioFileUnsupportedPropertyError; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::SetChannelLayout( UInt32 /*inDataSize*/, const AudioChannelLayout* /*inPropertyData*/) { return kAudioFileUnsupportedPropertyError; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::GetInfoDictionarySize( UInt32 *outDataSize, UInt32 *isWritable) { if (outDataSize) *outDataSize = sizeof(CFDictionaryRef); if (isWritable) *isWritable = 0; return noErr; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::GetInfoDictionary(CACFDictionary *infoDict) { Float64 fl; if (GetEstimatedDuration(&fl) == noErr) return AddDurationToInfoDictionary(infoDict, fl); return kAudioFileUnsupportedPropertyError; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::SetInfoDictionary(CACFDictionary *infoDict) { return kAudioFileUnsupportedPropertyError; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::GetEstimatedDuration(Float64* duration) { // calculate duration AudioStreamBasicDescription ASBD = GetDataFormat(); *duration = (ASBD.mFramesPerPacket != 0) ? (GetNumPackets() * ASBD.mFramesPerPacket) / ASBD.mSampleRate : 0.0; /* For now, assume that any ASBD that has zero in the frames per packet field has been subclassed for this method. i.e. A CAF file has a frame count in one of it's chunks. MP3 has been subclassed because it guesstimates a duration so the entire file does not need to be parsed in order to calculate the total frames. */ return noErr; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::GetPropertyInfo ( AudioFilePropertyID inPropertyID, UInt32 *outDataSize, UInt32 *isWritable) { OSStatus err = noErr; UInt32 writable = 0; switch (inPropertyID) { case kAudioFilePropertyDeferSizeUpdates : if (outDataSize) *outDataSize = sizeof(UInt32); writable = 1; break; case kAudioFilePropertyFileFormat: if (outDataSize) *outDataSize = sizeof(UInt32); writable = 0; break; case kAudioFilePropertyDataFormat: if (outDataSize) *outDataSize = sizeof(AudioStreamBasicDescription); writable = 1; break; case kAudioFilePropertyFormatList: err = GetFormatListInfo(*outDataSize, writable); break; case kAudioFilePropertyPacketSizeUpperBound: case kAudioFilePropertyIsOptimized: case kAudioFilePropertyMaximumPacketSize: if (outDataSize) *outDataSize = sizeof(UInt32); writable = 0; break; case kAudioFilePropertyAudioDataByteCount: case kAudioFilePropertyAudioDataPacketCount: writable = 1; if (outDataSize) *outDataSize = sizeof(SInt64); break; case kAudioFilePropertyDataOffset: writable = 0; if (outDataSize) *outDataSize = sizeof(SInt64); break; case kAudioFilePropertyBitRate: writable = 0; if (outDataSize) *outDataSize = sizeof(UInt32); break; case kAudioFilePropertyMagicCookieData: err = GetMagicCookieDataSize(outDataSize, &writable); break; case kAudioFilePropertyMarkerList : err = GetMarkerListSize(outDataSize, &writable); break; case kAudioFilePropertyRegionList : err = GetRegionListSize(outDataSize, &writable); break; case kAudioFilePropertyChannelLayout : err = GetChannelLayoutSize(outDataSize, &writable); break; case kAudioFilePropertyPacketToFrame : case kAudioFilePropertyFrameToPacket : if (outDataSize) *outDataSize = sizeof(AudioFramePacketTranslation); writable = 0; break; case kAudioFilePropertyPacketToByte : case kAudioFilePropertyByteToPacket : if (outDataSize) *outDataSize = sizeof(AudioBytePacketTranslation); writable = 0; break; case kAudioFilePropertyInfoDictionary : err = GetInfoDictionarySize(outDataSize, &writable); break; case kTEMPAudioFilePropertySoundCheckDictionary : err = GetSoundCheckDictionarySize(outDataSize, &writable); break; case kTEMPAudioFilePropertyGenerateLoudnessInfo : err = GetLoudnessInfoSize(outDataSize, &writable); writable = 0; break; case kTEMPAudioFilePropertyLoudnessInfo : err = GetLoudnessInfoSize(outDataSize, &writable); break; case kAudioFilePropertyEstimatedDuration : if (outDataSize) *outDataSize = sizeof(Float64); writable = 0; break; case 'LYRC': if (outDataSize) *outDataSize = sizeof(CFStringRef); if (isWritable) *isWritable = 0; break; case 'eof?': if (outDataSize) *outDataSize = sizeof(UInt32); if (isWritable) *isWritable = 0; break; case 'sbtd' /*kAudioFilePropertySourceBitDepth*/ : if (outDataSize) *outDataSize = sizeof(SInt32); if (isWritable) *isWritable = CanWrite(); break; default: writable = 0; err = kAudioFileUnsupportedPropertyError; break; } if (isWritable) *isWritable = writable; return (err); } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::GetProperty( AudioFilePropertyID inPropertyID, UInt32 *ioDataSize, void *ioPropertyData) { OSStatus err = noErr; UInt32 neededSize; UInt32 writable; switch (inPropertyID) { case kAudioFilePropertyFileFormat: FailWithAction(*ioDataSize != sizeof(UInt32), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); *(UInt32 *) ioPropertyData = GetFileType(); break; case kAudioFilePropertyFormatList: err = GetFormatList(*ioDataSize, (AudioFormatListItem*)ioPropertyData); break; case kAudioFilePropertyDataFormat: FailWithAction(*ioDataSize != sizeof(AudioStreamBasicDescription), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); memcpy(ioPropertyData, &mDataFormat, sizeof(AudioStreamBasicDescription)); break; case kAudioFilePropertyDataOffset: FailWithAction(*ioDataSize != sizeof(SInt64), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); *(SInt64 *) ioPropertyData = mDataOffset; break; case kAudioFilePropertyIsOptimized: FailWithAction(*ioDataSize != sizeof(UInt32), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); *(UInt32 *) ioPropertyData = mIsOptimized; break; case kAudioFilePropertyAudioDataByteCount: FailWithAction(*ioDataSize != sizeof(SInt64), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); *(SInt64 *)ioPropertyData = GetNumBytes(); break; case kAudioFilePropertyAudioDataPacketCount: FailWithAction(*ioDataSize != sizeof(SInt64), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); *(SInt64 *)ioPropertyData = GetNumPackets(); break; case kAudioFilePropertyPacketSizeUpperBound: FailWithAction(*ioDataSize != sizeof(UInt32), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); *(UInt32 *)ioPropertyData = GetPacketSizeUpperBound(); break; case kAudioFilePropertyMaximumPacketSize: FailWithAction(*ioDataSize != sizeof(UInt32), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); *(UInt32 *)ioPropertyData = FindMaximumPacketSize(); break; case kAudioFilePropertyBitRate: FailWithAction(*ioDataSize != sizeof(UInt32), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); err = GetBitRate((UInt32*)ioPropertyData); break; case kAudioFilePropertyMagicCookieData: err = GetMagicCookieData(ioDataSize, ioPropertyData); break; case kAudioFilePropertyMarkerList : err = GetMarkerList(ioDataSize, static_cast(ioPropertyData)); break; case kAudioFilePropertyRegionList : memset(ioPropertyData, 0, *ioDataSize); err = GetRegionList(ioDataSize, static_cast(ioPropertyData)); break; case kAudioFilePropertyChannelLayout : err = GetChannelLayoutSize(&neededSize, &writable); FailIf(err, Bail, "GetChannelLayoutSize failed"); FailWithAction(*ioDataSize != neededSize, err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); err = GetChannelLayout(ioDataSize, static_cast(ioPropertyData)); break; case kAudioFilePropertyDeferSizeUpdates : FailWithAction(*ioDataSize != sizeof(UInt32), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); *(UInt32 *) ioPropertyData = DeferSizeUpdates(); break; case kAudioFilePropertyPacketToFrame : { FailWithAction(*ioDataSize != sizeof(AudioFramePacketTranslation), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); AudioFramePacketTranslation* afpt = (AudioFramePacketTranslation*)ioPropertyData; err = PacketToFrame(afpt->mPacket, afpt->mFrame); break; } case kAudioFilePropertyFrameToPacket : { FailWithAction(*ioDataSize != sizeof(AudioFramePacketTranslation), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); AudioFramePacketTranslation* afpt = (AudioFramePacketTranslation*)ioPropertyData; err = FrameToPacket(afpt->mFrame, afpt->mPacket, afpt->mFrameOffsetInPacket); break; } case kAudioFilePropertyPacketToByte : { FailWithAction(*ioDataSize != sizeof(AudioBytePacketTranslation), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); AudioBytePacketTranslation* abpt = (AudioBytePacketTranslation*)ioPropertyData; err = PacketToByte(abpt); break; } case kAudioFilePropertyByteToPacket : { FailWithAction(*ioDataSize != sizeof(AudioBytePacketTranslation), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); AudioBytePacketTranslation* abpt = (AudioBytePacketTranslation*)ioPropertyData; err = ByteToPacket(abpt); break; } case kAudioFilePropertyInfoDictionary : { FailWithAction(*ioDataSize != sizeof(CFDictionaryRef), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); CACFDictionary afInfoDictionary(true); err = GetInfoDictionary(&afInfoDictionary); if (!err) { *(CFMutableDictionaryRef *)ioPropertyData = afInfoDictionary.CopyCFMutableDictionary(); } break; } case kTEMPAudioFilePropertySoundCheckDictionary : { FailWithAction(*ioDataSize != sizeof(CFDictionaryRef), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); CACFDictionary afInfoDictionary(true); err = GetSoundCheckDictionary(&afInfoDictionary); if (err) { OSStatus err2 = GetSoundCheckDictionaryFromLoudnessInfo(&afInfoDictionary); if (err2 == noErr) err = noErr; // else report original error from GetSoundCheckDictionary. } if (!err) { *(CFMutableDictionaryRef *)ioPropertyData = afInfoDictionary.CopyCFMutableDictionary(); } break; } case kTEMPAudioFilePropertyLoudnessInfo : { FailWithAction(*ioDataSize != sizeof(CFDictionaryRef), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); CACFDictionary afInfoDictionary(true); err = GetLoudnessInfo(&afInfoDictionary); if (err) { OSStatus err2 = GetLoudnessInfoFromSoundCheckDictionary(&afInfoDictionary); if (err2 == noErr) err = noErr; // else report original error from GetLoudnessInfo. } if (!err) { *(CFMutableDictionaryRef *)ioPropertyData = afInfoDictionary.CopyCFMutableDictionary(); } break; } case kTEMPAudioFilePropertyGenerateLoudnessInfo : { FailWithAction(*ioDataSize != sizeof(CFDictionaryRef), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); CACFDictionary afInfoDictionary(true); err = GenerateLoudnessInfo(&afInfoDictionary); if (!err) { *(CFMutableDictionaryRef *)ioPropertyData = afInfoDictionary.CopyCFMutableDictionary(); } break; } case kAudioFilePropertyEstimatedDuration : { FailWithAction(*ioDataSize != sizeof(Float64), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); err = GetEstimatedDuration((Float64*)ioPropertyData); break; } case 'LYRC' : if (*ioDataSize < sizeof(CFStringRef)) return kAudioFileBadPropertySizeError; *ioDataSize = sizeof(CFStringRef); err = GetLyrics((CFStringRef*) ioPropertyData); break; case 'eof?' : { if (*ioDataSize != sizeof(UInt32)) return kAudioFileBadPropertySizeError; SInt64 pos; err = GetDataSource()->GetPos(pos); if (err) break; SInt64 endOfData = GetDataOffset() + GetNumBytes(); *(UInt32*)ioPropertyData = pos >= endOfData; break; } case 'sbtd' /*kAudioFilePropertySourceBitDepth*/ : { if (*ioDataSize != sizeof(SInt32)) return kAudioFileBadPropertySizeError; SInt32 outValue = 0; err = GetSourceBitDepth(outValue); if ((err || outValue == 0) && GetDataFormat().mFormatID == kAudioFormatLinearPCM) { // if there was no stored source bit depth, and this file is LPCM, then report this file's bit depth. err = noErr; outValue = GetDataFormat().mBitsPerChannel; if (GetDataFormat().mFormatFlags & kAudioFormatFlagIsFloat) outValue = -outValue; } else if (err) break; *(SInt32 *) ioPropertyData = outValue; break; } default: err = kAudioFileUnsupportedPropertyError; break; } Bail: return err; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::SetProperty( AudioFilePropertyID inPropertyID, UInt32 inDataSize, const void *inPropertyData) { OSStatus err = noErr; switch (inPropertyID) { case kAudioFilePropertyDataFormat: FailWithAction(inDataSize != sizeof(AudioStreamBasicDescription), err = kAudioFileBadPropertySizeError, Bail, "Incorrect data size"); err = UpdateDataFormat((AudioStreamBasicDescription *) inPropertyData); break; case kAudioFilePropertyFormatList: err = SetFormatList(inDataSize, (AudioFormatListItem*)inPropertyData); break; case kAudioFilePropertyAudioDataByteCount: { FailWithAction(inDataSize != sizeof(SInt64), err = kAudioFileBadPropertySizeError, Bail, "Incorrect data size"); SInt64 numBytes = *(SInt64 *) inPropertyData; if (numBytes > GetNumBytes()) { // can't use this to increase data size. return kAudioFileOperationNotSupportedError; } UInt32 saveDefer = DeferSizeUpdates(); SetDeferSizeUpdates(0); // force an update. err = UpdateNumBytes(numBytes); SetDeferSizeUpdates(saveDefer); } break; case kAudioFilePropertyAudioDataPacketCount: { SInt64 numPackets = *(SInt64 *) inPropertyData; if (numPackets > GetNumPackets()) { // can't use this to increase data size. return kAudioFileOperationNotSupportedError; } err = UpdateNumPackets(numPackets); } break; case kAudioFilePropertyMagicCookieData: err = SetMagicCookieData(inDataSize, inPropertyData); break; case kAudioFilePropertyMarkerList : err = SetMarkerList(inDataSize, static_cast(inPropertyData)); break; case kAudioFilePropertyRegionList : err = SetRegionList(inDataSize, static_cast(inPropertyData)); break; case kAudioFilePropertyChannelLayout : err = SetChannelLayout(inDataSize, static_cast(inPropertyData)); break; case kAudioFilePropertyDeferSizeUpdates : FailWithAction(inDataSize != sizeof(UInt32), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); SetDeferSizeUpdates(*(UInt32 *) inPropertyData); break; case kAudioFilePropertyInfoDictionary : { FailWithAction(inDataSize != sizeof(CFDictionaryRef), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); // pass the SetInfoDictionary a CACFDictionary object made with the provided CFDictionaryRef // Let the caller release their own CFObject so pass false for th erelease parameter CACFDictionary afInfoDictionary(*(CFDictionaryRef *)inPropertyData, false); err = SetInfoDictionary(&afInfoDictionary); break; } case kTEMPAudioFilePropertySoundCheckDictionary : { FailWithAction(inDataSize != sizeof(CFDictionaryRef), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); // pass the SetInfoDictionary a CACFDictionary object made with the provided CFDictionaryRef // Let the caller release their own CFObject so pass false for the release parameter CACFDictionary afInfoDictionary(*(CFDictionaryRef *)inPropertyData, false); err = SetSoundCheckDictionary(&afInfoDictionary); break; } case kTEMPAudioFilePropertyLoudnessInfo : { FailWithAction(inDataSize != sizeof(CFDictionaryRef), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); // pass the SetInfoDictionary a CACFDictionary object made with the provided CFDictionaryRef // Let the caller release their own CFObject so pass false for the release parameter CACFDictionary afInfoDictionary(*(CFDictionaryRef *)inPropertyData, false); err = SetLoudnessInfo(&afInfoDictionary); break; } case 'sbtd' /*kAudioFilePropertySourceBitDepth*/ : { FailWithAction(inDataSize != sizeof(SInt32), err = kAudioFileBadPropertySizeError, Bail, "inDataSize is wrong"); SInt32 inValue = *(SInt32 *)inPropertyData; err = SetSourceBitDepth(inValue); break; } default: err = kAudioFileUnsupportedPropertyError; break; } Bail: return err; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::SetDataFormat(const AudioStreamBasicDescription* inStreamFormat) { OSStatus err = noErr; if (!IsDataFormatValid(inStreamFormat)) return kAudioFileUnsupportedDataFormatError; if (!IsDataFormatSupported(inStreamFormat)) return kAudioFileUnsupportedDataFormatError; UInt32 prevBytesPerPacket = mDataFormat.mBytesPerPacket; mDataFormat = *inStreamFormat; // if CBR and bytes per packet changes, we need to change the number of packets we think we have. if (mDataFormat.mBytesPerPacket && mDataFormat.mBytesPerPacket != prevBytesPerPacket) { SInt64 numPackets = GetNumBytes() / mDataFormat.mBytesPerPacket; SetNumPackets(numPackets); SetMaximumPacketSize(mDataFormat.mBytesPerPacket); if (!mFirstSetFormat) SizeChanged(); } mFirstSetFormat = false; return err; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::GetFormatListInfo( UInt32 &outDataSize, UInt32 &outWritable) { // default implementation is to just return the data format outDataSize = sizeof(AudioFormatListItem); outWritable = false; return noErr; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::GetFormatList( UInt32 &ioDataSize, AudioFormatListItem *ioPropertyData) { // default implementation is to just return the data format if (ioDataSize < sizeof(AudioFormatListItem)) return kAudioFileBadPropertySizeError; AudioFormatListItem afli; afli.mASBD = mDataFormat; AudioChannelLayoutTag layoutTag = /*kAudioChannelLayoutTag_Unknown*/ 0xFFFF0000 | mDataFormat.mChannelsPerFrame; UInt32 layoutSize, isWritable; OSStatus err = GetChannelLayoutSize(&layoutSize, &isWritable); if (err == noErr) { CAAutoFree layout; layout.allocBytes(layoutSize); err = GetChannelLayout(&layoutSize, layout()); if (err == noErr) { layoutTag = layout->mChannelLayoutTag; } } afli.mChannelLayoutTag = layoutTag; memcpy(ioPropertyData, &afli, sizeof(AudioFormatListItem)); ioDataSize = sizeof(AudioFormatListItem); return noErr; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::SetFormatList( UInt32 inDataSize, const AudioFormatListItem *inPropertyData) { return kAudioFileOperationNotSupportedError; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ OSStatus AudioFileObject::UpdateDataFormat(const AudioStreamBasicDescription* inStreamFormat) { return SetDataFormat(inStreamFormat); } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ Boolean AudioFileObject::IsDataFormatValid(AudioStreamBasicDescription const* inDesc) { if (inDesc->mSampleRate < 0.) return false; if (inDesc->mSampleRate > 3e6) return false; if (inDesc->mChannelsPerFrame < 1 || inDesc->mChannelsPerFrame > 0x000FFFFF) return false; if (inDesc->mFormatID == kAudioFormatLinearPCM) { if (inDesc->mBitsPerChannel < 1 || inDesc->mBitsPerChannel > 64) return false; if (inDesc->mFramesPerPacket != 1) return false; if (inDesc->mBytesPerPacket == 0) return false; if (inDesc->mBytesPerFrame != inDesc->mBytesPerPacket) return false; // [3605260] we assume here that a packet is an integer number of frames. UInt32 minimumBytesPerPacket = (inDesc->mBitsPerChannel * inDesc->mChannelsPerFrame + 7) / 8; if (inDesc->mBytesPerPacket < minimumBytesPerPacket) return false; } return true; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ void AudioFileObject::SetDataSource(DataSource* inDataSource) { if (mDataSource != inDataSource) { delete mDataSource; mDataSource = inDataSource; } } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ void AudioFileObject::SetURL (CFURLRef inURL) { if (mFileRef == inURL) return; if (inURL) CFRetain (inURL); if (mFileRef) CFRelease(mFileRef); mFileRef = inURL; } OSStatus AudioFileObject::OpenFile(SInt8 inPermissions, int inFD) { OSStatus err = noErr; SetDataSource(new Cached_DataSource(new UnixFile_DataSource(inFD, inPermissions, true))); mFileD = inFD; SetPermissions (inPermissions); return err; } OSStatus AudioFileObject::CreateDataFile (CFURLRef inFileRef, int &outFileD) { UInt8 fPath[FILENAME_MAX]; if (!CFURLGetFileSystemRepresentation (inFileRef, true, fPath, FILENAME_MAX)) return kAudio_FileNotFoundError; struct stat stbuf; if (stat ((const char*)fPath, &stbuf) == 0) return kAudioFilePermissionsError; #if TARGET_OS_WIN32 int filePerms = S_IREAD | S_IWRITE; int flags = O_CREAT | O_EXCL | O_RDWR | O_BINARY; #else mode_t filePerms = S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH; int flags = O_CREAT | O_EXCL | O_RDWR; #endif outFileD = open((const char*)fPath, flags, filePerms); if (outFileD < 0) return AudioFileTranslateErrno(errno); return noErr; } OSStatus AudioFileObject::AddDurationToInfoDictionary(CACFDictionary *infoDict, Float64 &inDuration) { #if !TARGET_OS_WIN32 CFLocaleRef currentLocale = CFLocaleGetSystem(); CFNumberFormatterRef numberFormatter = NULL; numberFormatter = CFNumberFormatterCreate(kCFAllocatorDefault, currentLocale, kCFNumberFormatterDecimalStyle); CFStringRef cfStr = CFNumberFormatterCreateStringWithValue( kCFAllocatorDefault, numberFormatter, kCFNumberFloat64Type, &inDuration); if (cfStr) { if (CFStringGetLength(cfStr) != 0) infoDict->AddString(CFSTR(kAFInfoDictionary_ApproximateDurationInSeconds), cfStr); CFRelease(cfStr); } CFRelease(numberFormatter); #endif return noErr; } OSStatus AudioFileObject::SizeChanged() { OSStatus err = noErr; if (mPermissions & kAudioFileWritePermission) { if (DeferSizeUpdates()) SetNeedsSizeUpdate(true); else err = UpdateSize(); } return err; } OSStatus AudioFileObject::UpdateSizeIfNeeded() { if (GetNeedsSizeUpdate()) { OSStatus err = UpdateSize(); if (err) return err; SetNeedsSizeUpdate(false); } return noErr; } OSStatus AudioFileObject::CountUserData( UInt32 /*inUserDataID*/, UInt32* /*outNumberItems*/) { return kAudioFileOperationNotSupportedError; } OSStatus AudioFileObject::GetUserDataSize( UInt32 /*inUserDataID*/, UInt32 /*inIndex*/, UInt32* /*outDataSize*/) { return kAudioFileOperationNotSupportedError; } OSStatus AudioFileObject::GetUserData( UInt32 /*inUserDataID*/, UInt32 /*inIndex*/, UInt32* /*ioDataSize*/, void* /*ioUserData*/) { return kAudioFileOperationNotSupportedError; } OSStatus AudioFileObject::SetUserData( UInt32 /*inUserDataID*/, UInt32 /*inIndex*/, UInt32 /*inDataSize*/, const void* /*inUserData*/) { return kAudioFileOperationNotSupportedError; } OSStatus AudioFileObject::RemoveUserData( UInt32 /*inUserDataID*/, UInt32 /*inIndex*/) { return kAudioFileOperationNotSupportedError; } OSStatus AudioFileObject::MoveData(SInt64 fromPos, SInt64 toPos, SInt64 size) { if (fromPos == toPos) return noErr; OSStatus err = noErr; CAAutoFree audioData(kCopySoundDataBufferSize, true); SInt64 bytesRemaining = size; if (fromPos < toPos) { while (bytesRemaining > 0) { // read from old file UInt32 byteCount; SInt64 count = (bytesRemaining < kCopySoundDataBufferSize) ? bytesRemaining : kCopySoundDataBufferSize; err = GetDataSource()->ReadBytes(SEEK_SET, fromPos+(bytesRemaining-count), (UInt32)count, audioData(), &byteCount); FailIf (err != noErr, Bail, "MoveData ReadBytes failed"); err = GetDataSource()->WriteBytes(SEEK_SET, toPos+(bytesRemaining-count), (UInt32)count, audioData(), &byteCount); FailIf (err != noErr, Bail, "WriteBytes failed"); bytesRemaining -= count; } } else { while (bytesRemaining > 0) { // read from old file UInt32 byteCount; SInt64 count = (bytesRemaining < kCopySoundDataBufferSize) ? bytesRemaining : kCopySoundDataBufferSize; err = GetDataSource()->ReadBytes(SEEK_SET, fromPos+(size - bytesRemaining), (UInt32)count, audioData(), &byteCount); FailIf (err != noErr, Bail, "MoveData ReadBytes failed"); err = GetDataSource()->WriteBytes(SEEK_SET, toPos+(size - bytesRemaining), (UInt32)count, audioData(), &byteCount); FailIf (err != noErr, Bail, "WriteBytes failed"); bytesRemaining -= count; } } Bail: return err; } // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~