2021-09-16 12:15:00 +00:00
|
|
|
#!/usr/bin/python2
|
|
|
|
# -*- coding: utf8 -*-
|
2015-12-07 16:21:31 +00:00
|
|
|
|
2021-09-16 12:15:00 +00:00
|
|
|
import pygame, numpy, sys, datetime, wave, time
|
|
|
|
from orao.cpu import CPU
|
2021-09-16 13:14:24 +00:00
|
|
|
from orao.keyboard import listener as orao_kbd_listener
|
2015-12-07 16:21:31 +00:00
|
|
|
|
2021-09-16 13:38:05 +00:00
|
|
|
MEM_LOAD_PRG = None
|
|
|
|
|
|
|
|
if len(sys.argv) == 2:
|
|
|
|
MEM_LOAD_PRG = sys.argv[1]
|
|
|
|
|
|
|
|
|
|
|
|
# pygame init
|
|
|
|
ratio, running = 0, True
|
|
|
|
|
2015-12-07 16:21:31 +00:00
|
|
|
pygame.mixer.pre_init(44100, 8, 1, buffer=2048)
|
|
|
|
pygame.init()
|
|
|
|
pygame.time.set_timer(pygame.USEREVENT + 1, 40)
|
|
|
|
|
2021-09-16 13:00:54 +00:00
|
|
|
# setup surfaces
|
|
|
|
screen = pygame.display.set_mode((800, 900))
|
|
|
|
background = pygame.image.load("pozadina.png").convert_alpha()
|
|
|
|
terminal = pygame.Surface((256, 256), pygame.SRCALPHA, depth=32)
|
|
|
|
terminal.fill((255, 255, 255))
|
|
|
|
|
|
|
|
# create CPU
|
|
|
|
cpu = CPU(bytearray([0xFF]*0xC000) + bytearray(open('ORAO13.ROM', 'rb').read()))
|
|
|
|
cpu.alphaarray = pygame.surfarray.pixels_alpha(terminal)
|
|
|
|
cpu.channel = pygame.mixer.Channel(0)
|
|
|
|
|
2015-12-07 16:21:31 +00:00
|
|
|
while running:
|
|
|
|
before, previous_loop_cycles = datetime.datetime.now(), cpu.cycles
|
|
|
|
time_elapsed = lambda: (datetime.datetime.now()-before).microseconds + 1
|
|
|
|
|
|
|
|
for i in range(5000):
|
|
|
|
cpu.step()
|
|
|
|
|
|
|
|
for event in pygame.event.get():
|
|
|
|
if event.type == pygame.QUIT:
|
|
|
|
running = False
|
|
|
|
|
|
|
|
if event.type == pygame.MOUSEBUTTONDOWN:
|
|
|
|
x, y = event.pos
|
|
|
|
if 650 < x < 700 and 720 < y < 790: # Reset button
|
|
|
|
cpu.__init__(cpu.memory[:]) # Warm reset
|
|
|
|
|
2021-09-16 13:14:24 +00:00
|
|
|
orao_kbd_listener(event, cpu)
|
2015-12-07 16:21:31 +00:00
|
|
|
|
2021-09-16 13:30:17 +00:00
|
|
|
if event.type in [pygame.KEYDOWN, pygame.KEYUP]:
|
2021-09-16 13:38:05 +00:00
|
|
|
pkeys = pygame.key.get_pressed()
|
|
|
|
if pkeys[pygame.K_ESCAPE]:
|
2021-09-16 13:30:17 +00:00
|
|
|
running = False
|
|
|
|
|
2021-09-16 13:38:05 +00:00
|
|
|
if pkeys[pygame.K_F8]:
|
|
|
|
print("LOADING: %s" % MEM_LOAD_PRG)
|
|
|
|
ba = bytearray(open(MEM_LOAD_PRG, "rb").read())
|
|
|
|
|
|
|
|
# read load address
|
|
|
|
addr = ba[1] * 256 + ba[0]
|
|
|
|
ba = ba[2:]
|
|
|
|
print('Loadaddr: %04x' % addr)
|
|
|
|
|
|
|
|
# load file to memory
|
|
|
|
for i in range(0, len(ba)):
|
|
|
|
cpu.memory[addr + i] = ba[i]
|
|
|
|
|
|
|
|
# run
|
|
|
|
cpu.pc = addr
|
|
|
|
# HACK: reset stack pointer
|
|
|
|
cpu.sp = 241
|
|
|
|
|
2015-12-07 16:21:31 +00:00
|
|
|
if event.type == pygame.USEREVENT + 1:
|
2021-09-16 13:00:54 +00:00
|
|
|
screen.blit(background, [0, 0])
|
|
|
|
screen.blit(pygame.transform.smoothscale(terminal, (512, 512)), [150, 140])
|
2015-12-07 16:21:31 +00:00
|
|
|
|
|
|
|
pygame.display.set_caption('({0:.2f} MHz) Orao Emulator v0.1'.format(ratio))
|
|
|
|
pygame.display.update()
|
|
|
|
|
|
|
|
cpu.tape_out = None if cpu.cycles - cpu.last_sound_cycles > 20000 else cpu.tape_out
|
|
|
|
|
|
|
|
if len(cpu.sndbuf) > 4096 or cpu.sndbuf and cpu.cycles - cpu.last_sound_cycles > 20000:
|
|
|
|
while cpu.channel.get_queue():
|
|
|
|
if time_elapsed() > 10000: break
|
|
|
|
|
|
|
|
cpu.channel.queue(pygame.sndarray.make_sound(numpy.uint8(cpu.sndbuf)))
|
|
|
|
cpu.sndbuf = []
|
|
|
|
|
|
|
|
overshoot = cpu.cycles - previous_loop_cycles - time_elapsed()
|
|
|
|
pygame.time.wait((overshoot > 0) * overshoot // 1000) # Pričekaj da budemo cycle exact
|
|
|
|
|
|
|
|
ratio = 1.0 * (cpu.cycles - previous_loop_cycles) / time_elapsed()
|
|
|
|
|
|
|
|
pygame.quit()
|