2017-05-02 13:59:39 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include <SDL.h>
|
|
|
|
|
2017-05-02 14:56:59 +00:00
|
|
|
#include "gr-sim.h"
|
|
|
|
|
2017-05-02 13:59:39 +00:00
|
|
|
#define XSIZE 40
|
|
|
|
#define YSIZE 48
|
|
|
|
|
|
|
|
#define PIXEL_X_SCALE 14
|
|
|
|
#define PIXEL_Y_SCALE 8
|
|
|
|
|
|
|
|
static int xsize=XSIZE*PIXEL_X_SCALE;
|
|
|
|
static int ysize=YSIZE*PIXEL_Y_SCALE;
|
|
|
|
|
2017-05-03 16:20:53 +00:00
|
|
|
static int debug=0;
|
2017-05-02 13:59:39 +00:00
|
|
|
|
2017-05-03 14:50:15 +00:00
|
|
|
|
2017-05-02 14:56:59 +00:00
|
|
|
/* 128kB of RAM */
|
2017-05-03 14:50:15 +00:00
|
|
|
#define RAMSIZE 128*1024
|
|
|
|
unsigned char ram[RAMSIZE];
|
2017-05-02 14:56:59 +00:00
|
|
|
|
2017-05-03 16:20:53 +00:00
|
|
|
/* Zero page addresses */
|
|
|
|
#define GBASL 40
|
|
|
|
#define GBASH 41
|
|
|
|
#define MASK 50
|
|
|
|
#define COLOR 56
|
|
|
|
|
|
|
|
|
2017-05-02 14:56:59 +00:00
|
|
|
|
|
|
|
static SDL_Surface *sdl_screen=NULL;
|
|
|
|
|
|
|
|
int grsim_input(void) {
|
2017-05-02 13:59:39 +00:00
|
|
|
|
|
|
|
SDL_Event event;
|
|
|
|
int keypressed;
|
|
|
|
|
|
|
|
|
|
|
|
while ( SDL_PollEvent(&event)) {
|
|
|
|
|
|
|
|
switch(event.type) {
|
|
|
|
|
|
|
|
case SDL_KEYDOWN:
|
|
|
|
keypressed=event.key.keysym.sym;
|
|
|
|
switch (keypressed) {
|
|
|
|
|
|
|
|
case SDLK_ESCAPE:
|
|
|
|
return 'q';
|
|
|
|
default:
|
|
|
|
return keypressed;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case SDL_JOYBUTTONDOWN:
|
|
|
|
case SDL_JOYAXISMOTION:
|
|
|
|
default:
|
|
|
|
printf("Unknown input action!\n");
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static unsigned int color[16]={
|
|
|
|
0, /* 0 black */
|
|
|
|
0xe31e60, /* 1 magenta */
|
|
|
|
0x604ebd, /* 2 dark blue */
|
|
|
|
0xff44fd, /* 3 purple */
|
|
|
|
0x00a360, /* 4 dark green */
|
|
|
|
0x9c9c9c, /* 5 grey 1 */
|
|
|
|
0x14cffd, /* 6 medium blue */
|
|
|
|
0xd0c3ff, /* 7 light blue */
|
|
|
|
0x607203, /* 8 brown */
|
|
|
|
0xff6a3c, /* 9 orange */
|
|
|
|
0x9d9d9d, /* 10 grey 2 */
|
|
|
|
0xffa0d0, /* 11 pink */
|
|
|
|
0x14f53c, /* 12 bright green */
|
|
|
|
0xd0dd8d, /* 13 yellow */
|
|
|
|
0x72ffd0, /* 14 aqua */
|
|
|
|
0xffffff, /* 15 white */
|
|
|
|
};
|
|
|
|
|
2017-05-03 16:20:53 +00:00
|
|
|
|
|
|
|
/* a = ycoord */
|
|
|
|
static int gbascalc(unsigned char a) {
|
|
|
|
|
|
|
|
unsigned char s,c;
|
|
|
|
|
|
|
|
/* input ABCD EFGH */
|
|
|
|
s=a; /* store a on stack */
|
|
|
|
c=a&1;
|
|
|
|
a=a>>1; /* lsr */
|
|
|
|
a=a&0x3; /* mask */
|
|
|
|
a=a|0x4; /* 00001FG */
|
|
|
|
ram[GBASH]=a;
|
|
|
|
a=s;
|
|
|
|
|
|
|
|
a=a&0x18; /* 000D E000 */
|
|
|
|
|
|
|
|
/* if odd */
|
|
|
|
if (c) {
|
|
|
|
a=a+0x7f+1;
|
|
|
|
}
|
|
|
|
ram[GBASL]=a;
|
|
|
|
a=a<<2;
|
|
|
|
a=a|ram[GBASL];
|
|
|
|
ram[GBASL]=a;
|
|
|
|
|
|
|
|
if (debug) printf("GBAS=%02X%02X\n",ram[GBASH],ram[GBASL]);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static short y_indirect(unsigned char base, unsigned char y) {
|
|
|
|
|
|
|
|
unsigned short addr;
|
|
|
|
|
|
|
|
addr=(((short)(ram[base+1]))<<8) | (short)ram[base];
|
|
|
|
|
|
|
|
if (debug) printf("Address=%x\n",addr+y);
|
|
|
|
|
|
|
|
return addr+y;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
int scrn(unsigned char xcoord, unsigned char ycoord) {
|
|
|
|
|
|
|
|
unsigned char a,y,c;
|
|
|
|
|
|
|
|
a=ycoord;
|
|
|
|
y=xcoord;
|
|
|
|
|
|
|
|
c=a&1;
|
|
|
|
a=a>>1;
|
|
|
|
gbascalc(a);
|
|
|
|
a=ram[y_indirect(GBASL,y)];
|
|
|
|
|
|
|
|
if (c) {
|
|
|
|
return a>>4;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return a&0xf;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-05-02 14:56:59 +00:00
|
|
|
int grsim_update(void) {
|
2017-05-02 13:59:39 +00:00
|
|
|
|
|
|
|
int x,y,i,j;
|
|
|
|
unsigned int *t_pointer;
|
|
|
|
|
2017-05-02 14:56:59 +00:00
|
|
|
t_pointer=((Uint32 *)sdl_screen->pixels);
|
2017-05-02 13:59:39 +00:00
|
|
|
|
|
|
|
for(y=0;y<YSIZE;y++) {
|
|
|
|
for(j=0;j<PIXEL_Y_SCALE;j++) {
|
|
|
|
for(x=0;x<XSIZE;x++) {
|
|
|
|
for(i=0;i<PIXEL_X_SCALE;i++) {
|
2017-05-03 16:20:53 +00:00
|
|
|
*t_pointer=color[scrn(x,y)];
|
2017-05-02 13:59:39 +00:00
|
|
|
t_pointer++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-02 14:56:59 +00:00
|
|
|
SDL_UpdateRect(sdl_screen, 0, 0, xsize, ysize);
|
2017-05-02 13:59:39 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-05-02 14:56:59 +00:00
|
|
|
int grsim_init(void) {
|
2017-05-02 13:59:39 +00:00
|
|
|
|
|
|
|
int mode;
|
2017-05-03 16:20:53 +00:00
|
|
|
int x;
|
2017-05-02 13:59:39 +00:00
|
|
|
|
|
|
|
mode=SDL_SWSURFACE|SDL_HWPALETTE|SDL_HWSURFACE;
|
|
|
|
|
|
|
|
if ( SDL_Init(SDL_INIT_VIDEO) < 0 ) {
|
|
|
|
fprintf(stderr,
|
|
|
|
"Couldn't initialize SDL: %s\n", SDL_GetError());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Clean up on exit */
|
|
|
|
atexit(SDL_Quit);
|
|
|
|
|
|
|
|
/* assume 32-bit color */
|
|
|
|
sdl_screen = SDL_SetVideoMode(xsize, ysize, 32, mode);
|
|
|
|
|
|
|
|
if ( sdl_screen == NULL ) {
|
|
|
|
fprintf(stderr, "ERROR! Couldn't set %dx%d video mode: %s\n",
|
|
|
|
xsize,ysize,SDL_GetError());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Init screen */
|
2017-05-03 16:20:53 +00:00
|
|
|
for(x=0x400;x<0x800;x++) ram[x]=0;
|
2017-05-02 13:59:39 +00:00
|
|
|
|
2017-05-02 14:56:59 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2017-05-02 13:59:39 +00:00
|
|
|
|
2017-05-02 14:56:59 +00:00
|
|
|
int color_equals(int new_color) {
|
2017-05-03 16:20:53 +00:00
|
|
|
|
|
|
|
/* Top and Bottom both have color */
|
|
|
|
ram[COLOR]=((new_color%16)<<4)|(new_color%16);
|
|
|
|
|
2017-05-02 14:56:59 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2017-05-02 13:59:39 +00:00
|
|
|
|
2017-05-03 16:20:53 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int plot(unsigned char xcoord, unsigned char ycoord) {
|
|
|
|
|
|
|
|
unsigned char c,a,y;
|
|
|
|
|
|
|
|
if (ycoord>40) {
|
|
|
|
printf("Y too big %d\n",ycoord);
|
2017-05-02 19:21:32 +00:00
|
|
|
return -1;
|
|
|
|
}
|
2017-05-03 16:20:53 +00:00
|
|
|
|
|
|
|
/* Applesoft Source Code */
|
|
|
|
/* F225 GET X,Y Values */
|
|
|
|
/* Y-coord in A */
|
|
|
|
/* X-coord in Y */
|
|
|
|
/* Check that X-coord<40 */
|
|
|
|
a=ycoord;
|
|
|
|
y=xcoord;
|
|
|
|
|
|
|
|
if (y>=40) {
|
|
|
|
printf("X too big %d\n",y);
|
2017-05-02 19:21:32 +00:00
|
|
|
return -1;
|
|
|
|
}
|
2017-05-03 16:20:53 +00:00
|
|
|
/* Call into Monitor $F800 */
|
|
|
|
|
|
|
|
c=a&1; /* save LSB in carry */
|
|
|
|
a=a>>1; /* lsr A */
|
|
|
|
gbascalc(a);
|
|
|
|
|
|
|
|
if (c) {
|
|
|
|
/* If odd, mask is 0xf0 */
|
|
|
|
ram[MASK]=0xf0;
|
2017-05-02 19:21:32 +00:00
|
|
|
}
|
2017-05-03 16:20:53 +00:00
|
|
|
else {
|
|
|
|
/* If even, mask is 0x0f */
|
|
|
|
ram[MASK]=0x0f;
|
2017-05-02 19:21:32 +00:00
|
|
|
}
|
|
|
|
|
2017-05-03 16:20:53 +00:00
|
|
|
a=ram[y_indirect(GBASL,y)];
|
|
|
|
|
|
|
|
a=a^ram[COLOR];
|
|
|
|
|
|
|
|
a=a&ram[MASK];
|
|
|
|
|
|
|
|
a=a^ram[y_indirect(GBASL,y)];
|
|
|
|
|
|
|
|
ram[y_indirect(GBASL,y)]=a;
|
|
|
|
|
2017-05-02 14:56:59 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2017-05-02 13:59:39 +00:00
|
|
|
|
2017-05-02 14:56:59 +00:00
|
|
|
int hlin(int x1, int x2, int at) {
|
|
|
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for(i=x1;i<x2;i++) plot(i,at);
|
2017-05-02 13:59:39 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2017-05-02 14:56:59 +00:00
|
|
|
|
|
|
|
int vlin(int y1, int y2, int at) {
|
|
|
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for(i=y1;i<y2;i++) plot(at,i);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-05-02 19:21:32 +00:00
|
|
|
int gr(void) {
|
2017-05-03 16:20:53 +00:00
|
|
|
int x;
|
2017-05-02 14:56:59 +00:00
|
|
|
|
2017-05-02 19:21:32 +00:00
|
|
|
/* Init screen */
|
2017-05-03 16:20:53 +00:00
|
|
|
for(x=0x400;x<0x800;x++) ram[x]=0;
|
2017-05-02 19:21:32 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2017-05-03 14:50:15 +00:00
|
|
|
|
|
|
|
int bload(char *filename, int address) {
|
|
|
|
|
|
|
|
FILE *fff;
|
|
|
|
int count=0,ch=0;
|
|
|
|
|
|
|
|
fff=fopen(filename,"r");
|
|
|
|
if (fff==NULL) {
|
|
|
|
fprintf(stderr,"Could not open %s\n",filename);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
while(1) {
|
|
|
|
|
|
|
|
if ((address+count)>RAMSIZE) {
|
|
|
|
fprintf(stderr,"ERROR ram too high\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ch=fgetc(fff);
|
|
|
|
if (ch<0) break;
|
|
|
|
|
|
|
|
ram[address+count]=ch;
|
|
|
|
count++;
|
|
|
|
}
|
|
|
|
fclose(fff);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|