/[VRac]/M6502/M6502.pm
This is repository of my old source code which isn't updated any more. Go to git.rot13.org for current projects!
ViewVC logotype

Annotation of /M6502/M6502.pm

Parent Directory Parent Directory | Revision Log Revision Log


Revision 51 - (hide annotations)
Tue Jul 31 12:35:02 2007 UTC (16 years, 9 months ago) by dpavlin
File size: 2700 byte(s)
re-add ram (needed for tests :-) but with slightly differenet semantic, more akin to poke_code
1 dpavlin 24 package M6502;
2    
3     use strict;
4     use warnings;
5    
6 dpavlin 27 use Data::Dump qw/dump/;
7 dpavlin 29 use Carp qw/confess/;
8 dpavlin 34 use Exporter 'import';
9 dpavlin 50 our @EXPORT = qw'@mem $PC $A $P $X $Y $S $IPeriod $run_for $debug';
10 dpavlin 27
11 dpavlin 29 =head1 NAME
12 dpavlin 24
13 dpavlin 29 M6502 - perl bindings for 6502 emulator
14    
15     =cut
16    
17 dpavlin 50 my $debug = 0;
18 dpavlin 24
19 dpavlin 29 our $VERSION = qw(0.0.1);
20 dpavlin 27
21 dpavlin 29 our @mem = (0xff) x 0x10000; # 64M
22    
23 dpavlin 25 # program counter
24 dpavlin 34 our $PC = 0xbeef;
25 dpavlin 25 # CPU registars
26 dpavlin 36 our ( $A, $P, $X, $Y, $S ) = (0) x 5;
27 dpavlin 25 # Set IPeriod to number of CPU cycles between calls to Loop6502
28 dpavlin 27 our $IPeriod = 1;
29 dpavlin 40 # Exec6502 cycles
30 dpavlin 42 our $run_for = 0;
31 dpavlin 24
32 dpavlin 26 =head1 init
33    
34     Called before C<Run6502>
35    
36     =cut
37    
38     sub init {
39 dpavlin 33 my $self = shift;
40     warn "inside init low-level M6502 from $self\n";
41 dpavlin 26 };
42 dpavlin 27
43     =head2 read
44    
45     Read from memory
46    
47     $byte = read( $address );
48    
49     =cut
50    
51     sub read {
52     my ($addr) = @_;
53     my $byte = $mem[$addr];
54 dpavlin 50 warn "## M6502::read(",dump(@_),") = ",dump( $byte ),"\n" if $debug;
55 dpavlin 27 return $byte;
56     }
57    
58     =head2 write
59    
60     Write into emory
61    
62     write( $address, $byte );
63    
64     =cut
65    
66     sub write {
67 dpavlin 50 warn "## M6502::write(",dump(@_),")\n" if $debug;
68 dpavlin 27 my ($addr,$byte) = @_;
69     $mem[$addr] = $byte;
70     }
71 dpavlin 29
72     =head2 poke_code
73    
74 dpavlin 42 Write series of bytes into memory passing through MMU (C<read> and C<write>)
75     functions. If you don't want to trigger MMU, use C<write_chunk>.
76 dpavlin 29
77     $emu->poke_code( 0xbeef, 0xff, 0x00, 0xff, 0x00, 0xaa );
78    
79     =cut
80    
81     sub poke_code {
82     my $self = shift;
83     my $addr = shift;
84 dpavlin 50 warn sprintf("## M6502::poke_code(%04x,%s)\n", $addr, dump( @_ )) if $self->debug;
85 dpavlin 42 #$mem[$addr++] = $_ foreach @_;
86 dpavlin 51 # call low-level write
87     Arch::write($addr++, $_) foreach @_;
88 dpavlin 29 }
89    
90 dpavlin 51 =head2 ram
91    
92     Read series of bytes into memory without MMU interaction
93    
94     my @code = $emu->ram( 0xc000, 0xc1000 );
95    
96     =cut
97    
98     sub ram {
99     my $self = shift;
100     my ( $from, $to ) = @_;
101     warn sprintf("## M6502::ram(%04x,%04x)\n", $from, $to) if $self->debug;
102     return @mem[ $from .. $to ];
103     }
104    
105 dpavlin 31 =head2 write_chunk
106    
107 dpavlin 42 Low-level update of memory, overriding user specified MMU functions C<read> and C<write>
108    
109 dpavlin 31 $emu->write_chunk( $address, $chunk_of_data );
110    
111     =cut
112    
113     sub write_chunk {
114     my ($self, $addr, $chunk) = @_;
115     my $len = length($chunk);
116     splice @mem, $addr, $len, unpack('C*', $chunk);
117     }
118    
119 dpavlin 42 =head2 prompt
120    
121     Call this after C<< $run_for >> cycles have been run on processor
122    
123     =cut
124    
125     sub prompt {
126     warn "prompt -- you should override this\n";
127     return 1;
128     }
129    
130 dpavlin 38 =head2 push_R
131    
132     called by C<perl.c> to push changes in registars back to perl variables
133    
134     =cut
135    
136     sub push_R {
137 dpavlin 50 warn "## M6502::push_R(",dump(@_),")\n" if $debug;
138 dpavlin 38 my ( $a, $p, $x, $y, $s, $pc ) = @_;
139     $PC = $pc;
140     $S=$s; $X=$x; $Y=$y; $P=$p; $A=$a;
141     dump_R();
142     }
143    
144     =head2 dump_R
145    
146     helper function which dumps registers
147    
148     =cut
149    
150     sub dump_R {
151 dpavlin 50 warn sprintf("## M6502::dump_R PC: %04x A:%02x P:%02x X:%02x Y:%02x S:%02x\n", $PC, $A, $P, $X, $Y, $S) if $debug;
152 dpavlin 38 }
153    
154 dpavlin 24 1;

  ViewVC Help
Powered by ViewVC 1.1.26