repo2/firmware/main.c @ 57
46 | markw | #include "integer.h"
|
|
#include "regs.h"
|
|||
#include "pause.h"
|
|||
#include "printf.h"
|
|||
49 | markw | #include "joystick.h"
|
|
46 | markw | ||
#include "simpledir.h"
|
|||
#include "simplefile.h"
|
|||
49 | markw | #include "fileselector.h"
|
|
46 | markw | ||
#include "atari_drive_emulator.h"
|
|||
// FUNCTIONS in here
|
|||
// i) pff init - NOT USED EVERYWHERE
|
|||
// ii) file selector - kind of crap, no fine scrolling - NOT USED EVERYWHERE
|
|||
// iii) cold reset atari (clears base ram...)
|
|||
// iv) start atari (begins paused)
|
|||
// v) freeze/resume atari - NOT USED EVERYWHERE!
|
|||
// vi) menu for various options - NOT USED EVERYWHERE!
|
|||
// vii) pause - TODO - base this on pokey clock...
|
|||
// standard ZPU IN/OUT use...
|
|||
// OUT1 - 6502 settings (pause,reset,speed)
|
|||
// pause_n: bit 0
|
|||
// reset_n: bit 1
|
|||
// turbo: bit 2-4: meaning... 0=1.79Mhz,1=3.58MHz,2=7.16MHz,3=14.32MHz,4=28.64MHz,5=57.28MHz,etc.
|
|||
// ram_select: bit 5-7:
|
|||
// RAM_SELECT : in std_logic_vector(2 downto 0); -- 64K,128K,320KB Compy, 320KB Rambo, 576K Compy, 576K Rambo, 1088K, 4MB
|
|||
// rom_select: bit 8-13:
|
|||
// ROM_SELECT : in std_logic_vector(5 downto 0); -- 16KB ROM Bank
|
|||
#define BIT_REG(op,mask,shift,name,reg) \
|
|||
int get_ ## name() \
|
|||
{ \
|
|||
int val = *reg; \
|
|||
return op((val>>shift)&mask); \
|
|||
} \
|
|||
void set_ ## name(int param) \
|
|||
{ \
|
|||
int val = *reg; \
|
|||
\
|
|||
val = (val&~(mask<<shift)); \
|
|||
val |= op(param)<<shift; \
|
|||
\
|
|||
*reg = val; \
|
|||
}
|
|||
47 | markw | #define BIT_REG_RO(op,mask,shift,name,reg) \
|
|
int get_ ## name() \
|
|||
{ \
|
|||
int val = *reg; \
|
|||
return op((val>>shift)&mask); \
|
|||
}
|
|||
46 | markw | BIT_REG(,0x1,0,pause_6502,zpu_out1)
|
|
BIT_REG(,0x1,1,reset_6502,zpu_out1)
|
|||
BIT_REG(,0x3f,2,turbo_6502,zpu_out1)
|
|||
BIT_REG(,0x7,8,ram_select,zpu_out1)
|
|||
BIT_REG(,0x3f,11,rom_select,zpu_out1)
|
|||
49 | markw | BIT_REG_RO(,0x1,4,hotkey_softboot,zpu_in1)
|
|
BIT_REG_RO(,0x1,5,hotkey_coldboot,zpu_in1)
|
|||
BIT_REG_RO(,0x1,6,hotkey_fileselect,zpu_in1)
|
|||
BIT_REG_RO(,0x1,7,hotkey_settings,zpu_in1)
|
|||
47 | markw | ||
46 | markw | void
|
|
wait_us(int unsigned num)
|
|||
{
|
|||
// pause counter runs at pokey frequency - should be 1.79MHz
|
|||
int unsigned cycles = (num*230)>>7;
|
|||
*zpu_pause = cycles;
|
|||
}
|
|||
void memset8(void * address, int value, int length)
|
|||
{
|
|||
char * mem = address;
|
|||
while (length--)
|
|||
*mem++=value;
|
|||
}
|
|||
void memset32(void * address, int value, int length)
|
|||
{
|
|||
int * mem = address;
|
|||
while (length--)
|
|||
*mem++=value;
|
|||
}
|
|||
void clear_64k_ram()
|
|||
{
|
|||
memset8((void *)0x200000, 0, 65536); // SRAM, if present (TODO)
|
|||
memset32((void *)0x800000, 0, 16384);
|
|||
}
|
|||
void
|
|||
47 | markw | reboot(int cold)
|
|
46 | markw | {
|
|
set_reset_6502(1);
|
|||
47 | markw | if (cold)
|
|
46 | markw | {
|
|
clear_64k_ram();
|
|||
}
|
|||
set_reset_6502(0);
|
|||
set_pause_6502(0);
|
|||
}
|
|||
unsigned char toatarichar(int val)
|
|||
{
|
|||
int inv = val>=128;
|
|||
if (inv)
|
|||
{
|
|||
val-=128;
|
|||
}
|
|||
if (val>='A' && val<='Z')
|
|||
{
|
|||
val+=-'A'+33;
|
|||
}
|
|||
else if (val>='a' && val<='z')
|
|||
{
|
|||
val+=-'a'+33+64;
|
|||
}
|
|||
else if (val>='0' && val<='9')
|
|||
{
|
|||
val+=-'0'+16;
|
|||
}
|
|||
else if (val>=32 && val<=47)
|
|||
{
|
|||
val+=-32;
|
|||
}
|
|||
else
|
|||
{
|
|||
val = 0;
|
|||
}
|
|||
if (inv)
|
|||
{
|
|||
val+=128;
|
|||
}
|
|||
return val;
|
|||
}
|
|||
int debug_pos;
|
|||
49 | markw | int debug_adjust;
|
|
46 | markw | unsigned char volatile * baseaddr;
|
|
49 | markw | void clearscreen()
|
|
{
|
|||
unsigned volatile char * screen;
|
|||
for (screen=(unsigned volatile char *)40000+atari_regbase; screen!=(unsigned volatile char *)(atari_regbase+40000+1024); ++screen)
|
|||
*screen = 0x00;
|
|||
}
|
|||
46 | markw | void char_out ( void* p, char c)
|
|
{
|
|||
unsigned char val = toatarichar(c);
|
|||
49 | markw | if (debug_pos>=0)
|
|
47 | markw | {
|
|
49 | markw | *(baseaddr+debug_pos) = val|debug_adjust;
|
|
47 | markw | ++debug_pos;
|
|
}
|
|||
46 | markw | }
|
|
// Memory usage...
|
|||
// 0x410000-0x41FFFF (0xc10000 in zpu space) = directory cache - 64k
|
|||
47 | markw | // 0x420000-0x43FFFF (0xc20000 in zpu space) = freeze backup
|
|
46 | markw | ||
55 | markw | struct SimpleFile * files[5];
|
|
49 | markw | ||
55 | markw | void loadrom(char const * path, int size, void * ram_address)
|
|
{
|
|||
ram_address += 0x800000;
|
|||
if (SimpleFile_OK == file_open_name(path, files[4]))
|
|||
{
|
|||
int read = 0;
|
|||
file_read(files[4], ram_address, size, &read);
|
|||
}
|
|||
}
|
|||
46 | markw | int main(void)
|
|
{
|
|||
49 | markw | int i;
|
|
55 | markw | for (i=0; i!=5; ++i)
|
|
49 | markw | {
|
|
files[i] = (struct SimpleFile *)alloca(file_struct_size());
|
|||
52 | markw | file_init(files[i]);
|
|
49 | markw | }
|
|
47 | markw | freeze_init((void*)0xc20000); // 128k
|
|
debug_pos = -1;
|
|||
49 | markw | debug_adjust = 0;
|
|
46 | markw | baseaddr = (unsigned char volatile *)(40000 + atari_regbase);
|
|
set_reset_6502(1);
|
|||
set_turbo_6502(1);
|
|||
55 | markw | set_rom_select(1);
|
|
46 | markw | ||
init_printf(0, char_out);
|
|||
// TODO...
|
|||
if (SimpleFile_OK == dir_init((void *)0xc10000, 65536))
|
|||
{
|
|||
55 | markw | // printf("DIR init ok\n");
|
|
47 | markw | init_drive_emulator();
|
|
55 | markw | ||
loadrom("xlorig.rom",0x4000, (void *)0x704000);
|
|||
loadrom("xlhias.rom",0x4000, (void *)0x708000);
|
|||
loadrom("ultimon.rom",0x4000, (void *)0x70c000);
|
|||
loadrom("osbhias.rom",0x4000, (void *)0x710000);
|
|||
loadrom("osborig.rom",0x2800, (void *)0x715800);
|
|||
loadrom("osaorig.rom",0x2800, (void *)0x719800);
|
|||
loadrom("ataribas.rom",0x2000,(void *)0x700000);
|
|||
//ROM = xlorig.rom,0x4000, (void *)0x704000
|
|||
//ROM = xlhias.rom,0x4000, (void *)0x708000
|
|||
//ROM = ultimon.rom,0x4000, (void *)0x70c000
|
|||
//ROM = osbhias.rom,0x4000, (void *)0x710000
|
|||
//ROM = osborig.rom,0x2800, (void *)0x715800
|
|||
//ROM = osaorig.rom,0x2800, (void *)0x719800
|
|||
//
|
|||
//ROM = ataribas.rom,0x2000,(void *)0x700000
|
|||
//--SDRAM_BASIC_ROM_ADDR <= "111"&"000000" &"00000000000000";
|
|||
//--SDRAM_OS_ROM_ADDR <= "111"&rom_select &"00000000000000";
|
|||
47 | markw | reboot(1);
|
|
run_drive_emulator();
|
|||
46 | markw | }
|
|
else
|
|||
{
|
|||
55 | markw | //printf("DIR init failed\n");
|
|
46 | markw | }
|
|
47 | markw | reboot(1);
|
|
for (;;) actions();
|
|||
46 | markw | }
|
|
47 | markw | // struct SimpleFile * file = alloca(file_struct_size());
|
|
// printf("Opening file\n");
|
|||
// if (SimpleFile_OK == file_open_name("GUNPOWDR.ATR",file))
|
|||
// {
|
|||
// printf("FILE open ok\n");
|
|||
//
|
|||
// set_drive_status(0,file);
|
|||
//
|
|||
// }
|
|||
// else
|
|||
// {
|
|||
// printf("FILE open failed\n");
|
|||
// }
|
|||
49 | markw | char const * get_ram()
|
|
{
|
|||
switch(get_ram_select())
|
|||
{
|
|||
case 0:
|
|||
return "64K";
|
|||
case 1:
|
|||
55 | markw | return "128K";
|
|
49 | markw | case 2:
|
|
return "320K(Compy)";
|
|||
case 3:
|
|||
return "320K(Rambo)";
|
|||
case 4:
|
|||
return "576K(Compy)";
|
|||
case 5:
|
|||
return "576K(Rambo)";
|
|||
case 6:
|
|||
return "1MB";
|
|||
case 7:
|
|||
return "4MB";
|
|||
}
|
|||
}
|
|||
55 | markw | int settings()
|
|
49 | markw | {
|
|
struct joystick_status joy;
|
|||
joy.x_ = joy.y_ = joy.fire_ = 0;
|
|||
int row = 0;
|
|||
52 | markw | int done = 0;
|
|
for (;!done;)
|
|||
49 | markw | {
|
|
// Render
|
|||
clearscreen();
|
|||
55 | markw | debug_pos = 0;
|
|
49 | markw | debug_adjust = 0;
|
|
55 | markw | printf("Se");
|
|
debug_adjust = 128;
|
|||
printf("ttings");
|
|||
49 | markw | debug_pos = 80;
|
|
debug_adjust = row==0 ? 128 : 0;
|
|||
printf("Turbo:%dx", get_turbo_6502());
|
|||
debug_pos = 120;
|
|||
debug_adjust = row==1 ? 128 : 0;
|
|||
printf("Ram:%s", get_ram());
|
|||
debug_pos = 160;
|
|||
debug_adjust = row==2 ? 128 : 0;
|
|||
printf("Rom bank:%d", get_rom_select());
|
|||
52 | markw | debug_pos = 240;
|
|
int i;
|
|||
for (i=1;i!=5;++i)
|
|||
{
|
|||
int temp = debug_pos;
|
|||
debug_adjust = row==i+2 ? 128 : 0;
|
|||
printf("Drive %d:%s", i, file_name(files[i-1]));
|
|||
debug_pos = temp+40;
|
|||
}
|
|||
49 | markw | ||
55 | markw | debug_pos = 400;
|
|
52 | markw | debug_adjust = row==7 ? 128 : 0;
|
|
55 | markw | printf("Cartridge 8k simple");
|
|
debug_pos = 480;
|
|||
debug_adjust = row==8 ? 128 : 0;
|
|||
52 | markw | printf("Exit");
|
|
49 | markw | // Slow it down a bit
|
|
wait_us(100000);
|
|||
// move
|
|||
joystick_wait(&joy,WAIT_QUIET);
|
|||
joystick_wait(&joy,WAIT_EITHER);
|
|||
row+=joy.y_;
|
|||
if (row<0) row = 0;
|
|||
55 | markw | if (row>8) row = 8;
|
|
49 | markw | switch (row)
|
|
{
|
|||
case 0:
|
|||
{
|
|||
int turbo = get_turbo_6502();
|
|||
if (joy.x_==1) turbo<<=1;
|
|||
if (joy.x_==-1) turbo>>=1;
|
|||
if (turbo>16) turbo = 16;
|
|||
if (turbo<1) turbo = 1;
|
|||
set_turbo_6502(turbo);
|
|||
}
|
|||
break;
|
|||
case 1:
|
|||
{
|
|||
int ram_select = get_ram_select();
|
|||
ram_select+=joy.x_;
|
|||
if (ram_select<0) ram_select = 0;
|
|||
if (ram_select>7) ram_select = 7;
|
|||
set_ram_select(ram_select);
|
|||
}
|
|||
break;
|
|||
case 2:
|
|||
{
|
|||
int rom_select = get_rom_select();
|
|||
rom_select+=joy.x_;
|
|||
55 | markw | if (rom_select<1) rom_select = 1;
|
|
if (rom_select>6) rom_select = 6; // TODO
|
|||
49 | markw | set_rom_select(rom_select);
|
|
}
|
|||
break;
|
|||
52 | markw | case 3:
|
|
case 4:
|
|||
case 5:
|
|||
case 6:
|
|||
{
|
|||
if (joy.x_>0)
|
|||
{
|
|||
// Choose new disk
|
|||
55 | markw | filter = filter_disks;
|
|
52 | markw | file_selector(files[row-3]);
|
|
set_drive_status(row-3,files[row-3]);
|
|||
}
|
|||
else if(joy.x_<0)
|
|||
{
|
|||
// Remove disk
|
|||
file_init(files[row-3]);
|
|||
set_drive_status(row-3,files[row-3]);
|
|||
}
|
|||
else if (joy.fire_)
|
|||
{
|
|||
struct SimpleFile * temp;
|
|||
temp = files[0];
|
|||
files[0] = files[row-3];
|
|||
files[row-3] = temp;
|
|||
set_drive_status(row-3,temp);
|
|||
set_drive_status(0,files[0]);
|
|||
}
|
|||
}
|
|||
break;
|
|||
case 7:
|
|||
55 | markw | {
|
|
if (joy.fire_)
|
|||
{
|
|||
filter = filter_roms;
|
|||
file_selector(files[4]);
|
|||
loadrom(file_name(files[4]),0x2000,(void *)0x700000);
|
|||
return 1;
|
|||
}
|
|||
}
|
|||
break;
|
|||
case 8:
|
|||
52 | markw | if (joy.fire_)
|
|
{
|
|||
done = 1;
|
|||
}
|
|||
break;
|
|||
49 | markw | }
|
|
}
|
|||
55 | markw | ||
return 0;
|
|||
49 | markw | }
|
|
46 | markw | void actions()
|
|
{
|
|||
47 | markw | // Show some activity!
|
|
49 | markw | //*atari_colbk = *atari_random;
|
|
47 | markw | ||
// Hot keys
|
|||
if (get_hotkey_softboot())
|
|||
{
|
|||
reboot(0);
|
|||
}
|
|||
else if (get_hotkey_coldboot())
|
|||
{
|
|||
reboot(1);
|
|||
}
|
|||
else if (get_hotkey_settings())
|
|||
{
|
|||
set_pause_6502(1);
|
|||
freeze();
|
|||
debug_pos = 0;
|
|||
55 | markw | int do_reboot = settings();
|
|
47 | markw | debug_pos = -1;
|
|
restore();
|
|||
55 | markw | if (do_reboot)
|
|
reboot(1);
|
|||
else
|
|||
set_pause_6502(0);
|
|||
47 | markw | }
|
|
else if (get_hotkey_fileselect())
|
|||
{
|
|||
set_pause_6502(1);
|
|||
freeze();
|
|||
55 | markw | filter = filter_disks;
|
|
49 | markw | file_selector(files[0]);
|
|
47 | markw | debug_pos = -1;
|
|
restore();
|
|||
49 | markw | set_drive_status(0,files[0]);
|
|
reboot(1);
|
|||
47 | markw | }
|
|
46 | markw | }
|
|