mirror of
https://github.com/TomHarte/CLK.git
synced 2025-01-13 22:32:03 +00:00
f10be2a18a
Explicit cause: dealloc calls close_output(). That may decide to flush work, indiscriminately. Some of the flushed work might be audio generation. Audio generation might cause the audio queue to react with an out-of-data announcement. Which would cause a fresh attempt to update the CSMachine.
187 lines
5.6 KiB
Plaintext
187 lines
5.6 KiB
Plaintext
//
|
|
// CSMachine.m
|
|
// Clock Signal
|
|
//
|
|
// Created by Thomas Harte on 04/01/2016.
|
|
// Copyright © 2016 Thomas Harte. All rights reserved.
|
|
//
|
|
|
|
#import "CSMachine.h"
|
|
#import "CSMachine+Subclassing.h"
|
|
#import "CSMachine+Target.h"
|
|
|
|
#include "Typer.hpp"
|
|
#include "ConfigurationTarget.hpp"
|
|
|
|
@interface CSMachine()
|
|
- (void)speaker:(Outputs::Speaker *)speaker didCompleteSamples:(const int16_t *)samples length:(int)length;
|
|
- (void)machineDidChangeClockRate;
|
|
- (void)machineDidChangeClockIsUnlimited;
|
|
@end
|
|
|
|
struct LockProtectedDelegate {
|
|
// Contractual promise is: machine — the pointer **and** the object ** — may be accessed only
|
|
// in sections protected by the machineAccessLock;
|
|
NSLock *machineAccessLock;
|
|
__unsafe_unretained CSMachine *machine;
|
|
};
|
|
|
|
struct SpeakerDelegate: public Outputs::Speaker::Delegate, public LockProtectedDelegate {
|
|
void speaker_did_complete_samples(Outputs::Speaker *speaker, const std::vector<int16_t> &buffer) {
|
|
[machineAccessLock lock];
|
|
[machine speaker:speaker didCompleteSamples:buffer.data() length:(int)buffer.size()];
|
|
[machineAccessLock unlock];
|
|
}
|
|
};
|
|
|
|
struct MachineDelegate: CRTMachine::Machine::Delegate, public LockProtectedDelegate {
|
|
void machine_did_change_clock_rate(CRTMachine::Machine *sender) {
|
|
[machineAccessLock lock];
|
|
[machine machineDidChangeClockRate];
|
|
[machineAccessLock unlock];
|
|
}
|
|
void machine_did_change_clock_is_unlimited(CRTMachine::Machine *sender) {
|
|
[machineAccessLock lock];
|
|
[machine machineDidChangeClockIsUnlimited];
|
|
[machineAccessLock unlock];
|
|
}
|
|
};
|
|
|
|
@implementation CSMachine {
|
|
SpeakerDelegate _speakerDelegate;
|
|
MachineDelegate _machineDelegate;
|
|
NSLock *_delegateMachineAccessLock;
|
|
CRTMachine::Machine *_machine;
|
|
}
|
|
|
|
- (instancetype)initWithMachine:(void *)machine {
|
|
self = [super init];
|
|
if(self) {
|
|
_machine = (CRTMachine::Machine *)machine;
|
|
_delegateMachineAccessLock = [[NSLock alloc] init];
|
|
|
|
_machineDelegate.machine = self;
|
|
_speakerDelegate.machine = self;
|
|
_machineDelegate.machineAccessLock = _delegateMachineAccessLock;
|
|
_speakerDelegate.machineAccessLock = _delegateMachineAccessLock;
|
|
|
|
_machine->set_delegate(&_machineDelegate);
|
|
}
|
|
return self;
|
|
}
|
|
|
|
- (void)speaker:(Outputs::Speaker *)speaker didCompleteSamples:(const int16_t *)samples length:(int)length {
|
|
[self.audioQueue enqueueAudioBuffer:samples numberOfSamples:(unsigned int)length];
|
|
}
|
|
|
|
- (void)machineDidChangeClockRate {
|
|
[self.delegate machineDidChangeClockRate:self];
|
|
}
|
|
|
|
- (void)machineDidChangeClockIsUnlimited {
|
|
[self.delegate machineDidChangeClockIsUnlimited:self];
|
|
}
|
|
|
|
- (void)dealloc {
|
|
// The two delegate's references to this machine are nilled out here because close_output may result
|
|
// in a data flush, which might cause an audio callback, which could cause the audio queue to decide
|
|
// that it's out of data, resulting in an attempt further to run the machine while it is dealloc'ing.
|
|
//
|
|
// They are nilled inside an explicit lock because that allows the delegates to protect their entire
|
|
// call into the machine, not just the pointer access.
|
|
[_delegateMachineAccessLock lock];
|
|
_machineDelegate.machine = nil;
|
|
_speakerDelegate.machine = nil;
|
|
[_delegateMachineAccessLock unlock];
|
|
|
|
[_view performWithGLContext:^{
|
|
@synchronized(self) {
|
|
_machine->close_output();
|
|
}
|
|
}];
|
|
}
|
|
|
|
- (float)idealSamplingRateFromRange:(NSRange)range {
|
|
@synchronized(self) {
|
|
std::shared_ptr<Outputs::Speaker> speaker = _machine->get_speaker();
|
|
if(speaker) {
|
|
return speaker->get_ideal_clock_rate_in_range((float)range.location, (float)(range.location + range.length));
|
|
}
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
- (void)setAudioSamplingRate:(float)samplingRate bufferSize:(NSUInteger)bufferSize {
|
|
@synchronized(self) {
|
|
[self setSpeakerDelegate:&_speakerDelegate sampleRate:samplingRate bufferSize:bufferSize];
|
|
}
|
|
}
|
|
|
|
- (BOOL)setSpeakerDelegate:(Outputs::Speaker::Delegate *)delegate sampleRate:(float)sampleRate bufferSize:(NSUInteger)bufferSize {
|
|
@synchronized(self) {
|
|
std::shared_ptr<Outputs::Speaker> speaker = _machine->get_speaker();
|
|
if(speaker) {
|
|
speaker->set_output_rate(sampleRate, (int)bufferSize);
|
|
speaker->set_delegate(delegate);
|
|
return YES;
|
|
}
|
|
return NO;
|
|
}
|
|
}
|
|
|
|
- (void)runForNumberOfCycles:(int)numberOfCycles {
|
|
@synchronized(self) {
|
|
_machine->run_for(Cycles(numberOfCycles));
|
|
}
|
|
}
|
|
|
|
- (void)setView:(CSOpenGLView *)view aspectRatio:(float)aspectRatio {
|
|
_view = view;
|
|
[view performWithGLContext:^{
|
|
[self setupOutputWithAspectRatio:aspectRatio];
|
|
}];
|
|
}
|
|
|
|
- (void)setupOutputWithAspectRatio:(float)aspectRatio {
|
|
_machine->setup_output(aspectRatio);
|
|
|
|
// Since OS X v10.6, Macs have had a gamma of 2.2.
|
|
_machine->get_crt()->set_output_gamma(2.2f);
|
|
}
|
|
|
|
- (void)drawViewForPixelSize:(CGSize)pixelSize onlyIfDirty:(BOOL)onlyIfDirty {
|
|
_machine->get_crt()->draw_frame((unsigned int)pixelSize.width, (unsigned int)pixelSize.height, onlyIfDirty ? true : false);
|
|
}
|
|
|
|
- (double)clockRate {
|
|
return _machine->get_clock_rate();
|
|
}
|
|
|
|
- (BOOL)clockIsUnlimited {
|
|
return _machine->get_clock_is_unlimited() ? YES : NO;
|
|
}
|
|
|
|
- (void)paste:(NSString *)paste {
|
|
Utility::TypeRecipient *typeRecipient = dynamic_cast<Utility::TypeRecipient *>(_machine);
|
|
if(typeRecipient)
|
|
typeRecipient->set_typer_for_string([paste UTF8String]);
|
|
}
|
|
|
|
- (void)applyTarget:(const StaticAnalyser::Target &)target {
|
|
@synchronized(self) {
|
|
ConfigurationTarget::Machine *const configurationTarget =
|
|
dynamic_cast<ConfigurationTarget::Machine *>(_machine);
|
|
if(configurationTarget) configurationTarget->configure_as_target(target);
|
|
}
|
|
}
|
|
|
|
- (void)applyMedia:(const StaticAnalyser::Media &)media {
|
|
@synchronized(self) {
|
|
ConfigurationTarget::Machine *const configurationTarget =
|
|
dynamic_cast<ConfigurationTarget::Machine *>(_machine);
|
|
if(configurationTarget) configurationTarget->insert_media(media);
|
|
}
|
|
}
|
|
|
|
@end
|