OpenCores
URL https://opencores.org/ocsvn/w11/w11/trunk

Subversion Repositories w11

[/] [w11/] [tags/] [w11a_V0.74/] [tools/] [bin/] [tmuconv] - Blame information for rev 38

Details | Compare with Previous | View Log

Line No. Rev Author Line
1 2 wfjm
#!/usr/bin/perl -w
2 34 wfjm
# $Id: tmuconv 712 2015-11-01 22:53:45Z mueller $
3 2 wfjm
#
4 30 wfjm
# Copyright 2008-2015 by Walter F.J. Mueller 
5 2 wfjm
#
6
# This program is free software; you may redistribute and/or modify it under
7
# the terms of the GNU General Public License as published by the Free
8
# Software Foundation, either version 2, or at your option any later version.
9
#
10
# This program is distributed in the hope that it will be useful, but
11
# WITHOUT ANY WARRANTY, without even the implied warranty of MERCHANTABILITY
12
# or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
13
# for complete details.
14
#
15
#  Revision History:
16
# Date         Rev Version  Comment
17 34 wfjm
# 2015-11-01   712   1.1.1  BUGFIX: fix '.' handling for br/sob instructions
18
#                           BUGFIX: correct xor (now r,dst, and not src,r)
19
#                           br/sob offsets now octal; assume --t_id if no opts
20
# 2015-07-03   697   1.1    adapt to new DM_STAT_(SY|VM); add rhrp vector
21 8 wfjm
# 2010-10-22   334   1.0.9  adapt to ibus V2 signals: req,we,dip->aval,re,we,rmw
22 2 wfjm
# 2010-06-26   309   1.0.8  add ibimres.cacc/racc handling
23
# 2010-04-26   284   1.0.7  add error check for GetOptions
24
# 2009-09-19   240   1.0.6  add more VFETCH addr defs; add 2nd DL11 defs
25
# 2009-06-04   223   1.0.5  add IIST and PC11 defs
26
# 2009-05-03   212   1.0.4  add defs for mmu par/pdr's and some unibus dev's
27
# 2008-12-14   177   1.0.3  add -t_ru; use dp_ireg_we_last; add ibus names
28
# 2008-11-30   174   1.0.2  SPUSH and VFETCH tags for em cycles; psw in id lines
29
# 2008-04-25   138   1.0.1  show ccc/scc for code 000257/000277 in disassembler
30
# 2008-04-19   137   1.0    Initial version
31
#
32
# Current fields in tmu_ofile:
33
#   clkcycle:d
34
#   cpu:o
35
#   dp.pc:o
36
#   dp.psw:o
37
#   dp.ireg:o
38
#   dp.ireg_we:b
39
#   dp.ireg_we_last:b
40
#   dp.dsrc:o
41
#   dp.ddst:o
42
#   dp.dtmp:o
43
#   dp.dres:o
44
#   dp.gpr_adst:o
45
#   dp.gpr_mode:o
46
#   dp.gpr_bytop:b
47
#   dp.gpr_we:b
48 8 wfjm
#   vm.ibmreq.aval:b
49
#   vm.ibmreq.re:b
50 2 wfjm
#   vm.ibmreq.we:b
51 8 wfjm
#   vm.ibmreq.rmw:b
52 2 wfjm
#   vm.ibmreq.be0:b
53
#   vm.ibmreq.be1:b
54
#   vm.ibmreq.cacc:b
55
#   vm.ibmreq.racc:b
56
#   vm.ibmreq.addr:o
57
#   vm.ibmreq.din:o
58
#   vm.ibsres.ack:b
59
#   vm.ibsres.busy:b
60
#   vm.ibsres.dout:o
61 34 wfjm
#   vm.emmreq.req:b
62
#   vm.emmreq.we:b
63
#   vm.emmreq.be:b
64
#   vm.emmreq.cancel:b
65
#   vm.emmreq.addr:o
66
#   vm.emmreq.din:o
67
#   vm.emsres.ack_r:b
68
#   vm.emsres.ack_w:b
69
#   vm.emsres.dout:o
70 2 wfjm
#   co.cpugo:b
71 30 wfjm
#   co.cpususp:b
72
#   co.suspint:b
73
#   co.suspext:b
74 2 wfjm
#   sy.chit:b
75
#
76
 
77
use 5.005;                                  # require Perl 5.005 or higher
78
use strict;                                 # require strict checking
79
use FileHandle;
80
 
81
use Getopt::Long;
82
 
83
my %opts = ();
84
 
85
GetOptions(\%opts, "help", "dump", "cdump",
86
           "t_id", "t_ru", "t_em", "t_ib")
87
  or die "bad options";
88
 
89
sub print_help;
90
sub do_file;
91
sub code2mnemo;
92
sub regmod;
93
 
94
my @var_name;
95
my @var_type;
96
my @var_dec;
97
my @var_oct;
98
my %name;
99
 
100
my @val_curr_text;
101
my @val_curr;
102
my @val_last;
103
 
104
my @reg_05 = ("------","------","------","------","------","------",   # set 0
105
              "------","------","------","------","------","------",); # set 1
106
my @reg_sp = ("------","------","------","------");        # ksp,ssp,???,usp
107
 
108
my $ind_dp_pc;
109
my $ind_dp_psw;
110
my $ind_dp_ireg;
111
my $ind_dp_ireg_we;
112
my $ind_dp_ireg_we_last;
113
my $ind_dp_dres;
114
my $ind_dp_gpr_adst;
115
my $ind_dp_gpr_mode;
116
my $ind_dp_gpr_bytop;
117
my $ind_dp_gpr_we;
118
 
119 8 wfjm
my $ind_vm_ibmreq_aval;
120
my $ind_vm_ibmreq_re;
121 2 wfjm
my $ind_vm_ibmreq_we;
122 8 wfjm
my $ind_vm_ibmreq_rmw;
123 2 wfjm
my $ind_vm_ibmreq_be0;
124
my $ind_vm_ibmreq_be1;
125
my $ind_vm_ibmreq_cacc;
126
my $ind_vm_ibmreq_racc;
127
my $ind_vm_ibmreq_addr;
128
my $ind_vm_ibmreq_din;
129
my $ind_vm_ibsres_ack;
130
my $ind_vm_ibsres_busy;
131
my $ind_vm_ibsres_dout;
132
 
