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