mirror of
https://github.com/TomHarte/CLK.git
synced 2025-01-12 15:31:09 +00:00
A full communication pathway now leads to Atari2600.mm (for now) being in possession of a frame and receiving a command to draw, with a suitable OpenGL context being active and whatever is drawn subsequently appearing.
This commit is contained in:
parent
0611646181
commit
7276c77027
@ -8,9 +8,18 @@
|
|||||||
|
|
||||||
#import <Foundation/Foundation.h>
|
#import <Foundation/Foundation.h>
|
||||||
|
|
||||||
|
@class CSAtari2600;
|
||||||
|
@protocol CSAtari2600Delegate
|
||||||
|
- (void)atari2600NeedsRedraw:(CSAtari2600 *)atari2600;
|
||||||
|
@end
|
||||||
|
|
||||||
@interface CSAtari2600 : NSObject
|
@interface CSAtari2600 : NSObject
|
||||||
|
|
||||||
|
@property (nonatomic, weak) id <CSAtari2600Delegate> delegate;
|
||||||
|
|
||||||
- (void)runForNumberOfCycles:(int)cycles;
|
- (void)runForNumberOfCycles:(int)cycles;
|
||||||
- (void)setROM:(NSData *)rom;
|
- (void)setROM:(NSData *)rom;
|
||||||
|
|
||||||
|
- (void)draw;
|
||||||
|
|
||||||
@end
|
@end
|
||||||
|
@ -8,6 +8,7 @@
|
|||||||
|
|
||||||
#import "Atari2600.h"
|
#import "Atari2600.h"
|
||||||
#import "Atari2600.hpp"
|
#import "Atari2600.hpp"
|
||||||
|
#import <OpenGL/gl.h>
|
||||||
|
|
||||||
@interface CSAtari2600 (Callbacks)
|
@interface CSAtari2600 (Callbacks)
|
||||||
- (void)crtDidEndFrame:(Outputs::CRTFrame *)frame;
|
- (void)crtDidEndFrame:(Outputs::CRTFrame *)frame;
|
||||||
@ -23,18 +24,35 @@ struct Atari2600CRTDelegate: public Outputs::CRT::CRTDelegate {
|
|||||||
Atari2600CRTDelegate _crtDelegate;
|
Atari2600CRTDelegate _crtDelegate;
|
||||||
|
|
||||||
dispatch_queue_t _serialDispatchQueue;
|
dispatch_queue_t _serialDispatchQueue;
|
||||||
|
Outputs::CRTFrame *_queuedFrame;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)crtDidEndFrame:(Outputs::CRTFrame *)frame {
|
- (void)crtDidEndFrame:(Outputs::CRTFrame *)frame {
|
||||||
|
|
||||||
dispatch_async(dispatch_get_main_queue(), ^{
|
dispatch_async(dispatch_get_main_queue(), ^{
|
||||||
|
if(_queuedFrame) {
|
||||||
|
dispatch_async(_serialDispatchQueue, ^{
|
||||||
|
_atari2600.get_crt()->return_frame();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
_queuedFrame = frame;
|
||||||
|
|
||||||
printf("\n\n===\n\n");
|
[self.delegate atari2600NeedsRedraw:self];
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
- (void)draw {
|
||||||
|
|
||||||
|
if(_queuedFrame)
|
||||||
|
{
|
||||||
|
glClearColor(1.0, 0.0, 0.0, 1.0);
|
||||||
|
glClear(GL_COLOR_BUFFER_BIT);
|
||||||
|
/* printf("\n\n===\n\n");
|
||||||
int c = 0;
|
int c = 0;
|
||||||
for(int run = 0; run < frame->number_of_runs; run++)
|
for(int run = 0; run < _queuedFrame->number_of_runs; run++)
|
||||||
{
|
{
|
||||||
char character = ' ';
|
char character = ' ';
|
||||||
switch(frame->runs[run].type)
|
switch(_queuedFrame->runs[run].type)
|
||||||
{
|
{
|
||||||
case Outputs::CRTRun::Type::Sync: character = '<'; break;
|
case Outputs::CRTRun::Type::Sync: character = '<'; break;
|
||||||
case Outputs::CRTRun::Type::Level: character = '_'; break;
|
case Outputs::CRTRun::Type::Level: character = '_'; break;
|
||||||
@ -42,14 +60,14 @@ struct Atari2600CRTDelegate: public Outputs::CRT::CRTDelegate {
|
|||||||
case Outputs::CRTRun::Type::Blank: character = ' '; break;
|
case Outputs::CRTRun::Type::Blank: character = ' '; break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(frame->runs[run].start_point.dst_x < 1.0 / 224.0)
|
if(_queuedFrame->runs[run].start_point.dst_x < 1.0 / 224.0)
|
||||||
{
|
{
|
||||||
printf("\n[%0.2f]: ", frame->runs[run].start_point.dst_y);
|
printf("\n[%0.2f]: ", _queuedFrame->runs[run].start_point.dst_y);
|
||||||
c++;
|
c++;
|
||||||
}
|
}
|
||||||
|
|
||||||
printf("(%0.2f): ", frame->runs[run].start_point.dst_x);
|
printf("(%0.2f): ", _queuedFrame->runs[run].start_point.dst_x);
|
||||||
float length = fabsf(frame->runs[run].end_point.dst_x - frame->runs[run].start_point.dst_x);
|
float length = fabsf(_queuedFrame->runs[run].end_point.dst_x - _queuedFrame->runs[run].start_point.dst_x);
|
||||||
int iLength = (int)(length * 64.0);
|
int iLength = (int)(length * 64.0);
|
||||||
for(int c = 0; c < iLength; c++)
|
for(int c = 0; c < iLength; c++)
|
||||||
{
|
{
|
||||||
@ -57,12 +75,8 @@ struct Atari2600CRTDelegate: public Outputs::CRT::CRTDelegate {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
printf("\n\n[%d]\n\n", c);
|
printf("\n\n[%d]\n\n", c);*/
|
||||||
|
}
|
||||||
dispatch_async(_serialDispatchQueue, ^{
|
|
||||||
_atari2600.get_crt()->return_frame();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)runForNumberOfCycles:(int)cycles {
|
- (void)runForNumberOfCycles:(int)cycles {
|
||||||
|
@ -8,7 +8,7 @@
|
|||||||
|
|
||||||
import Cocoa
|
import Cocoa
|
||||||
|
|
||||||
class Atari2600Document: NSDocument, CSOpenGLViewDelegate {
|
class Atari2600Document: NSDocument, CSOpenGLViewDelegate, CSAtari2600Delegate {
|
||||||
|
|
||||||
override init() {
|
override init() {
|
||||||
super.init()
|
super.init()
|
||||||
@ -44,9 +44,12 @@ class Atari2600Document: NSDocument, CSOpenGLViewDelegate {
|
|||||||
// You can also choose to override readFromFileWrapper:ofType:error: or readFromURL:ofType:error: instead.
|
// You can also choose to override readFromFileWrapper:ofType:error: or readFromURL:ofType:error: instead.
|
||||||
// If you override either of these, you should also override -isEntireFileLoaded to return false if the contents are lazily loaded.
|
// If you override either of these, you should also override -isEntireFileLoaded to return false if the contents are lazily loaded.
|
||||||
atari2600 = CSAtari2600()
|
atari2600 = CSAtari2600()
|
||||||
atari2600?.setROM(data)
|
atari2600!.setROM(data)
|
||||||
|
atari2600!.delegate = self
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// MARK: CSOpenGLViewDelegate
|
||||||
|
|
||||||
private var lastCycleCount: Int64?
|
private var lastCycleCount: Int64?
|
||||||
func openGLView(view: CSOpenGLView!, didUpdateToTime time: CVTimeStamp) {
|
func openGLView(view: CSOpenGLView!, didUpdateToTime time: CVTimeStamp) {
|
||||||
|
|
||||||
@ -66,4 +69,14 @@ class Atari2600Document: NSDocument, CSOpenGLViewDelegate {
|
|||||||
}
|
}
|
||||||
lastCycleCount = cycleCount
|
lastCycleCount = cycleCount
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func openGLViewDrawView(view: CSOpenGLView!) {
|
||||||
|
atari2600!.draw()
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: CSAtari2600Delegate
|
||||||
|
|
||||||
|
func atari2600NeedsRedraw(atari2600: CSAtari2600!) {
|
||||||
|
openGLView?.needsDisplay = true
|
||||||
|
}
|
||||||
}
|
}
|
@ -13,6 +13,7 @@
|
|||||||
|
|
||||||
@protocol CSOpenGLViewDelegate
|
@protocol CSOpenGLViewDelegate
|
||||||
- (void)openGLView:(CSOpenGLView *)view didUpdateToTime:(CVTimeStamp)time;
|
- (void)openGLView:(CSOpenGLView *)view didUpdateToTime:(CVTimeStamp)time;
|
||||||
|
- (void)openGLViewDrawView:(CSOpenGLView *)view;
|
||||||
@end
|
@end
|
||||||
|
|
||||||
@interface CSOpenGLView : NSOpenGLView
|
@interface CSOpenGLView : NSOpenGLView
|
||||||
|
@ -8,6 +8,7 @@
|
|||||||
|
|
||||||
#import "OpenGLView.h"
|
#import "OpenGLView.h"
|
||||||
@import CoreVideo;
|
@import CoreVideo;
|
||||||
|
#import <OpenGL/gl.h>
|
||||||
|
|
||||||
@implementation CSOpenGLView {
|
@implementation CSOpenGLView {
|
||||||
CVDisplayLinkRef displayLink;
|
CVDisplayLinkRef displayLink;
|
||||||
@ -23,7 +24,7 @@
|
|||||||
CVDisplayLinkCreateWithActiveCGDisplays(&displayLink);
|
CVDisplayLinkCreateWithActiveCGDisplays(&displayLink);
|
||||||
|
|
||||||
// Set the renderer output callback function
|
// Set the renderer output callback function
|
||||||
CVDisplayLinkSetOutputCallback(displayLink, &MyDisplayLinkCallback, (__bridge void * __nullable)(self));
|
CVDisplayLinkSetOutputCallback(displayLink, DisplayLinkCallback, (__bridge void * __nullable)(self));
|
||||||
|
|
||||||
// Set the display link for the current renderer
|
// Set the display link for the current renderer
|
||||||
CGLContextObj cglContext = [[self openGLContext] CGLContextObj];
|
CGLContextObj cglContext = [[self openGLContext] CGLContextObj];
|
||||||
@ -34,7 +35,7 @@
|
|||||||
CVDisplayLinkStart(displayLink);
|
CVDisplayLinkStart(displayLink);
|
||||||
}
|
}
|
||||||
|
|
||||||
static CVReturn MyDisplayLinkCallback(CVDisplayLinkRef displayLink, const CVTimeStamp* now, const CVTimeStamp* outputTime, CVOptionFlags flagsIn, CVOptionFlags* flagsOut, void* displayLinkContext)
|
static CVReturn DisplayLinkCallback(CVDisplayLinkRef displayLink, const CVTimeStamp* now, const CVTimeStamp* outputTime, CVOptionFlags flagsIn, CVOptionFlags* flagsOut, void* displayLinkContext)
|
||||||
{
|
{
|
||||||
CSOpenGLView *view = (__bridge CSOpenGLView *)displayLinkContext;
|
CSOpenGLView *view = (__bridge CSOpenGLView *)displayLinkContext;
|
||||||
[view.delegate openGLView:view didUpdateToTime:*now];
|
[view.delegate openGLView:view didUpdateToTime:*now];
|
||||||
@ -47,4 +48,15 @@ static CVReturn MyDisplayLinkCallback(CVDisplayLinkRef displayLink, const CVTime
|
|||||||
CVDisplayLinkRelease(displayLink);
|
CVDisplayLinkRelease(displayLink);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
- (void)drawRect:(NSRect)dirtyRect
|
||||||
|
{
|
||||||
|
[self.openGLContext makeCurrentContext];
|
||||||
|
|
||||||
|
CGSize viewSize = [self convertSize:self.bounds.size toView:self];
|
||||||
|
glViewport((GLint)0, (GLint)0, (GLsizei)viewSize.width, (GLsizei)viewSize.height);
|
||||||
|
[self.delegate openGLViewDrawView:self];
|
||||||
|
|
||||||
|
glSwapAPPLE();
|
||||||
|
}
|
||||||
|
|
||||||
@end
|
@end
|
||||||
|
Loading…
x
Reference in New Issue
Block a user