133 34 wfjm
my $ind_vm_emmreq_req;
134
my $ind_vm_emmreq_we;
135
my $ind_vm_emmreq_be;
136
my $ind_vm_emmreq_cancel;
137
my $ind_vm_emmreq_addr;
138
my $ind_vm_emmreq_din;
139
my $ind_vm_emsres_ack_r;
140
my $ind_vm_emsres_ack_w;
141
my $ind_vm_emsres_dout;
142
 
143 2 wfjm
my $ind_sy_chit;
144
 
145
my @pdp11_opcode_tbl = (
146
    {code=>0000000, mask=>0000000, name=>"halt", type=>"0arg"},
147
    {code=>0000001, mask=>0000000, name=>"wait", type=>"0arg"},
148
    {code=>0000002, mask=>0000000, name=>"rti ", type=>"0arg"},
149
    {code=>0000003, mask=>0000000, name=>"bpt ", type=>"0arg"},
150
    {code=>0000004, mask=>0000000, name=>"iot ", type=>"0arg"},
151
    {code=>0000005, mask=>0000000, name=>"reset",type=>"0arg"},
152
    {code=>0000006, mask=>0000000, name=>"rtt ", type=>"0arg"},
153
    {code=>0000007, mask=>0000000, name=>"!!mfpt", type=>"0arg"},
154
    {code=>0000100, mask=>0000077, name=>"jmp ", type=>"1arg"},
155
    {code=>0000200, mask=>0000007, name=>"rts ", type=>"1reg"},
156
    {code=>0000230, mask=>0000007, name=>"spl ", type=>"spl"},
157
    {code=>0000240, mask=>0000017, name=>"cl",   type=>"ccop"},
158
    {code=>0000260, mask=>0000017, name=>"se",   type=>"ccop"},
159 34 wfjm
    {code=>0000300, mask=>0000077, name=>"swab", type=>"1arg"},
160 2 wfjm
    {code=>0000400, mask=>0000377, name=>"br  ", type=>"br"},
161
    {code=>0001000, mask=>0000377, name=>"bne ", type=>"br"},
162
    {code=>0001400, mask=>0000377, name=>"beq ", type=>"br"},
163
    {code=>0002000, mask=>0000377, name=>"bge ", type=>"br"},
164
    {code=>0002400, mask=>0000377, name=>"blt ", type=>"br"},
165
    {code=>0003000, mask=>0000377, name=>"bgt ", type=>"br"},
166
    {code=>0003400, mask=>0000377, name=>"ble ", type=>"br"},
167 34 wfjm
    {code=>0004000, mask=>0000777, name=>"jsr ", type=>"rsrc"},
168 2 wfjm
    {code=>0005000, mask=>0000077, name=>"clr ", type=>"1arg"},
169
    {code=>0005100, mask=>0000077, name=>"com ", type=>"1arg"},
170
    {code=>0005200, mask=>0000077, name=>"inc ", type=>"1arg"},
171
    {code=>0005300, mask=>0000077, name=>"dec ", type=>"1arg"},
172
    {code=>0005400, mask=>0000077, name=>"neg ", type=>"1arg"},
173
    {code=>0005500, mask=>0000077, name=>"adc ", type=>"1arg"},
174
    {code=>0005600, mask=>0000077, name=>"sbc ", type=>"1arg"},
175
    {code=>0005700, mask=>0000077, name=>"tst ", type=>"1arg"},
176
    {code=>0006000, mask=>0000077, name=>"ror ", type=>"1arg"},
177
    {code=>0006100, mask=>0000077, name=>"rol ", type=>"1arg"},
178
    {code=>0006200, mask=>0000077, name=>"asr ", type=>"1arg"},
179
    {code=>0006300, mask=>0000077, name=>"asl ", type=>"1arg"},
180
    {code=>0006400, mask=>0000077, name=>"mark", type=>"mark"},
181
    {code=>0006500, mask=>0000077, name=>"mfpi", type=>"1arg"},
182
    {code=>0006600, mask=>0000077, name=>"mtpi", type=>"1arg"},
183
    {code=>0006700, mask=>0000077, name=>"sxt ", type=>"1arg"},
184
    {code=>0007000, mask=>0000077, name=>"!!csm",  type=>"1arg"},
185
    {code=>0007200, mask=>0000077, name=>"!!tstset",type=>"1arg"},
186
    {code=>0007300, mask=>0000077, name=>"!!wrtlck",type=>"1arg"},
187
    {code=>0010000, mask=>0007777, name=>"mov ", type=>"2arg"},
188
    {code=>0020000, mask=>0007777, name=>"cmp ", type=>"2arg"},
189
    {code=>0030000, mask=>0007777, name=>"bit ", type=>"2arg"},
190
    {code=>0040000, mask=>0007777, name=>"bic ", type=>"2arg"},
191
    {code=>0050000, mask=>0007777, name=>"bis ", type=>"2arg"},
192
    {code=>0060000, mask=>0007777, name=>"add ", type=>"2arg"},
193
    {code=>0070000, mask=>0000777, name=>"mul ", type=>"rdst"},
194
    {code=>0071000, mask=>0000777, name=>"div ", type=>"rdst"},
195
    {code=>0072000, mask=>0000777, name=>"ash ", type=>"rdst"},
196
    {code=>0073000, mask=>0000777, name=>"ashc", type=>"rdst"},
197 34 wfjm
    {code=>0074000, mask=>0000777, name=>"xor ", type=>"rsrc"},
198 2 wfjm
    {code=>0077000, mask=>0000777, name=>"sob ", type=>"sob"},
199
    {code=>0100000, mask=>0000377, name=>"bpl ", type=>"br"},
200
    {code=>0100400, mask=>0000377, name=>"bmi ", type=>"br"},
201
    {code=>0101000, mask=>0000377, name=>"bhi ", type=>"br"},
202
    {code=>0101400, mask=>0000377, name=>"blos", type=>"br"},
203
    {code=>0102000, mask=>0000377, name=>"bvc ", type=>"br"},
204
    {code=>0102400, mask=>0000377, name=>"bvs ", type=>"br"},
205
    {code=>0103000, mask=>0000377, name=>"bcc ", type=>"br"},
206
    {code=>0103400, mask=>0000377, name=>"bcs ", type=>"br"},
207
    {code=>0104000, mask=>0000377, name=>"emt ", type=>"trap"},
208
    {code=>0104400, mask=>0000377, name=>"trap", type=>"trap"},
209
    {code=>0105000, mask=>0000077, name=>"clrb", type=>"1arg"},
210
    {code=>0105100, mask=>0000077, name=>"comb", type=>"1arg"},
211
    {code=>0105200, mask=>0000077, name=>"incb", type=>"1arg"},
212
    {code=>0105300, mask=>0000077, name=>"decb", type=>"1arg"},
213
    {code=>0105400, mask=>0000077, name=>"negb", type=>"1arg"},
214
    {code=>0105500, mask=>0000077, name=>"adcb", type=>"1arg"},
215
    {code=>0105600, mask=>0000077, name=>"sbcb", type=>"1arg"},
216
    {code=>0105700, mask=>0000077, name=>"tstb", type=>"1arg"},
217
    {code=>0106000, mask=>0000077, name=>"rorb", type=>"1arg"},
218
    {code=>0106100, mask=>0000077, name=>"rolb", type=>"1arg"},
219
    {code=>0106200, mask=>0000077, name=>"asrb", type=>"1arg"},
220
    {code=>0106300, mask=>0000077, name=>"aslb", type=>"1arg"},
221
    {code=>0106400, mask=>0000077, name=>"!!mtps", type=>"1arg"},
222
    {code=>0106500, mask=>0000077, name=>"mfpd", type=>"1arg"},
223
    {code=>0106600, mask=>0000077, name=>"mtpd", type=>"1arg"},
224
    {code=>0106700, mask=>0000077, name=>"!!mfps", type=>"1arg"},
225
    {code=>0110000, mask=>0007777, name=>"movb", type=>"2arg"},
226
    {code=>0120000, mask=>0007777, name=>"cmpb", type=>"2arg"},
227
    {code=>0130000, mask=>0007777, name=>"bitb", type=>"2arg"},
228
    {code=>0140000, mask=>0007777, name=>"bicb", type=>"2arg"},
229
    {code=>0150000, mask=>0007777, name=>"bisb", type=>"2arg"},
230
    {code=>0160000, mask=>0007777, name=>"sub ", type=>"2arg"},
231
    {code=>0170000, mask=>0000000, name=>"!!cfcc", type=>"0arg"},
232
    {code=>0170001, mask=>0000000, name=>"!!setf", type=>"0arg"},
233
    {code=>0170011, mask=>0000000, name=>"!!setd", type=>"0arg"},
234
    {code=>0170002, mask=>0000000, name=>"!!seti", type=>"0arg"},
235
    {code=>0170012, mask=>0000000, name=>"!!setl", type=>"0arg"},
236
    {code=>0170100, mask=>0000077, name=>"!!ldfps",type=>"1fpp"},
237
    {code=>0170200, mask=>0000077, name=>"!!stfps",type=>"1fpp"},
238
    {code=>0170300, mask=>0000077, name=>"!!stst", type=>"1fpp"},
239
    {code=>0170400, mask=>0000077, name=>"!!clrf", type=>"1fpp"},
240
    {code=>0170500, mask=>0000077, name=>"!!tstf", type=>"1fpp"},
241
    {code=>0170600, mask=>0000077, name=>"!!absf", type=>"1fpp"},
242
    {code=>0170700, mask=>0000077, name=>"!!negf", type=>"1fpp"},
243
    {code=>0171000, mask=>0000377, name=>"!!mulf", type=>"rfpp"},
244
    {code=>0171400, mask=>0000377, name=>"!!modf", type=>"rfpp"},
245
    {code=>0172000, mask=>0000377, name=>"!!addf", type=>"rfpp"},
246
    {code=>0172400, mask=>0000377, name=>"!!ldf",  type=>"rfpp"},
247
    {code=>0173000, mask=>0000377, name=>"!!subf", type=>"rfpp"},
248
    {code=>0173400, mask=>0000377, name=>"!!cmpf", type=>"rfpp"},
249
    {code=>0174000, mask=>0000377, name=>"!!stf",  type=>"rfpp"},
250
    {code=>0174400, mask=>0000377, name=>"!!divf", type=>"rfpp"},
251
    {code=>0175000, mask=>0000377, name=>"!!stexp",type=>"rfpp"},
252
    {code=>0175400, mask=>0000377, name=>"!!stcif",type=>"rfpp"},
253
    {code=>0176000, mask=>0000377, name=>"!!stcfd",type=>"rfpp"},
254
    {code=>0176400, mask=>0000377, name=>"!!ldexp",type=>"rfpp"},
255
    {code=>0177000, mask=>0000377, name=>"!!ldcif",type=>"rfpp"},
256
    {code=>0177400, mask=>0000377, name=>"!!ldcdf",type=>"rfpp"}
257
  );
