- Keyboard 2 Mouse

- FPS set back to 30
- 16 bit PCM
- Smoother speaker level fade out
- Faster keyboard poll query -- This is for checking only, should be either removed or the entire ioRead should based on binary search, whatever is faster
This commit is contained in:
tudnai 2020-05-24 12:14:09 -07:00
parent 0aea0b9c84
commit ab120dc459
6 changed files with 147 additions and 146 deletions

View File

@ -14,10 +14,13 @@
</PersistentStrings>
</ContextState>
<ContextState
contextName = "specialized closure #1 in ViewController.Update():ViewController.swift">
contextName = "BRA:6502_instr_branch.h">
<PersistentStrings>
<PersistentString
value = "self.HiRes.frame">
value = "">
</PersistentString>
<PersistentString
value = "RAM[0x346f]">
</PersistentString>
</PersistentStrings>
</ContextState>
@ -60,6 +63,14 @@
</PersistentString>
</PersistentStrings>
</ContextState>
<ContextState
contextName = "closure #1 in ViewController.Update():ViewController.swift">
<PersistentStrings>
<PersistentString
value = "self.shadowTxt">
</PersistentString>
</PersistentStrings>
</ContextState>
<ContextState
contextName = "m6502_Step:6502.c">
<PersistentStrings>
@ -175,6 +186,20 @@
</PersistentString>
</PersistentStrings>
</ContextState>
<ContextState
contextName = "spkr_update:speaker.c">
<PersistentStrings>
<PersistentString
value = "(uint8_t)spkr_samples[451]">
</PersistentString>
<PersistentString
value = "(uint8_t)spkr_samples[452]">
</PersistentString>
<PersistentString
value = "(uint8_t)spkr_samples[455]">
</PersistentString>
</PersistentStrings>
</ContextState>
<ContextState
contextName = "spkr_playUpd:speaker.c">
</ContextState>
@ -216,7 +241,7 @@
value = "pdl_value[pdl]">
</PersistentString>
<PersistentString
value = "normalized_time &gt;= pdl_value[pdl] ? 255 : 0">
value = "normalized_time">
</PersistentString>
<PersistentString
value = "(3300 * 255/3300)">
@ -228,7 +253,7 @@
value = "1 * 512 * (1 - ( 3300 / 3300.0 ))">
</PersistentString>
<PersistentString
value = "normalized_time">
value = "normalized_time &gt;= pdl_value[pdl] ? 255 : 0">
</PersistentString>
</PersistentStrings>
</ContextState>
@ -245,13 +270,13 @@
value = "textLines">
</PersistentString>
<PersistentString
value = "hires.layer">
value = "hires.frame">
</PersistentString>
<PersistentString
value = "view">
</PersistentString>
<PersistentString
value = "hires.frame">
value = "hires.layer">
</PersistentString>
</PersistentStrings>
</ContextState>
@ -278,10 +303,10 @@
contextName = "set_flags_NZC:common.h">
<PersistentStrings>
<PersistentString
value = "(unsigned)test ">
value = "(unsigned)0xFF">
</PersistentString>
<PersistentString
value = "(unsigned)0xFF">
value = "(unsigned)test ">
</PersistentString>
</PersistentStrings>
</ContextState>
@ -405,19 +430,25 @@
<PersistentString
value = "(clkfrm / (default_MHz_6502 / spkr_sample_rate))">
</PersistentString>
<PersistentString
value = "spkr_level">
</PersistentString>
<PersistentString
value = "spkr_sample_idx">
</PersistentString>
<PersistentString
value = "(int16_t)28000 + (int16_t)-32768">
</PersistentString>
</PersistentStrings>
</ContextState>
<ContextState
contextName = "HiRes.draw(_:):HiRes.swift">
<PersistentStrings>
<PersistentString
value = "linAddr">
value = "ctx?.bitsPerComponent">
</PersistentString>
<PersistentString
value = "shadowScreen">
value = "ctx?.data">
</PersistentString>
<PersistentString
value = "ctx?.width">
@ -434,21 +465,21 @@
<PersistentString
value = "ctx?.bitmapInfo">
</PersistentString>
<PersistentString
value = "shadowScreen">
</PersistentString>
<PersistentString
value = "(blockH7 | ( block &amp; bitMask ))">
</PersistentString>
<PersistentString
value = "HiRes.blockCols">
</PersistentString>
<PersistentString
value = "linAddr">
</PersistentString>
<PersistentString
value = "ctx?.height">
</PersistentString>
<PersistentString
value = "ctx?.bitsPerComponent">
</PersistentString>
<PersistentString
value = "ctx?.data">
</PersistentString>
</PersistentStrings>
</ContextState>
<ContextState
@ -463,13 +494,13 @@
contextName = "HiRes.compute():HiRes.swift">
<PersistentStrings>
<PersistentString
value = "computePipelineState.maxTotalThreadsPerThreadgroup">
value = "UnsafeRawBufferPointer(result)">
</PersistentString>
<PersistentString
value = "result[2]">
</PersistentString>
<PersistentString
value = "UnsafeRawBufferPointer(result)">
value = "computePipelineState.maxTotalThreadsPerThreadgroup">
</PersistentString>
</PersistentStrings>
</ContextState>
@ -491,15 +522,15 @@
<PersistentString
value = "textLines">
</PersistentString>
<PersistentString
value = "clkfrm">
</PersistentString>
<PersistentString
value = "mouseLocation">
</PersistentString>
<PersistentString
value = "txtArr">
</PersistentString>
<PersistentString
value = "clkfrm">
</PersistentString>
</PersistentStrings>
</ContextState>
<ContextState
@ -542,10 +573,10 @@
value = "Apple2_64K_AUX + 0xC600">
</PersistentString>
<PersistentString
value = "(void*)rom">
value = "Apple2_64K_RAM + 0xC600">
</PersistentString>
<PersistentString
value = "Apple2_64K_RAM + 0xC600">
value = "(void*)rom">
</PersistentString>
<PersistentString
value = "strlen(fullPath)">
@ -562,13 +593,10 @@
contextName = "BNE:6502_instr_branch.h">
</ContextState>
<ContextState
contextName = "BRA:6502_instr_branch.h">
contextName = "specialized closure #1 in ViewController.Update():ViewController.swift">
<PersistentStrings>
<PersistentString
value = "">
</PersistentString>
<PersistentString
value = "RAM[0x346f]">
value = "self.HiRes.frame">
</PersistentString>
</PersistentStrings>
</ContextState>

