2016-01-05 04:12:47 +00:00
|
|
|
//
|
|
|
|
// 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"
|
2016-09-01 00:43:29 +00:00
|
|
|
#import "CSMachine+Target.h"
|
2016-09-08 09:32:17 +00:00
|
|
|
|
2016-06-19 20:35:04 +00:00
|
|
|
#include "Typer.hpp"
|
2016-09-08 09:32:17 +00:00
|
|
|
#include "ConfigurationTarget.hpp"
|
2016-01-05 04:12:47 +00:00
|
|
|
|
2016-06-01 02:16:20 +00:00
|
|
|
@interface CSMachine()
|
|
|
|
- (void)speaker:(Outputs::Speaker *)speaker didCompleteSamples:(const int16_t *)samples length:(int)length;
|
2016-06-21 01:47:27 +00:00
|
|
|
- (void)machineDidChangeClockRate;
|
2016-09-13 02:06:03 +00:00
|
|
|
- (void)machineDidChangeClockIsUnlimited;
|
2016-06-01 02:16:20 +00:00
|
|
|
@end
|
|
|
|
|
2016-01-14 03:38:59 +00:00
|
|
|
struct SpeakerDelegate: public Outputs::Speaker::Delegate {
|
|
|
|
__weak CSMachine *machine;
|
2017-07-16 19:01:39 +00:00
|
|
|
void speaker_did_complete_samples(Outputs::Speaker *speaker, const std::vector<int16_t> &buffer) {
|
|
|
|
[machine speaker:speaker didCompleteSamples:buffer.data() length:(int)buffer.size()];
|
2016-01-14 03:38:59 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-06-21 01:47:27 +00:00
|
|
|
struct MachineDelegate: CRTMachine::Machine::Delegate {
|
|
|
|
__weak CSMachine *machine;
|
|
|
|
void machine_did_change_clock_rate(CRTMachine::Machine *sender) {
|
|
|
|
[machine machineDidChangeClockRate];
|
|
|
|
}
|
2016-09-13 02:06:03 +00:00
|
|
|
void machine_did_change_clock_is_unlimited(CRTMachine::Machine *sender) {
|
|
|
|
[machine machineDidChangeClockIsUnlimited];
|
|
|
|
}
|
2016-06-21 01:47:27 +00:00
|
|
|
};
|
|
|
|
|
2016-01-05 04:12:47 +00:00
|
|
|
@implementation CSMachine {
|
2016-01-14 03:38:59 +00:00
|
|
|
SpeakerDelegate _speakerDelegate;
|
2016-06-21 01:47:27 +00:00
|
|
|
MachineDelegate _machineDelegate;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (instancetype)init {
|
|
|
|
self = [super init];
|
2017-08-01 02:32:26 +00:00
|
|
|
if(self) {
|
2016-06-21 01:47:27 +00:00
|
|
|
_machineDelegate.machine = self;
|
|
|
|
self.machine->set_delegate(&_machineDelegate);
|
|
|
|
_speakerDelegate.machine = self;
|
|
|
|
}
|
|
|
|
return self;
|
2016-01-05 04:12:47 +00:00
|
|
|
}
|
|
|
|
|
2016-01-15 01:33:22 +00:00
|
|
|
- (void)speaker:(Outputs::Speaker *)speaker didCompleteSamples:(const int16_t *)samples length:(int)length {
|
|
|
|
[self.audioQueue enqueueAudioBuffer:samples numberOfSamples:(unsigned int)length];
|
2016-01-14 03:38:59 +00:00
|
|
|
}
|
|
|
|
|
2016-06-21 01:47:27 +00:00
|
|
|
- (void)machineDidChangeClockRate {
|
|
|
|
[self.delegate machineDidChangeClockRate:self];
|
|
|
|
}
|
|
|
|
|
2016-09-13 02:06:03 +00:00
|
|
|
- (void)machineDidChangeClockIsUnlimited {
|
|
|
|
[self.delegate machineDidChangeClockIsUnlimited:self];
|
|
|
|
}
|
|
|
|
|
2016-04-25 00:34:25 +00:00
|
|
|
- (void)dealloc {
|
|
|
|
[_view performWithGLContext:^{
|
2016-06-01 02:16:20 +00:00
|
|
|
@synchronized(self) {
|
|
|
|
self.machine->close_output();
|
|
|
|
}
|
2016-04-25 00:34:25 +00:00
|
|
|
}];
|
|
|
|
}
|
|
|
|
|
2016-06-13 23:30:41 +00:00
|
|
|
- (float)idealSamplingRateFromRange:(NSRange)range {
|
2016-06-01 23:04:07 +00:00
|
|
|
@synchronized(self) {
|
2016-07-04 23:33:55 +00:00
|
|
|
std::shared_ptr<Outputs::Speaker> speaker = self.machine->get_speaker();
|
2016-06-01 23:04:07 +00:00
|
|
|
if(speaker)
|
|
|
|
{
|
2016-06-13 23:30:41 +00:00
|
|
|
return speaker->get_ideal_clock_rate_in_range((float)range.location, (float)(range.location + range.length));
|
2016-06-01 23:04:07 +00:00
|
|
|
}
|
2016-06-05 15:20:05 +00:00
|
|
|
return 0;
|
2016-06-01 23:04:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-17 00:51:35 +00:00
|
|
|
- (void)setAudioSamplingRate:(float)samplingRate bufferSize:(NSUInteger)bufferSize {
|
2016-06-01 23:04:07 +00:00
|
|
|
@synchronized(self) {
|
2016-06-17 00:51:35 +00:00
|
|
|
[self setSpeakerDelegate:&_speakerDelegate sampleRate:samplingRate bufferSize:bufferSize];
|
2016-06-01 23:04:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-17 00:51:35 +00:00
|
|
|
- (BOOL)setSpeakerDelegate:(Outputs::Speaker::Delegate *)delegate sampleRate:(float)sampleRate bufferSize:(NSUInteger)bufferSize {
|
2016-06-01 02:16:20 +00:00
|
|
|
@synchronized(self) {
|
2016-07-04 23:33:55 +00:00
|
|
|
std::shared_ptr<Outputs::Speaker> speaker = self.machine->get_speaker();
|
2016-06-01 02:16:20 +00:00
|
|
|
if(speaker)
|
|
|
|
{
|
2016-06-17 00:51:35 +00:00
|
|
|
speaker->set_output_rate(sampleRate, (int)bufferSize);
|
2016-06-01 02:16:20 +00:00
|
|
|
speaker->set_delegate(delegate);
|
|
|
|
return YES;
|
|
|
|
}
|
|
|
|
return NO;
|
|
|
|
}
|
2016-01-14 03:38:59 +00:00
|
|
|
}
|
2016-03-20 02:46:17 +00:00
|
|
|
|
2016-06-01 02:16:20 +00:00
|
|
|
- (void)runForNumberOfCycles:(int)numberOfCycles {
|
|
|
|
@synchronized(self) {
|
2017-07-23 02:17:29 +00:00
|
|
|
self.machine->run_for(Cycles(numberOfCycles));
|
2016-06-01 02:16:20 +00:00
|
|
|
}
|
|
|
|
}
|
2016-03-20 02:46:17 +00:00
|
|
|
|
2016-04-12 03:12:56 +00:00
|
|
|
- (void)setView:(CSOpenGLView *)view aspectRatio:(float)aspectRatio {
|
2016-04-25 00:34:25 +00:00
|
|
|
_view = view;
|
2016-03-20 17:50:13 +00:00
|
|
|
[view performWithGLContext:^{
|
2016-04-12 03:12:56 +00:00
|
|
|
[self setupOutputWithAspectRatio:aspectRatio];
|
2016-03-20 17:50:13 +00:00
|
|
|
}];
|
|
|
|
}
|
|
|
|
|
2016-06-01 02:16:20 +00:00
|
|
|
- (void)setupOutputWithAspectRatio:(float)aspectRatio {
|
|
|
|
self.machine->setup_output(aspectRatio);
|
2017-08-10 19:17:08 +00:00
|
|
|
|
|
|
|
// Since OS X v10.6, Macs have had a gamma of 2.2.
|
|
|
|
self.machine->get_crt()->set_output_gamma(2.2f);
|
2016-06-01 02:16:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
- (void)drawViewForPixelSize:(CGSize)pixelSize onlyIfDirty:(BOOL)onlyIfDirty {
|
|
|
|
self.machine->get_crt()->draw_frame((unsigned int)pixelSize.width, (unsigned int)pixelSize.height, onlyIfDirty ? true : false);
|
|
|
|
}
|
|
|
|
|
2016-06-17 00:39:46 +00:00
|
|
|
- (double)clockRate {
|
|
|
|
return self.machine->get_clock_rate();
|
|
|
|
}
|
|
|
|
|
2016-09-13 02:06:03 +00:00
|
|
|
- (BOOL)clockIsUnlimited {
|
|
|
|
return self.machine->get_clock_is_unlimited() ? YES : NO;
|
|
|
|
}
|
|
|
|
|
2016-06-19 20:35:04 +00:00
|
|
|
- (void)paste:(NSString *)paste {
|
|
|
|
Utility::TypeRecipient *typeRecipient = dynamic_cast<Utility::TypeRecipient *>(self.machine);
|
|
|
|
if(typeRecipient)
|
|
|
|
typeRecipient->set_typer_for_string([paste UTF8String]);
|
|
|
|
}
|
|
|
|
|
2017-08-17 15:00:08 +00:00
|
|
|
- (void)applyTarget:(const StaticAnalyser::Target &)target {
|
2016-09-08 09:32:17 +00:00
|
|
|
@synchronized(self) {
|
|
|
|
ConfigurationTarget::Machine *const configurationTarget =
|
|
|
|
dynamic_cast<ConfigurationTarget::Machine *>(self.machine);
|
|
|
|
if(configurationTarget) configurationTarget->configure_as_target(target);
|
|
|
|
}
|
|
|
|
}
|
2016-09-01 00:43:29 +00:00
|
|
|
|
2017-08-17 15:00:08 +00:00
|
|
|
- (void)applyMedia:(const StaticAnalyser::Media &)media {
|
|
|
|
@synchronized(self) {
|
|
|
|
ConfigurationTarget::Machine *const configurationTarget =
|
|
|
|
dynamic_cast<ConfigurationTarget::Machine *>(self.machine);
|
|
|
|
if(configurationTarget) configurationTarget->insert_media(media);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-05 04:12:47 +00:00
|
|
|
@end
|