258
 
259
my %pdp11_regs = (                          # use simh naming convention
260
     177776=> "psw",
261
     177774=> "stklim",
262
     177772=> "pirq",
263
     177770=> "mbrk",
264
     177766=> "cpuerr",
265
     177764=> "sysid",
266
     177600=> "uipdr0",
267
     177602=> "uipdr1",
268
     177604=> "uipdr2",
269
     177606=> "uipdr3",
270
     177610=> "uipdr4",
271
     177612=> "uipdr5",
272
     177614=> "uipdr6",
273
     177616=> "uipdr7",
274
     177620=> "udpdr0",
275
     177622=> "udpdr1",
276
     177624=> "udpdr2",
277
     177626=> "udpdr3",
278
     177630=> "udpdr4",
279
     177632=> "udpdr5",
280
     177634=> "udpdr6",
281
     177636=> "udpdr7",
282
     177640=> "uipar0",
283
     177642=> "uipar1",
284
     177644=> "uipar2",
285
     177646=> "uipar3",
286
     177650=> "uipar4",
287
     177652=> "uipar5",
288
     177654=> "uipar6",
289
     177656=> "uipar7",
290
     177660=> "udpar0",
291
     177662=> "udpar1",
292
     177664=> "udpar2",
293
     177666=> "udpar3",
294
     177670=> "udpar4",
295
     177672=> "udpar5",
296
     177674=> "udpar6",
297
     177676=> "udpar7",
298
     177576=> "mmr2",
299
     177574=> "mmr1",
300
     177572=> "mmr0",
301
     177570=> "sdreg",                      # not a simh name !!
302 34 wfjm
     177560=> "tia.csr",
303
     177562=> "tia.buf",
304
     177564=> "toa.csr",
305
     177566=> "toa.buf",
306 2 wfjm
     177550=> "pr.csr",
307
     177552=> "pr.buf",
308
     177554=> "pp.csr",
309
     177556=> "pp.buf",
310
     177546=> "kl.csr",
311
     177514=> "lp.csr",
312
     177516=> "lp.buf",
313
     177500=> "ii.acr",
314
     177502=> "ii.adr",
315
     177400=> "rk.ds ",
316
     177402=> "rk.er ",
317
     177404=> "rk.cs ",
318
     177406=> "rk.wc ",
319
     177410=> "rk.ba ",
320
     177412=> "rk.da ",
321
     177414=> "rk.mr ",
322
     177416=> "rk.db ",
323
     177060=> "xor.cs",                     # XOR Tester
324 34 wfjm
     176700=> "rp.cs1",
325
     176702=> "rp.wc ",
326
     176704=> "rp.ba ",
327
     176706=> "rp.da ",
328
     176710=> "rp.cs2",
329
     176712=> "rp.ds ",
330
     176714=> "rp.er1",
331
     176716=> "rp.as ",
332
     176720=> "rp.la ",
333
     176722=> "rp.db ",
334
     176724=> "rp.mr1",
335
     176726=> "rp.dt ",
336
     176730=> "rp.sn ",
337
     176732=> "rp.of ",
338
     176734=> "rp.dc ",
339
     176736=> "rp.m13",
340
     176740=> "rp.m14",
341
     176742=> "rp.m15",
342
     176744=> "rp.ec1",
343
     176746=> "rp.ec2",
344
     176750=> "rp.bae",
345
     176752=> "rp.cs3",
346
     176500=> "tib.cs",
347
     176502=> "tib.bu",
348
     176504=> "tob.cs",
349
     176506=> "tob.bu",
350 2 wfjm
     174400=> "rl.cs ",
351
     174402=> "rl.ba ",
352
     174404=> "rl.da ",
353
     174406=> "rl.mp ",
354
     172540=> "kp.csr",
355
     172542=> "kp.buf",
356
     172544=> "kp.cnt",
357 34 wfjm
     172520=> "tm.sr",
358
     172522=> "tm.cr",
359
     172524=> "tm.bc",
360
     172526=> "tm.ba",
361
     172530=> "tm.db",
362
     172532=> "tm.rl",
363 2 wfjm
     172516=> "mmr3",
364
     172200=> "sipdr0",
365
     172202=> "sipdr1",
366
     172204=> "sipdr2",
367
     172206=> "sipdr3",
368
     172210=> "sipdr4",
369
     172212=> "sipdr5",
370
     172214=> "sipdr6",
371
     172216=> "sipdr7",
372
     172220=> "sdpdr0",
373
     172222=> "sdpdr1",
374
     172224=> "sdpdr2",
375
     172226=> "sdpdr3",
376
     172230=> "sdpdr4",
377
     172232=> "sdpdr5",
378
     172234=> "sdpdr6",
379
     172236=> "sdpdr7",
380
     172240=> "sipar0",
381
     172242=> "sipar1",
382
     172244=> "sipar2",
383
     172246=> "sipar3",
384
     172250=> "sipar4",
385
     172252=> "sipar5",
386
     172254=> "sipar6",
387
     172256=> "sipar7",
388
     172260=> "sdpar0",
389
     172262=> "sdpar1",
390
     172264=> "sdpar2",
391
     172266=> "sdpar3",
392
     172270=> "sdpar4",
393
     172272=> "sdpar5",
394
     172274=> "sdpar6",
395
     172276=> "sdpar7",
396
     172300=> "kipdr0",
397
     172302=> "kipdr1",
398
     172304=> "kipdr2",
399
     172306=> "kipdr3",
400
     172310=> "kipdr4",
401
     172312=> "kipdr5",
402
     172314=> "kipdr6",
403
     172316=> "kipdr7",
404
     172320=> "kdpdr0",
405
     172322=> "kdpdr1",
406
     172324=> "kdpdr2",
407
     172326=> "kdpdr3",
408
     172330=> "kdpdr4",
409
     172332=> "kdpdr5",
410
     172334=> "kdpdr6",
411
     172336=> "kdpdr7",
412
     172340=> "kipar0",
413
     172342=> "kipar1",
414
     172344=> "kipar2",
415
     172346=> "kipar3",
416
     172350=> "kipar4",
417
     172352=> "kipar5",
418
     172354=> "kipar6",
419
     172356=> "kipar7",
420
     172360=> "kdpar0",
421
     172362=> "kdpar1",
422
     172364=> "kdpar2",
423
     172366=> "kdpar3",
424
     172370=> "kdpar4",
425
     172372=> "kdpar5",
426
     172374=> "kdpar6",
427
     172376=> "kdpar7",
428
     160100=> "dz.csr",
429
     160102=> "dz.mp2",
430
     160104=> "dz.tcr",
431
     160106=> "dz.mp6"
432
);
433
 
