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

Contents of /M6502/perl.c

Parent Directory Parent Directory | Revision Log Revision Log


Revision 36 - (show annotations)
Mon Jul 30 22:06:13 2007 UTC (16 years, 9 months ago) by dpavlin
File MIME type: text/plain
File size: 4292 byte(s)
more tweaks
1 #include <EXTERN.h>
2 #include <perl.h>
3 #include "M6502.h"
4 #include "config.h"
5
6 #if DEBUGF
7 #define debugf(x) do { \
8 PerlIO_stdoutf x ; \
9 } while (0)
10 #else
11 #define debugf(x)
12 #endif
13
14 static PerlInterpreter *my_perl;
15
16 static M6502 *R;
17
18 #if DUMP_R
19 #define dump_R printf("# PC: %04x A:%02x P:%02x X:%02x Y:%02x S:%02x\n", R->PC.W, R->A, R->P, R->X, R->Y, R->S );
20 #else
21 #define dump_R
22 #endif
23
24 void update_R(M6502 *R) {
25 R->A = SvIV( get_sv("M6502::A", FALSE) );
26 printf("--%d", get_sv("M6502::A", FALSE) );
27 R->P = SvIV( get_sv("M6502::P", FALSE) );
28 R->X = atoi( SvPV_nolen( get_sv("M6502::X", FALSE) ) );
29 R->Y = atoi( SvPV_nolen( get_sv("M6502::Y", FALSE) ) );
30 R->S = atoi( SvPV_nolen( get_sv("M6502::S", FALSE) ) );
31 R->PC.W = atoi( SvPV_nolen( get_sv("M6502::PC", FALSE) ) );
32 R->IPeriod = atoi( SvPV_nolen( get_sv("M6502::IPeriod", FALSE) ) );
33 // ICount IRequest IAutoReset TrapBadOps Trap Trace
34 printf("update_r finished\n");
35 dump_R;
36 }
37
38 /** Rd6502()/Wr6502/Op6502() *********************************/
39 /** These functions are called when access to RAM occurs. **/
40 /** They allow to control memory access. Op6502 is the same **/
41 /** as Rd6502, but used to read *opcodes* only, when many **/
42 /** checks can be skipped to make it fast. It is only **/
43 /** required if there is a #define FAST_RDOP. **/
44 /************************************ TO BE WRITTEN BY USER **/
45
46 byte mem(word Addr) {
47 byte byte;
48 int count;
49 dSP;
50 ENTER;
51 SAVETMPS;
52 PUSHMARK(SP);
53 XPUSHs( sv_2mortal( newSViv( Addr ) ) );
54 PUTBACK;
55 count = call_pv("Arch::read", G_ARRAY );
56 if ( count != 1 ) {
57 printf("expect 1 return value, got %d", count);
58 exit(1);
59 }
60 //debugf(("got %d values\n", count));
61 SPAGAIN;
62 SV *sv;
63 sv = POPs;
64 byte = SvIV(sv);
65 FREETMPS;
66 LEAVE;
67 //debugf(("mem(%04x) = %02x\n", Addr, byte));
68 return byte;
69 }
70
71 byte Rd6502(register word Addr) {
72 byte Value;
73 Value = mem(Addr);
74 debugf(("Rd6502(%04x) = %02x\n", Addr, Value));
75 return Value;
76 }
77
78 void Wr6502(register word Addr,register byte Value) {
79 debugf(("Wr6502(%04x,%02x)\n", Addr, Value));
80 dSP;
81 ENTER;
82 SAVETMPS;
83 PUSHMARK(SP);
84 XPUSHs( sv_2mortal( newSViv( Addr ) ) );
85 XPUSHs( sv_2mortal( newSViv( Value ) ) );
86 PUTBACK;
87 call_pv("Arch::write", G_DISCARD );
88 FREETMPS;
89 LEAVE;
90 }
91
92 byte Op6502(register word Addr) {
93 byte Op;
94 Op = mem(Addr);
95 debugf(("Op6502(%04x,%02x) PC:%04x\n", Addr, Op, R->PC.W));
96 }
97
98 /** Loop6502() ***********************************************/
99 /** 6502 emulation calls this function periodically to **/
100 /** check if the system hardware requires any interrupts. **/
101 /** This function must return one of following values: **/
102 /** INT_NONE, INT_IRQ, INT_NMI, or INT_QUIT to exit the **/
103 /** emulation loop. **/
104 /************************************ TO BE WRITTEN BY USER **/
105 byte Loop6502(register M6502 *R) {
106 debugf(("Loop6502\n"));
107 dump_R;
108 return INT_NONE;
109 }
110
111 /** Patch6502() **********************************************/
112 /** Emulation calls this function when it encounters an **/
113 /** unknown opcode. This can be used to patch the code to **/
114 /** emulate BIOS calls, such as disk and tape access. The **/
115 /** function should return 1 if the exception was handled, **/
116 /** or 0 if the opcode was truly illegal. **/
117 /************************************ TO BE WRITTEN BY USER **/
118 byte Patch6502(register byte Op,register M6502 *R) {
119 debugf(("Patch6502(%02x)\n", Op));
120 dump_R;
121 return 0;
122 }
123
124 /**
125 * main code
126 *
127 **/
128
129 int main(int argc, char **argv) {
130 char *command_line[] = {"", "-e", EMU_START };
131 my_perl = perl_alloc();
132 perl_construct(my_perl);
133 if (perl_parse(my_perl, xs_init, 3, command_line, (char **)NULL)) {
134 printf("Failed to parse initial: %s\n", EMU_START );
135 return 0;
136 }
137 perl_run(my_perl);
138 if (SvTRUE(ERRSV)) {
139 printf("Failed to execute\n");
140 return 0;
141 } else {
142 R = malloc(sizeof(M6502));
143 if (!R) {
144 printf("can't alloc %d bytes for M6502", sizeof(M6502));
145 exit(1);
146 }
147
148 printf("reset CPU\n");
149 Reset6502(R);
150
151 printf("call Arch::init\n");
152 dSP;
153 PUSHMARK(SP);
154 call_pv("Arch::init", G_DISCARD | G_NOARGS );
155
156 update_R(R);
157 printf("run CPU\n");
158 Run6502(R);
159 printf("end of CPU run\n");
160 }
161 free(R);
162 perl_destruct(my_perl);
163 perl_free(my_perl);
164 return 0;
165 }

  ViewVC Help
Powered by ViewVC 1.1.26