2017-01-03 20:01:18 +00:00
# include <stdio.h>
# include <stdlib.h>
2017-12-18 09:15:02 -08:00
# include <unistd.h>
2017-01-03 20:01:18 +00:00
# include <termios.h>
# include <time.h>
# include "6502.h"
# include "6850.h"
2017-01-03 20:18:36 +00:00
struct termios initial_termios ;
2017-01-03 20:01:18 +00:00
void step_delay ( )
{
struct timespec req , rem ;
req . tv_sec = 0 ;
req . tv_nsec = STEP_DURATION ;
nanosleep ( & req , & rem ) ;
}
2017-12-19 14:55:24 -08:00
void run_cpu ( long cycle_stop , int verbose , int mem_dump , int break_pc )
2017-01-03 20:01:18 +00:00
{
2017-12-19 10:06:59 -08:00
long cycles = 0 ;
2017-01-08 16:28:32 +00:00
int cycles_per_step = ( CPU_FREQ / ( ONE_SECOND / STEP_DURATION ) ) ;
for ( ; ; ) {
for ( cycles % = cycles_per_step ; cycles < cycles_per_step ; ) {
2017-12-18 09:15:02 -08:00
if ( mem_dump ) save_memory ( NULL ) ;
cycles + = step_cpu ( verbose ) ;
2017-12-19 10:06:59 -08:00
if ( ( cycle_stop > 0 ) & & ( total_cycles > = cycle_stop ) ) goto end ;
2017-01-03 20:01:18 +00:00
step_uart ( ) ;
2017-12-19 14:55:24 -08:00
if ( break_pc > = 0 & & PC = = ( uint16_t ) break_pc ) {
fprintf ( stderr , " break at %04x \n " , break_pc ) ;
save_memory ( NULL ) ;
goto end ;
}
2017-01-03 20:01:18 +00:00
}
step_delay ( ) ; // remove this for more speed
}
2017-12-18 09:52:11 -08:00
end :
return ;
2017-01-03 20:01:18 +00:00
}
void restore_stdin ( )
{
2017-01-03 20:18:36 +00:00
tcsetattr ( 0 , TCSANOW , & initial_termios ) ;
2017-01-03 20:01:18 +00:00
}
void raw_stdin ( )
{
2017-01-03 20:18:36 +00:00
struct termios new_termios ;
2017-01-03 20:01:18 +00:00
2017-01-03 20:18:36 +00:00
tcgetattr ( 0 , & initial_termios ) ;
new_termios = initial_termios ;
cfmakeraw ( & new_termios ) ;
tcsetattr ( 0 , TCSANOW , & new_termios ) ;
2017-01-03 20:01:18 +00:00
atexit ( restore_stdin ) ;
}
2017-12-18 09:15:02 -08:00
int hextoint ( char * str ) {
int val ;
if ( * str = = ' $ ' ) str + + ;
val = strtol ( str , NULL , 16 ) ;
return val ;
}
2017-01-03 20:01:18 +00:00
int main ( int argc , char * argv [ ] )
{
2017-12-19 11:44:17 -08:00
int a , x , y , sp , sr , pc , load_addr ;
2017-12-19 14:55:24 -08:00
int verbose , interactive , mem_dump , break_pc ;
2017-12-19 10:06:59 -08:00
long cycles ;
2017-12-18 09:15:02 -08:00
int opt ;
verbose = 0 ;
interactive = 0 ;
mem_dump = 0 ;
2017-12-18 09:52:11 -08:00
cycles = 0 ;
2017-12-19 11:44:17 -08:00
load_addr = 0xC000 ;
2017-12-19 14:55:24 -08:00
break_pc = - 1 ;
2017-12-18 09:15:02 -08:00
a = 0 ;
x = 0 ;
y = 0 ;
sp = 0 ;
sr = 0 ;
pc = - RST_VEC ; // negative implies indirect
2017-12-19 14:55:24 -08:00
while ( ( opt = getopt ( argc , argv , " vima:b:x:y:r:p:s:g:c:l: " ) ) ! = - 1 ) {
2017-12-18 09:15:02 -08:00
switch ( opt ) {
case ' v ' :
verbose = 1 ;
break ;
case ' i ' :
interactive = 1 ;
break ;
case ' m ' :
mem_dump = 1 ;
break ;
2017-12-19 14:55:24 -08:00
case ' b ' :
break_pc = hextoint ( optarg ) ;
break ;
2017-12-18 09:15:02 -08:00
case ' a ' :
a = hextoint ( optarg ) ;
break ;
case ' x ' :
x = hextoint ( optarg ) ;
break ;
case ' y ' :
y = hextoint ( optarg ) ;
break ;
case ' s ' :
sp = hextoint ( optarg ) ;
break ;
case ' p ' :
sr = hextoint ( optarg ) ;
break ;
case ' r ' :
case ' g ' :
pc = hextoint ( optarg ) ;
break ;
2017-12-18 09:52:11 -08:00
case ' c ' :
2017-12-19 10:06:59 -08:00
cycles = atol ( optarg ) ;
2017-12-18 09:52:11 -08:00
break ;
2017-12-19 11:44:17 -08:00
case ' l ' :
load_addr = hextoint ( optarg ) ;
break ;
2017-12-18 09:15:02 -08:00
default : /* '?' */
fprintf ( stderr , " Usage: %s [-v] [-i] [-a HEX] [-x HEX] [-y HEX] [-s HEX] [-p HEX] [-g|-r ADDR] file.rom \n The first 16k of \" file.rom \" is loaded into the last 16k of memory. \n " ,
argv [ 0 ] ) ;
exit ( EXIT_FAILURE ) ;
}
2017-01-03 20:01:18 +00:00
}
2017-12-18 09:15:02 -08:00
if ( optind > = argc ) {
fprintf ( stderr , " Expected argument after options \n " ) ;
exit ( EXIT_FAILURE ) ;
}
2017-12-19 11:44:17 -08:00
if ( load_rom ( argv [ optind ] , load_addr ) ! = 0 ) {
2017-12-18 09:15:02 -08:00
printf ( " Error loading \" %s \" . \n " , argv [ optind ] ) ;
2017-01-03 20:01:18 +00:00
return EXIT_FAILURE ;
}
2017-12-18 09:15:02 -08:00
if ( interactive ) raw_stdin ( ) ; // allow individual keystrokes to be detected
2017-01-03 20:01:18 +00:00
init_tables ( ) ;
init_uart ( ) ;
2017-12-18 09:15:02 -08:00
reset_cpu ( a , x , y , sp , sr , pc ) ;
2017-12-19 14:55:24 -08:00
run_cpu ( cycles , verbose , mem_dump , break_pc ) ;
2017-01-03 20:01:18 +00:00
return EXIT_SUCCESS ;
}