434
autoflush STDOUT 1 if (-p STDOUT);          # autoflush if output into pipe
435
 
436
if (exists $opts{help}) {
437
  print_help;
438
  exit 0;
439
}
440
 
441 34 wfjm
my $nopts = 0;                              # count options
442
$nopts += 1 if $opts{dump};
443
$nopts += 1 if $opts{cdump};
444
$nopts += 1 if $opts{t_id};
445
$nopts += 1 if $opts{t_ru};
446
$nopts += 1 if $opts{t_em};
447
$nopts += 1 if $opts{t_ib};
448
 
449
$opts{t_id} = 1 if $nopts == 0;             # if no opts, assume t_id
450
 
451 2 wfjm
foreach my $file (@ARGV) {
452
  do_file($file);
453
}
454
 
455
 
456
#-------------------------------------------------------------------------------
457
 
458
sub do_file {
459
  my ($file) = @_;
460
 
461
  open IFILE,"<$file" or die "failed to open $file";
462
 
463
  my $idec_cyc = 0;
464
  my $change_cyc = 0;
465
  my $emreq_cyc = 0;
466
  my $emreq_str = "";
467
  my $ibreq_cyc = 0;
468
  my $ibreq_typ = "";
469
  my $ibreq_str = "";
470
  my $ibreq_nam = "";
471
 
472
  my $emcurr_we   = 0;                      # curr em write enable (or undef)
473
  my $emcurr_addr = undef;                  # curr em address
474
  my $emlast_we   = 0;                      # prev em write enable (or undef)
475
  my $emlast_addr = undef;                  # prev em address
476
 
477
  while () {
478
    chomp;
479
    if (/^#\s+/) {
480
      @var_name = ();
481
      @var_type = ();
482
      my $dsc_str  = $';
483
      my @dsc_list = split /\s+/,$dsc_str;
484
      foreach my $dsc (@dsc_list) {
485
        if ($dsc =~ /^(.*):([bdo])$/) {
486
          my $ind = scalar(@var_name);
487
          $name{$1} = {ind=>$ind,
488
                       typ=>$2};
489
          push @var_name, $1;
490
          push @var_type, $2;
491
          push @var_dec, $ind if $2 eq "d";
492
          push @var_oct, $ind if $2 eq "o";
493
        } else {
494
          print "tmuconv-E: bad descriptor $dsc\n";
495
        }
496
      }
497
 
498
      $ind_dp_pc            = $name{'dp.pc'}->{ind};
499
      $ind_dp_psw           = $name{'dp.psw'}->{ind};
500
      $ind_dp_ireg          = $name{'dp.ireg'}->{ind};
501
      $ind_dp_ireg_we       = $name{'dp.ireg_we'}->{ind};
502
      $ind_dp_ireg_we_last  = $name{'dp.ireg_we_last'}->{ind};
503
      $ind_dp_dres          = $name{'dp.dres'}->{ind};
504
      $ind_dp_gpr_adst      = $name{'dp.gpr_adst'}->{ind};
505
      $ind_dp_gpr_mode      = $name{'dp.gpr_mode'}->{ind};
506
      $ind_dp_gpr_bytop     = $name{'dp.gpr_bytop'}->{ind};
507
      $ind_dp_gpr_we        = $name{'dp.gpr_we'}->{ind};
508
 
509 8 wfjm
      $ind_vm_ibmreq_aval   = $name{'vm.ibmreq.aval'}->{ind};
510
      $ind_vm_ibmreq_re     = $name{'vm.ibmreq.re'}->{ind};
511 2 wfjm
      $ind_vm_ibmreq_we     = $name{'vm.ibmreq.we'}->{ind};
512 8 wfjm
      $ind_vm_ibmreq_rmw    = $name{'vm.ibmreq.rmw'}->{ind};
513 2 wfjm
      $ind_vm_ibmreq_be0    = $name{'vm.ibmreq.be0'}->{ind};
514
      $ind_vm_ibmreq_be1    = $name{'vm.ibmreq.be1'}->{ind};
515
      $ind_vm_ibmreq_cacc   = $name{'vm.ibmreq.cacc'}->{ind};
516
      $ind_vm_ibmreq_racc   = $name{'vm.ibmreq.racc'}->{ind};
517
      $ind_vm_ibmreq_addr   = $name{'vm.ibmreq.addr'}->{ind};
518
      $ind_vm_ibmreq_din    = $name{'vm.ibmreq.din'}->{ind};
519
      $ind_vm_ibsres_ack    = $name{'vm.ibsres.ack'}->{ind};
520
      $ind_vm_ibsres_busy   = $name{'vm.ibsres.busy'}->{ind};
521
      $ind_vm_ibsres_dout   = $name{'vm.ibsres.dout'}->{ind};
522
 
523 34 wfjm
      $ind_vm_emmreq_req    = $name{'vm.emmreq.req'}->{ind};
524
      $ind_vm_emmreq_we     = $name{'vm.emmreq.we'}->{ind};
525
      $ind_vm_emmreq_be     = $name{'vm.emmreq.be'}->{ind};
526
      $ind_vm_emmreq_cancel = $name{'vm.emmreq.cancel'}->{ind};
527
      $ind_vm_emmreq_addr   = $name{'vm.emmreq.addr'}->{ind};
528
      $ind_vm_emmreq_din    = $name{'vm.emmreq.din'}->{ind};
529
      $ind_vm_emsres_ack_r  = $name{'vm.emsres.ack_r'}->{ind};
530
      $ind_vm_emsres_ack_w  = $name{'vm.emsres.ack_w'}->{ind};
531
      $ind_vm_emsres_dout   = $name{'vm.emsres.dout'}->{ind};
532
 
533 2 wfjm
      $ind_sy_chit          = $name{'sy.chit'}->{ind};
534
 
535
    } else {
536
      @val_last = @val_curr;
537
      my $notfirst = scalar(@val_last) > 0;
538
 
539
      $_ =~ s/^\s*//;
540
      $_ =~ s/\s*$//;
541
      @val_curr = split /\s+/,$_;
542
      if (scalar(@val_curr) != scalar(@var_name)) {
543
        printf "tmuconv-E: value list length mismatch, seen %d, expected %d\n",
544
          scalar(@val_curr), scalar(@var_name);
545
        for (my $i=0; $i
546
          printf "%3d: %s\n", $i,$val_curr[$i];
547
        }
548
        next;
549
      }
550
 
551
      @val_curr_text = @val_curr  if exists $opts{dump} || exists $opts{cdump};
552
 
553
      my $cyc_curr = int $val_curr[0];
554
      my $cyc_str  = sprintf "%8d", $cyc_curr;
555
 
556
      foreach my $ind (@var_dec) {
557
        $val_curr[$ind] = int ($val_curr[$ind]);
558
      }
559
      foreach my $ind (@var_oct) {
560
        $val_curr[$ind] = oct ($val_curr[$ind]);
561
      }
562
 
563
      my $id_str = "";
564
      my $ru_str = "";
565
      my $emres_str = "";
566
      my $emtyp_str = "";
567
      my $ibres_str = "";
568
      my $ibreq_we  = 0;
569
      my $ibreq_act = 0;
570
 
571
      if (exists $opts{dump} || exists $opts{cdump}) {
572
        my @val_change;
573
        my $any_change;
574
 
575
        for (my $i=1; $i
576
          my $change = (not $notfirst) || ($val_curr[$i] != $val_last[$i]);
577
          $val_change[$i] = $change;
578
          $any_change |= $change;
579
        }
580
 
581
        if (exists $opts{dump} || $any_change) {
582
          printf "cycle $cyc_str %s", "-" x 32;
583
          if ($notfirst && exists $opts{cdump}) {
584
            printf " (%d)",$cyc_curr-$change_cyc;
585
          }
586
          print "\n";
587
 
588
          for (my $i=1; $i
589
            my $oper = $val_change[$i] ? "<=" : " =";
590
            if (exists $opts{dump} || $val_change[$i]) {
591
              printf "   %-16s:%s %s %s\n", $var_name[$i], $var_type[$i],
592
                                            $oper, $val_curr_text[$i];
593
            }
594
          }
595
          $change_cyc = $cyc_curr;
596
        }
597
      }
598
#
599
# handle t_id
600
#   uses cycles with dp_ireg_we = '1'
601
#
602
      if (exists $opts{t_id} and $notfirst) {
603
        if ($val_curr[$ind_dp_ireg_we_last]) {
604
          my $pc   = $val_curr[$ind_dp_pc] - 2;
605
          my $psw  = $val_curr[$ind_dp_psw];
606
          my $ireg = $val_curr[$ind_dp_ireg];
607
          my $code = code2mnemo($ireg);
608
          $id_str = sprintf "       %6.6o %6.6o %6.6o  %s",
609
                            $pc, $psw, $ireg, $code;
610 34 wfjm
          $id_str .= " " x (22-length($code));
611 2 wfjm
          $id_str .= sprintf " (%d)",$cyc_curr-$idec_cyc;
612
          $idec_cyc = $cyc_curr;
613
        }
614
      }
615
#
616
 
617
#    1706 ru  0 06   000002 000002 000002 000002 000002 000002 000002  ksp
618
#    1694 id         002012 000340 010036  mov  r0,@(sp)+       (8)
619
 
620
 
621
# handle t_ru
622
#   uses cycles with dp_gpr_we = '1'
623
#
624
      if (exists $opts{t_ru}) {
625
        if ($val_curr[$ind_dp_gpr_we]) {
626
          my $adst  = $val_curr[$ind_dp_gpr_adst];
627
          my $mode  = $val_curr[$ind_dp_gpr_mode];
628
          my $bytop = $val_curr[$ind_dp_gpr_bytop];
629
          my $psw   = $val_curr[$ind_dp_psw];
630
          my $dres  = $val_curr[$ind_dp_dres];
631
          my $rset  = $psw>>11 & 01;
632
          $ru_str  = sprintf "%o %o%o   %6.6o", $bytop, $rset, $adst, $dres;
633
          $ru_str .= " ";
634
          if ($adst eq "7") {
635
            $ru_str .= "pc";
636
          } elsif ($adst eq "6") {
637
            $reg_sp[$mode] = sprintf "%6.6o",$dres;
638
            $ru_str .= $reg_sp[0];
639
            $ru_str .= ($mode == 0) ? "*" : " ";
640
            $ru_str .= $reg_sp[1];
641
            $ru_str .= ($mode == 1) ? "*" : " ";
642
            $ru_str .= $reg_sp[3];
643
            $ru_str .= ($mode == 3) ? "*" : " ";
644
            $ru_str .= " ksp" if $mode eq "0";
645
            $ru_str .= " ssp" if $mode eq "1";
646
            $ru_str .= " usp" if $mode eq "3";
647
          } else {
648
            my $rbase = ($rset==0) ? 0 : 6;
649
            $reg_05[$rbase+$adst] = sprintf "%6.6o",$dres;
650
            for (my $i=0; $i<6; $i++) {
651
              $ru_str .= $reg_05[$rbase+$i];
652
              $ru_str .= ($adst==$i) ? "*" : " ";
653
            }
654
            $ru_str .= sprintf " r%o%o", $rset, $adst;
655
          }
656
        }
657
      }
658
#
659
# handle t_em
660 34 wfjm
#   uses cycles with vm_emmreq_req = '1'
661
#                    vm_emsres_ack_r = '1'
662
#                    vm_emsres_ack_w = '1'
663
#                    vm_emsreq_cancel = '1'
664 2 wfjm
#
665
      if (exists $opts{t_em}) {
666 34 wfjm
        if ($val_curr[$ind_vm_emmreq_req]) {
667 2 wfjm
          $emreq_cyc = $cyc_curr;
668
          $emreq_str = sprintf "%s %s %8.8o",
669 34 wfjm
                         ($val_curr[$ind_vm_emmreq_we] ? "w" : "r"),
670
                         $val_curr[$ind_vm_emmreq_be],
671
                         $val_curr[$ind_vm_emmreq_addr];
672
          $emcurr_we   = $val_curr[$ind_vm_emmreq_we];
673
          $emcurr_addr = $val_curr[$ind_vm_emmreq_addr];
674 2 wfjm
          if ($emcurr_we) {
675 34 wfjm
            $emreq_str .= sprintf " %6.6o", $val_curr[$ind_vm_emmreq_din];
676 2 wfjm
          } else {
677
            $emreq_str .= " " x 7;
678
          }
679
        }
680 34 wfjm
        if ($val_curr[$ind_vm_emsres_ack_r] ||
681
            $val_curr[$ind_vm_emsres_ack_w] ||
682
            $val_curr[$ind_vm_emmreq_cancel]) {
683 2 wfjm
          $emres_str = sprintf "%s%s%s%s",
684 34 wfjm
            $val_curr[$ind_vm_emmreq_cancel],
685
            $val_curr[$ind_vm_emsres_ack_r],
686
            $val_curr[$ind_vm_emsres_ack_w],
687 2 wfjm
            $val_curr[$ind_sy_chit];
688 34 wfjm
          if ($val_curr[$ind_vm_emmreq_cancel]) {
689 2 wfjm
            $emreq_str .= " cancel";
690
            $emcurr_we = undef;
691
          } else {
692 34 wfjm
            if ($val_curr[$ind_vm_emsres_ack_r]) {
693
              $emreq_str .= sprintf " %6.6o", $val_curr[$ind_vm_emsres_dout];
694 2 wfjm
            } else {
695
              $emreq_str .= " " x 7;
696
            }
697
            if (defined $emlast_we && $emcurr_we == $emlast_we) {
698
              if ($emcurr_we && $emcurr_addr == $emlast_addr-2) {
699
                $emtyp_str = "SPUSH";
700
              } elsif ((not $emcurr_we) && $emcurr_addr == $emlast_addr+2 &&
701
                      $emcurr_addr < 0400 && ($emcurr_addr % 04) == 02) {
702
                $emtyp_str = "VFETCH";
703
                $emtyp_str .= " 004 ill.inst"   if ($emlast_addr == 0004);
704
                $emtyp_str .= " 010 res.inst"   if ($emlast_addr == 0010);
705
                $emtyp_str .= " 014 BPT"        if ($emlast_addr == 0014);
706
                $emtyp_str .= " 020 IOT"        if ($emlast_addr == 0020);
707
                $emtyp_str .= " 030 EMT"        if ($emlast_addr == 0030);
708
                $emtyp_str .= " 034 TRAP"       if ($emlast_addr == 0034);
709
                $emtyp_str .= " 060 DL11-TTI"   if ($emlast_addr == 0060);
710
                $emtyp_str .= " 064 DL11-TTO"   if ($emlast_addr == 0064);
711
                $emtyp_str .= " 070 PC11-PTR"   if ($emlast_addr == 0070);
712
                $emtyp_str .= " 074 PC11-PTP"   if ($emlast_addr == 0074);
713
                $emtyp_str .= " 100 KW11-L"     if ($emlast_addr == 0100);
714
                $emtyp_str .= " 104 KW11-P"     if ($emlast_addr == 0104);
715
                $emtyp_str .= " 160 RL11"       if ($emlast_addr == 0160);
716
                $emtyp_str .= " 200 LP11"       if ($emlast_addr == 0200);
717
                $emtyp_str .= " 220 RK11"       if ($emlast_addr == 0220);
718
                $emtyp_str .= " 224 TM11"       if ($emlast_addr == 0224);
719
                $emtyp_str .= " 240 PIRQ"       if ($emlast_addr == 0240);
720
                $emtyp_str .= " 244 FPP exp"    if ($emlast_addr == 0244);
721
                $emtyp_str .= " 250 MMU trap"   if ($emlast_addr == 0250);
722 34 wfjm
                $emtyp_str .= " 254 RHRP"       if ($emlast_addr == 0254);
723 2 wfjm
                $emtyp_str .= " 260 IIST"       if ($emlast_addr == 0260);
724
                $emtyp_str .= " 300 DL11-2-TTI" if ($emlast_addr == 0300);
725
                $emtyp_str .= " 304 DL11-2-TTO" if ($emlast_addr == 0304);
726
              }
727
            }
728
          }
729
          $emlast_we   = $emcurr_we;
730
          $emlast_addr = $emcurr_addr;
731
        }
732
      }
733
#
734
# handle t_ib
735 8 wfjm
#   uses cycles with sy_ibmreq_re = '1' or sy_ibmreq_we = '1'
736 2 wfjm
#                    sy_ibsres_ack = '1'
737
#                    vm_ibsres_busy '1' -> '0' transition
738
#
739
      if (exists $opts{t_ib}) {
740 8 wfjm
        if ($val_curr[$ind_vm_ibmreq_re] || $val_curr[$ind_vm_ibmreq_we]) {
741 2 wfjm
          my $addr_str = sprintf "%6.6o", $val_curr[$ind_vm_ibmreq_addr];
742
          $ibreq_cyc = $cyc_curr;
743
          $ibreq_typ = sprintf "%s%s",
744
                         ($val_curr[$ind_vm_ibmreq_cacc] ? "c" : "-"),
745
                         ($val_curr[$ind_vm_ibmreq_racc] ? "r" : "-");
746 8 wfjm
          $ibreq_str = sprintf "%s%s%s%s   %s",
747
                         ($val_curr[$ind_vm_ibmreq_we]  ? "w" : "r"),
748
                         ($val_curr[$ind_vm_ibmreq_rmw] ? "m" : " "),
749 2 wfjm
                         $val_curr[$ind_vm_ibmreq_be1],
750
                         $val_curr[$ind_vm_ibmreq_be0],
751
                         $addr_str;
752
          $ibreq_we  = $val_curr[$ind_vm_ibmreq_we];
753
          $ibreq_act = 1;
754
          if ($ibreq_we) {
755
            $ibreq_str .= sprintf " %6.6o", $val_curr[$ind_vm_ibmreq_din];
756
          } else {
757
            $ibreq_str .= " " x 7;
758
          }
759
          $ibreq_nam = $pdp11_regs{$addr_str};
760
          $ibreq_nam = "" if not defined $ibreq_nam;
761
        }
762
 
763
        if ($val_curr[$ind_vm_ibsres_ack]) {
764
          $ibreq_act = 0;
765
          $ibres_str .= sprintf "   %s", $val_curr[$ind_vm_ibsres_ack];
766
          if (not $ibreq_we) {
767
            $ibreq_str .= sprintf " %6.6o", $val_curr[$ind_vm_ibsres_dout];
768
          } else {
769
            $ibreq_str .= " " x 7;
770
          }
771
        }
772
 
773
        if ($ibreq_act && $val_curr[$ind_vm_ibsres_busy]==0) {
774
          $ibres_str .= "no ACK, no BUSY";
775
          $ibreq_act = 0;
776
        }
777
      }
778
 
779
      print "$cyc_str id    $id_str\n"     if $id_str;
780
      print "$cyc_str ru    $ru_str\n"     if $ru_str;
781
      if ($emres_str) {
782
        printf "$cyc_str em    $emreq_str  $emres_str (%d) $emtyp_str\n",
783
          $cyc_curr-$emreq_cyc;
784
      }
785
      if ($ibres_str) {
786
        printf "$cyc_str ib %s $ibreq_str  $ibres_str (%d) $ibreq_nam\n",
787
          $ibreq_typ, $cyc_curr-$ibreq_cyc;
788
      }
789
    }
790
  }
791
 
792
  close IFILE;
793
}
794
 
