Add znver1 processor
[deliverable/binutils-gdb.git] / sim / erc32 / func.c
CommitLineData
c906108c
SS
1/*
2 * func.c, misc simulator functions. This file is part of SIS.
3 *
4 * SIS, SPARC instruction simulator V1.8 Copyright (C) 1995 Jiri Gaisler,
5 * European Space Agency
6 *
7 * This program is free software; you can redistribute it and/or modify it under
8 * the terms of the GNU General Public License as published by the Free
3fd725ef 9 * Software Foundation; either version 3 of the License, or (at your option)
c906108c
SS
10 * any later version.
11 *
12 * This program is distributed in the hope that it will be useful, but WITHOUT
13 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
14 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
15 * more details.
16 *
17 * You should have received a copy of the GNU General Public License along with
51b318de 18 * this program; if not, see <http://www.gnu.org/licenses/>.
c906108c
SS
19 *
20 */
21
5272643f 22#include "config.h"
c906108c
SS
23#include <signal.h>
24#include <string.h>
25#include <stdio.h>
26#include <stdlib.h>
27#include <ctype.h>
28#include "sis.h"
c906108c
SS
29#include <dis-asm.h>
30#include "sim-config.h"
31
32
33#define VAL(x) strtoul(x,(char **)NULL,0)
34
35extern int current_target_byte_order;
36struct disassemble_info dinfo;
37struct pstate sregs;
38extern struct estate ebase;
39int ctrl_c = 0;
40int sis_verbose = 0;
41char *sis_version = "2.7.5";
42int nfp = 0;
43int ift = 0;
44int wrp = 0;
45int rom8 = 0;
46int uben = 0;
47int termsave;
48int sparclite = 0; /* emulating SPARClite instructions? */
49int sparclite_board = 0; /* emulating SPARClite board RAM? */
50char uart_dev1[128] = "";
51char uart_dev2[128] = "";
52extern int ext_irl;
53uint32 last_load_addr = 0;
54
55#ifdef ERRINJ
56uint32 errcnt = 0;
57uint32 errper = 0;
58uint32 errtt = 0;
59uint32 errftt = 0;
60uint32 errmec = 0;
61#endif
62
63/* Forward declarations */
64
bdca5ee4
TT
65static int batch (struct pstate *sregs, char *fname);
66static void set_rega (struct pstate *sregs, char *reg, uint32 rval);
67static void disp_reg (struct pstate *sregs, char *reg);
68static uint32 limcalc (float32 freq);
69static void int_handler (int32 sig);
70static void init_event (void);
71static int disp_fpu (struct pstate *sregs);
72static void disp_regs (struct pstate *sregs, int cwp);
73static void disp_ctrl (struct pstate *sregs);
74static void disp_mem (uint32 addr, uint32 len);
c906108c
SS
75
76static int
77batch(sregs, fname)
78 struct pstate *sregs;
79 char *fname;
80{
81 FILE *fp;
82 char lbuf[1024];
83
84 if ((fp = fopen(fname, "r")) == NULL) {
85 fprintf(stderr, "couldn't open batch file %s\n", fname);
86 return (0);
87 }
88 while (!feof(fp)) {
89 lbuf[0] = 0;
90 fgets(lbuf, 1023, fp);
91 if ((strlen(lbuf) > 0) && (lbuf[strlen(lbuf) - 1] == '\n'))
92 lbuf[strlen(lbuf) - 1] = 0;
93 printf("sis> %s\n", lbuf);
94 exec_cmd(sregs, lbuf);
95 }
96 fclose(fp);
97 return (1);
98}
99
100void
101set_regi(sregs, reg, rval)
102 struct pstate *sregs;
103 int32 reg;
104 uint32 rval;
105{
106 uint32 cwp;
107
108 cwp = ((sregs->psr & 0x7) << 4);
109 if ((reg > 0) && (reg < 8)) {
110 sregs->g[reg] = rval;
111 } else if ((reg >= 8) && (reg < 32)) {
112 sregs->r[(cwp + reg) & 0x7f] = rval;
113 } else if ((reg >= 32) && (reg < 64)) {
114 sregs->fsi[reg - 32] = rval;
115 } else {
116 switch (reg) {
117 case 64:
118 sregs->y = rval;
119 break;
120 case 65:
121 sregs->psr = rval;
122 break;
123 case 66:
124 sregs->wim = rval;
125 break;
126 case 67:
127 sregs->tbr = rval;
128 break;
129 case 68:
130 sregs->pc = rval;
131 break;
132 case 69:
133 sregs->npc = rval;
134 break;
135 case 70:
136 sregs->fsr = rval;
137 set_fsr(rval);
138 break;
139 default:break;
140 }
141 }
142}
143
144void
145get_regi(struct pstate * sregs, int32 reg, char *buf)
146{
147 uint32 cwp;
148 uint32 rval = 0;
149
150 cwp = ((sregs->psr & 0x7) << 4);
151 if ((reg >= 0) && (reg < 8)) {
152 rval = sregs->g[reg];
153 } else if ((reg >= 8) && (reg < 32)) {
154 rval = sregs->r[(cwp + reg) & 0x7f];
155 } else if ((reg >= 32) && (reg < 64)) {
156 rval = sregs->fsi[reg - 32];
157 } else {
158 switch (reg) {
159 case 64:
160 rval = sregs->y;
161 break;
162 case 65:
163 rval = sregs->psr;
164 break;
165 case 66:
166 rval = sregs->wim;
167 break;
168 case 67:
169 rval = sregs->tbr;
170 break;
171 case 68:
172 rval = sregs->pc;
173 break;
174 case 69:
175 rval = sregs->npc;
176 break;
177 case 70:
178 rval = sregs->fsr;
179 break;
180 default:break;
181 }
182 }
183 if (current_target_byte_order == BIG_ENDIAN) {
184 buf[0] = (rval >> 24) & 0x0ff;
185 buf[1] = (rval >> 16) & 0x0ff;
186 buf[2] = (rval >> 8) & 0x0ff;
187 buf[3] = rval & 0x0ff;
188 }
189 else {
190 buf[3] = (rval >> 24) & 0x0ff;
191 buf[2] = (rval >> 16) & 0x0ff;
192 buf[1] = (rval >> 8) & 0x0ff;
193 buf[0] = rval & 0x0ff;
194 }
195}
196
197
198static void
199set_rega(sregs, reg, rval)
200 struct pstate *sregs;
201 char *reg;
202 uint32 rval;
203{
204 uint32 cwp;
205 int32 err = 0;
206
207 cwp = ((sregs->psr & 0x7) << 4);
208 if (strcmp(reg, "psr") == 0)
209 sregs->psr = (rval = (rval & 0x00f03fff));
210 else if (strcmp(reg, "tbr") == 0)
211 sregs->tbr = (rval = (rval & 0xfffffff0));
212 else if (strcmp(reg, "wim") == 0)
213 sregs->wim = (rval = (rval & 0x0ff));
214 else if (strcmp(reg, "y") == 0)
215 sregs->y = rval;
216 else if (strcmp(reg, "pc") == 0)
217 sregs->pc = rval;
218 else if (strcmp(reg, "npc") == 0)
219 sregs->npc = rval;
220 else if (strcmp(reg, "fsr") == 0) {
221 sregs->fsr = rval;
222 set_fsr(rval);
223 } else if (strcmp(reg, "g0") == 0)
224 err = 2;
225 else if (strcmp(reg, "g1") == 0)
226 sregs->g[1] = rval;
227 else if (strcmp(reg, "g2") == 0)
228 sregs->g[2] = rval;
229 else if (strcmp(reg, "g3") == 0)
230 sregs->g[3] = rval;
231 else if (strcmp(reg, "g4") == 0)
232 sregs->g[4] = rval;
233 else if (strcmp(reg, "g5") == 0)
234 sregs->g[5] = rval;
235 else if (strcmp(reg, "g6") == 0)
236 sregs->g[6] = rval;
237 else if (strcmp(reg, "g7") == 0)
238 sregs->g[7] = rval;
239 else if (strcmp(reg, "o0") == 0)
240 sregs->r[(cwp + 8) & 0x7f] = rval;
241 else if (strcmp(reg, "o1") == 0)
242 sregs->r[(cwp + 9) & 0x7f] = rval;
243 else if (strcmp(reg, "o2") == 0)
244 sregs->r[(cwp + 10) & 0x7f] = rval;
245 else if (strcmp(reg, "o3") == 0)
246 sregs->r[(cwp + 11) & 0x7f] = rval;
247 else if (strcmp(reg, "o4") == 0)
248 sregs->r[(cwp + 12) & 0x7f] = rval;
249 else if (strcmp(reg, "o5") == 0)
250 sregs->r[(cwp + 13) & 0x7f] = rval;
251 else if (strcmp(reg, "o6") == 0)
252 sregs->r[(cwp + 14) & 0x7f] = rval;
253 else if (strcmp(reg, "o7") == 0)
254 sregs->r[(cwp + 15) & 0x7f] = rval;
255 else if (strcmp(reg, "l0") == 0)
256 sregs->r[(cwp + 16) & 0x7f] = rval;
257 else if (strcmp(reg, "l1") == 0)
258 sregs->r[(cwp + 17) & 0x7f] = rval;
259 else if (strcmp(reg, "l2") == 0)
260 sregs->r[(cwp + 18) & 0x7f] = rval;
261 else if (strcmp(reg, "l3") == 0)
262 sregs->r[(cwp + 19) & 0x7f] = rval;
263 else if (strcmp(reg, "l4") == 0)
264 sregs->r[(cwp + 20) & 0x7f] = rval;
265 else if (strcmp(reg, "l5") == 0)
266 sregs->r[(cwp + 21) & 0x7f] = rval;
267 else if (strcmp(reg, "l6") == 0)
268 sregs->r[(cwp + 22) & 0x7f] = rval;
269 else if (strcmp(reg, "l7") == 0)
270 sregs->r[(cwp + 23) & 0x7f] = rval;
271 else if (strcmp(reg, "i0") == 0)
272 sregs->r[(cwp + 24) & 0x7f] = rval;
273 else if (strcmp(reg, "i1") == 0)
274 sregs->r[(cwp + 25) & 0x7f] = rval;
275 else if (strcmp(reg, "i2") == 0)
276 sregs->r[(cwp + 26) & 0x7f] = rval;
277 else if (strcmp(reg, "i3") == 0)
278 sregs->r[(cwp + 27) & 0x7f] = rval;
279 else if (strcmp(reg, "i4") == 0)
280 sregs->r[(cwp + 28) & 0x7f] = rval;
281 else if (strcmp(reg, "i5") == 0)
282 sregs->r[(cwp + 29) & 0x7f] = rval;
283 else if (strcmp(reg, "i6") == 0)
284 sregs->r[(cwp + 30) & 0x7f] = rval;
285 else if (strcmp(reg, "i7") == 0)
286 sregs->r[(cwp + 31) & 0x7f] = rval;
287 else
288 err = 1;
289 switch (err) {
290 case 0:
291 printf("%s = %d (0x%08x)\n", reg, rval, rval);
292 break;
293 case 1:
294 printf("no such regiser: %s\n", reg);
295 break;
296 case 2:
297 printf("cannot set g0\n");
298 break;
299 default:
300 break;
301 }
302
303}
304
305static void
306disp_reg(sregs, reg)
307 struct pstate *sregs;
308 char *reg;
309{
310 if (strncmp(reg, "w",1) == 0)
311 disp_regs(sregs, VAL(&reg[1]));
312}
313
314#ifdef ERRINJ
315
316void
317errinj()
318{
319 int err;
320
321 switch (err = (random() % 12)) {
322 case 0: errtt = 0x61; break;
323 case 1: errtt = 0x62; break;
324 case 2: errtt = 0x63; break;
325 case 3: errtt = 0x64; break;
326 case 4: errtt = 0x65; break;
327 case 5:
328 case 6:
329 case 7: errftt = err;
330 break;
331 case 8: errmec = 1; break;
332 case 9: errmec = 2; break;
333 case 10: errmec = 5; break;
334 case 11: errmec = 6; break;
335 }
336 errcnt++;
337 if (errper) event(errinj, 0, (random()%errper));
338}
339
340void
341errinjstart()
342{
343 if (errper) event(errinj, 0, (random()%errper));
344}
345
346#endif
347
348static uint32
349limcalc (freq)
350 float32 freq;
351{
352 uint32 unit, lim;
353 double flim;
354 char *cmd1, *cmd2;
355
356 unit = 1;
357 lim = -1;
358 if ((cmd1 = strtok(NULL, " \t\n\r")) != NULL) {
359 lim = VAL(cmd1);
360 if ((cmd2 = strtok(NULL, " \t\n\r")) != NULL) {
361 if (strcmp(cmd2,"us")==0) unit = 1;
362 if (strcmp(cmd2,"ms")==0) unit = 1000;
363 if (strcmp(cmd2,"s")==0) unit = 1000000;
364 }
365 flim = (double) lim * (double) unit * (double) freq +
366 (double) ebase.simtime;
367 if ((flim > ebase.simtime) && (flim < 4294967296.0)) {
368 lim = (uint32) flim;
369 } else {
370 printf("error in expression\n");
371 lim = -1;
372 }
373 }
374 return (lim);
375}
376
377int
378exec_cmd(sregs, cmd)
379 char *cmd;
380 struct pstate *sregs;
381{
382 char *cmd1, *cmd2;
383 int32 stat;
384 uint32 len, i, clen, j;
385 static uint32 daddr = 0;
386 char *cmdsave;
387
388 stat = OK;
389 cmdsave = strdup(cmd);
390 if ((cmd1 = strtok(cmd, " \t")) != NULL) {
391 clen = strlen(cmd1);
392 if (strncmp(cmd1, "bp", clen) == 0) {
393 for (i = 0; i < sregs->bptnum; i++) {
394 printf(" %d : 0x%08x\n", i + 1, sregs->bpts[i]);
395 }
396 } else if (strncmp(cmd1, "+bp", clen) == 0) {
397 if ((cmd1 = strtok(NULL, " \t\n\r")) != NULL) {
398 sregs->bpts[sregs->bptnum] = VAL(cmd1) & ~0x3;
399 printf("added breakpoint %d at 0x%08x\n",
400 sregs->bptnum + 1, sregs->bpts[sregs->bptnum]);
401 sregs->bptnum += 1;
402 }
403 } else if (strncmp(cmd1, "-bp", clen) == 0) {
404 if ((cmd1 = strtok(NULL, " \t\n\r")) != NULL) {
405 i = VAL(cmd1) - 1;
406 if ((i >= 0) && (i < sregs->bptnum)) {
407 printf("deleted breakpoint %d at 0x%08x\n", i + 1,
408 sregs->bpts[i]);
409 for (; i < sregs->bptnum - 1; i++) {
410 sregs->bpts[i] = sregs->bpts[i + 1];
411 }
412 sregs->bptnum -= 1;
413 }
414 }
415 } else if (strncmp(cmd1, "batch", clen) == 0) {
416 if ((cmd1 = strtok(NULL, " \t\n\r")) == NULL) {
417 printf("no file specified\n");
418 } else {
419 batch(sregs, cmd1);
420 }
421 } else if (strncmp(cmd1, "cont", clen) == 0) {
422 if ((cmd1 = strtok(NULL, " \t\n\r")) == NULL) {
94110024 423 stat = run_sim(sregs, UINT64_MAX, 0);
c906108c
SS
424 } else {
425 stat = run_sim(sregs, VAL(cmd1), 0);
426 }
427 daddr = sregs->pc;
428 sim_halt();
429 } else if (strncmp(cmd1, "debug", clen) == 0) {
430 if ((cmd1 = strtok(NULL, " \t\n\r")) != NULL) {
431 sis_verbose = VAL(cmd1);
432 }
433 printf("Debug level = %d\n",sis_verbose);
434 } else if (strncmp(cmd1, "dis", clen) == 0) {
435 if ((cmd1 = strtok(NULL, " \t\n\r")) != NULL) {
436 daddr = VAL(cmd1);
437 }
438 if ((cmd2 = strtok(NULL, " \t\n\r")) != NULL) {
439 len = VAL(cmd2);
440 } else
441 len = 16;
442 printf("\n");
443 dis_mem(daddr, len, &dinfo);
444 printf("\n");
445 daddr += len * 4;
446 } else if (strncmp(cmd1, "echo", clen) == 0) {
447 if ((cmd1 = strtok(NULL, " \t\n\r")) != NULL) {
448 printf("%s\n", (&cmdsave[clen+1]));
449 }
450#ifdef ERRINJ
451 } else if (strncmp(cmd1, "error", clen) == 0) {
452 if ((cmd1 = strtok(NULL, " \t\n\r")) != NULL) {
453 errper = VAL(cmd1);
454 if (errper) {
455 event(errinj, 0, (len = (random()%errper)));
456 printf("Error injection started with period %d\n",len);
457 }
458 } else printf("Injected errors: %d\n",errcnt);
459#endif
460 } else if (strncmp(cmd1, "float", clen) == 0) {
461 stat = disp_fpu(sregs);
462 } else if (strncmp(cmd1, "go", clen) == 0) {
463 if ((cmd1 = strtok(NULL, " \t\n\r")) == NULL) {
464 len = last_load_addr;
465 } else {
466 len = VAL(cmd1);
467 }
468 sregs->pc = len & ~3;
469 sregs->npc = sregs->pc + 4;
20a0ffe3
JG
470 if ((sregs->pc != 0) && (ebase.simtime == 0))
471 boot_init();
c906108c
SS
472 printf("resuming at 0x%08x\n",sregs->pc);
473 if ((cmd2 = strtok(NULL, " \t\n\r")) != NULL) {
474 stat = run_sim(sregs, VAL(cmd2), 0);
475 } else {
94110024 476 stat = run_sim(sregs, UINT64_MAX, 0);
c906108c
SS
477 }
478 daddr = sregs->pc;
479 sim_halt();
480 } else if (strncmp(cmd1, "help", clen) == 0) {
481 gen_help();
482 } else if (strncmp(cmd1, "history", clen) == 0) {
483 if ((cmd1 = strtok(NULL, " \t\n\r")) != NULL) {
484 sregs->histlen = VAL(cmd1);
485 if (sregs->histbuf != NULL)
486 free(sregs->histbuf);
487 sregs->histbuf = (struct histype *) calloc(sregs->histlen, sizeof(struct histype));
488 printf("trace history length = %d\n\r", sregs->histlen);
489 sregs->histind = 0;
490
491 } else {
492 j = sregs->histind;
493 for (i = 0; i < sregs->histlen; i++) {
494 if (j >= sregs->histlen)
495 j = 0;
496 printf(" %8d ", sregs->histbuf[j].time);
497 dis_mem(sregs->histbuf[j].addr, 1, &dinfo);
498 j++;
499 }
500 }
501
502 } else if (strncmp(cmd1, "load", clen) == 0) {
503 if ((cmd1 = strtok(NULL, " \t\n\r")) != NULL) {
504 last_load_addr = bfd_load(cmd1);
505 while ((cmd1 = strtok(NULL, " \t\n\r")) != NULL)
506 last_load_addr = bfd_load(cmd1);
507 } else {
508 printf("load: no file specified\n");
509 }
510 } else if (strncmp(cmd1, "mem", clen) == 0) {
511 if ((cmd1 = strtok(NULL, " \t\n\r")) != NULL)
512 daddr = VAL(cmd1);
513 if ((cmd2 = strtok(NULL, " \t\n\r")) != NULL)
514 len = VAL(cmd2);
515 else
516 len = 64;
517 disp_mem(daddr, len);
518 daddr += len;
519 } else if (strncmp(cmd1, "perf", clen) == 0) {
520 cmd1 = strtok(NULL, " \t\n\r");
521 if ((cmd1 != NULL) &&
522 (strncmp(cmd1, "reset", strlen(cmd1)) == 0)) {
523 reset_stat(sregs);
524 } else
525 show_stat(sregs);
526 } else if (strncmp(cmd1, "quit", clen) == 0) {
527 exit(0);
528 } else if (strncmp(cmd1, "reg", clen) == 0) {
529 cmd1 = strtok(NULL, " \t\n\r");
530 cmd2 = strtok(NULL, " \t\n\r");
531 if (cmd2 != NULL)
532 set_rega(sregs, cmd1, VAL(cmd2));
533 else if (cmd1 != NULL)
534 disp_reg(sregs, cmd1);
535 else {
536 disp_regs(sregs,sregs->psr);
537 disp_ctrl(sregs);
538 }
539 } else if (strncmp(cmd1, "reset", clen) == 0) {
540 ebase.simtime = 0;
541 reset_all();
542 reset_stat(sregs);
543 } else if (strncmp(cmd1, "run", clen) == 0) {
544 ebase.simtime = 0;
545 reset_all();
546 reset_stat(sregs);
547 if ((cmd1 = strtok(NULL, " \t\n\r")) == NULL) {
94110024 548 stat = run_sim(sregs, UINT64_MAX, 0);
c906108c
SS
549 } else {
550 stat = run_sim(sregs, VAL(cmd1), 0);
551 }
552 daddr = sregs->pc;
553 sim_halt();
554 } else if (strncmp(cmd1, "shell", clen) == 0) {
555 if ((cmd1 = strtok(NULL, " \t\n\r")) != NULL) {
556 system(&cmdsave[clen]);
557 }
558 } else if (strncmp(cmd1, "step", clen) == 0) {
559 stat = run_sim(sregs, 1, 1);
560 daddr = sregs->pc;
561 sim_halt();
562 } else if (strncmp(cmd1, "tcont", clen) == 0) {
563 sregs->tlimit = limcalc(sregs->freq);
94110024 564 stat = run_sim(sregs, UINT64_MAX, 0);
c906108c
SS
565 daddr = sregs->pc;
566 sim_halt();
567 } else if (strncmp(cmd1, "tgo", clen) == 0) {
568 if ((cmd1 = strtok(NULL, " \t\n\r")) == NULL) {
569 len = last_load_addr;
570 } else {
571 len = VAL(cmd1);
572 sregs->tlimit = limcalc(sregs->freq);
573 }
574 sregs->pc = len & ~3;
575 sregs->npc = sregs->pc + 4;
576 printf("resuming at 0x%08x\n",sregs->pc);
94110024 577 stat = run_sim(sregs, UINT64_MAX, 0);
c906108c
SS
578 daddr = sregs->pc;
579 sim_halt();
580 } else if (strncmp(cmd1, "tlimit", clen) == 0) {
581 sregs->tlimit = limcalc(sregs->freq);
582 if (sregs->tlimit != (uint32) -1)
583 printf("simulation limit = %u (%.3f ms)\n",(uint32) sregs->tlimit,
584 sregs->tlimit / sregs->freq / 1000);
585 } else if (strncmp(cmd1, "tra", clen) == 0) {
586 if ((cmd1 = strtok(NULL, " \t\n\r")) == NULL) {
94110024 587 stat = run_sim(sregs, UINT64_MAX, 1);
c906108c
SS
588 } else {
589 stat = run_sim(sregs, VAL(cmd1), 1);
590 }
591 printf("\n");
592 daddr = sregs->pc;
593 sim_halt();
594 } else if (strncmp(cmd1, "trun", clen) == 0) {
595 ebase.simtime = 0;
596 reset_all();
597 reset_stat(sregs);
598 sregs->tlimit = limcalc(sregs->freq);
94110024 599 stat = run_sim(sregs, UINT64_MAX, 0);
c906108c
SS
600 daddr = sregs->pc;
601 sim_halt();
602 } else
603 printf("syntax error\n");
604 }
605 if (cmdsave != NULL)
606 free(cmdsave);
607 return (stat);
608}
609
610
611void
612reset_stat(sregs)
613 struct pstate *sregs;
614{
96d67095 615 sregs->tottime = 0.0;
c906108c
SS
616 sregs->pwdtime = 0;
617 sregs->ninst = 0;
618 sregs->fholdt = 0;
619 sregs->holdt = 0;
620 sregs->icntt = 0;
621 sregs->finst = 0;
622 sregs->nstore = 0;
623 sregs->nload = 0;
624 sregs->nbranch = 0;
625 sregs->simstart = ebase.simtime;
626
627}
628
629void
630show_stat(sregs)
631 struct pstate *sregs;
632{
633 uint32 iinst;
96d67095 634 uint32 stime;
c906108c 635
96d67095
JG
636 if (sregs->tottime == 0.0)
637 sregs->tottime += 1E-6;
c906108c
SS
638 stime = ebase.simtime - sregs->simstart; /* Total simulated time */
639#ifdef STAT
640
641 iinst = sregs->ninst - sregs->finst - sregs->nload - sregs->nstore -
642 sregs->nbranch;
643#endif
644
645 printf("\n Cycles : %9d\n\r", ebase.simtime - sregs->simstart);
646 printf(" Instructions : %9d\n", sregs->ninst);
647
648#ifdef STAT
649 printf(" integer : %9.2f %%\n", 100.0 * (float) iinst / (float) sregs->ninst);
650 printf(" load : %9.2f %%\n",
651 100.0 * (float) sregs->nload / (float) sregs->ninst);
652 printf(" store : %9.2f %%\n",
653 100.0 * (float) sregs->nstore / (float) sregs->ninst);
654 printf(" branch : %9.2f %%\n",
655 100.0 * (float) sregs->nbranch / (float) sregs->ninst);
656 printf(" float : %9.2f %%\n",
657 100.0 * (float) sregs->finst / (float) sregs->ninst);
658 printf(" Integer CPI : %9.2f\n",
659 ((float) (stime - sregs->pwdtime - sregs->fholdt - sregs->finst))
660 /
661 (float) (sregs->ninst - sregs->finst));
662 printf(" Float CPI : %9.2f\n",
663 ((float) sregs->fholdt / (float) sregs->finst) + 1.0);
664#endif
665 printf(" Overall CPI : %9.2f\n",
666 (float) (stime - sregs->pwdtime) / (float) sregs->ninst);
667 printf("\n ERC32 performance (%4.1f MHz): %5.2f MOPS (%5.2f MIPS, %5.2f MFLOPS)\n",
668 sregs->freq, sregs->freq * (float) sregs->ninst / (float) (stime - sregs->pwdtime),
669 sregs->freq * (float) (sregs->ninst - sregs->finst) /
670 (float) (stime - sregs->pwdtime),
671 sregs->freq * (float) sregs->finst / (float) (stime - sregs->pwdtime));
96d67095
JG
672 printf(" Simulated ERC32 time : %.2f s\n",
673 (float) (ebase.simtime - sregs->simstart) / 1000000.0 / sregs->freq);
674 printf(" Processor utilisation : %.2f %%\n",
675 100.0 * (1.0 - ((float) sregs->pwdtime / (float) stime)));
676 printf(" Real-time performance : %.2f %%\n",
677 100.0 / (sregs->tottime / ((double) (stime) / (sregs->freq * 1.0E6))));
678 printf(" Simulator performance : %.2f MIPS\n",
679 (double)(sregs->ninst) / sregs->tottime / 1E6);
680 printf(" Used time (sys + user) : %.2f s\n\n", sregs->tottime);
c906108c
SS
681}
682
683
684
685void
686init_bpt(sregs)
687 struct pstate *sregs;
688{
689 sregs->bptnum = 0;
690 sregs->histlen = 0;
691 sregs->histind = 0;
692 sregs->histbuf = NULL;
693 sregs->tlimit = -1;
694}
695
696static void
697int_handler(sig)
698 int32 sig;
699{
700 if (sig != 2)
701 printf("\n\n Signal handler error (%d)\n\n", sig);
702 ctrl_c = 1;
703}
704
705void
706init_signals()
707{
708 typedef void (*PFI) ();
709 static PFI int_tab[2];
710
711 int_tab[0] = signal(SIGTERM, int_handler);
712 int_tab[1] = signal(SIGINT, int_handler);
713}
714
715
716extern struct disassemble_info dinfo;
717
718struct estate ebase;
719struct evcell evbuf[EVENT_MAX];
720struct irqcell irqarr[16];
721
722static int
723disp_fpu(sregs)
724 struct pstate *sregs;
725{
726
727 int i;
728 float t;
729
730 printf("\n fsr: %08X\n\n", sregs->fsr);
731
9c5f41df 732#ifdef HOST_LITTLE_ENDIAN
c906108c
SS
733 for (i = 0; i < 32; i++)
734 sregs->fdp[i ^ 1] = sregs->fs[i];
735#endif
736
737 for (i = 0; i < 32; i++) {
738 t = sregs->fs[i];
739 printf(" f%02d %08x %14e ", i, sregs->fsi[i], sregs->fs[i]);
740 if (!(i & 1))
741 printf("%14e\n", sregs->fd[i >> 1]);
742 else
743 printf("\n");
744 }
745 printf("\n");
746 return (OK);
747}
748
749static void
750disp_regs(sregs,cwp)
751 struct pstate *sregs;
752 int cwp;
753{
754
755 int i;
756
757 cwp = ((cwp & 0x7) << 4);
758 printf("\n\t INS LOCALS OUTS GLOBALS\n");
759 for (i = 0; i < 8; i++) {
760 printf(" %d: %08X %08X %08X %08X\n", i,
761 sregs->r[(cwp + i + 24) & 0x7f],
762 sregs->r[(cwp + i + 16) & 0x7f], sregs->r[(cwp + i + 8) & 0x7f],
763 sregs->g[i]);
764 }
765}
766
53b5af48
JG
767static void print_insn_sparc_sis(uint32 addr, struct disassemble_info *info)
768{
769 unsigned char i[4];
770
771 sis_memory_read(addr, i, 4);
772 dinfo.buffer_vma = addr;
773 dinfo.buffer_length = 4;
774 dinfo.buffer = i;
775 print_insn_sparc(addr, info);
776}
777
c906108c
SS
778static void
779disp_ctrl(sregs)
780 struct pstate *sregs;
781{
782
783 unsigned char i[4];
784
785 printf("\n psr: %08X wim: %08X tbr: %08X y: %08X\n",
786 sregs->psr, sregs->wim, sregs->tbr, sregs->y);
787 sis_memory_read(sregs->pc, i, 4);
788 printf("\n pc: %08X = %02X%02X%02X%02X ", sregs->pc,i[0],i[1],i[2],i[3]);
53b5af48 789 print_insn_sparc_sis(sregs->pc, &dinfo);
c906108c
SS
790 sis_memory_read(sregs->npc, i, 4);
791 printf("\n npc: %08X = %02X%02X%02X%02X ",sregs->npc,i[0],i[1],i[2],i[3]);
53b5af48 792 print_insn_sparc_sis(sregs->npc, &dinfo);
c906108c
SS
793 if (sregs->err_mode)
794 printf("\n IU in error mode");
795 printf("\n\n");
796}
797
798static void
799disp_mem(addr, len)
800 uint32 addr;
801 uint32 len;
802{
803
804 uint32 i;
805 unsigned char data[4];
806 uint32 mem[4], j;
807 char *p;
808
809 for (i = addr & ~3; i < ((addr + len) & ~3); i += 16) {
810 printf("\n %8X ", i);
811 for (j = 0; j < 4; j++) {
812 sis_memory_read((i + (j * 4)), data, 4);
813 printf("%02x%02x%02x%02x ", data[0],data[1],data[2],data[3]);
814 mem[j] = *((int *) &data);
815 }
816 printf(" ");
817 p = (char *) mem;
818 for (j = 0; j < 16; j++) {
819 if (isprint(p[j]))
820 putchar(p[j]);
821 else
822 putchar('.');
823 }
824 }
825 printf("\n\n");
826}
827
828void
829dis_mem(addr, len, info)
830 uint32 addr;
831 uint32 len;
832 struct disassemble_info *info;
833{
834 uint32 i;
835 unsigned char data[4];
836
837 for (i = addr & -3; i < ((addr & -3) + (len << 2)); i += 4) {
838 sis_memory_read(i, data, 4);
839 printf(" %08x %02x%02x%02x%02x ", i, data[0],data[1],data[2],data[3]);
53b5af48 840 print_insn_sparc_sis(i, info);
c906108c
SS
841 if (i >= 0xfffffffc) break;
842 printf("\n");
843 }
844}
845
c906108c
SS
846/* Add event to event queue */
847
848void
849event(cfunc, arg, delta)
850 void (*cfunc) ();
851 int32 arg;
94110024 852 uint64 delta;
c906108c
SS
853{
854 struct evcell *ev1, *evins;
855
856 if (ebase.freeq == NULL) {
857 printf("Error, too many events in event queue\n");
858 return;
859 }
860 ev1 = &ebase.eq;
861 delta += ebase.simtime;
862 while ((ev1->nxt != NULL) && (ev1->nxt->time <= delta)) {
863 ev1 = ev1->nxt;
864 }
865 if (ev1->nxt == NULL) {
866 ev1->nxt = ebase.freeq;
867 ebase.freeq = ebase.freeq->nxt;
868 ev1->nxt->nxt = NULL;
869 } else {
870 evins = ebase.freeq;
871 ebase.freeq = ebase.freeq->nxt;
872 evins->nxt = ev1->nxt;
873 ev1->nxt = evins;
874 }
875 ev1->nxt->time = delta;
876 ev1->nxt->cfunc = cfunc;
877 ev1->nxt->arg = arg;
878}
879
880#if 0 /* apparently not used */
881void
882stop_event()
883{
884}
885#endif
886
887void
888init_event()
889{
890 int32 i;
891
892 ebase.eq.nxt = NULL;
893 ebase.freeq = evbuf;
894 for (i = 0; i < EVENT_MAX; i++) {
895 evbuf[i].nxt = &evbuf[i + 1];
896 }
897 evbuf[EVENT_MAX - 1].nxt = NULL;
898}
899
900void
901set_int(level, callback, arg)
902 int32 level;
903 void (*callback) ();
904 int32 arg;
905{
906 irqarr[level & 0x0f].callback = callback;
907 irqarr[level & 0x0f].arg = arg;
908}
909
910/* Advance simulator time */
911
912void
913advance_time(sregs)
914 struct pstate *sregs;
915{
916
917 struct evcell *evrem;
918 void (*cfunc) ();
94110024
JS
919 uint32 arg;
920 uint64 endtime;
c906108c
SS
921
922#ifdef STAT
923 sregs->fholdt += sregs->fhold;
924 sregs->holdt += sregs->hold;
925 sregs->icntt += sregs->icnt;
926#endif
927
928 endtime = ebase.simtime + sregs->icnt + sregs->hold + sregs->fhold;
929
930 while ((ebase.eq.nxt->time <= (endtime)) && (ebase.eq.nxt != NULL)) {
931 ebase.simtime = ebase.eq.nxt->time;
932 cfunc = ebase.eq.nxt->cfunc;
933 arg = ebase.eq.nxt->arg;
934 evrem = ebase.eq.nxt;
935 ebase.eq.nxt = ebase.eq.nxt->nxt;
936 evrem->nxt = ebase.freeq;
937 ebase.freeq = evrem;
938 cfunc(arg);
939 }
940 ebase.simtime = endtime;
941
942}
943
944uint32
945now()
946{
947 return(ebase.simtime);
948}
949
950
951/* Advance time until an external interrupt is seen */
952
953int
954wait_for_irq()
955{
956 struct evcell *evrem;
957 void (*cfunc) ();
94110024
JS
958 int32 arg;
959 uint64 endtime;
c906108c
SS
960
961 if (ebase.eq.nxt == NULL)
962 printf("Warning: event queue empty - power-down mode not entered\n");
963 endtime = ebase.simtime;
964 while (!ext_irl && (ebase.eq.nxt != NULL)) {
965 ebase.simtime = ebase.eq.nxt->time;
966 cfunc = ebase.eq.nxt->cfunc;
967 arg = ebase.eq.nxt->arg;
968 evrem = ebase.eq.nxt;
969 ebase.eq.nxt = ebase.eq.nxt->nxt;
970 evrem->nxt = ebase.freeq;
971 ebase.freeq = evrem;
972 cfunc(arg);
973 if (ctrl_c) {
974 printf("\bwarning: power-down mode interrupted\n");
975 break;
976 }
977 }
978 sregs.pwdtime += ebase.simtime - endtime;
979 return (ebase.simtime - endtime);
980}
981
982int
983check_bpt(sregs)
984 struct pstate *sregs;
985{
986 int32 i;
987
988 if ((sregs->bphit) || (sregs->annul))
989 return (0);
990 for (i = 0; i < (int32) sregs->bptnum; i++) {
991 if (sregs->pc == sregs->bpts[i])
992 return (BPT_HIT);
993 }
994 return (0);
995}
996
997void
998reset_all()
999{
1000 init_event(); /* Clear event queue */
1001 init_regs(&sregs);
1002 reset();
1003#ifdef ERRINJ
1004 errinjstart();
1005#endif
1006}
1007
1008void
1009sys_reset()
1010{
1011 reset_all();
1012 sregs.trap = 256; /* Force fake reset trap */
1013}
1014
1015void
1016sys_halt()
1017{
1018 sregs.trap = 257; /* Force fake halt trap */
1019}
1020
1021#include "ansidecl.h"
1022
c906108c 1023#include <stdarg.h>
c906108c
SS
1024
1025#include "libiberty.h"
1026#include "bfd.h"
1027
1028#define min(A, B) (((A) < (B)) ? (A) : (B))
1029#define LOAD_ADDRESS 0
1030
1031int
1032bfd_load(fname)
1033 char *fname;
1034{
1035 asection *section;
1036 bfd *pbfd;
1037 const bfd_arch_info_type *arch;
1038
1039 pbfd = bfd_openr(fname, 0);
1040
1041 if (pbfd == NULL) {
1042 printf("open of %s failed\n", fname);
1043 return (-1);
1044 }
1045 if (!bfd_check_format(pbfd, bfd_object)) {
1046 printf("file %s doesn't seem to be an object file\n", fname);
1047 return (-1);
1048 }
1049
1050 arch = bfd_get_arch_info (pbfd);
1051 if (bfd_little_endian (pbfd) || arch->mach == bfd_mach_sparc_sparclite_le)
1052 current_target_byte_order = LITTLE_ENDIAN;
1053 else
1054 current_target_byte_order = BIG_ENDIAN;
1055 if (sis_verbose)
1056 printf("file %s is %s-endian.\n", fname,
1057 current_target_byte_order == BIG_ENDIAN ? "big" : "little");
1058
1059 if (sis_verbose)
1060 printf("loading %s:", fname);
1061 for (section = pbfd->sections; section; section = section->next) {
1062 if (bfd_get_section_flags(pbfd, section) & SEC_ALLOC) {
1063 bfd_vma section_address;
1064 unsigned long section_size;
1065 const char *section_name;
1066
1067 section_name = bfd_get_section_name(pbfd, section);
1068
1069 section_address = bfd_get_section_vma(pbfd, section);
1070 /*
1071 * Adjust sections from a.out files, since they don't carry their
1072 * addresses with.
1073 */
1074 if (bfd_get_flavour(pbfd) == bfd_target_aout_flavour) {
1075 if (strcmp (section_name, ".text") == 0)
1076 section_address = bfd_get_start_address (pbfd);
1077 else if (strcmp (section_name, ".data") == 0) {
1078 /* Read the first 8 bytes of the data section.
1079 There should be the string 'DaTa' followed by
1080 a word containing the actual section address. */
1081 struct data_marker
1082 {
1083 char signature[4]; /* 'DaTa' */
1084 unsigned char sdata[4]; /* &sdata */
1085 } marker;
1086 bfd_get_section_contents (pbfd, section, &marker, 0,
1087 sizeof (marker));
1088 if (strncmp (marker.signature, "DaTa", 4) == 0)
1089 {
1090 if (current_target_byte_order == BIG_ENDIAN)
1091 section_address = bfd_getb32 (marker.sdata);
1092 else
1093 section_address = bfd_getl32 (marker.sdata);
1094 }
1095 }
1096 }
1097
1098 section_size = bfd_section_size(pbfd, section);
1099
1100 if (sis_verbose)
1101 printf("\nsection %s at 0x%08lx (0x%lx bytes)",
1102 section_name, section_address, section_size);
1103
1104 /* Text, data or lit */
1105 if (bfd_get_section_flags(pbfd, section) & SEC_LOAD) {
1106 file_ptr fptr;
1107
1108 fptr = 0;
1109
1110 while (section_size > 0) {
1111 char buffer[1024];
1112 int count;
1113
1114 count = min(section_size, 1024);
1115
1116 bfd_get_section_contents(pbfd, section, buffer, fptr, count);
1117
1118 sis_memory_write(section_address, buffer, count);
1119
1120 section_address += count;
1121 fptr += count;
1122 section_size -= count;
1123 }
1124 } else /* BSS */
1125 if (sis_verbose)
1126 printf("(not loaded)");
1127 }
1128 }
1129 if (sis_verbose)
1130 printf("\n");
1131
1132 return(bfd_get_start_address (pbfd));
1133}
96d67095
JG
1134
1135double get_time (void)
1136{
1137 double usec;
1138
1139 struct timeval tm;
1140
1141 gettimeofday (&tm, NULL);
1142 usec = ((double) tm.tv_sec) * 1E6 + ((double) tm.tv_usec);
1143 return (usec / 1E6);
1144}
This page took 0.738693 seconds and 4 git commands to generate.