View File

@ -285,20 +285,33 @@ class ViewController: NSViewController {
switch keyCode {
case leftArrowKey:
kbdInput(0x08)
setIO(0xC064, 0);
// setIO(0xC064, 0);
print("LEFT", ddd);
// Keyboard 2 JoyStick (Game Controller / Paddle)
pdl_valarr[0] = 0
case rightArrowKey:
kbdInput(0x15)
setIO(0xC064, 255);
// setIO(0xC064, 255);
print("RIGHT")
// Keyboard 2 JoyStick (Game Controller / Paddle)
pdl_valarr[0] = 1
case downArrowKey:
kbdInput(0x0B)
setIO(0xC065, 255);
// setIO(0xC065, 255);
print("DOWN")
// Keyboard 2 JoyStick (Game Controller / Paddle)
pdl_valarr[1] = 0
case upArrowKey:
kbdInput(0x0A)
setIO(0xC065, 0);
// setIO(0xC065, 0);
print("UP")
// Keyboard 2 JoyStick (Game Controller / Paddle)
pdl_valarr[1] = 1
default:
// print("keycode: %d", keyCode)
if let chars = event.characters {
@ -335,20 +348,32 @@ class ViewController: NSViewController {
switch keyCode {
case leftArrowKey:
// kbdInput(0x08)
setIO(0xC064, 127);
// setIO(0xC064, 127);
print("left")
// Keyboard 2 JoyStick (Game Controller / Paddle)
pdl_valarr[0] = 0.5
case rightArrowKey:
// kbdInput(0x15)
setIO(0xC064, 128);
// setIO(0xC064, 128);
print("right")
// Keyboard 2 JoyStick (Game Controller / Paddle)
pdl_valarr[0] = 0.5
case downArrowKey:
// kbdInput(0x0B)
setIO(0xC065, 127);
// setIO(0xC065, 127);
print("down")
// Keyboard 2 JoyStick (Game Controller / Paddle)
pdl_valarr[1] = 0.5
case upArrowKey:
// kbdInput(0x0A)
setIO(0xC065, 128);
// setIO(0xC065, 128);
print("up")
// Keyboard 2 JoyStick (Game Controller / Paddle)
pdl_valarr[1] = 0.5
default:
// print("keycode: %d", keyCode)
// if let chars = event.characters {
@ -425,6 +450,8 @@ class ViewController: NSViewController {
var mouseLocation = NSPoint.zero
var shadowTxt : String = ""
func Update() {
// clk_6502_per_frm_max = 0
@ -466,9 +493,8 @@ class ViewController: NSViewController {
// 1. We can update UI elements
// 2. it is independent of the simulation, de that is running in the background thread while we are busy with rendering...
DispatchQueue.main.async {
var txt : String = ""
var fromLines = 0
var toLines = self.textLines
@ -543,7 +569,10 @@ class ViewController: NSViewController {
}
}
self.display.stringValue = txt
if ( self.shadowTxt != txt ) {
self.shadowTxt = txt
self.display.stringValue = txt
}
// self.display.stringValue = "testing\nit\nout"
if ( (mhz < 1.5) && (mhz != floor(mhz)) ) {
@ -589,16 +618,16 @@ class ViewController: NSViewController {
#if SPEEDTEST
#else
if ( !halted ) {
// Mouse 2 JoyStick (Game Controller / Paddle)
mouseLocation = view.window!.mouseLocationOutsideOfEventStream
pdl_prevarr[0] = pdl_valarr[0]
pdl_valarr[0] = Double(mouseLocation.x / (displayField.frame.width) )
pdl_diffarr[0] = pdl_valarr[0] - pdl_prevarr[0]
pdl_prevarr[1] = pdl_valarr[1]
pdl_valarr[1] = 1 - Double(mouseLocation.y / (displayField.frame.height) )
pdl_diffarr[1] = pdl_valarr[1] - pdl_prevarr[1]
// // Mouse 2 JoyStick (Game Controller / Paddle)
// mouseLocation = view.window!.mouseLocationOutsideOfEventStream
//
// pdl_prevarr[0] = pdl_valarr[0]
// pdl_valarr[0] = Double(mouseLocation.x / (displayField.frame.width) )
// pdl_diffarr[0] = pdl_valarr[0] - pdl_prevarr[0]
//
// pdl_prevarr[1] = pdl_valarr[1]
// pdl_valarr[1] = 1 - Double(mouseLocation.y / (displayField.frame.height) )
// pdl_diffarr[1] = pdl_valarr[1] - pdl_prevarr[1]
m6502_Run()
}
@ -627,7 +656,6 @@ class ViewController: NSViewController {
hires.clearScreen();
woz_loadFile( Bundle.main.resourcePath, "Apple DOS 3.3 January 1983.woz" )
let spk_up_path = Bundle.main.path(forResource: "spk_up", ofType:"wav")!
let spk_up_url = URL(fileURLWithPath: spk_up_path)

View File

@ -157,7 +157,7 @@ extern double * pdl_diffarr;
extern double mips;
extern double mhz;
#define fps 60U
#define fps 30U
extern void rom_loadFile( const char * bundlePath, const char * filename );
extern void woz_loadFile( const char * bundlePath, const char * filename );

View File

@ -70,7 +70,7 @@ const int spkr_seconds = 1;
const unsigned spkr_sample_rate = 44100;
unsigned spkr_extra_buf = 780 / fps;
const unsigned spkr_buf_size = spkr_seconds * spkr_sample_rate * 2 / spkr_fps;
char spkr_samples [ spkr_buf_size * spkr_fps * BUFFER_COUNT * 2]; // stereo
int16_t spkr_samples [ spkr_buf_size * spkr_fps * BUFFER_COUNT * 2]; // stereo
unsigned spkr_sample_idx = 0;
const unsigned spkr_play_timeout = 8;
@ -90,7 +90,7 @@ void spkr_init() {
alcMakeContextCurrent(ctx);
// Fill buffer with zeros
memset( spkr_samples, spkr_level, spkr_buf_size );
memset( spkr_samples, spkr_level, spkr_buf_size * sizeof(spkr_samples[0]) );
// Create buffer to store samples
alGenBuffers(BUFFER_COUNT, spkr_buffers);
@ -146,7 +146,7 @@ void spkr_toggle() {
// (we will play the entire buffer at the end of the frame)
spkr_sample_idx = (clkfrm / (default_MHz_6502 / spkr_sample_rate)) * 2;
if ( spkr_level > SPKR_LEVEL_MIN ) {
if ( spkr_level > SPKR_LEVEL_ZERO ) {
// down edge
while( (spkr_level = (spkr_level + SPKR_LEVEL_MIN) / 2 ) > SPKR_LEVEL_MIN + 1 ) {
spkr_samples[ spkr_sample_idx++ ] = spkr_level;
@ -163,19 +163,11 @@ void spkr_toggle() {
spkr_level = SPKR_LEVEL_MAX;
}
//spkr_samples[sample_idx] = spkr_level;
memset(spkr_samples + spkr_sample_idx, spkr_level, spkr_buf_size);
for ( int i = spkr_sample_idx; i < spkr_buf_size + spkr_extra_buf; i++ ) {
spkr_samples[i] = spkr_level;
}
// memset(spkr_samples + spkr_sample_idx, spkr_level, spkr_buf_size * sizeof(spkr_samples[0]));
//ViewController_spk_up_play();
// case io_VID_CLR80VID:
// videoMode.col80 = 0;
// break;
//
// case io_VID_SET80VID:
// videoMode.col80 = 1;
// break;
//
}
@ -186,29 +178,8 @@ ALint freeBuffers = BUFFER_COUNT;
void spkr_update() {
if ( spkr_play_time ) {
// printf("freeBuffers: %d", freeBuffers);
// if ( spkr_src ) {
// alSourceStop(spkr_src);
// al_check_error();
// alSourcei(spkr_src, AL_BUFFER, 0);
// al_check_error();
//
// spkr_src = 0;
// }
//
// if ( spkr_buf ) {
// alDeleteBuffers(1, &spkr_buf);
// al_check_error();
// spkr_buf = 0;
// }
// ALint queued = 0;
ALint processed = 0;
// alGetSourcei (spkr_src, AL_BUFFERS_QUEUED, &queued);
// alGetSourcei (spkr_src, AL_BUFFERS_PROCESSED, &processed);
//
// printf("freeBuffers:%d queued:%d processed:%d\n", freeBuffers, queued,processed);
do {
@ -221,8 +192,6 @@ void spkr_update() {
freeBuffers += processed;
}
// usleep(100);
// if ( freeBuffers <= 0 ) printf("No Free Buffer\n");
} while( freeBuffers <= 0 );
freeBuffers = clamp( 1, freeBuffers, BUFFER_COUNT );
@ -233,33 +202,33 @@ void spkr_update() {
alGetSourcei( spkr_src, AL_SOURCE_STATE, &state );
// al_check_error();
// spkr_samples[0] = 0;
// spkr_samples[1] = 255;
// spkr_samples[2] = 0;
// spkr_samples[3] = 255;
// spkr_samples[4] = 0;
// Download buffer to OpenAL
////////// check if there is no sound generated for long time, and fade out speaker level to avoid pops and crackles
if ( --spkr_play_time == 0 ) {
// we need to soft mute the speaker to eliminate clicking noise
// simple linear mute
int step = (SPKR_LEVEL_ZERO - (int)spkr_level) / 128;
const int steepness = 64;
float step = (SPKR_LEVEL_ZERO - (float)spkr_level) / steepness;
float fadeLevel = spkr_level - SPKR_LEVEL_ZERO;
if ( step != 0 ) {
for ( spkr_sample_idx = 0; spkr_level != SPKR_LEVEL_ZERO; spkr_level += step) {
for ( int i = 0; i < 4; i++ ) {
spkr_samples[ spkr_sample_idx++ ] = spkr_level;
spkr_samples[ spkr_sample_idx++ ] = spkr_level;
}
spkr_sample_idx = 0;
while ( ( fadeLevel < -1 ) || ( fadeLevel > 1 ) ) {
spkr_samples[ spkr_sample_idx++ ] = SPKR_LEVEL_ZERO + fadeLevel;
spkr_samples[ spkr_sample_idx++ ] = SPKR_LEVEL_ZERO + fadeLevel;
// how smooth we want the speeker to decay, so we will hear no pops and crackles
fadeLevel *= 0.999;
}
spkr_level = SPKR_LEVEL_ZERO;
spkr_samples[ spkr_sample_idx++ ] = spkr_level;
spkr_samples[ spkr_sample_idx++ ] = spkr_level;
spkr_level = SPKR_LEVEL_ZERO + fadeLevel;
//spkr_samples[sample_idx] = spkr_level;
memset(spkr_samples + spkr_sample_idx, spkr_level, spkr_extra_buf);
memset(spkr_samples + spkr_sample_idx, SPKR_LEVEL_ZERO, spkr_extra_buf * sizeof(spkr_samples[0]));
freeBuffers--;
alBufferData(spkr_buffers[freeBuffers], AL_FORMAT_STEREO8, spkr_samples, spkr_sample_idx, spkr_sample_rate);
alBufferData(spkr_buffers[freeBuffers], AL_FORMAT_STEREO16, spkr_samples, spkr_sample_idx * sizeof(spkr_samples[0]), spkr_sample_rate);
al_check_error();
alSourceQueueBuffers(spkr_src, 1, &spkr_buffers[freeBuffers]);
al_check_error();
@ -267,7 +236,7 @@ void spkr_update() {
}
else {
freeBuffers--;
alBufferData(spkr_buffers[freeBuffers], AL_FORMAT_STEREO8, spkr_samples, spkr_buf_size + spkr_extra_buf, spkr_sample_rate);
alBufferData(spkr_buffers[freeBuffers], AL_FORMAT_STEREO16, spkr_samples, (spkr_buf_size + spkr_extra_buf) * sizeof(spkr_samples[0]), spkr_sample_rate);
al_check_error();
alSourceQueueBuffers(spkr_src, 1, &spkr_buffers[freeBuffers]);
al_check_error();
@ -289,45 +258,15 @@ void spkr_update() {
}
// clear the slack buffer , so we can fill it up by new data
memset(spkr_samples, spkr_level, spkr_buf_size + spkr_extra_buf);
for ( int i = 0; i < spkr_buf_size + spkr_extra_buf; i++ ) {
spkr_samples[i] = spkr_level;
}
// memset(spkr_samples, spkr_level, (spkr_buf_size + spkr_extra_buf) * sizeof(spkr_samples[0]));
// start from the beginning
spkr_sample_idx = 0;
}
else {
// TODO: Need better speaker turn off logic to avoid click noise
//// if ( spkr_src ) {
//// alSourceStop(spkr_src);
//// al_check_error();
//// alSourceUnqueueBuffers( spkr_src, BUFFER_COUNT, spkr_buffers);
//// al_check_error();
//// alSourcei(spkr_src, AL_BUFFER, 0);
//// al_check_error();
//// spkr_src = 0;
//// // clear the buffer
//// memset(spkr_samples, spkr_level, spkr_buf_size);
//// }
//// printf("freeBuffers_nosound: %d\n", freeBuffers);
}
}
//void spkr_Update() {
// if ( spkr_src && spkr_buf ) {
// if ( spkr_src ) {
// alSourcePause(spkr_src);
// al_check_error();
// alSourcei(spkr_src, AL_BUFFER, 0);
// al_check_error();
// }
//
// // Download buffer to OpenAL
// alBufferData(spkr_buf, AL_FORMAT_MONO8, spkr_samples, spkr_buf_size, spkr_sample_rate);
// al_check_error();
//
// alSourcei( spkr_src, AL_BYTE_OFFSET, 0 );
// al_check_error();
// alSourcePlay(spkr_src);
// }
//}

View File

@ -11,15 +11,15 @@
#include <stdio.h>
#define SPKR_LEVEL_MIN 32
#define SPKR_LEVEL_MAX 223
#define SPKR_LEVEL_ZERO 128 // as defined in OpenAL documentation for 8bit PCM
#define SPKR_LEVEL_MIN (-28000)
#define SPKR_LEVEL_MAX (+28000)
#define SPKR_LEVEL_ZERO 0 // as defined in OpenAL documentation for 8bit PCM
extern const unsigned spkr_sample_rate;
extern const unsigned spkr_buf_size;
extern const int spkr_fps;
extern char spkr_samples [];
extern int16_t spkr_samples [];
extern unsigned spkr_sample_idx;
extern int spkr_level;
extern int freeBuffers;

View File

@ -407,6 +407,12 @@ INLINE uint8_t ioRead( uint16_t addr ) {
uint8_t currentMagnet = 0;
// TODO: This is for checking only, should be either removed or the entire ioRead should based on binary search, whatever is faster
if ( addr == io_KBD ) {
// clk_6502_per_frm_max = clk_6502_per_frm_max > 32768 ? clk_6502_per_frm_max - 32768 : 0; // ECO Mode!
return Apple2_64K_RAM[io_KBD];
}
switch ( (uint8_t)addr ) {
case (uint8_t)io_KBD:
// if ( RAM[io_KBD] > 0x7F ) printf("io_KBD:%04X\n", addr);