795
#-------------------------------------------------------------------------------
796
 
797
sub code2mnemo {
798
  my ($code) = @_;
799
 
800
  foreach my $ele (@pdp11_opcode_tbl) {
801
    if (($code & (~($ele->{mask})) ) == $ele->{code}) {
802
      my $name = $ele->{name};
803
      my $type = $ele->{type};
804
      my $str  = $name;
805
      if ($type eq "0arg") {
806
        return $name;
807
 
808
      } elsif ($type eq "1arg" or $type eq "1fpp") {
809
        my $dst = $code & 077;
810
        my $dst_str = regmod($dst);
811
        return "$name $dst_str";
812
 
813
      } elsif ($type eq "2arg") {
814
        my $src = ($code>>6) & 077;
815
        my $dst = $code & 077;
816
        my $src_str = regmod($src);
817
        my $dst_str = regmod($dst);
818
        return "$name $src_str,$dst_str";
819
 
820
      } elsif ($type eq "rdst") {
821
        my $reg = ($code>>6) & 07;
822
        my $src = $code & 077;
823
        my $src_str = regmod($src);
824
        return "$name $src_str,r$reg";
825
 
826
      } elsif ($type eq "1reg") {
827
        my $reg = $code & 07;
828
        my $reg_str = "r$reg";
829
        $reg_str = "sp" if $reg == 6;
830
        $reg_str = "pc" if $reg == 7;
831
        return "$name $reg_str";
832
 
833
      } elsif ($type eq "br")   {
834 34 wfjm
        # Note: in MACRO-11 syntax . refers to the address of the instruction
835
        #       the opcode has offset relative to PC after instruction fetch
836
        #       so   000776  --> br .-2
837
        #            000777  --> br .
838
        #            000400  --> br .+2
839
        #
840 2 wfjm
        my $off  = $code & 0177;
841 34 wfjm
        my $sign = "?";
842
        if ($code & 0200) {                 # negative offsets
843 2 wfjm
          $sign = "-";
844 34 wfjm
          $off  = ((~$off) & 0177)+1;
845
          $off  = $off - 1;                   # refer to address of instruction
846
        } else {                            # positive offsets
847
          $sign = "+";
848
          $off  = $off + 1;                   # refer to address of instruction
849
        }
850
        return sprintf "$name .%s%o", $sign, abs(2*$off);
851 30 wfjm
 
852 2 wfjm
      } elsif ($type eq "sob")  {
853 34 wfjm
        # Note: like in br type instructions, asm syntax and opcode differ by one
854 2 wfjm
        my $reg = ($code>>6) & 07;
855
        my $off = $code & 077;
856 34 wfjm
        return sprintf "$name r%d,.-%o", $reg, 2*($off-1);
857 30 wfjm
 
858 2 wfjm
      } elsif ($type eq "trap") {
859
        my $off = $code & 0377;
860
        return sprintf "$name %3.3o", $off;
861 30 wfjm
 
862 2 wfjm
      } elsif ($type eq "spl")  {
863
        my $off = $code & 07;
864
        return sprintf "$name %d", $off;
865 30 wfjm
 
866 2 wfjm
      } elsif ($type eq "ccop")  {
867
        my $cc = $code & 017;
868
        return "nop" if ($cc == 0);
869
        return "ccc" if ($code == 0257);
870
        return "scc" if ($code == 0277);
871
        my $str = "";
872
        my $del = "";
873
        if ($code & 010) { $str .= $del . $name . "n", $del = "+" }
874
        if ($code & 004) { $str .= $del . $name . "z", $del = "+" }
875
        if ($code & 002) { $str .= $del . $name . "v", $del = "+" }
876
        if ($code & 001) { $str .= $del . $name . "c", $del = "+" }
877
        return $str;
878 30 wfjm
 
879 34 wfjm
      } elsif ($type eq "rsrc")  {
880 2 wfjm
        my $reg = ($code>>6) & 07;
881
        my $dst = $code & 077;
882
        my $dst_str = regmod($dst);
883
        return "$name r$reg,$dst_str";
884 30 wfjm
 
885 2 wfjm
      } elsif ($type eq "mark") {
886
        my $off = $code & 077;
887
        return sprintf "$name %3.3o", $off;
888 30 wfjm
 
889 2 wfjm
      } elsif ($type eq "rfpp") {
890
        my $reg = ($code>>6) & 03;
891
        my $dst = $code & 077;
892
        my $dst_str = regmod($dst,"f");
893
        return "$name f$reg,$dst_str";
894
 
895
      } else {
896
        return "?type?";
897
      }
898
    }
899
  }
900
  return "=inval=";
901
}
902
 
