1998-02-06 02:29:22 +00:00
|
|
|
/* Copyright (C) 1998, Cygnus Solutions
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "sim-main.h"
|
1998-02-14 05:34:08 +00:00
|
|
|
#include "sim-endian.h"
|
1998-02-06 02:29:22 +00:00
|
|
|
|
1998-02-06 03:09:03 +00:00
|
|
|
#include "sky-device.h"
|
|
|
|
#include "sky-vu1.h"
|
|
|
|
#include "sky-libvpe.h"
|
|
|
|
#include "sky-vu.h"
|
1998-02-16 21:44:45 +00:00
|
|
|
#include "sky-bits.h"
|
|
|
|
|
|
|
|
#include <assert.h>
|
1998-02-06 02:29:22 +00:00
|
|
|
|
|
|
|
VectorUnitState vu1_state;
|
|
|
|
|
1998-02-14 05:34:08 +00:00
|
|
|
static char vu1_umem_buffer[VU1_MEM0_SIZE] __attribute__ ((aligned(16)));
|
|
|
|
static char vu1_mem_buffer[VU1_MEM1_SIZE] __attribute__ ((aligned(16)));
|
1998-02-06 02:29:22 +00:00
|
|
|
|
1998-02-10 20:08:16 +00:00
|
|
|
void init_vu1(void);
|
1998-02-06 02:29:22 +00:00
|
|
|
void init_vu(VectorUnitState *state, char* umem_buffer, char* mem_buffer);
|
|
|
|
|
|
|
|
void
|
1998-02-10 20:08:16 +00:00
|
|
|
vu1_issue(void)
|
1998-02-06 02:29:22 +00:00
|
|
|
{
|
|
|
|
if (vu1_state.runState == VU_RUN)
|
|
|
|
vpecallms_cycle(&vu1_state);
|
|
|
|
}
|
|
|
|
|
1998-02-10 20:08:16 +00:00
|
|
|
static int
|
1998-02-06 02:29:22 +00:00
|
|
|
vu1_io_read_register_window(device *me,
|
|
|
|
void *dest,
|
|
|
|
int space,
|
|
|
|
address_word addr,
|
|
|
|
unsigned nr_bytes,
|
|
|
|
sim_cpu *processor,
|
|
|
|
sim_cia cia)
|
|
|
|
{
|
|
|
|
/* Slow and crappy hack ... */
|
|
|
|
|
1998-02-16 21:44:45 +00:00
|
|
|
|
1998-02-06 02:29:22 +00:00
|
|
|
int i;
|
|
|
|
|
|
|
|
char source_buffer[VU1_REGISTER_WINDOW_SIZE];
|
|
|
|
char* src;
|
|
|
|
|
1998-02-16 21:44:45 +00:00
|
|
|
assert(nr_bytes == 1 || nr_bytes == 2 || nr_bytes == 4 || nr_bytes == 8 || nr_bytes == 16);
|
|
|
|
|
1998-02-06 02:29:22 +00:00
|
|
|
memcpy(source_buffer, &vu1_state.regs.VF[0][0], 0x200); /* copy VF registers */
|
|
|
|
for (i = 0; i<16; i++ ) {
|
|
|
|
*(short*)&source_buffer[0x200 + i*16] = vu1_state.regs.VI[i];
|
|
|
|
}
|
1998-02-16 21:44:45 +00:00
|
|
|
*(u_long*)&source_buffer[VU1_MST - VU1_REGISTER_WINDOW_START] = vu1_state.regs.MST;
|
|
|
|
*(u_long*)&source_buffer[VU1_MMC - VU1_REGISTER_WINDOW_START] = vu1_state.regs.MMC;
|
|
|
|
*(u_long*)&source_buffer[VU1_MCP - VU1_REGISTER_WINDOW_START] = vu1_state.regs.MCP;
|
|
|
|
*(u_long*)&source_buffer[VU1_MR - VU1_REGISTER_WINDOW_START] = vu1_state.regs.MR;
|
|
|
|
*(u_long*)&source_buffer[VU1_MI - VU1_REGISTER_WINDOW_START] = vu1_state.regs.MI;
|
|
|
|
*(u_long*)&source_buffer[VU1_MQ - VU1_REGISTER_WINDOW_START] = vu1_state.regs.MQ;
|
|
|
|
*(u_long*)&source_buffer[VU1_MP - VU1_REGISTER_WINDOW_START] = vu1_state.regs.MP;
|
|
|
|
*(u_long*)&source_buffer[VU1_MTPC - VU1_REGISTER_WINDOW_START] = vu1_state.regs.MTPC;
|
|
|
|
|
|
|
|
{
|
|
|
|
u_long stat;
|
|
|
|
stat = 0;
|
|
|
|
if (vu1_state.runState == VU_RUN || vu1_state.runState == VU_BREAK)
|
|
|
|
SET_BIT(stat, VPU_STAT_VBS1_BIT);
|
|
|
|
|
|
|
|
*(u_long*)&source_buffer[VPE1_STAT - VU1_REGISTER_WINDOW_START] = stat;
|
|
|
|
}
|
1998-02-06 02:29:22 +00:00
|
|
|
|
1998-02-14 05:34:08 +00:00
|
|
|
#if 0
|
1998-02-10 20:08:16 +00:00
|
|
|
printf("%s: Read: %x, %d, dest: %x, space: %d, %x!\n", me->name, (int)addr, nr_bytes, (int)dest, space, *(int*)&(vu1_state.regs.VPE_STAT));
|
|
|
|
printf(" vu1_state.regs.VPE_STAT = %x\n", *(int*)&(vu1_state.regs.VPE_STAT));
|
1998-02-14 05:34:08 +00:00
|
|
|
#endif
|
1998-02-06 02:29:22 +00:00
|
|
|
|
|
|
|
if (addr + nr_bytes > VU1_REGISTER_WINDOW_END) {
|
|
|
|
fprintf(stderr, "Error: Read past end of vu1 register window!!!\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
src = &source_buffer[0] + (addr - VU1_REGISTER_WINDOW_START);
|
|
|
|
memcpy(dest, src, nr_bytes);
|
|
|
|
return nr_bytes;
|
|
|
|
}
|
|
|
|
|
1998-02-10 20:08:16 +00:00
|
|
|
static int
|
1998-02-06 02:29:22 +00:00
|
|
|
vu1_io_write_register_window(device *me,
|
|
|
|
const void *source,
|
|
|
|
int space,
|
|
|
|
address_word addr,
|
|
|
|
unsigned nr_bytes,
|
|
|
|
sim_cpu *processor,
|
|
|
|
sim_cia cia)
|
|
|
|
{
|
|
|
|
char *dest;
|
|
|
|
|
|
|
|
if (addr == VPE1_STAT && nr_bytes == 4) {
|
|
|
|
/* Magic to switch VU to run state, until other methods are available. */
|
|
|
|
vu1_state.runState = VU_RUN;
|
1998-02-14 05:34:08 +00:00
|
|
|
vu1_state.junk.eflag = 0;
|
|
|
|
vu1_state.junk.peflag = 0;
|
|
|
|
/*printf("Magic start run...\n");*/
|
1998-02-06 02:29:22 +00:00
|
|
|
return nr_bytes;
|
1998-02-14 05:34:08 +00:00
|
|
|
} else if (addr == VU1_MST && nr_bytes == 4) {
|
|
|
|
/* Magic switch to set _TOP register */
|
|
|
|
/*printf("Magic set TOP register to %d\n", T2H_4(*(int*)source));*/
|
|
|
|
vu1_state.junk._TOP = T2H_4(*(int*)source);
|
|
|
|
return nr_bytes;
|
|
|
|
}
|
1998-02-06 02:29:22 +00:00
|
|
|
|
1998-02-14 05:34:08 +00:00
|
|
|
/*printf("%s: Write: %x, %d, source: %x, space: %d!\n", me->name, (int)addr, nr_bytes, (int)source, space);*/
|
1998-02-06 02:29:22 +00:00
|
|
|
|
|
|
|
if (addr + nr_bytes > VU1_REGISTER_WINDOW_END) {
|
|
|
|
fprintf(stderr, "Error: Read past end of vu1 register window!!!\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
dest = ((char*) (&vu1_state.regs)) + (addr - VU1_REGISTER_WINDOW_START);
|
|
|
|
|
|
|
|
memcpy(dest, source, nr_bytes);
|
|
|
|
|
|
|
|
return nr_bytes;
|
|
|
|
}
|
|
|
|
|
|
|
|
device vu1_device =
|
|
|
|
{
|
|
|
|
"vu1",
|
|
|
|
&vu1_io_read_register_window,
|
|
|
|
&vu1_io_write_register_window
|
|
|
|
};
|
|
|
|
|
|
|
|
void
|
|
|
|
vu1_init(SIM_DESC sd)
|
|
|
|
{
|
|
|
|
|
|
|
|
sim_core_attach (sd,
|
|
|
|
NULL,
|
|
|
|
0 /*level*/,
|
|
|
|
access_read_write,
|
|
|
|
0 /*space ???*/,
|
|
|
|
VU1_REGISTER_WINDOW_START,
|
|
|
|
VU1_REGISTER_WINDOW_SIZE /*nr_bytes*/,
|
|
|
|
0 /*modulo*/,
|
|
|
|
&vu1_device,
|
|
|
|
NULL /*buffer*/);
|
|
|
|
|
|
|
|
sim_core_attach (sd,
|
|
|
|
NULL,
|
|
|
|
0 /*level*/,
|
|
|
|
access_read_write,
|
|
|
|
0 /*space ???*/,
|
|
|
|
VU1_MEM0_WINDOW_START,
|
|
|
|
VU1_MEM0_SIZE /*nr_bytes*/,
|
|
|
|
0 /*modulo*/,
|
|
|
|
0 /*device*/,
|
|
|
|
&vu1_umem_buffer /*buffer*/);
|
|
|
|
|
|
|
|
sim_core_attach (sd,
|
|
|
|
NULL,
|
|
|
|
0 /*level*/,
|
|
|
|
access_read_write,
|
|
|
|
0 /*space ???*/,
|
|
|
|
VU1_MEM1_WINDOW_START,
|
|
|
|
VU1_MEM1_SIZE /*nr_bytes*/,
|
|
|
|
0 /*modulo*/,
|
|
|
|
0 /*device*/,
|
|
|
|
&vu1_mem_buffer /*buffer*/);
|
|
|
|
|
|
|
|
init_vu1();
|
|
|
|
/*initvpe();*/
|
|
|
|
vpecallms_init(&vu1_state);
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************/
|
|
|
|
/* */
|
|
|
|
/* Sony Computer Entertainment CONFIDENTIAL */
|
|
|
|
/* (C) 1997 Sony Computer Entertainment Inc. All Rights Reserved */
|
|
|
|
/* */
|
|
|
|
/* VPE1 simulator */
|
|
|
|
/* */
|
|
|
|
/****************************************************************************/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <strings.h>
|
1998-02-06 03:11:44 +00:00
|
|
|
#include "sky-libvpe.h"
|
1998-02-06 02:29:22 +00:00
|
|
|
|
|
|
|
char ifilename[64] = "vu.bin";
|
|
|
|
char ofilename[64] = "";
|
|
|
|
char pfilename[64] = "";
|
|
|
|
|
1998-02-10 20:08:16 +00:00
|
|
|
static void abend2(char *fmt, char* p) {
|
1998-02-06 02:29:22 +00:00
|
|
|
fprintf(stderr, fmt, p);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
1998-02-14 05:34:08 +00:00
|
|
|
void getoption(VectorUnitState* state);
|
1998-02-06 02:29:22 +00:00
|
|
|
|
1998-02-10 20:08:16 +00:00
|
|
|
void init_vu1(void) {
|
1998-02-06 02:29:22 +00:00
|
|
|
init_vu(&vu1_state, &vu1_umem_buffer[0], &vu1_mem_buffer[0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
void init_vu(VectorUnitState *state, char* umem_buffer, char* mem_buffer)
|
|
|
|
{
|
|
|
|
FILE *fp;
|
|
|
|
int i, j;
|
|
|
|
u_long data[4];
|
|
|
|
|
|
|
|
/* set up memory buffers */
|
|
|
|
state->uMEM = (uMEM_Entry_Type *) umem_buffer;
|
|
|
|
state->MEM = (MEM_Entry_Type*) mem_buffer;
|
|
|
|
|
|
|
|
/* set up run state */
|
|
|
|
state->runState = VU_READY;
|
|
|
|
|
|
|
|
/* read option */
|
1998-02-14 05:34:08 +00:00
|
|
|
getoption(state);
|
1998-02-06 02:29:22 +00:00
|
|
|
|
|
|
|
/* read instruction file (mandatory) */
|
|
|
|
if (*ifilename) {
|
1998-02-10 00:13:54 +00:00
|
|
|
if((fp = fopen(ifilename, "r")) != NULL) {
|
|
|
|
for (i = 0; fread(&data[0], 4, 1, fp) != 0; i++) {
|
|
|
|
fread(&data[1], 4, 1, fp);
|
|
|
|
LoadMMem(state, i, data, 1);
|
|
|
|
}
|
|
|
|
fclose(fp);
|
1998-02-06 02:29:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* PKE dirven simvpe */
|
|
|
|
if (*pfilename) {
|
|
|
|
/* initpke(pfilename); */
|
|
|
|
initvpe(&vu1_state);
|
|
|
|
/* while (simpke() != -1)
|
|
|
|
simvpe(); */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* conventional simvpe */
|
|
|
|
else {
|
|
|
|
initvpe(&vu1_state);
|
|
|
|
/*simvpe();*/
|
|
|
|
}
|
|
|
|
|
|
|
|
/* write result memory image (optional) */
|
|
|
|
if (*ofilename) {
|
|
|
|
if((fp = fopen(ofilename, "w")) == NULL)
|
|
|
|
abend2("%s: can not open\n", ofilename);
|
|
|
|
|
|
|
|
for(i = 0; i < 2048; i++){
|
1998-02-10 20:08:16 +00:00
|
|
|
StoreVUMem(state, i, data, 1);
|
1998-02-06 02:29:22 +00:00
|
|
|
for(j = 0; j < 4; j++)
|
|
|
|
fwrite(&data[j], 4, 1, fp);
|
|
|
|
}
|
|
|
|
fclose(fp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1998-02-10 20:08:16 +00:00
|
|
|
#if 0
|
1998-02-06 02:29:22 +00:00
|
|
|
static void Usage(void)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "Usage: simvpe [options]\n");
|
|
|
|
fprintf(stderr, "\t\t-i instruction-file\n");
|
|
|
|
fprintf(stderr, "\t\t-o output-memory-file\n");
|
|
|
|
fprintf(stderr, "\t\t-t PKE-file (text type)\n");
|
|
|
|
fprintf(stderr, "\t\t-s start-address [default = 0]\n");
|
|
|
|
fprintf(stderr, "\t\t-d [interactive mode enable: default desable]\n");
|
|
|
|
fprintf(stderr, "\t\t-v [statistics mode enable: default desable]\n");
|
|
|
|
fprintf(stderr, "\t\t-p [debug print mode enable: default desable]\n");
|
|
|
|
}
|
1998-02-10 20:08:16 +00:00
|
|
|
#endif
|
1998-02-06 02:29:22 +00:00
|
|
|
|
1998-02-14 05:34:08 +00:00
|
|
|
void getoption(VectorUnitState* state)
|
1998-02-06 02:29:22 +00:00
|
|
|
{
|
1998-02-10 20:08:16 +00:00
|
|
|
#if 0
|
1998-02-06 02:29:22 +00:00
|
|
|
int startline = 0;
|
|
|
|
int count = 1;
|
1998-02-10 20:08:16 +00:00
|
|
|
#endif
|
1998-02-06 02:29:22 +00:00
|
|
|
|
1998-02-14 05:34:08 +00:00
|
|
|
state->junk._is_dbg = 1;
|
|
|
|
state->junk._vpepc = 0;
|
|
|
|
state->junk._is_verb = 0;
|
|
|
|
state->junk._is_dump = 0;
|
|
|
|
state->junk._pgpuif = 2;
|
|
|
|
state->junk._ITOP = 20;
|
|
|
|
state->junk._TOP = 10;
|
1998-02-06 02:29:22 +00:00
|
|
|
|
|
|
|
#if 0
|
|
|
|
while(argc - count){
|
|
|
|
if(argv[count][0] == '-'){
|
|
|
|
switch(argv[count][1]){
|
|
|
|
case 'i':
|
|
|
|
strcpy(ifilename, argv[count+1]);
|
|
|
|
count += 2;
|
|
|
|
break;
|
|
|
|
case 'o':
|
|
|
|
strcpy(ofilename, argv[count+1]);
|
|
|
|
count += 2;
|
|
|
|
break;
|
|
|
|
case 't':
|
|
|
|
strcpy(pfilename, argv[count+1]);
|
|
|
|
count += 2;
|
|
|
|
break;
|
|
|
|
case 's':
|
|
|
|
sscanf(argv[count+1], "%d", &startline);
|
1998-02-14 05:34:08 +00:00
|
|
|
state->junk._vpepc = startline;
|
1998-02-06 02:29:22 +00:00
|
|
|
count += 2;
|
|
|
|
break;
|
|
|
|
case 'd':
|
1998-02-14 05:34:08 +00:00
|
|
|
state->junk._is_dbg = 1;
|
1998-02-06 02:29:22 +00:00
|
|
|
count += 1;
|
|
|
|
break;
|
|
|
|
case 'v':
|
1998-02-14 05:34:08 +00:00
|
|
|
state->junk._is_verb = 1;
|
1998-02-06 02:29:22 +00:00
|
|
|
count += 1;
|
|
|
|
break;
|
|
|
|
case 'p':
|
1998-02-14 05:34:08 +00:00
|
|
|
state->junk._is_dump = 1;
|
1998-02-06 02:29:22 +00:00
|
|
|
count += 1;
|
|
|
|
break;
|
|
|
|
case 'h':
|
|
|
|
case '?':
|
|
|
|
Usage();
|
|
|
|
exit(1);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
Usage();
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}else{
|
|
|
|
Usage();
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|