cog/Plugins/CoreAudio/CoreAudioDecoder.m

271 lines
6.5 KiB
Matlab
Raw Normal View History

2006-05-07 13:19:23 +00:00
/*
* $Id$
*
* Copyright (C) 2006 Stephen F. Booth <me@sbooth.org>
*
* 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
*/
2006-05-29 22:02:59 +00:00
#include <unistd.h>
#import "CoreAudioDecoder.h"
2006-05-07 13:19:23 +00:00
#import "Logging.h"
@interface CoreAudioDecoder (Private)
2006-05-13 04:50:06 +00:00
- (BOOL) readInfoFromExtAudioFileRef;
2006-05-07 13:19:23 +00:00
@end
static OSStatus readProc(void* clientData,
SInt64 position,
UInt32 requestCount,
void* buffer,
UInt32* actualCount)
{
NSObject* _handle = (__bridge NSObject *)(clientData);
CoreAudioDecoder * __unsafe_unretained pSelf = (id) _handle;
id<CogSource> source = pSelf->_audioSource;
if (position != pSelf->_lastPosition) {
if ([source seekable])
[source seek:position whence:SEEK_SET];
else
return seekErr;
}
size_t bytesRead = [source read:buffer amount:requestCount];
pSelf->_lastPosition = position + bytesRead;
if(actualCount)
*actualCount = (UInt32) bytesRead;
return noErr;
}
static SInt64 getSizeProc(void* clientData) {
NSObject* _handle = (__bridge NSObject *)(clientData);
CoreAudioDecoder * __unsafe_unretained pSelf = (id) _handle;
id<CogSource> source = pSelf->_audioSource;
SInt64 size;
if ([source seekable]) {
[source seek:0 whence:SEEK_END];
size = [source tell];
[source seek:pSelf->_lastPosition whence:SEEK_SET];
}
else {
size = INT64_MAX;
}
return size;
}
@implementation CoreAudioDecoder
2006-05-07 13:19:23 +00:00
- (void) close
{
2006-05-13 04:50:06 +00:00
OSStatus err;
2006-05-29 22:02:59 +00:00
2006-05-13 04:50:06 +00:00
err = ExtAudioFileDispose(_in);
if(noErr != err) {
DLog(@"Error closing ExtAudioFile");
2006-05-12 01:13:00 +00:00
}
err = AudioFileClose(_audioFile);
if(noErr != err) {
DLog(@"Error closing AudioFile");
}
_audioSource = nil;
2006-05-07 13:19:23 +00:00
}
- (void) dealloc
{
[self close];
}
- (BOOL)open:(id<CogSource>)source;
2006-05-07 13:19:23 +00:00
{
OSStatus err;
_audioSource = source;
_lastPosition = [source tell];
err = AudioFileOpenWithCallbacks((__bridge void *)self, readProc, 0, getSizeProc, 0, 0, &_audioFile);
if(noErr != err) {
ALog(@"Error opening callback interface to file: %d", err);
return NO;
}
err = ExtAudioFileWrapAudioFileID(_audioFile, false, &_in);
2006-05-12 01:13:00 +00:00
if(noErr != err) {
ALog(@"Error opening file: %d", err);
2006-05-12 01:13:00 +00:00
return NO;
}
2006-05-13 04:50:06 +00:00
return [self readInfoFromExtAudioFileRef];
2006-05-07 13:19:23 +00:00
}
2006-05-13 04:50:06 +00:00
- (BOOL) readInfoFromExtAudioFileRef
2006-05-07 13:19:23 +00:00
{
OSStatus err;
UInt32 size;
AudioStreamBasicDescription asbd;
2006-05-29 22:02:59 +00:00
2006-05-07 13:19:23 +00:00
// Get input file information
size = sizeof(asbd);
2006-05-13 04:50:06 +00:00
err = ExtAudioFileGetProperty(_in, kExtAudioFileProperty_FileDataFormat, &size, &asbd);
2006-05-07 13:19:23 +00:00
if(err != noErr) {
2006-05-13 04:50:06 +00:00
err = ExtAudioFileDispose(_in);
2006-05-07 13:19:23 +00:00
return NO;
}
2006-05-29 22:02:59 +00:00
SInt64 total;
size = sizeof(total);
err = ExtAudioFileGetProperty(_in, kExtAudioFileProperty_FileLengthFrames, &size, &total);
2006-05-12 00:06:50 +00:00
if(err != noErr) {
2006-05-13 04:50:06 +00:00
err = ExtAudioFileDispose(_in);
2006-05-12 00:06:50 +00:00
return NO;
}
totalFrames = total;
2006-05-29 22:02:59 +00:00
2006-05-13 16:50:52 +00:00
//Is there a way to get bitrate with extAudioFile?
2006-06-19 00:39:41 +00:00
bitrate = 0;
2006-05-13 16:50:52 +00:00
2006-05-07 13:19:23 +00:00
// Set our properties
bitsPerSample = asbd.mBitsPerChannel;
channels = asbd.mChannelsPerFrame;
frequency = asbd.mSampleRate;
floatingPoint = NO;
2006-05-07 13:19:23 +00:00
// mBitsPerChannel will only be set for lpcm formats
if(0 == bitsPerSample) {
bitsPerSample = 32;
floatingPoint = YES;
2006-05-07 13:19:23 +00:00
}
2006-05-12 00:06:50 +00:00
// Set output format
2006-05-13 04:50:06 +00:00
AudioStreamBasicDescription result;
2006-05-07 13:19:23 +00:00
bzero(&result, sizeof(AudioStreamBasicDescription));
result.mFormatID = kAudioFormatLinearPCM;
if (floatingPoint) {
result.mFormatFlags = kAudioFormatFlagIsFloat | kAudioFormatFlagIsPacked;
}
else {
result.mFormatFlags = kAudioFormatFlagIsSignedInteger | kLinearPCMFormatFlagIsBigEndian;
}
2006-05-07 13:19:23 +00:00
result.mSampleRate = frequency;
result.mChannelsPerFrame = channels;
result.mBitsPerChannel = bitsPerSample;
result.mBytesPerPacket = channels * (bitsPerSample / 8);
result.mFramesPerPacket = 1;
result.mBytesPerFrame = channels * (bitsPerSample / 8);
2006-05-29 22:02:59 +00:00
2006-05-13 04:50:06 +00:00
err = ExtAudioFileSetProperty(_in, kExtAudioFileProperty_ClientDataFormat, sizeof(result), &result);
if(noErr != err) {
err = ExtAudioFileDispose(_in);
2006-05-12 00:06:50 +00:00
return NO;
}
2006-05-07 13:19:23 +00:00
[self willChangeValueForKey:@"properties"];
[self didChangeValueForKey:@"properties"];
2006-05-07 13:19:23 +00:00
return YES;
}
- (int) readAudio:(void *)buf frames:(UInt32)frames
2006-05-07 13:19:23 +00:00
{
2006-05-13 04:50:06 +00:00
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);
2006-05-07 13:19:23 +00:00
// Read a chunk of PCM input (converted from whatever format)
frameCount = frames;
2006-05-13 04:50:06 +00:00
err = ExtAudioFileRead(_in, &frameCount, &bufferList);
2006-05-07 13:19:23 +00:00
if(err != noErr) {
return 0;
}
2006-05-13 04:50:06 +00:00
return frameCount;
2006-05-07 13:19:23 +00:00
}
- (long) seek:(long)frame
2006-05-07 13:19:23 +00:00
{
2006-05-13 04:50:06 +00:00
OSStatus err;
2006-05-29 22:02:59 +00:00
err = ExtAudioFileSeek(_in, frame);
2006-05-13 04:50:06 +00:00
if(noErr != err) {
return -1;
2006-05-13 04:50:06 +00:00
}
2006-05-07 13:19:23 +00:00
return frame;
2006-05-07 13:19:23 +00:00
}
+ (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;
}
+ (NSArray *)mimeTypes
{
return nil;
}
+ (float)priority
{
return 1.0;
}
- (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:[_audioSource seekable]], @"seekable",
floatingPoint ? @"host" : @"big", @"endian",
nil];
}
2006-05-07 13:19:23 +00:00
@end