903
#-------------------------------------------------------------------------------
904
sub regmod {
905
  my ($regmod,$pref) = @_;
906
  my $mod = ($regmod>>3) & 07;
907
  my $reg = $regmod & 07;
908
 
909
  $pref = "r" if not defined $pref or $reg>5;
910
 
911
  my $reg_str = "r$reg";
912
  $reg_str = "sp" if $reg == 6;
913
  $reg_str = "pc" if $reg == 7;
914
 
915
  if ($mod == 0) {                      # mode 0:    Rx  { Fx for float }
916
    $reg_str = "f$reg" if defined $pref && $pref eq "f" && $reg<=5;
917
    return $reg_str;
918
  } elsif ($mod == 1) {                 # mode 1:    (Rx)
919
    return "($reg_str)";
920
  } elsif ($mod == 2 || $mod == 3) {    # mode 2/3:  (Rx)+   @(Rx)+
921
    my $ind = ($mod == 3) ? "@" : "";
922
    if ($reg != 7) {                      # if reg != pc
923
      return "$ind($reg_str)+";
924
    } else {                              # if reg == pc
925
      my $str = sprintf "$ind#nnn";     # 27 -> #nnn;  37 -> @#nnn
926
      return $str;
927
    }
928
  } elsif ($mod == 4 || $mod == 5) {    # mode 4/5:  -(Rx)   @-(Rx)
929
    my $ind = ($mod == 5) ? "@" : "";
930
    return "$ind-($reg_str)";
931
  } elsif ($mod == 6 || $mod == 7) {    # mode 6/7:  nn(Rx)  @nn(Rx)
932
    my $ind = ($mod == 7) ? "@" : "";
933
    return "${ind}nnn($reg_str)";
934
  }
935
}
936
 
937
#-------------------------------------------------------------------------------
938
 
939
sub print_help {
940
  print "usage: tmuconf  file\n";
941
  print "  --help           this message\n";
942
  print "  --dump           dump all information\n";
943
  print "  --cdump          dump only changes relative to prev cycle\n";
944
  print "  --t_id           trace instruction decodes\n";
945
  print "  --t_ru           trace register updates\n";
946
  print "  --t_em           trace em transactions\n";
947
  print "  --t_ib           trace ib transactions\n";
948
}

powered by: WebSVN 2.1.0

© copyright 1999-2024 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.