/* * $Id$ * * Copyright (C) 2006 Stephen F. Booth * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include #import "CoreAudioDecoder.h" @interface CoreAudioDecoder (Private) - (BOOL) readInfoFromExtAudioFileRef; @end @implementation CoreAudioDecoder - (void) close { OSStatus err; err = ExtAudioFileDispose(_in); if(noErr != err) { NSLog(@"Error closing ExtAudioFile"); } } - (BOOL)open:(id)source; { OSStatus err; NSURL *url = [source url]; [source close]; //There's no room for your kind around here! err = ExtAudioFileOpenURL((CFURLRef)url, &_in); if(noErr != err) { NSLog(@"Error opening file: %d", err); return NO; } return [self readInfoFromExtAudioFileRef]; } - (BOOL) readInfoFromExtAudioFileRef { OSStatus err; UInt32 size; AudioStreamBasicDescription asbd; // Get input file information size = sizeof(asbd); err = ExtAudioFileGetProperty(_in, kExtAudioFileProperty_FileDataFormat, &size, &asbd); if(err != noErr) { err = ExtAudioFileDispose(_in); return NO; } SInt64 total; size = sizeof(total); err = ExtAudioFileGetProperty(_in, kExtAudioFileProperty_FileLengthFrames, &size, &total); if(err != noErr) { err = ExtAudioFileDispose(_in); return NO; } totalFrames = total; //Is there a way to get bitrate with extAudioFile? bitrate = 0; // Set our properties bitsPerSample = asbd.mBitsPerChannel; channels = asbd.mChannelsPerFrame; frequency = asbd.mSampleRate; floatingPoint = NO; // mBitsPerChannel will only be set for lpcm formats if(0 == bitsPerSample) { bitsPerSample = 32; floatingPoint = YES; } // Set output format AudioStreamBasicDescription result; bzero(&result, sizeof(AudioStreamBasicDescription)); result.mFormatID = kAudioFormatLinearPCM; if (floatingPoint) { result.mFormatFlags = kAudioFormatFlagIsFloat | kAudioFormatFlagIsPacked; } else { result.mFormatFlags = kAudioFormatFlagIsSignedInteger | kLinearPCMFormatFlagIsBigEndian; } result.mSampleRate = frequency; result.mChannelsPerFrame = channels; result.mBitsPerChannel = bitsPerSample; result.mBytesPerPacket = channels * (bitsPerSample / 8); result.mFramesPerPacket = 1; result.mBytesPerFrame = channels * (bitsPerSample / 8); err = ExtAudioFileSetProperty(_in, kExtAudioFileProperty_ClientDataFormat, sizeof(result), &result); if(noErr != err) { err = ExtAudioFileDispose(_in); return NO; } [self willChangeValueForKey:@"properties"]; [self didChangeValueForKey:@"properties"]; return YES; } - (int) readAudio:(void *)buf frames:(UInt32)frames { OSStatus err; AudioBufferList bufferList; UInt32 frameCount; // Set up the AudioBufferList bufferList.mNumberBuffers = 1; bufferList.mBuffers[0].mNumberChannels = channels; bufferList.mBuffers[0].mData = buf; bufferList.mBuffers[0].mDataByteSize = frames * channels * (bitsPerSample/8); // Read a chunk of PCM input (converted from whatever format) frameCount = frames; err = ExtAudioFileRead(_in, &frameCount, &bufferList); if(err != noErr) { return 0; } return frameCount; } - (long) seek:(long)frame { OSStatus err; err = ExtAudioFileSeek(_in, frame); if(noErr != err) { return -1; } return frame; } + (NSArray *)fileTypes { OSStatus err; UInt32 size; NSArray *sAudioExtensions; size = sizeof(sAudioExtensions); err = AudioFileGetGlobalInfo(kAudioFileGlobalInfo_AllExtensions, 0, NULL, &size, &sAudioExtensions); if(noErr != err) { return nil; } return [sAudioExtensions autorelease]; } + (NSArray *)mimeTypes { return nil; } - (NSDictionary *)properties { return [NSDictionary dictionaryWithObjectsAndKeys: [NSNumber numberWithInt:channels],@"channels", [NSNumber numberWithInt:bitsPerSample],@"bitsPerSample", [NSNumber numberWithBool:floatingPoint],@"floatingPoint", [NSNumber numberWithInt:bitrate],@"bitrate", [NSNumber numberWithFloat:frequency],@"sampleRate", [NSNumber numberWithLong:totalFrames],@"totalFrames", [NSNumber numberWithBool:YES], @"seekable", floatingPoint ? @"host" : @"big", @"endian", nil]; } @end