wok-current rev 17124

runcom: misc enhancements
author Pascal Bellard <pascal.bellard@slitaz.org>
date Sun Sep 07 21:21:07 2014 +0200 (2014-09-07)
parents 77c872ed3f55
children 984910ac3270
files runcom/stuff/runcom.c
line diff
     1.1 --- a/runcom/stuff/runcom.c	Sat Sep 06 22:48:52 2014 +0300
     1.2 +++ b/runcom/stuff/runcom.c	Sun Sep 07 21:21:07 2014 +0200
     1.3 @@ -13,10 +13,13 @@
     1.4  #include <sys/time.h>
     1.5  #include <sys/types.h>
     1.6  #include <sys/stat.h>
     1.7 +#include <sys/statfs.h>
     1.8  #include <signal.h>
     1.9  #include <errno.h>
    1.10  #include <ctype.h>
    1.11  #include <termios.h>
    1.12 +#include <dirent.h>
    1.13 +#include <fnmatch.h>
    1.14  
    1.15  #include <sys/syscall.h>
    1.16  #include <asm/vm86.h>
    1.17 @@ -87,6 +90,9 @@
    1.18      fprintf(stderr,"\n");
    1.19  }
    1.20  
    1.21 +static int argflags;
    1.22 +#define DEBUG 1
    1.23 +
    1.24  #define DOS_FD_MAX 256
    1.25  typedef struct {
    1.26      int fd; /* -1 means closed */
    1.27 @@ -94,6 +100,8 @@
    1.28  
    1.29  DOSFile dos_files[DOS_FD_MAX];
    1.30  uint16_t cur_psp;
    1.31 +uint16_t cur_dta_seg;
    1.32 +uint16_t cur_dta_ofs;
    1.33  
    1.34  void dos_init(void)
    1.35  {
    1.36 @@ -155,6 +163,8 @@
    1.37          offset++;
    1.38      }
    1.39      *q = '\0';
    1.40 +    if (buf[1] == ':')
    1.41 +    	strcpy(buf, buf+2);
    1.42      return buf;
    1.43  } 
    1.44  
    1.45 @@ -163,6 +173,15 @@
    1.46      return get_filename1(r, buf, buf_size, r->ds, r->edx & 0xffff);
    1.47  }
    1.48  
    1.49 +static char *upcase(const char *s)
    1.50 +{
    1.51 +    static char buffer[80];
    1.52 +    int i;
    1.53 +    for (i = 0; i < sizeof(buffer)-1; i++, s++)
    1.54 +	buffer[i] = (*s >= 'a' && *s <= 'z') ? *s + 'A' - 'a' : *s;
    1.55 +    return buffer;
    1.56 +}
    1.57 +
    1.58  typedef struct __attribute__((packed)) {
    1.59      uint8_t drive_num;
    1.60      uint8_t file_name[8];
    1.61 @@ -187,6 +206,22 @@
    1.62      uint16_t ip, cs;
    1.63  } ExecParamBlock;
    1.64  
    1.65 +typedef struct __attribute__((packed)) {
    1.66 +    /* internals */
    1.67 +	uint8_t attr;		/* 00 */
    1.68 +	uint8_t drive_letter;	/* 01 */
    1.69 +	uint8_t template[11];	/* 02 */
    1.70 +	uint16_t entry_count;	/* 0D */
    1.71 +	uint32_t dta_address;	/* 0F */
    1.72 +	uint16_t cluster_parent_dir;	/* 13 */
    1.73 +    /* output */
    1.74 +	uint8_t attr_found;	/* 15 */
    1.75 +	uint16_t file_time;	/* 16 */
    1.76 +	uint16_t file_date;	/* 18 */
    1.77 +	uint32_t file_size;	/* 1A */
    1.78 +	uint8_t filename[13];	/* 1E */
    1.79 +} dirdta;
    1.80 +
    1.81  typedef struct MemBlock {
    1.82      struct MemBlock *next;
    1.83      uint16_t seg;
    1.84 @@ -340,7 +375,8 @@
    1.85      header.min_extra_paragraphs += (size-1)/16;
    1.86      
    1.87      /* address of last segment allocated */
    1.88 -    *(uint16_t *)seg_to_linear(psp, 2) = psp + header.min_extra_paragraphs;
    1.89 +    //*(uint16_t *)seg_to_linear(psp, 2) = psp + header.min_extra_paragraphs;
    1.90 +    *(uint16_t *)seg_to_linear(psp, 2) = 0x9fff;
    1.91      
    1.92      if (pfile_size)
    1.93          *pfile_size = size;
    1.94 @@ -378,6 +414,8 @@
    1.95  
    1.96      /* load the MSDOS .com executable */
    1.97      fd = open(filename, O_RDONLY);
    1.98 +    if (fd < 0)
    1.99 +        fd = open(upcase(filename), O_RDONLY);
   1.100      if (fd < 0) {
   1.101          return -1;
   1.102      }
   1.103 @@ -394,10 +432,10 @@
   1.104      /* reset the PSP */
   1.105      memset(seg_to_linear(psp, 0), 0, 0x100);
   1.106  
   1.107 -    *seg_to_linear(psp, 0) = 0xcd; /* int $0x20 */
   1.108 -    *seg_to_linear(psp, 1) = 0x20;
   1.109 +    * (uint16_t *) seg_to_linear(psp, 0) = 0x20CD; /* int $0x20 */
   1.110      /* address of last segment allocated */
   1.111 -    *(uint16_t *)seg_to_linear(psp, 2) = psp + 0xfff;
   1.112 +    //*(uint16_t *)seg_to_linear(psp, 2) = psp + 0xfff;
   1.113 +    *(uint16_t *)seg_to_linear(psp, 2) = 0x9fff;
   1.114  
   1.115      if (argc) {
   1.116          int i, p;
   1.117 @@ -499,18 +537,98 @@
   1.118      }
   1.119  }
   1.120  
   1.121 +#define ESC "\033"
   1.122  void do_int10(struct vm86_regs *r)
   1.123  {
   1.124      uint8_t ah;
   1.125 +    char buf[20];
   1.126 +    static unsigned cursorlines = 0x0607;
   1.127 +    static unsigned activepage = 0;
   1.128 +    static uint8_t cursrow, curscol;
   1.129      
   1.130      ah = (r->eax >> 8);
   1.131      switch(ah) {
   1.132 +    case 0x02: /* set cursor position (BH == page number) */
   1.133 +        cursrow = r->edx >> 8;
   1.134 +        curscol = r->edx;
   1.135 +        * (uint16_t *) seg_to_linear(0x40, 0x50 + 2*((r->ebx >> 8) & 0xFF)) = r->edx;
   1.136 +	sprintf(buf,ESC"[%u;%uH",cursrow + 1, curscol + 1);
   1.137 +	write(1, buf, strlen(buf));
   1.138 +        break;
   1.139 +    case 0x03: /* get cursor position (BH == page number) */
   1.140 +	r->eax = 0;
   1.141 +	r->ecx = cursorlines;
   1.142 +	r->edx &= ~0xFFFF;
   1.143 +        r->edx |= * (uint16_t *) seg_to_linear(0x40, 0x50 + 2*((r->ebx >> 8) & 0xFF));
   1.144 +	sprintf(buf,ESC"[%u;%uH",cursrow + 1, curscol + 1);
   1.145 +	write(1, buf, strlen(buf));
   1.146 +        break;
   1.147 +    case 0x05: /* set active page */
   1.148 +	activepage = r->eax & 0xFF;
   1.149 +        break;
   1.150 +    case 0x06: /* scroll up */
   1.151 +    case 0x07: /* scroll down */
   1.152 +	{
   1.153 +	    int i = r->eax & 0xFF;
   1.154 +	    if (i == 0) i = 50;
   1.155 +    	/* FIXME assume full row, ignore colums in CL, DL */
   1.156 +	    sprintf(buf,ESC"[%u;%ur",1+(r->ecx >> 8) & 0xFF, 1+(r->edx >> 8) & 0xFF);
   1.157 +	    write(1, buf, strlen(buf));
   1.158 +	    buf[2] = (ah != 6) ? 'T' : 'S';
   1.159 +	    while (i--) write(1,buf,3);
   1.160 +	}
   1.161 +        break;
   1.162 +    case 0x09: /* write char and attribute at cursor position (BH == page number) */
   1.163 +        {
   1.164 +            static char color[8] = "04261537";
   1.165 +            char extra[5], *s = extra;
   1.166 +            uint8_t c = r->eax;
   1.167 +            uint16_t n = r->ecx;
   1.168 +            int i;
   1.169 +
   1.170 +	    if (r->ebx & 0x8) { *s++ = '1'; *s++ = ';'; } // bold
   1.171 +	    if (r->ebx & 0x80) { *s++ = '5'; *s++ = ';'; } // blink
   1.172 +	    *s = 0;
   1.173 +	    sprintf(buf,ESC"[0;%s4%c;3%cm",extra,
   1.174 +	    	   color[(r->ebx & 0x70) >> 4],color[r->ebx & 0x7]);
   1.175 +	    write(1, buf, strlen(buf));
   1.176 +            for (i = 0; i < n; i++) 
   1.177 +		write(1, &c, 1);
   1.178 +	    write(1, ESC"[0m", 4); /* restore attributes */
   1.179 +        }
   1.180 +        break;
   1.181      case 0x0E: /* write char */
   1.182          {
   1.183              uint8_t c = r->eax;
   1.184              write(1, &c, 1);
   1.185          }
   1.186          break;
   1.187 +    case 0x0F: /* get current video mode */
   1.188 +        {
   1.189 +	    r->eax &= ~0xFFFF;
   1.190 +	    r->eax |= 0x5003; /* color or 5007 mono */
   1.191 +	    r->ebx &= ~0xFF00;
   1.192 +	    r->ebx |= activepage << 8;
   1.193 +        }
   1.194 +        break;
   1.195 +    case 0x11: /* get window coordonates  */
   1.196 +        r->ecx &= ~0xFFFF;
   1.197 +        r->edx &= ~0xFFFF;
   1.198 +        r->edx |= ~0x1950; /* 80x25 */
   1.199 +        break;
   1.200 +    case 0x12: /* get blanking attribute (for scroll) */
   1.201 +        r->ebx &= ~0xFF00;
   1.202 +        break;
   1.203 +    case 0x1A: /* get display combination code */
   1.204 +#if 0
   1.205 +        set_error(r, 1);
   1.206 +#else
   1.207 +        r->eax &= ~0xFF;
   1.208 +        r->eax |= ~0x1A;
   1.209 +        r->ebx &= ~0xFFFF;
   1.210 +        r->ebx |= ~0x0202; // CGA + color display
   1.211 +#endif
   1.212 +        break;
   1.213      default:
   1.214          unsupported_function(r, 0x10, ah);
   1.215      }
   1.216 @@ -684,6 +802,8 @@
   1.217  void do_int21(struct vm86_regs *r)
   1.218  {
   1.219      uint8_t ah;
   1.220 +    DIR *dirp;
   1.221 +    dirdta *dta;
   1.222      
   1.223      ah = (r->eax >> 8);
   1.224      switch(ah) {
   1.225 @@ -695,6 +815,11 @@
   1.226              write(1, &c, 1);
   1.227          }
   1.228          break;
   1.229 +    case 0x08: /* read stdin */
   1.230 +        {
   1.231 +            read(0,&r->eax,1);
   1.232 +        }
   1.233 +        break;
   1.234      case 0x09: /* write string */
   1.235          {
   1.236              uint8_t c;
   1.237 @@ -738,6 +863,34 @@
   1.238              *seg_to_linear(r->ds, off + 2 + cur_len) = '\r';
   1.239          }
   1.240          break;
   1.241 +    case 0x0b: /* get stdin status */
   1.242 +        {
   1.243 +	    r->eax &= ~0xFF; /* no character available */
   1.244 +        }
   1.245 +        break;
   1.246 +    case 0x0d: /* disk reset */
   1.247 +        {
   1.248 +            sync();
   1.249 +        }
   1.250 +        break;
   1.251 +    case 0x0e: /* select default disk */
   1.252 +        {
   1.253 +            r->eax &= ~0xFF;
   1.254 +            r->eax |= 3; /* A: B: & C: valid */
   1.255 +        }
   1.256 +        break;
   1.257 +    case 0x19: /* get current default drive */
   1.258 +        {
   1.259 +	    r->eax &= ~0xFF;
   1.260 +	    r->eax |= 2; /* C: */
   1.261 +        }
   1.262 +        break;
   1.263 +    case 0x1a: /* set DTA (disk transfert address) */
   1.264 +        {
   1.265 +	    cur_dta_seg = r->ds;
   1.266 +	    cur_dta_ofs = r->edx;
   1.267 +        }
   1.268 +        break;
   1.269      case 0x25: /* set interrupt vector */
   1.270          {
   1.271              uint16_t *ptr;
   1.272 @@ -851,6 +1004,12 @@
   1.273              r->edx = (tim.tv_sec * 256) + tim.tv_usec/10000;
   1.274          }
   1.275          break;
   1.276 +    case 0x2f: /* get DTA (disk transfert address */
   1.277 +        {
   1.278 +	    r->es = cur_dta_seg;
   1.279 +	    r->ebx = cur_dta_ofs;
   1.280 +        }
   1.281 +        break;
   1.282      case 0x30: /* get dos version */
   1.283          {
   1.284              int major, minor, serial, oem;
   1.285 @@ -864,6 +1023,11 @@
   1.286              r->ebx = (r->ebx & ~0xffff) | (serial & 0xff) | (0x66 << 8);
   1.287          }
   1.288          break;
   1.289 +    case 0x33: /* extended break checking */
   1.290 +        {
   1.291 +	    r->edx &= ~0xFFFF;
   1.292 +        }
   1.293 +        break;
   1.294      case 0x35: /* get interrupt vector */
   1.295          {
   1.296              uint16_t *ptr;
   1.297 @@ -872,6 +1036,25 @@
   1.298              r->es = ptr[1];
   1.299          }
   1.300          break;
   1.301 +    case 0x36: /* get free disk space */
   1.302 +        {
   1.303 +            struct statfs buf;
   1.304 +            
   1.305 +            if (statfs(".", &buf)) {
   1.306 +		r->eax |= 0xFFFF;
   1.307 +            }
   1.308 +            else {
   1.309 +		r->eax &= ~0xFFFF;
   1.310 +		r->eax |= buf.f_bsize / 512; /* sectors per cluster */
   1.311 +		r->ebx &= ~0xFFFF;
   1.312 +		r->ebx |= buf.f_bavail;
   1.313 +		r->ecx &= ~0xFFFF;
   1.314 +		r->ecx |= 512; /* bytes per sector */
   1.315 +		r->edx &= ~0xFFFF;
   1.316 +		r->edx |= buf.f_blocks;
   1.317 +            }
   1.318 +        }
   1.319 +        break;
   1.320      case 0x37: 
   1.321          {
   1.322              switch(r->eax & 0xff) {
   1.323 @@ -884,6 +1067,15 @@
   1.324              }
   1.325          }
   1.326          break;
   1.327 +    case 0x3B: 
   1.328 +        {
   1.329 +            char filename[1024];
   1.330 +        
   1.331 +            get_filename(r, filename, sizeof(filename));
   1.332 +            if (chdir(filename))
   1.333 +                set_error(r, 0x03); /* path not found */
   1.334 +        }
   1.335 +        break;
   1.336      case 0x3c: /* create or truncate file */
   1.337          {
   1.338              char filename[1024];
   1.339 @@ -899,6 +1091,8 @@
   1.340                  else
   1.341                      flags = 0777;
   1.342                  fd = open(filename, O_RDWR | O_TRUNC | O_CREAT, flags);
   1.343 +                if (fd < 0)
   1.344 +                    fd = open(upcase(filename), O_RDWR | O_TRUNC | O_CREAT, flags);
   1.345  #ifdef DUMP_INT21
   1.346                  printf("int21: create: file='%s' cx=0x%04x ret=%d\n", 
   1.347                         filename, (int)(r->ecx & 0xffff), h);
   1.348 @@ -923,11 +1117,9 @@
   1.349                  set_error(r, 0x04); /* too many open files */
   1.350              } else {
   1.351                  get_filename(r, filename, sizeof(filename));
   1.352 -#ifdef DUMP_INT21
   1.353 -                printf("int21: open: file='%s' al=0x%02x ret=%d\n", 
   1.354 -                       filename, (int)(r->eax & 0xff), h);
   1.355 -#endif
   1.356                  fd = open(filename, r->eax & 3);
   1.357 +                if (fd < 1)
   1.358 +                    fd = open(upcase(filename), r->eax & 3);
   1.359                  if (fd < 0) {
   1.360                      set_error(r, 0x02); /* file not found */
   1.361                  } else {
   1.362 @@ -1031,7 +1223,7 @@
   1.363          {
   1.364              char filename[1024];
   1.365              get_filename(r, filename, sizeof(filename));
   1.366 -            if (unlink(filename) < 0) {
   1.367 +            if (unlink(filename) < 0 && unlink(upcase(filename))) {
   1.368                  set_error(r, 0x02); /* file not found */
   1.369              } else {
   1.370                  set_error(r, 0);
   1.371 @@ -1062,6 +1254,20 @@
   1.372              }
   1.373          }
   1.374          break;
   1.375 +    case 0x43: /* get attribute */
   1.376 +        {
   1.377 +	    struct stat statbuf;
   1.378 +            char filename[1024];
   1.379 +            get_filename(r, filename, sizeof(filename));
   1.380 +            if (stat(filename, &statbuf) && stat(upcase(filename), &statbuf)) {
   1.381 +		set_error(r, 5);
   1.382 +            }
   1.383 +            else {
   1.384 +            	r->ecx &= ~0xFFFF;
   1.385 +		if (S_ISDIR(statbuf.st_mode)) r->ecx |= 0x10;
   1.386 +            }
   1.387 +        }
   1.388 +        break;
   1.389      case 0x44: /* ioctl */
   1.390          switch(r->eax & 0xff) {
   1.391          case 0x00: /* get device information */
   1.392 @@ -1084,11 +1290,23 @@
   1.393                      set_error(r, 0);
   1.394                  }
   1.395              }
   1.396 +        case 0x01: /* set device information */
   1.397              break;
   1.398          default:
   1.399              goto unsupported;
   1.400          }
   1.401          break;
   1.402 +    case 0x47: /* get current directory (DL drive)*/
   1.403 +        {
   1.404 +            char *s = seg_to_linear(r->ds, r->esi);
   1.405 +            getcwd(s,64);
   1.406 +            strcpy(s,s+1);
   1.407 +            while (*s)
   1.408 +        	if (*s++ == '/')
   1.409 +        	    s[-1] = '\\';
   1.410 +            r->eax = 0x100;
   1.411 +        }
   1.412 +        break;
   1.413      case 0x48: /* allocate memory */
   1.414          {
   1.415              int ret, max_size;
   1.416 @@ -1144,10 +1362,13 @@
   1.417              get_filename(r, filename, sizeof(filename));
   1.418              blk = (ExecParamBlock *)seg_to_linear(r->es, r->ebx);
   1.419              ret = load_com(blk, filename, NULL, 0, NULL);
   1.420 +            if (ret < 0)
   1.421 +                ret = load_com(blk, upcase(filename), NULL, 0, NULL);
   1.422              if (ret < 0) {
   1.423                  set_error(r, 0x02); /* file not found */
   1.424              } else {
   1.425 -                cur_psp = ret;
   1.426 +                cur_dta_seg = cur_psp = ret;
   1.427 +                cur_dta_ofs = 0x80;
   1.428                  set_error(r, 0);
   1.429              }
   1.430          }
   1.431 @@ -1155,6 +1376,53 @@
   1.432      case 0x4c: /* exit with return code */
   1.433          exit(r->eax & 0xff);
   1.434          break;
   1.435 +    case 0x4e: /* find first matching file */
   1.436 +// TODO AL input support
   1.437 +	dirp = opendir(".");
   1.438 +	if (dirp == NULL) {
   1.439 +		set_error(r, (errno == ENOTDIR) ? 0x03 /* path not found */
   1.440 +						: 0x02 /* file not found */ );
   1.441 +		goto pattern_found;		
   1.442 +	}
   1.443 +	else {
   1.444 +		struct dirent *dp;
   1.445 +		char *s;
   1.446 +		
   1.447 +		dta = (dirdta *) seg_to_linear(cur_dta_seg, cur_dta_ofs);
   1.448 +		dta->attr = r->ecx;
   1.449 +    		* (DIR **) &dta->entry_count = dirp;
   1.450 +    		s = seg_to_linear(r->ds, r->edx);
   1.451 +    		if (s[1] == ':') s+= 2;
   1.452 +    		if (s[0] == '\\') s++;
   1.453 +		strncpy(dta->template, s, 11);
   1.454 +        // NO break;
   1.455 +    case 0x4f: /* find next matching file */
   1.456 +		dta = (dirdta *) seg_to_linear(cur_dta_seg, cur_dta_ofs);
   1.457 +    		dirp = * (DIR **) &dta->entry_count;
   1.458 +		while ((dp = readdir(dirp)) != NULL) {
   1.459 +			if (!fnmatch(dta->template, dp->d_name, 0)) {
   1.460 +				struct stat statbuf;
   1.461 +				
   1.462 +				r->eflags &= ~CF_MASK;
   1.463 +				strncpy(dta->filename, dp->d_name, 13);
   1.464 +				stat(dp->d_name, &statbuf);
   1.465 +				dta->file_size = statbuf.st_size;
   1.466 +				dta->file_date = 0; //DOSDATE(statbuf.st_mtime);
   1.467 +				dta->file_time = 0; //DOSIME(statbuf.st_mtime);
   1.468 +				dta->attr_found = S_ISDIR(statbuf.st_mode) ?
   1.469 +					0x10 /*aDvshr*/ : 0x20 /*Advshr*/;
   1.470 +#if 0
   1.471 +				if ((dta->attr_found ^ dta->attr) & 0x16)
   1.472 +					continue;
   1.473 +#endif
   1.474 +				goto pattern_found;		
   1.475 +			}
   1.476 +		}
   1.477 +	}
   1.478 +	closedir(dirp);
   1.479 +	set_error(r, 0x12 /* no more files */);
   1.480 +pattern_found:
   1.481 +        break;
   1.482      case 0x50: /* set PSP address */
   1.483  #ifdef DUMP_INT21
   1.484          printf("int21: set PSP: 0x%04x\n", (uint16_t)r->ebx);
   1.485 @@ -1182,6 +1450,11 @@
   1.486              r->eax = (r->eax & ~0xff);
   1.487          }
   1.488          break;
   1.489 +    case 0x56: /* rename file (CL attribute mask) */
   1.490 +        if (rename((char *) seg_to_linear(r->ds, r->edx),
   1.491 +	           (char *) seg_to_linear(r->es, r->edi)))
   1.492 +	    set_error(r, 0x5 /* access denied or 2,3,0x11 */);
   1.493 +        break;
   1.494      default:
   1.495      unsupported:
   1.496          unsupported_function(r, 0x21, ah);
   1.497 @@ -1194,26 +1467,45 @@
   1.498      write(1, &c, 1);
   1.499  }
   1.500  
   1.501 +static int int8pending;
   1.502 +
   1.503  void raise_interrupt(int number)
   1.504  {
   1.505      if (* (uint32_t *) seg_to_linear(0, number * 4) == 0)
   1.506          return;
   1.507 -    // FIXME VM86_SIGNAL
   1.508 +    int8pending++;
   1.509  }
   1.510  
   1.511  void biosclock()
   1.512  {
   1.513 -    uint32_t *timer = (uint32_t *) seg_to_linear(0, 0x46C);
   1.514 -    ++*timer;
   1.515 +    //uint32_t *timer = (uint32_t *) seg_to_linear(0, 0x46C);
   1.516 +    //++*timer;
   1.517      raise_interrupt(8);
   1.518 -    raise_interrupt(0x1C);
   1.519 +    //raise_interrupt(0x1C);
   1.520 +}
   1.521 +
   1.522 +static void exec_int(struct vm86_regs *r, unsigned num)
   1.523 +{
   1.524 +    uint16_t *int_vector;
   1.525 +    uint32_t eflags;
   1.526 +
   1.527 +    eflags = r->eflags & ~IF_MASK;
   1.528 +    if (r->eflags & VIF_MASK)
   1.529 +        eflags |= IF_MASK;
   1.530 +    pushw(r, eflags);
   1.531 +    pushw(r, r->cs);
   1.532 +    pushw(r, r->eip);
   1.533 +    int_vector = (uint16_t *)seg_to_linear(0, num * 4);
   1.534 +    r->eip = int_vector[0];
   1.535 +    r->cs = int_vector[1];
   1.536 +    r->eflags &= ~(VIF_MASK | TF_MASK | AC_MASK);
   1.537  }
   1.538  
   1.539  int main(int argc, char **argv)
   1.540  {
   1.541      uint8_t *vm86_mem;
   1.542      const char *filename;
   1.543 -    int ret;
   1.544 +    int i, ret;
   1.545      uint32_t file_size;
   1.546      struct sigaction sa; 
   1.547      struct itimerval timerval;
   1.548 @@ -1221,6 +1513,16 @@
   1.549      struct vm86_regs *r;
   1.550      ExecParamBlock blk1, *blk = &blk1;
   1.551  
   1.552 +    for (argflags = 0; *argv[1] == '-'; argv++) {
   1.553 +    	char *s = argv[1];
   1.554 +
   1.555 +    	while (1)
   1.556 +    	switch (*++s) {
   1.557 +    	case 'd' : argflags |= DEBUG; break;
   1.558 +    	case 0 : goto nextargv;
   1.559 +    	}
   1.560 +nextargv:;
   1.561 +    }
   1.562      if (argc < 2)
   1.563          usage();
   1.564      filename = argv[1];
   1.565 @@ -1253,7 +1555,8 @@
   1.566              perror(filename);
   1.567              exit(1);
   1.568          }
   1.569 -        cur_psp = ret;
   1.570 +        cur_dta_seg = cur_psp = ret;
   1.571 +        cur_dta_ofs = 0x80;
   1.572  
   1.573          /* init basic registers */
   1.574          r->eip = blk->ip;
   1.575 @@ -1291,6 +1594,13 @@
   1.576          timerval.it_interval.tv_usec = timerval.it_value.tv_usec = 10000000 / 182;
   1.577          setitimer (ITIMER_REAL, &timerval, NULL);
   1.578      }
   1.579 +    *(uint8_t *)seg_to_linear(0xF000, 0) = 0xCF;
   1.580 +    for (i = 0; i < 16; i++) 
   1.581 +	*(uint32_t *)seg_to_linear(0, i * 4) = 0xF0000000;
   1.582 +    *(uint32_t *)seg_to_linear(0, 0x18 * 4) = 0xF0000000; /* Basic */
   1.583 +    *(uint32_t *)seg_to_linear(0, 0x1B * 4) = 0xF0000000; /* Keyboard Ctrl-Break */
   1.584 +    *(uint32_t *)seg_to_linear(0, 0x23 * 4) = 0xF0000000; /* DOS Ctrl-Break */
   1.585 +    *(uint32_t *)seg_to_linear(0, 0x24 * 4) = 0xF0000000; /* Critical error */
   1.586  
   1.587      for(;;) {
   1.588          ret = vm86(VM86_ENTER, &ctx);
   1.589 @@ -1300,6 +1610,9 @@
   1.590                  int int_num;
   1.591  
   1.592                  int_num = VM86_ARG(ret);
   1.593 +		if (argflags & 1)
   1.594 +			fprintf(stderr,"Int%02X: CS:IP=%04X:%04X AX=%04X\n",
   1.595 +				int_num, r->cs, r->eip, r->eax);
   1.596                  switch(int_num) {
   1.597                  case 0x10:
   1.598                      do_int10(r);
   1.599 @@ -1334,27 +1647,32 @@
   1.600              break;
   1.601          case VM86_SIGNAL:
   1.602              /* a signal came, we just ignore that */
   1.603 +            if (int8pending) {
   1.604 +		int8pending--;
   1.605 +		exec_int(r, 8);
   1.606 +            }
   1.607              break;
   1.608          case VM86_STI:
   1.609              break;
   1.610          case VM86_TRAP:
   1.611              /* just executes the interruption */
   1.612 -            {
   1.613 -                uint16_t *int_vector;
   1.614 -                uint32_t eflags;
   1.615 -                
   1.616 -                eflags = r->eflags & ~IF_MASK;
   1.617 -                if (r->eflags & VIF_MASK)
   1.618 -                    eflags |= IF_MASK;
   1.619 -                pushw(r, eflags);
   1.620 -                pushw(r, r->cs);
   1.621 -                pushw(r, r->eip);
   1.622 -                int_vector = (uint16_t *)seg_to_linear(0, VM86_ARG(ret) * 4);
   1.623 -                r->eip = int_vector[0];
   1.624 -                r->cs = int_vector[1];
   1.625 -                r->eflags &= ~(VIF_MASK | TF_MASK | AC_MASK);
   1.626 +            exec_int(r, VM86_ARG(ret));
   1.627 +            break;
   1.628 +        case VM86_UNKNOWN:
   1.629 +            switch ( *(uint8_t *)seg_to_linear(r->cs, r->eip) ) {
   1.630 +            case 0xE4: /* inx portb,al */
   1.631 +            case 0xE5: /* in portb,ax */
   1.632 +            case 0xE6: /* out al,portb */
   1.633 +            case 0xE7: /* out ax,portb */
   1.634 +            	r->eip += 2;
   1.635 +            	continue;
   1.636 +            case 0xEC: /* in dx,al */
   1.637 +            case 0xED: /* in dx,ax */
   1.638 +            case 0xEE: /* out al,dx */
   1.639 +            case 0xEF: /* out ax,dx */
   1.640 +            	r->eip++;
   1.641 +            	continue;
   1.642              }
   1.643 -            break;
   1.644          default:
   1.645              fprintf(stderr, "unhandled vm86 return code (0x%x)\n", ret);
   1.646              dump_regs(&ctx.regs);