Source file: /~heha/hs/borg.zip/DISIO.CPP

/************************************************************************
*		 disio.cpp						*
* This is a fairly big class, and yet it was originally part of the	*
* larger disasm class. I finally got around to splitting off some of	*
* 3500 lines of code from that class into here. This represents the	*
* disasm I/O functions for window I/O and file I/O. Version 2.11 was	*
* when this was split. There are still a huge number of confusing	*
* functions and code in here which probably needs more work to clean it *
* up, and better split/define the classes. Some of the functions are	*
* friends of disasm because it has been difficult to split the classes	*
* effectively and both access the complex main disassembly structures	*
************************************************************************/

#include <windows.h>
#include <stdio.h>	      // for sprintf, for float printing

#include "disio.h"
#include "disasm.h"
#include "data.h"
#include "dasm.h"
#include "schedule.h"
#include "gname.h"
#include "xref.h"
#include "range.h"
#include "debug.h"

/************************************************************************
* globals								*
* - actually some constants used in file i/o as a header		*
************************************************************************/
char hdr[] =";\t      Created by Borg Disassembler\r\n";
char hdr2[]=";\t      written by Cronos + h#s\r\n";
disio *cvd;	// pointer to data of focused view

/************************************************************************
* constructor function							*
* - this just enables window updates and sets the subline to null	*
* - the subline (subitem) is basically the line in the disassembly that *
*   you see for any given loc. As a loc may refer to many lines (like	*
*   segheader, comments, xref, actual instruction are all separate	*
*   line, and subline says which of these it is)			*
************************************************************************/
disio::disio(HWND w) {
 wnd=w;
 subitem=dsmnull;
// bufferbusy=false;   
}

/************************************************************************
* destructor function							*
* - currently null							*
* - note that disio has no particular structs the same way most of the	*
*   other classes in Borg do						*
************************************************************************/
disio::~disio()
{
}

/************************************************************************
* argoverdec								*
* - disio also acts as a go between for the user and the disasm engine	*
* - here we translate the users current line of the disassembly and	*
*   their request for a decimal override into a call to the disasm	*
*   engine to add the override to the instruction that is there		*
* - note that these kind of calls come from the scheduler and are part	*
*   of the secondary thread						*
************************************************************************/
void disio::argoverdec(void)
{ lptr outhere;
  findcurrentaddr(&outhere);
  dsm->disargoverdec(outhere);
}

/************************************************************************
* argoversingle								*
* - very similar to argoverdec, for a single (float) override		*
************************************************************************/
void disio::argoversingle(void)
{ lptr outhere;
  findcurrentaddr(&outhere);
  dsm->disargoversingle(outhere);
}

/************************************************************************
* argovernegate								*
* - very similar to argoverdec, for an argument negation override	*
************************************************************************/
void disio::arg_negate(void)
{ lptr outhere;
  findcurrentaddr(&outhere);
  dsm->disargnegate(outhere);
}

/************************************************************************
* argoverhex								*
* - very similar to argoverdec, for a hex override			*
************************************************************************/
void disio::argoverhex(void)
{ lptr outhere;
  findcurrentaddr(&outhere);
  dsm->disargoverhex(outhere);
}

/************************************************************************
* argoveroffsetdseg							*
* - very similar to argoverdec, for a dseg override			*
************************************************************************/
void disio::argoveroffsetdseg(void)
{ lptr outhere;
  findcurrentaddr(&outhere);
  dsm->disargoveroffsetdseg(outhere);
}

/************************************************************************
* argoverchar								*
* - very similar to argoverdec, for a char override			*
************************************************************************/
void disio::argoverchar(void)
{ lptr outhere;
  findcurrentaddr(&outhere);
  dsm->disargoverchar(outhere);
}

/************************************************************************
* makedword								*
* - disio acts as a go between for the user and the disasm engine	*
* - here we translate the users current line of the disassembly and	*
*   their request for a dword into a call to the disasm engine to	*
*   disassemble a dword at the current point				*
* - note that these kind of calls come from the scheduler and are part	*
*   of the secondary thread						*
************************************************************************/
void disio::makedword(void)
{ lptr outhere;
  findcurrentaddr(&outhere);
  if(outhere==curraddr)
    subitem=dsmnull;
  dsm->disdatadword(outhere);
}

/************************************************************************
* makesingle								*
* - very similar to makedword, this time to disassemble a single	*
*   (float)								*
************************************************************************/
void disio::makesingle(void) {
 lptr outhere;
 findcurrentaddr(&outhere);
 if (outhere==curraddr) subitem=dsmnull;
 dsm->disdatasingle(outhere);
}

/************************************************************************
* makedouble								*
* - very similar to makedword, this time to disassemble a double	*
*   (double float)							*
************************************************************************/
void disio::makedouble(void)
{ lptr outhere;
  findcurrentaddr(&outhere);
  if(outhere==curraddr)
    subitem=dsmnull;
  dsm->disdatadouble(outhere);
}

/************************************************************************
* makelongdouble							*
* - very similar to makedword, this time to disassemble a long double	*
*   (long double)							*
************************************************************************/
void disio::makelongdouble(void)
{ lptr outhere;
  findcurrentaddr(&outhere);
  if(outhere==curraddr)
    subitem=dsmnull;
  dsm->disdatalongdouble(outhere);
}

/************************************************************************
* makeword								*
* - very similar to makedword, this time to disassemble a word		*
************************************************************************/
void disio::makeword(void)
{ lptr outhere;
  findcurrentaddr(&outhere);
  if(outhere==curraddr)
    subitem=dsmnull;
  dsm->disdataword(outhere);
}

/************************************************************************
* makestring								*
* - very similar to makedword, this time to disassemble a string	*
*   (standard C string)							*
************************************************************************/
void disio::makestring(void)
{ lptr outhere;
  findcurrentaddr(&outhere);
  if(outhere==curraddr)
    subitem=dsmnull;
  dsm->disdatastring(outhere);
}

/************************************************************************
* pascalstring								*
* - very similar to makedword, this time to disassemble a string	*
*   (standard pascal string)						*
************************************************************************/
void disio::pascalstring(void)
{ lptr outhere;
  findcurrentaddr(&outhere);
  if(outhere==curraddr)
    subitem=dsmnull;
  dsm->disdatapstring(outhere);
}

/************************************************************************
* ucstring								*
* - very similar to makedword, this time to disassemble a string	*
*   (unicode C string)							*
************************************************************************/
void disio::ucstring(void)
{ lptr outhere;
  findcurrentaddr(&outhere);
  if(outhere==curraddr)
    subitem=dsmnull;
  dsm->disdataucstring(outhere);
}

/************************************************************************
* upstring								*
* - very similar to makedword, this time to disassemble a string	*
*   (unicode pascal string)						*
************************************************************************/
void disio::upstring(void)
{ lptr outhere;
  findcurrentaddr(&outhere);
  if(outhere==curraddr)
    subitem=dsmnull;
  dsm->disdataupstring(outhere);
}

/************************************************************************
* dosstring								*
* - very similar to makedword, this time to disassemble a string	*
*   (dos style string)							*
************************************************************************/
void disio::dosstring(void)
{ lptr outhere;
  findcurrentaddr(&outhere);
  if(outhere==curraddr)
    subitem=dsmnull;
  dsm->disdatadosstring(outhere);
}

/************************************************************************
* generalstring								*
* - very similar to makedword, this time to disassemble a string	*
*   (general string)							*
************************************************************************/
void disio::generalstring(void)
{ lptr outhere;
  findcurrentaddr(&outhere);
  if(outhere==curraddr)
    subitem=dsmnull;
  dsm->disdatageneralstring(outhere);
}

/************************************************************************
* makecode								*
* - very similar to makedword, but this time to disassemble as code	*
*   from the current location, and continue disassembling		*
************************************************************************/
void disio::makecode(void)
{ lptr outhere;
  findcurrentaddr(&outhere);
  if(outhere==curraddr)
    subitem=dsmnull;
  dsm->disblock(outhere);
}

/************************************************************************
* vertsetpos								*
* - this function takes a position of the vertical scroll bar, from 0	*
*   to the VERTSCROLLRANGE and it sets the current address to that	*
*   point								*
************************************************************************/
void disio::vertsetpos(int pos) {
 unsigned long location;
#ifdef WIN32
 location=(unsigned)MulDiv(pos,(int)total_data_size,VERTSCROLLRANGE);
#else
 location=(unsigned long)((float)pos/VERTSCROLLRANGE*total_data_size);
#endif
 setcuraddr(dta->getlocpos(location));	// linear to seg:ofs
}

// reverse function, sets scroll bar to seg:ofs given
// Vertical scroll bar has no PAGESIZE feedback yet
void disio::setscrollpos(lptr loc) {
 DWORD current_data_pos=dta->datagetpos(loc);	// seg:ofs to linear
#ifdef WIN32
 int sbarpos=MulDiv((int)current_data_pos,VERTSCROLLRANGE,(int)total_data_size+1);
#else
 int sbarpos=(int)((float)current_data_pos/(total_data_size+1)*VERTSCROLLRANGE;
#endif
 SetScrollPos(wnd,SB_VERT,sbarpos,true);
}

/************************************************************************
* setcuraddr								*
* - sets the address as a location for the screen output to start at	*
* - also sets the scroll bar position for the vertical scroll bar	*
* - also ensures that its not set to a mid-instruction address and	*
*   sets the subline							*
* - called from many places in the source code				*
************************************************************************/
void disio::setcuraddr(lptr loc) {
 dsmitem titem,*tblock;
 dsegitem *l_ds;
  // check for non-existant address added v2.20
 l_ds=dta->findseg(loc);
 if (!l_ds) return;
 curraddr=loc;		// set top line address
 titem.addr=curraddr;
 titem.type=dsmnull;
 dsm->findnext(&titem);
 tblock=dsm->nextiterator();
 if (tblock) subitem=tblock->type;
 else subitem=dsmnull;
 tblock=dsm->nextiter_code(tblock);
 if (tblock && loc.between(tblock->addr,tblock->addr+tblock->length-1))
   curraddr=tblock->addr;
 outend=loc+50;
 sel=0;
 updatewindow();
 setscrollpos(loc);
}

// calculates the rectangle for item at line <item> using CharBox.y
void disio::GetItemRect(int item, PRECT R) {
 GetClientRect(wnd,R);
 R->top=item*CharBox.y;
 R->bottom=R->top+CharBox.y;
}

/************************************************************************
* setpos								*
* - when the left mouse button is pressed the line is changed to the	*
*   line the mouse points at. This routine sets the line, and asks for	*
*   a screen update if needed (the screen update method here is a	*
*   primary thread request that simply invalidates the window rather	*
*   than requesting through the scheduler...				*
************************************************************************/
void disio::setpos(int ypos) {
 if (ypos!=sel) {
  RECT R;
  GetItemRect(sel,&R);
  sel=ypos;
  InvalidateRect(wnd,&R,TRUE);	// let remove marker
  GetItemRect(sel,&R);
  InvalidateRect(wnd,&R,TRUE);	// let set marker
 }
}

/************************************************************************
* printlineheader							*
* - prints an appropriate line header with the address if needed and	*
*   sets the cursor position for the next LastPrintBuff call		*
************************************************************************/
void disio::printlineheader(lptr loc,bool printaddrs) {
 if (options.mode32) {
  if (printaddrs) PrintBuff(/*"%04lx:"*/"%08lx",/*loc.segm,*/loc.offs);
  else PrintBuff("");
  LastPrintBuffEpos(BYTEPOS-5);
 }else{
  if (printaddrs) PrintBuff("%04lx:%04lx",loc.segm,loc.offs);
  else PrintBuff("");
  LastPrintBuffEpos(BYTEPOS-4);
 }
}

/************************************************************************
* outinst								*
* - this is the routine which results in the output of an address, hex	*
*   bytes, instruction and arguments					*
************************************************************************/
void disio::outinst(dsmitem *inst,bool printaddrs) {
 dsegitem *dblock;
 int i;
 int prefixptr;
 byte pbyte;
 printlineheader(inst->addr,printaddrs);
 dblock=dta->findseg(inst->addr);      // find segment item.
 switch (inst->type) {
  case dsmcode: {
   i=inst->length;
   if (printaddrs) {
    while (i) {
     if (dblock) {
      if (dblock->typ==uninitdata) LastPrintBuff("??");
      else LastPrintBuff("%02x",inst->data[inst->length-i]);
     }else LastPrintBuff("%02x",inst->data[inst->length-i]);
     i--;
     if (i+8<inst->length && inst->length>10) {
      LastPrintBuff("..");
      break;
     }
    }
   }
   if (options.mode32) LastPrintBuffEpos(ASMPOS+4);
   else	LastPrintBuffEpos(ASMPOS);
   if (inst->flags&FLAGS_PREFIX) {
    prefixptr=0;
    while (!isprefix(inst->data[prefixptr]) && prefixptr<15) prefixptr++;
    pbyte=inst->data[prefixptr];
    outprefix(pbyte);
   }
   LastPrintBuff(DSMITEM_NAME(inst));
   LastPrintBuff(" ");
   if (options.mode32) LastPrintBuffEpos(ARGPOS+4);
   else	LastPrintBuffEpos(ARGPOS);
   if (dblock) {
    if (dblock->typ==uninitdata) LastPrintBuff("?");
    else{
     outargs(inst,DSMITEM_ARG1(inst));
     if (DSMITEM_ARG2(inst)!=ARG_NONE) {
      LastPrintBuff(", ");
      outargs(inst,DSMITEM_ARG2(inst));
     }
     if (DSMITEM_ARG3(inst)!=ARG_NONE) {
      LastPrintBuff(", ");
      outargs(inst,DSMITEM_ARG3(inst));
     }
    }
   }else{
    outargs(inst,DSMITEM_ARG1(inst));
    if (DSMITEM_ARG2(inst)!=ARG_NONE) {
     LastPrintBuff(", ");
     outargs(inst,DSMITEM_ARG2(inst));
    }
    if (DSMITEM_ARG3(inst)!=ARG_NONE) {
     LastPrintBuff(", ");
     outargs(inst,DSMITEM_ARG3(inst));
    }
   }
  }break;
 }
}

/************************************************************************
* outdb									*
* - this is similar to outinst, but when there is no disassembly for a	*
*   loc we call this to output a db xxh or a db ? if its uninitdata    *
************************************************************************/
void disio::outdb(lptr *lp,bool printaddrs) {
 dsegitem *dblock;
 dword aaddr;
 byte *mcode;
 printlineheader(*lp,printaddrs);
 dblock=dta->findseg(*lp);	// find segment item.
 if (!dblock) {
  if (printaddrs) LastPrintBuff("??");
 }else if (dblock->typ==uninitdata) {
  if (printaddrs) LastPrintBuff("??");
 }else{
  aaddr=(*lp-dblock->addr);
  mcode=dblock->data+aaddr;
  if (printaddrs) LastPrintBuff("%02x",mcode[0]);
 }
 if (options.mode32) LastPrintBuffEpos(ASMPOS+4);
 else LastPrintBuffEpos(ASMPOS);

 LastPrintBuff("db");

 if (options.mode32) LastPrintBuffEpos(ARGPOS+4);
 else LastPrintBuffEpos(ARGPOS);
  // changed to single ? - 2.25
 if (!dblock) LastPrintBuff("?");
 else if (dblock->typ==uninitdata) {
  LastPrintBuff("?");
 }else{
  LastPrintBuffHexValue(mcode[0]);
  if (isprint(mcode[0])) {
   LastPrintBuffEpos(COMMENTPOS);
   LastPrintBuff(";'%c'",mcode[0]);
  }
 }
}

/************************************************************************
* issegprefix								*
* - returns true if byte is a segment prefix valid value		*
************************************************************************/
bool disio::issegprefix(byte byt) {
 switch (byt) {
  case 0x2e:
  case 0x36:
  case 0x3e:
  case 0x26:
  case 0x64:
  case 0x65: return true;
 }
 return false;
}

/************************************************************************
* isprefix								*
* - returns true if byte is a prefix valid value (rep/repne/lock)	*
************************************************************************/
bool disio::isprefix(byte byt) {
 switch (byt) {
  case 0xf0:
  case 0xf2:
  case 0xf3: return true;
 }
 return false;
}

/************************************************************************
* outprefix								*
* - here we output a prefix segment override				*
************************************************************************/
void disio::outprefix(byte prefixbyte) {
 char *s;
 switch (prefixbyte) {
  case 0x2e: s="cs:";	break;
  case 0x36: s="ss:";	break;
  case 0x3e: s="ds:";	break;
  case 0x26: s="es:";	break;
  case 0x64: s="fs:";	break;
  case 0x65: s="gs:";	break;
  case 0xf0: s="lock ";	break;
  case 0xf2: s="repne ";break;
  case 0xf3: s="repe ";	break;
  default: s="err:";
 }
 LastPrintBuff(s);
}

/************************************************************************
* jumpback								*
* - when the user presses ESC, or selects jump back we get the last	*
*   address from the top of the address stack and call setcuraddr and	*
*   update the window and so the disassembly listing flicks back	*
************************************************************************/
void disio::jumpback(void) {
 if (!retstack.stacktop) return;
 setcuraddr(retstack.pop());
 sel=selstack[retstack.stacktop];
 if ((unsigned)sel>=(unsigned)size.y) sel=0;	// for invalid values
 updatewindow();
}

/************************************************************************
* jumpto								*
* - more complex than the jumpback is the jumpto. The complexity lies	*
*   in deciding where we are jumping to and what the arguments value is *
*   and if the location exists. Actually making the jump consists of	*
*   saving the current location to the address stack and then changing	*
*   the curraddr for output, and updating the window			*
* - most of this routine is a complex decipherment of a modrm address	*
*   to jump to								*
* NB I need to stick this in a function of its own at some point, as it *
* would be quite useful to just get an argument address in several	*
* places in the code							*
************************************************************************/
void disio::jumpto(bool arg1) {
 dsmitem *tblock;
 lptr outhere;
 byte *data;
 bool madejump;
 byte modrm,sib;
 word rm;
 tblock=findcurrentline();
 outhere.assign(0,0);
 if (tblock && tblock->type!=dsmcode) return;
 madejump=false;
 if (tblock) {
  if (arg1) {
   switch (DSMITEM_ARG1(tblock)) {
    case ARG_FADDR: {
     data=tblock->data+tblock->length;
     if (tblock->mode32) {
      data-=6;
      outhere.assign(((word *)(&data[4]))[0],((dword *)(&data[0]))[0]);
     }else{
      data-=4;
      outhere.assign(((word *)(&data[2]))[0],((word *)(&data[0]))[0]);
     }
     if (follow(outhere)) madejump=true;
    }break;
    case ARG_IMM32: {
     if (tblock->override!=over_dsoffset) break;
     data=tblock->data+tblock->length;
     data-=4;
     outhere.assign(options.dseg,((dword *)(&data[0]))[0]);
     if (follow(outhere)) madejump=true;
    }break;
    case ARG_MEMLOC: {
     data=tblock->data+tblock->length;
     if (options.mode32) {
      data-=4;
      outhere.assign(tblock->addr.segm,((dword *)data)[0]);
     }else{
      data-=2;
      outhere.assign(tblock->addr.segm,((word *)data)[0]);
     }
     if (follow(outhere)) madejump=true;
    }break;
    case ARG_IMM: {
     if (tblock->override!=over_dsoffset) break;
     if (options.mode32) {
      data=tblock->data+tblock->length;
      data-=4;
      outhere.assign(options.dseg,((dword *)(&data[0]))[0]);
      if (follow(outhere)) madejump=true;
     }
    }break;
    case ARG_RELIMM: {
     data=tblock->data+tblock->length;
     outhere=tblock->addr;
     if (tblock->mode32) {
      data-=4;
      outhere+=((dword *)data)[0]+tblock->length;
     }else{
      data-=2;
      outhere+=(word)(((word *)data)[0]+tblock->length);
     }
     if (follow(outhere)) madejump=true;
    }break;
    case ARG_RELIMM8: {
     data=tblock->data+tblock->length-1;
     outhere=tblock->addr;
     if (tblock->mode32) {
      if (data[0]&0x80) outhere+=(dword)(data[0]+0xffffff00+tblock->length);
      else		outhere+=(dword)(data[0]+tblock->length);
     }else{
      if (data[0]&0x80) outhere+=(word)(data[0]+0xff00+tblock->length);
      else		outhere+=(word)(data[0]+tblock->length);
     }
     if (follow(outhere)) madejump=true;
    }break;
    case ARG_MMXMODRM:
    case ARG_XMMMODRM:
    case ARG_MODRM_S:
    case ARG_MODRMM512:
    case ARG_MODRMQ:
    case ARG_MODRM_SREAL:
    case ARG_MODRM_PTR:
    case ARG_MODRM_WORD:
    case ARG_MODRM_SINT:
    case ARG_MODRM_EREAL:
    case ARG_MODRM_DREAL:
    case ARG_MODRM_WINT:
    case ARG_MODRM_LINT:
    case ARG_MODRM_BCD:
    case ARG_MODRM_FPTR:
    case ARG_MODRM: {
     data=tblock->data+tblock->modrm;
     rm=(byte)((data[0]&0xc0)>>6);
     modrm=(byte)(data[0]&0x07);
     switch (rm) {
      case 0: {
       if (options.mode32) {
        if (modrm==5) {
         outhere.assign(tblock->addr.segm,((dword *)(&data[1]))[0]);
        }else if (modrm==4) {	 // case 4=sib
 	 sib=data[1];
	 if ((sib&0x07)==5) { // disp32
	  outhere.assign(tblock->addr.segm,((dword *)(&data[2]))[0]);
	 }
        }
       }else{
        if (modrm==6) {
         outhere.assign(tblock->addr.segm,((word *)(&data[1]))[0]);
        }
       }
      }break;
      case 1: break;
      case 2: {
       if (options.mode32) {
        outhere.assign(tblock->addr.segm,((dword *)(&data[1]))[0]);
	if (modrm==4) {	    // case 4=sib
	 outhere.assign(tblock->addr.segm,((dword *)(&data[2]))[0]);
	}
       }else{
        outhere.assign(tblock->addr.segm,((word *)(&data[1]))[0]);
       }
      }break;
      case 3: break;
     }/*switch(rm)*/
     if (follow(outhere)) madejump=true;
    }break;
   }/*switch(DSM_ITEM...)*/
  }/*if(arg1)*/
  if (!madejump) {
   switch (DSMITEM_ARG2(tblock)) {
    case ARG_IMM32: {
     if (tblock->override!=over_dsoffset) break;
     data=tblock->data+tblock->length;
     data-=4;
     outhere.assign(options.dseg,((dword *)(&data[0]))[0]);
     follow(outhere);
    }break;
    case ARG_IMM: {
     if (tblock->override!=over_dsoffset) break;
     if (options.mode32) {
      data=tblock->data+tblock->length;
      data-=4;
      outhere.assign(options.dseg,((dword *)(&data[0]))[0]);
      follow(outhere);
     }
    }break;
    case ARG_MEMLOC: {
     data=tblock->data+tblock->length;
     if (options.mode32) {
      data-=4;
      outhere.assign(tblock->addr.segm,((dword *)data)[0]);
     }else{
      data-=2;
      outhere.assign(tblock->addr.segm,((word *)data)[0]);
     }
     follow(outhere);
    }break;
    case ARG_MMXMODRM:
    case ARG_XMMMODRM:
    case ARG_MODRM_S:
    case ARG_MODRMM512:
    case ARG_MODRMQ:
    case ARG_MODRM_SREAL:
    case ARG_MODRM_PTR:
    case ARG_MODRM_WORD:
    case ARG_MODRM_SINT:
    case ARG_MODRM_EREAL:
    case ARG_MODRM_DREAL:
    case ARG_MODRM_WINT:
    case ARG_MODRM_LINT:
    case ARG_MODRM_BCD:
    case ARG_MODRM_FPTR:
    case ARG_MODRM: {
     data=tblock->data+tblock->modrm;
     rm=(byte)((data[0]&0xc0)>>6);
     modrm=(byte)(data[0]&0x07);
     switch (rm) {
      case 0:
	      if(options.mode32)
	      { if(modrm==5)
		{ outhere.assign(tblock->addr.segm,((dword *)(&data[1]))[0]);
		}
		else if(modrm==4)	 // case 4=sib
		{ sib=data[1];
		  if((sib&0x07)==5) // disp32
		  { outhere.assign(tblock->addr.segm,((dword *)(&data[2]))[0]);
		  }
		}
	      }
	      else
	      { if(modrm==6)
		{ outhere.assign(tblock->addr.segm,((word *)(&data[1]))[0]);
		}
	      }
	      break;
      case 1:
	      break;
      case 2:
	      if(options.mode32)
	      { outhere.assign(tblock->addr.segm,((dword *)(&data[1]))[0]);
		if(modrm==4)	    // case 4=sib
		{ outhere.assign(tblock->addr.segm,((dword *)(&data[2]))[0]);
		}
	      }
	      else
	      { outhere.assign(tblock->addr.segm,((word *)(&data[1]))[0]);
	      }
	      break;
      case 3:
	      break;
     }/*switch(rm)*/
     follow(outhere);
    }break;
   }/*switch(DSMITEM...)*/
  }
 }
}

/************************************************************************
* findcurrentline							*
* - this routine finds the current screen address and output line in	*
*   the disassembly database and from there works out the disassembly	*
*   item on the currently selected line.				*
* - it is used by jumpto, when the user presses return to follow a jump *
* - comments added to the procedure, it is a useful one to follow and	*
*   see the strategy employed, which is a fairly common Borg strategy	*
************************************************************************/
dsmitem *disio::findcurrentline(void) {
 dsmitem titem,*tblock;
 lptr outhere;
 int i;
  // strategy
  // - use pointer to first item if available (so comments,etc included in list
  // - otherwise use address.
 titem.addr=curraddr;
 titem.type=subitem;
  // hunt for current addr and subitem
 tblock=dsm->find(&titem);
 if (tblock) tblock=dsm->nextiterator();
  // on overlap - reset the curraddr.
  // [on the spot error correction]
 if (tblock) {
  if(curraddr.between(tblock->addr,tblock->addr+tblock->length-1))
     curraddr.offs=tblock->addr.offs;
 }
  // ensure we point to the right item, or the next one
 if (tblock) {
  if((tblock->addr<curraddr)||((tblock->addr==curraddr)&&(tblock->type<subitem)))
      tblock=dsm->nextiterator();
 }
  // now at the top of the screen, the next loop moves down to the user selection line
 outhere=curraddr;
 for (i=0;i<sel;i++) {
  if (tblock) {
   if (outhere==tblock->addr) {
    outhere+=tblock->length;
    tblock=dsm->nextiterator();
   }else outhere++;
  }else outhere++;
    // check if gone beyond seg, get next seg.
  if (dta->beyondseg(outhere)) {
   outhere.offs--;
   dta->nextseg(&outhere);
  }
  if (!outhere.segm) break;
 }
  // now we either have the line we are pointing to, in the database
  // or we have moved beyond the database and have a null
  // or we have an address which would be a db
 if (tblock && outhere!=tblock->addr) return NULL;
 return tblock;
}

/************************************************************************
* findcurrentaddr							*
* - this is very similar to findcurrentline, but it instead finds just	*
*   the location. the search strategy is the same			*
************************************************************************/
void disio::findcurrentaddr(lptr *loc) {
 dsmitem titem,*tblock;
 lptr outhere;
 int i;
  // strategy
  // - use pointer to first item if available (so comments,etc included in list
  // - otherwise use address.
 titem.addr=curraddr;
 titem.type=subitem;
 tblock=dsm->find(&titem);
 if (tblock) tblock=dsm->nextiterator();
 if (tblock && curraddr.between(tblock->addr,tblock->addr+tblock->length-1))
   curraddr.offs=tblock->addr.offs;
 if (tblock && tblock->addr<curraddr) tblock=dsm->nextiterator();
  // added 2.25 - bugfix
  // - wasnt finding correct lines when in mid-line......
 if (tblock) {
  while (tblock->addr==curraddr && tblock->type<subitem) {
   tblock=dsm->nextiterator();
   if (!tblock)	break;
  }
 }
 outhere=curraddr;
 for (i=0;i<sel;i++) {
  if (tblock && outhere==tblock->addr) {
   outhere+=tblock->length;
   tblock=dsm->nextiterator();
  }else outhere++;
    // check if gone beyond seg, get next seg.
  if (dta->beyondseg(outhere)) {
   outhere.offs--;
   dta->nextseg(&outhere);
  }
  if (!outhere.segm) break;
 }
 if (outhere.segm) *loc=outhere;
 else *loc=curraddr;
}

/************************************************************************
* findaddr								*
* Ich verstehe diesen Quelltext nicht, aber hier versuche ich eine
* Zeile im Bildschirm zu finden, die die angegebene Adresse enthΣlt	*
************************************************************************/
int disio::findaddr(const lptr &loc) {
 dsmitem titem,*tblock;
 lptr outhere;
 int i;
  // strategy
  // - use pointer to first item if available (so comments,etc included in list
  // - otherwise use address.
 titem.addr=curraddr;
 titem.type=subitem;
 tblock=dsm->find(&titem);
 if (tblock) tblock=dsm->nextiterator();
 if (tblock && curraddr.between(tblock->addr,tblock->addr+tblock->length-1))
   curraddr.offs=tblock->addr.offs;
 if (tblock && tblock->addr<curraddr) tblock=dsm->nextiterator();
  // added 2.25 - bugfix
  // - wasnt finding correct lines when in mid-line......
 if (tblock) {
  while (tblock->addr==curraddr && tblock->type<subitem) {
   tblock=dsm->nextiterator();
   if (!tblock)	break;
  }
 }
 outhere=curraddr;
 for (i=0;i<size.y;i++) {
  if (tblock && outhere==tblock->addr) {
   outhere+=tblock->length;
   tblock=dsm->nextiterator();
  }else outhere++;
    // check if gone beyond seg, get next seg.
  if (dta->beyondseg(outhere)) {
   outhere.offs--;
   dta->nextseg(&outhere);
  }
  if (outhere==loc) return i+1;		// wei▀ nicht warum das +1 ...
  if (!outhere.segm) break;
 }
 return -1;
}

/************************************************************************
* savecuraddr								*
* - this simply saves the window top line location to the return stack	*
* - this is called when the user selects to jump somewhere (like to a	*
*   named location) rather than the 'jumpto' routine used to jump to a	*
*   location specified by a disassembly argument			*
************************************************************************/
void disio::savecuraddr(void) {
 if (retstack.stacktop==CALLSTACKSIZE) {
	// if stack is full, move down its content (damn, no STL)
  memmove(selstack,selstack+1,sizeof(selstack)-sizeof(selstack[0]));
 }
 retstack.push(curraddr);
 selstack[retstack.stacktop-1]=sel;
}

// same as ^F in Turbo Debugger
bool disio::follow(const lptr &loc) {
 if (!dta->findseg(loc)) return false;
 savecuraddr();
 int i=findaddr(loc);
 if (i>=0) sel=i;	// NICHT ROLLEN wenn im Bildschirm!!!
 else setcuraddr(loc);
 updatewindow();
 return true;
}

/************************************************************************
* updatewindowifwithinrange						*
* - adds a scheduler task for a window update if the current window	*
*   overlaps with the specified range					*
************************************************************************/
void disio::updatewindowifwithinrange(lptr loc_start,lptr loc_end) {
 if (loc_end>=curraddr && loc_start<=outend)
   scheduler.addtask(windowupdate,priority_window,nlptr,NULL);
}

/************************************************************************
* updatewindowifinrange							*
* - adds a scheduler task for a window update if the current window	*
*   contains the loc specified						*
************************************************************************/
void disio::updatewindowifinrange(lptr loc) {
 if (loc.between(curraddr,outend))
   scheduler.addtask(windowupdate,priority_window,nlptr,NULL);
}

/************************************************************************
* updatewindow								*
* - this function rewrites the buffers for the disassembly output to	*
*   the window, and then requests a screen refresh which will paint the *
*   buffers into the window. The ClearBuff and DoneBuff functions	*
*   when the buffers are full and it is safe to refresh the screen with *
*   a repaint								*
************************************************************************/
// this functions job to output a window
// of disassembly to the buffer.
void disio::updatewindow(void) {
 dsmitem titem,*tblock;
 lptr outhere;
 int i;
  // find current position.
 titem.addr=curraddr;
 titem.type=subitem;
 tblock=dsm->find(&titem);
 if (tblock) tblock=dsm->nextiterator();
  // now tblock= current position, or previous one.
  // check if in middle of instruction
 if (tblock) {
 // bugfix 2.27
  if (tblock->length>1 
  && curraddr.between(tblock->addr,tblock->addr+tblock->length-1)) {
   curraddr.offs=tblock->addr.offs;
   subitem=tblock->type;
  }
 }
  // check if previous one. get next
 if (tblock) {
  if (tblock->addr<curraddr
  || tblock->addr==curraddr && tblock->type<subitem)
    tblock=dsm->nextiterator();
 }
  // tblock is now top of the page.
 outhere=curraddr;
 ClearBuff();			      // start again
 for (i=0; i<size.y;i++) {
  if (tblock) {
   if (outhere==tblock->addr) {
    switch (tblock->type) {
     case dsmcode: {
      outinst(tblock,true);
     }break;
     case dsmnameloc: {
      printlineheader(tblock->addr,true);
      LastPrintBuff("%s:",tblock->data);
     }break;
     case dsmxref: {
      printlineheader(tblock->addr,true);
      LastPrintBuff(";");
      LastPrintBuffEpos(COMMENTPOS);
      LastPrintBuff("XREFS First: ");
      xrefs->printfirst(tblock->addr);
     }break;
     default: {
      printlineheader(tblock->addr,true);
      outcomment(tblock);
     }
    }
    outhere+=tblock->length;
    tblock=dsm->nextiterator();
   }else{
    outdb(&outhere,true);
    outhere++;
   }
  }else{
   outdb(&outhere,true);
   outhere++;
  }
    // check if gone beyond seg, get next seg.
  if (dta->beyondseg(outhere)) {
   outhere.offs--;
   dta->nextseg(&outhere);
  }
  if (!outhere.segm) break;
  else outend=outhere;
 }
 DoneBuff();			      // mark as done
 InvalidateRect(wnd,NULL,true);
}

/************************************************************************
* scroller								*
* - this routine controls simple vertical scrolls. A vertical scroll is *
*   a movement of the currently selected line. Only if this line moves	*
*   beyond the boundaries of the window do we need to move the windowed *
*   disassembly output itself. Simple moves are handled quickly, and if *
*   we need to regenerate the buffers then we first recalculate the top *
*   position and then we call windowupdate to handle the rest.		*
************************************************************************/
void disio::scroller(int amount) {
 dsmitem titem,*tblock;
 int newpos=sel+amount;
  // simple move bar on screen
 if ((unsigned)newpos<(unsigned)(size.y-1)) {
  setpos(newpos);
  return;
 }
  // find top of page
 titem.addr=curraddr;
 titem.type=subitem;
 tblock=dsm->find(&titem);
 // tblock is now top of page or previous item.
 // if moving up find previous inst
 if ((unsigned)amount<100 && tblock) tblock=dsm->nextiterator();
 if ((unsigned)amount>100 && tblock 
 && tblock->addr==titem.addr && tblock->type==subitem)
   tblock=dsm->lastiterator();
 // move up - tblock=previous inst.
 // if moving down, check if tblock=previous item.
 if (tblock && (unsigned)amount<100) {
  if (tblock->addr<curraddr
  || tblock->addr==curraddr && tblock->type<subitem)
    tblock=dsm->nextiterator();
 }
 // moving down - tblock=current top.
 if ((unsigned)amount<100) while (amount) {
  if (sel<=size.y-3) sel++;
  else{
   if (tblock) {
    if (curraddr==tblock->addr) {
     curraddr+=tblock->length;
     tblock=dsm->nextiterator();
     if (tblock && curraddr==tblock->addr) subitem=tblock->type;
     else subitem=dsmcode;
    }else{
     subitem=dsmnull;
     curraddr++;
    }
   }else{
    subitem=dsmnull;
    curraddr++;
   }
// check if gone beyond seg, get next seg.
   if (dta->beyondseg(curraddr)) {
    curraddr.offs--;
    dta->nextseg(&curraddr);
    subitem=dsmnull;
   }
   if (!curraddr.segm) break;
   titem.addr=curraddr;
  }
  amount--;
 }else{
  while (amount) {
   if (sel) sel--;
   else{
    if (tblock) {
     if (curraddr==tblock->addr+tblock->length) {
      curraddr.offs-=tblock->length;
      subitem=tblock->type;
      tblock=dsm->lastiterator();
     }else{
      subitem=dsmcode;
      curraddr.offs--;
     }
    }else{
     subitem=dsmnull;
     curraddr.offs--;
    }
// check if gone beyond seg, get previous seg.
    if (dta->beyondseg(curraddr)) {
     curraddr++;
     dta->lastseg(&curraddr);
    }
    if (!curraddr.segm) break;
    titem.addr=curraddr;
   }
   amount++;
  }
 }
 if (!curraddr.segm) curraddr=titem.addr;
 updatewindow();
 setscrollpos(curraddr);
//  current_data_pos=dta->datagetpos();
//  sbarpos=((float)current_data_pos)/((float)total_data_size+(float)1.0)*(float)VERTSCROLLRANGE;
//  SetScrollPos(MainWnd,SB_VERT,(int)sbarpos,true);
}

BOOL PrintFoundName(lptr loc) {
 if (name->isname(loc))		name->printname(loc);
 else if (import->isname(loc))	import->printname(loc);
 else if(expt->isname(loc))	expt->printname(loc);
 else return FALSE;
 return TRUE;
}

void PrintLongHex(lptr loc) {
 if (!PrintFoundName(loc)) cvd->LastPrintBuffLongHexValue(loc.offs);
}

void PrintOffset(lptr loc) {
 cvd->LastPrintBuff("offset ");
 PrintLongHex(loc);
}

void PrintMemVar(lptr loc) {
 cvd->LastPrintBuff("[");
 PrintLongHex(loc);
 cvd->LastPrintBuff("]");
}

/************************************************************************
* outargs								*
* - this is a very long routine which handles every kind of argument	*
*   that we have set up in the processor tables. It outputs the ascii	*
*   form of the instructions arguments to the buffer. It handles	*
*   complex modrm and sib encodings, the display of names and locations *
*   which must be decoded, etc.						*
************************************************************************/
void disio::outargs(dsmitem *inst,argtype a) {
 byte *dta,modrm,sib;
  // rm extended to word. build 15. re M Ogden and VC++ warnings.
 word rm;
 argtype a1,a2;
 dword targetd;
 dword targetw;
 lptr loc;
 char fbuff[40];
 int i,sp;	      // sp= string printed
 byte pbyte; // prefix byte.
 int prefixptr;
 if(inst->flags&FLAGS_SEGPREFIX) {
  prefixptr=0;
  while (!issegprefix(inst->data[prefixptr]) && prefixptr<15)
    prefixptr++;
  pbyte=inst->data[prefixptr];
 }
 if (inst->flags&FLAGS_ADDRPREFIX) options.mode32=!options.mode32;
 switch (a) {
  case ARG_REG_AX: {
   if (inst->mode32) LastPrintBuff("eax");
   else		     LastPrintBuff("ax");
  }break;
  case ARG_REG_BX: {
   if (inst->mode32) LastPrintBuff("ebx");
   else		     LastPrintBuff("bx");
  }break;
  case ARG_REG_CX: {
   if (inst->mode32) LastPrintBuff("ecx");
   else		     LastPrintBuff("cx");
  }break;
  case ARG_REG_DX: {
   if(inst->mode32)  LastPrintBuff("edx");
   else		     LastPrintBuff("dx");
  }break;
  case ARG_16REG_DX: LastPrintBuff("dx"); break;
  case ARG_REG_SP: {
   if (inst->mode32) LastPrintBuff("esp");
   else		     LastPrintBuff("sp");
  }break;
  case ARG_REG_BP: {
   if (inst->mode32) LastPrintBuff("ebp");
   else		     LastPrintBuff("bp");
  }break;
  case ARG_REG_SI: {
   if (inst->mode32) LastPrintBuff("esi");
   else		     LastPrintBuff("si");
  }break;
  case ARG_REG_DI: {
   if (inst->mode32) LastPrintBuff("edi");
   else		     LastPrintBuff("di");
  }break;
  case ARG_REG_AL:   LastPrintBuff("al"); break;
  case ARG_REG_AH:   LastPrintBuff("ah"); break;
  case ARG_REG_BL:   LastPrintBuff("bl"); break;
  case ARG_REG_BH:   LastPrintBuff("bh"); break;
  case ARG_REG_CL:   LastPrintBuff("cl"); break;
  case ARG_REG_CH:   LastPrintBuff("ch"); break;
  case ARG_REG_DL:   LastPrintBuff("dl"); break;
  case ARG_REG_DH:   LastPrintBuff("dh"); break;
  case ARG_REG_ST0:  LastPrintBuff("st(0)"); break;
  case ARG_REG_ES:   LastPrintBuff("es"); break;
  case ARG_REG_CS:   LastPrintBuff("cs"); break;
  case ARG_REG_DS:   LastPrintBuff("ds"); break;
  case ARG_REG_SS:   LastPrintBuff("ss"); break;
  case ARG_REG_FS:   LastPrintBuff("fs"); break;
  case ARG_REG_GS:   LastPrintBuff("gs"); break;
  case ARG_REG_A:    LastPrintBuff("a"); break;
  case ARG_REG_B:    LastPrintBuff("b"); break;
  case ARG_REG_C:    LastPrintBuff("c"); break;
  case ARG_REG_D:    LastPrintBuff("d"); break;
  case ARG_REG_E:    LastPrintBuff("e"); break;
  case ARG_REG_H:    LastPrintBuff("h"); break;
  case ARG_REG_L:    LastPrintBuff("l"); break;
  case ARG_REG_I:    LastPrintBuff("i"); break;
  case ARG_REG_R:    LastPrintBuff("r"); break;
  case ARG_REG_HL_IND: LastPrintBuff("(hl)"); break;
  case ARG_REG_BC:   LastPrintBuff("bc"); break;
  case ARG_REG_DE:   LastPrintBuff("de"); break;
  case ARG_REG_HL:   LastPrintBuff("hl"); break;
  case ARG_REG_BC_IND: LastPrintBuff("(bc)"); break;
  case ARG_REG_DE_IND: LastPrintBuff("(de)"); break;
  case ARG_REG_SP_IND: LastPrintBuff("(sp)"); break;
  case ARG_REG_IX:   LastPrintBuff("ix"); break;
  case ARG_REG_IX_IND: {
   LastPrintBuff("(ix");
   if (inst->flags&FLAGS_INDEXREG) {
    dta=inst->data+2;
    LastPrintBuff("+");
    LastPrintBuffLongHexValue((word)(dta[0]));
   }LastPrintBuff(")");
  }break;
  case ARG_REG_IY:   LastPrintBuff("iy"); break;
  case ARG_REG_IY_IND: {
   LastPrintBuff("(iy");
   if (inst->flags&FLAGS_INDEXREG) {
    dta=inst->data+2;
    LastPrintBuff("+");
    LastPrintBuffLongHexValue((word)(dta[0]));
   }
   LastPrintBuff(")");
  }break;
  case ARG_REG_C_IND:LastPrintBuff("(c)"); break;
  case ARG_REG_AF:   LastPrintBuff("af"); break;
  case ARG_REG_AF2:  LastPrintBuff("af'"); break;
  case ARG_IMM: {
   dta=inst->data+inst->length;
   if (inst->mode32) {
    dta-=4;
    switch (inst->override) {
     case over_decimal: {
      if (inst->displayflags&DISPFLAG_NEGATE) LastPrintBuff("-%02lu",0-*(dword *)dta);
      else				      LastPrintBuff("%02lu",*(dword *)dta);
     }break;
     case over_char: {
      LastPrintBuff("'");
      for (i=3;i>=0;i--) if (dta[i]) LastPrintBuff("%c",dta[i]);
      LastPrintBuff("'");
     }break;
     case over_dsoffset: {
      loc.assign(inst->addr.segm,*(dword *)dta);
      PrintOffset(loc);
     }break;
     case over_single: {
      sprintf(fbuff,"(float)%g",*(float *)dta);
      LastPrintBuff(fbuff);
     }break;
     default: {
      if (inst->displayflags&DISPFLAG_NEGATE) {
       LastPrintBuff("-");
       LastPrintBuffLongHexValue(0-*(dword *)dta);
      }else LastPrintBuffLongHexValue(*(dword *)dta);
     }break;
    }/*switch*/
   }else{ dta-=2;
    if (inst->override==over_decimal) {
     if (inst->displayflags&DISPFLAG_NEGATE)
      LastPrintBuff("-%02lu",0x10000-((word *)dta)[0]);
     else LastPrintBuff("%02lu",((word *)dta)[0]);
    }else if(inst->override==over_char)	{
     LastPrintBuff("\"");
     for (i=1;i>=0;i--) if (dta[i]) LastPrintBuff("%c",dta[i]);
     LastPrintBuff("\"");
    }else{
     if (inst->displayflags&DISPFLAG_NEGATE) {
      LastPrintBuff("-");
      LastPrintBuffLongHexValue(0x10000-((word *)dta)[0]);
     }else LastPrintBuffLongHexValue(((word *)dta)[0]);
    }
   }
  }break;
  case ARG_IMM_SINGLE:
   dta=inst->data+inst->length-4;
   sprintf(fbuff,"%g",*(float *)dta);
   LastPrintBuff(fbuff);
   break;
  case ARG_IMM_DOUBLE:
   dta=inst->data+inst->length-8;
   sprintf(fbuff,"%g",*(double *)dta);
   LastPrintBuff(fbuff);
   break;
  case ARG_IMM_LONGDOUBLE:
   dta=inst->data+inst->length-10;
   sprintf(fbuff,"%Lg",*(long double *)dta);
   LastPrintBuff(fbuff);
   break;
  case ARG_IMM32: {
   dta=inst->data+inst->length-4;
   switch (inst->override) {
    case over_decimal:
     if (inst->displayflags&DISPFLAG_NEGATE) LastPrintBuff("-%02lu",0-((dword *)dta)[0]);
     else				     LastPrintBuff( "%02lu",((dword *)dta)[0]);
     break;
    case over_char:
     LastPrintBuff("\'");
     for(i=3;i>=0;i--) if(dta[i]) LastPrintBuff("%c",dta[i]);
     LastPrintBuff("\'");
     break;
    case over_dsoffset:
     loc.assign(inst->addr.segm,*(dword *)dta);
     PrintOffset(loc);
     break;
    case over_single:
     sprintf(fbuff,"(float)%g",*(float *)dta);
     LastPrintBuff(fbuff);
     break;
    default:
     if (inst->displayflags&DISPFLAG_NEGATE) {
      LastPrintBuff("-");
      LastPrintBuffLongHexValue(0-((dword *)dta)[0]);
     }else LastPrintBuffLongHexValue(((dword *)dta)[0]);
     break;
   }/*switch*/
  }break;
  case ARG_STRING: {
   rm=0;
   LastPrintBuff("\"");
   sp=0;
   while (inst->data[rm]) {
    if (inst->data[rm]<32) {
     LastPrintBuff("\",");
     LastPrintBuffHexValue(inst->data[rm]);
     LastPrintBuff(",\"");
    }else LastPrintBuff("%c",inst->data[rm]);
    rm++;
    sp++;
    if (sp>MAX_STRINGPRINT) break;
   }
   LastPrintBuff("\",00h");
  }break;
  case ARG_PSTRING: {
   dta=inst->data;
   rm=dta[0];
   dta++;
   LastPrintBuffHexValue((byte)rm);
   LastPrintBuff(",\"");
   sp=0;
   while (rm) {
    if (dta[0]<32) {
     LastPrintBuff("\",");
     LastPrintBuffHexValue(*dta);
     LastPrintBuff(",\"");
    }else LastPrintBuff("%c",*dta);
    dta++;
    rm--;
    sp++;
    if (sp>MAX_STRINGPRINT) break;
   }
   LastPrintBuff("\"");
  }break;
  case ARG_DOSSTRING: {
   dta=inst->data;
   rm=inst->length;
   rm--;
   LastPrintBuff("\"");
   sp=0;
   while (rm) {
    if (dta[0]<32) {
     LastPrintBuff("\",");
     LastPrintBuffHexValue(dta[0]);
     LastPrintBuff(",\"");
    }else LastPrintBuff("%c",dta[0]);
    dta++;
    rm--;
    sp++;
    if (sp>MAX_STRINGPRINT) break;
   }
   LastPrintBuff("\"");
  }break;
  case ARG_CUNICODESTRING:	// missing DW instead DB (?)
   dta=inst->data;
   rm=inst->length;
   rm-=(byte)2;
   LastPrintBuff("\"");
   sp=0;
   while (rm) {
    if (dta[0]<32) {
     LastPrintBuff("\",");
     LastPrintBuffHexValue(dta[0]);
     LastPrintBuff(",\"");
    }else LastPrintBuff("%c",dta[0]);
    dta+=2;
    rm-=(byte)2;
    sp++;
    if (sp>MAX_STRINGPRINT) break;
   }
   LastPrintBuff("\"");
   break;
  case ARG_PUNICODESTRING:
   dta=inst->data+2;
   rm=inst->length;
   rm-=(byte)2;
   sp=0;
   LastPrintBuffHexValue((byte)(rm/2));
   LastPrintBuff(",\"");
   while (rm) {
    if (dta[0]<32) {
     LastPrintBuff("\",");
     LastPrintBuffHexValue(dta[0]);
     LastPrintBuff(",\"");
    }else LastPrintBuff("%c",dta[0]);
    dta+=2;
    rm-=(byte)2;
    sp++;
    if (sp>MAX_STRINGPRINT) break;
   }
   LastPrintBuff("\"");
   break;
  case ARG_MEMLOC:
   if (inst->flags&FLAGS_SEGPREFIX) outprefix(pbyte);
   dta=inst->data+inst->length;
   if (options.mode32) {
    dta-=4;
    loc.assign(inst->addr.segm,*(dword *)dta);
    if (inst->flags&FLAGS_8BIT) LastPrintBuff("byte ptr");
    else if (inst->flags&FLAGS_ADDRPREFIX) LastPrintBuff("word ptr");
    else LastPrintBuff("dword ptr");
    PrintMemVar(loc);
   }else{
    dta-=2;
    loc.assign(inst->addr.segm,((word *)dta)[0]);
    if (inst->flags&FLAGS_8BIT) LastPrintBuff("byte ptr");
    if (inst->flags&FLAGS_ADDRPREFIX) LastPrintBuff("dword ptr");
    else LastPrintBuff("word ptr");
    PrintMemVar(loc);
   }
   break;
  case ARG_MEMLOC16:
   if (inst->flags&FLAGS_SEGPREFIX) outprefix(pbyte);
   dta=inst->data+inst->length-2;
   if (options.processor==PROC_Z80) {
    loc.assign(inst->addr.segm,*(dword *)dta);
    PrintMemVar(loc);
   }else{
    loc.assign(inst->addr.segm,*(word *)dta);
    PrintMemVar(loc);
   }
   break;
  case ARG_SIMM8:
   dta=inst->data+inst->length-1;
   if (inst->override==over_char) {
    LastPrintBuff("\"");
    for(i=0;i>=0;i--) if(dta[i]) LastPrintBuff("%c",dta[i]);
    LastPrintBuff("\"");
   }else if (dta[0]&0x80) {
    if (inst->override==over_decimal) LastPrintBuff("%02lu",(word)(0x100-dta[0]));
    else{
     LastPrintBuff("-");
     LastPrintBuffLongHexValue((word)(0x100-dta[0]));
    }
   }else{
    if (inst->override==over_decimal) LastPrintBuff("%02lu",(word)(dta[0]));
    else LastPrintBuffLongHexValue((word)(dta[0]));
   }
   break;
  case ARG_IMM8:
      dta=inst->data+inst->length-1;
      if(inst->override==over_decimal)
      { if(inst->displayflags&DISPFLAG_NEGATE)
	  LastPrintBuff("-%02lu",0x100-(word)(dta[0]));
	else
	  LastPrintBuff("%02lu",(word)(dta[0]));
      }
      else if(inst->override==over_char)
      { LastPrintBuff("\"");
	for(i=0;i>=0;i--)
	  if(dta[i])
	    LastPrintBuff("%c",dta[i]);
	LastPrintBuff("\"");
      }
      else
      { if(inst->displayflags&DISPFLAG_NEGATE)
	{  LastPrintBuff("-");
	   LastPrintBuffLongHexValue(0x100-(word)(dta[0]));
	}
	else
	  LastPrintBuffLongHexValue((word)(dta[0]));
      }
      break;
  case ARG_IMM8_IND:
      dta=inst->data+inst->length-1;
      LastPrintBuff("(");
      LastPrintBuffLongHexValue((word)(dta[0]));
      LastPrintBuff(")");
      break;
  case ARG_IMM16:
      dta=inst->data+inst->length-2;
      if(inst->override==over_decimal)
      { if(inst->displayflags&DISPFLAG_NEGATE)
	  LastPrintBuff("-%02lu",0x10000-((word *)dta)[0]);
	else
	  LastPrintBuff("%02lu",((word *)dta)[0]);
      }
      else if(inst->override==over_char)
      { LastPrintBuff("\"");
	for(i=1;i>=0;i--)
	  if(dta[i])
	    LastPrintBuff("%c",dta[i]);
	LastPrintBuff("\"");
      }
      else
      { if(inst->displayflags&DISPFLAG_NEGATE)
	{  LastPrintBuff("-");
	   LastPrintBuffLongHexValue(0x10000-((word *)dta)[0]);
	}
	else
	  LastPrintBuffLongHexValue(((word *)dta)[0]);
      }
      break;
  case ARG_IMM16_A:
      dta=inst->data+inst->length-3;
      LastPrintBuffLongHexValue(((word *)dta)[0]);
      break;
  case ARG_RELIMM8:
      dta=inst->data+inst->length-1;
      if(inst->mode32) {
       targetd=(dword)((signed char)dta[0]+inst->addr.offs+inst->length);
       loc.assign(inst->addr.segm,targetd);
       PrintLongHex(loc);
      }else{
       targetw=(word)((signed char)dta[0]+inst->addr.offs+inst->length);
       loc.assign(inst->addr.segm,targetw);
       PrintLongHex(loc);
      }
      break;
  case ARG_RELIMM:
      dta=inst->data+inst->length;
      if(inst->mode32)
      { dta-=4;
	targetd=((dword *)dta)[0]+inst->addr.offs+inst->length;
	loc.assign(inst->addr.segm,targetd);
        PrintLongHex(loc);
      }
      else
      { dta-=2;
	targetw=(word)(((word *)dta)[0]+inst->addr.offs+inst->length);
	loc.assign(inst->addr.segm,targetw);
        PrintLongHex(loc);
      }
      break;
  case ARG_REG:
      dta=inst->data+inst->modrm;
      if(options.processor==PROC_Z80)
	LastPrintBuff(regzascii[dta[0]&0x07]);
      else if(((asminstdata *)(inst->tptr))->flags&FLAGS_8BIT)
	LastPrintBuff(reg8ascii[(dta[0]>>3)&0x07]);
      else if(inst->mode32)
	LastPrintBuff(reg32ascii[(dta[0]>>3)&0x07]);
      else
	LastPrintBuff(reg16ascii[(dta[0]>>3)&0x07]);
      break;
  case ARG_MREG:
      dta=inst->data+inst->modrm;
      LastPrintBuff(regmascii[(dta[0]>>3)&0x07]);
      break;
  case ARG_XREG:
      dta=inst->data+inst->modrm;
      LastPrintBuff(regxascii[(dta[0]>>3)&0x07]);
      break;
  case ARG_FREG:
      dta=inst->data+inst->modrm;
      LastPrintBuff(regfascii[dta[0]&0x07]);
      break;
  case ARG_SREG:
      dta=inst->data+inst->modrm;
      LastPrintBuff(regsascii[(dta[0]>>3)&0x07]);
      break;
  case ARG_CREG:
      dta=inst->data+inst->modrm;
      LastPrintBuff(regcascii[(dta[0]>>3)&0x07]);
      break;
  case ARG_DREG:
      dta=inst->data+inst->modrm;
      LastPrintBuff(regdascii[(dta[0]>>3)&0x07]);
      break;
  case ARG_TREG:
    case ARG_TREG_67:
      dta=inst->data+inst->modrm;
      LastPrintBuff(regtascii[(dta[0]>>3)&0x07]);
      break;
  case ARG_MODREG:
  case ARG_MMXMODRM:
  case ARG_XMMMODRM:
  case ARG_MODRM8:
  case ARG_MODRM16:
  case ARG_MODRM_S:
  case ARG_MODRMM512:
  case ARG_MODRMQ:
  case ARG_MODRM_SREAL:
  case ARG_MODRM_PTR:
  case ARG_MODRM_WORD:
  case ARG_MODRM_BCD:
  case ARG_MODRM_SINT:
  case ARG_MODRM_EREAL:
  case ARG_MODRM_DREAL:
  case ARG_MODRM_WINT:
  case ARG_MODRM_LINT:
  case ARG_MODRM_FPTR:
  case ARG_MODRM: {
   dta=inst->data+inst->modrm;
   rm=(byte)((dta[0]&0xc0)>>6);
   modrm=(byte)(dta[0]&0x07);
   a1=DSMITEM_ARG1(inst);
   a2=DSMITEM_ARG2(inst);
   sib=dta[1];
   if (a1==ARG_IMM || a2==ARG_IMM || a1==ARG_IMM8 || a2==ARG_IMM8 || a2==ARG_NONE
   ||a1==ARG_SIMM8 || a2==ARG_SIMM8 || modrm==5 && rm==0
   || modrm==4 && rm==2 && sib&0x07==5
   || modrm==4 && rm==0 && sib&0x07==5) {
    if (rm<3) {
     switch (a) {
      case ARG_MODRM8:		LastPrintBuff("byte ptr ");	break;
      case ARG_MODRM16:
      case ARG_MODRM_WORD:	LastPrintBuff("word ptr ");	break;
      case ARG_MMXMODRM:
      case ARG_XMMMODRM:	LastPrintBuff("dword ptr ");	break;
      case ARG_MODRMQ:		LastPrintBuff("qword ptr ");	break;
      case ARG_MODRM_S:		LastPrintBuff("fword ptr ");	break;	// 6 bytes=fword
      case ARG_MODRM_SREAL:	LastPrintBuff("dword ptr ");	break;	// single real=4 bytes=dword
      case ARG_MODRM_BCD:	LastPrintBuff("tbyte ptr ");	break;	// packed bcd=10 bytes=tbyte
      case ARG_MODRM_SINT:	LastPrintBuff("dword ptr ");	break;	// short int=4 bytes
      case ARG_MODRM_WINT:	LastPrintBuff("word ptr ");	break;	// word int =2 bytes
      case ARG_MODRM_LINT:	LastPrintBuff("qword ptr ");	break;	// long int = 8 bytes
      case ARG_MODRMM512:	LastPrintBuff("byte ptr ");	break;	// points to 512 bits=64 bytes of memory......
      case ARG_MODRM_EREAL:	LastPrintBuff("tbyte ptr ");	break;	// extended real=10 bytes
      case ARG_MODRM_DREAL:	LastPrintBuff("qword ptr ");	break;	// double real=8 bytes
      case ARG_MODRM:
       if (inst->flags&FLAGS_8BIT)	LastPrintBuff("byte ptr ");
       else if (inst->mode32)		LastPrintBuff("dword ptr ");
       else				LastPrintBuff("word ptr ");
       break;
     }/*switch(a)*/
    }
   }else if (a1==ARG_REG || a2==ARG_REG) {
    if (rm<3) {
     switch (a) {		    // re movzx, movsx type instructions
      case ARG_MODRM8:		LastPrintBuff("byte ptr ");	break;
      case ARG_MODRM16:		LastPrintBuff("word ptr ");	break;
     }/*switch(a)*/
    }
   }
   switch (rm) {
    case 0:
	  if(inst->flags&FLAGS_SEGPREFIX)
	    outprefix(pbyte);
	  if(options.mode32)
	  { if(modrm==5)
	    { loc.assign(inst->addr.segm,((dword *)(&dta[1]))[0]);
	      PrintMemVar(loc);
	    }
	    else if(modrm==4)	     // case 4=sib
	    { sib=dta[1];
	      if((sib&0x07)==5) // disp32
	      { loc.assign(inst->addr.segm,((dword *)(&dta[2]))[0]);
	        PrintMemVar(loc);
	      }
	      else
	      { LastPrintBuff("[%s]",reg32ascii[sib&0x07]);
	      }
	      if(((sib>>3)&0x07)==4) // no scaled index reg
	      {
	      }
	      else
	      { LastPrintBuff("[%s",reg32ascii[(sib>>3)&0x07]);
		switch(sib>>6)
		{ case 0:
		    LastPrintBuff("]");
		    break;
		  case 1:
		    LastPrintBuff("*2]");
		    break;
		  case 2:
		    LastPrintBuff("*4]");
		    break;
		  case 3:
		    LastPrintBuff("*8]");
		    break;
		}
	      }
	    }
	    else
	      LastPrintBuff("[%s]",reg32ascii[dta[0]&0x07]);
	  }
	  else
	  { if(modrm==6)
	    { loc.assign(inst->addr.segm,((word *)(&dta[1]))[0]);
	      PrintMemVar(loc);
	    }
	    else
	      LastPrintBuff("[%s]",regind16ascii[dta[0]&0x07]);
	  }
	  break;
    case 1:
	  if(inst->flags&FLAGS_SEGPREFIX)
	    outprefix(pbyte);
	  if(options.mode32)
	  { if(modrm==4)	// case 4=sib
	    { sib=dta[1];
	      if(dta[2]&0x80)
	      { LastPrintBuff("[%s-",reg32ascii[dta[1]&0x07]);
		LastPrintBuffHexValue((byte)(-dta[2]));
	      }
	      else
	      { LastPrintBuff("[%s+",reg32ascii[dta[1]&0x07]);
		LastPrintBuffHexValue(dta[2]);
	      }
	      if(((sib>>3)&0x07)==4) // no scaled index reg
		LastPrintBuff("]");
	      else
	      { LastPrintBuff("][%s",reg32ascii[(sib>>3)&0x07]);
		switch(sib>>6)
		{ case 0:
		    LastPrintBuff("]");
		    break;
		  case 1:
		    LastPrintBuff("*2]");
		    break;
		  case 2:
		    LastPrintBuff("*4]");
		    break;
		  case 3:
		    LastPrintBuff("*8]");
		    break;
		}
	      }
	    }
	    else if(dta[1]&0x80)
	    { LastPrintBuff("[%s-",reg32ascii[dta[0]&0x07]);
	      LastPrintBuffHexValue((byte)(-dta[1]));
	      LastPrintBuff("]");
	    }
	    else
	    { LastPrintBuff("[%s+",reg32ascii[dta[0]&0x07]);
	      LastPrintBuffHexValue(dta[1]);
	      LastPrintBuff("]");
	    }
	  }
	  else
	  { if(dta[1]&0x80)
	    { LastPrintBuff("[%s-",regind16ascii[dta[0]&0x07]);
	      LastPrintBuffHexValue((byte)(-dta[1]));
	      LastPrintBuff("]");
	    }
	    else
	    { LastPrintBuff("[%s+",regind16ascii[dta[0]&0x07]);
	      LastPrintBuffHexValue(dta[1]);
	      LastPrintBuff("]");
	    }
	  }
	  break;
    case 2:
     if (inst->flags&FLAGS_SEGPREFIX) outprefix(pbyte);
     if (options.mode32) {
      loc.assign(inst->addr.segm,((dword *)(&dta[1]))[0]);
      if (modrm==4) {	// case 4=sib
       sib=dta[1];
       loc.assign(inst->addr.segm,((dword *)(&dta[2]))[0]);
       LastPrintBuff("[");
       if (name->isname(loc)) {
        LastPrintBuff("%s+",reg32ascii[sib&0x07]);
	name->printname(loc);
       }else if (import->isname(loc)) {
        LastPrintBuff("%s+",reg32ascii[sib&0x07]);
	import->printname(loc);
       }else if(expt->isname(loc)) {
        LastPrintBuff("%s+",reg32ascii[sib&0x07]);
	expt->printname(loc);
       }else if(dta[5]&0x80) {
        LastPrintBuff("%s-",reg32ascii[sib&0x07]);
	LastPrintBuffLongHexValue(0-((dword *)(&dta[2]))[0]);
       }else{
        LastPrintBuff("%s+",reg32ascii[sib&0x07]);
	LastPrintBuffLongHexValue(((dword *)(&dta[2]))[0]);
       }
       if (((sib>>3)&0x07)==4) // no scaled index reg
	 LastPrintBuff("]");
       else{
        LastPrintBuff("][%s",reg32ascii[(sib>>3)&0x07]);
	switch (sib>>6)	{
	 case 0: LastPrintBuff("]");	break;
	 case 1: LastPrintBuff("*2]");	break;
	 case 2: LastPrintBuff("*4]");	break;
	 case 3: LastPrintBuff("*8]");	break;
	}
       }
      }else if (name->isname(loc)) {
       name->printname(loc);
       LastPrintBuff("[%s]",reg32ascii[dta[0]&0x07]);
      }else if (import->isname(loc)) {
       import->printname(loc);
       LastPrintBuff("[%s]",reg32ascii[dta[0]&0x07]);
      }else if (expt->isname(loc)) {
       expt->printname(loc);
       LastPrintBuff("[%s]",reg32ascii[dta[0]&0x07]);
      }else if (dta[4]&0x80) {
       LastPrintBuff("[%s-",reg32ascii[dta[0]&0x07]);
       LastPrintBuffLongHexValue(0-*(dword*)(dta+1));
       LastPrintBuff("]");
      }else{
       LastPrintBuff("[%s+",reg32ascii[dta[0]&0x07]);
       LastPrintBuffLongHexValue(*(dword*)(dta+1));
       LastPrintBuff("]");
      }
     }else{
      loc.assign(inst->addr.segm,((word *)(&dta[1]))[0]);
      if (name->isname(loc)) {
       name->printname(loc);
       LastPrintBuff("[%s]",regind16ascii[*dta&0x07]);
      }else if (import->isname(loc)) {
       import->printname(loc);
       LastPrintBuff("[%s]",regind16ascii[*dta&0x07]);
      }else if (expt->isname(loc)) {
       expt->printname(loc);
       LastPrintBuff("[%s]",regind16ascii[*dta&0x07]);
      }else if (dta[2]&0x80) {
       LastPrintBuff("[%s-",regind16ascii[*dta&0x07]);
       LastPrintBuffLongHexValue(0x10000-((word *)(&dta[1]))[0]);
       LastPrintBuff("]");
      }else{
       LastPrintBuff("[%s+",regind16ascii[dta[0]&0x07]);
       LastPrintBuffLongHexValue(((word *)(&dta[1]))[0]);
       LastPrintBuff("]");
      }
     }
    break;
    case 3:
     if (a==ARG_MMXMODRM) LastPrintBuff(regmascii[dta[0]&0x07]);
     else if(a==ARG_XMMMODRM)
       LastPrintBuff(regxascii[dta[0]&0x07]);
     else if((((asminstdata *)(inst->tptr))->flags&FLAGS_8BIT)||(a==ARG_MODRM8))
       LastPrintBuff(reg8ascii[dta[0]&0x07]);
     else if((inst->mode32)&&(a!=ARG_MODRM16))
       LastPrintBuff(reg32ascii[dta[0]&0x07]);
     else LastPrintBuff(reg16ascii[dta[0]&0x07]);
   }/*switch(rm)*/
  }break;
  case ARG_IMM_1:
      if(inst->override==over_decimal)
	LastPrintBuff("1");
      else
	LastPrintBuff("1h");
      break;
  case ARG_FADDR:
      dta=inst->data+inst->length;
      if(options.mode32)
      { dta-=6;
	loc.assign(((word *)(&dta[4]))[0],((dword *)(&dta[0]))[0]);
        if (!PrintFoundName(loc)) LastPrintBuff("%04x:%08lxh",loc.segm,loc.offs);
      }
      else
      { dta-=4;
	loc.assign(((word *)(&dta[2]))[0],((word *)(&dta[0]))[0]);
        if (!PrintFoundName(loc)) LastPrintBuff("%04x:%04xh",loc.segm,loc.offs);
      }
      break;
  case ARG_BIT:
      dta=inst->data+inst->length-1;
      LastPrintBuff("%x",(dta[0]>>3)&7);
      break;
  case ARG_NONE:
      if(inst->flags&FLAGS_SEGPREFIX)
	outprefix(pbyte);
      break;
  case ARG_NONEBYTE: break;
 }
 if (inst->flags&FLAGS_ADDRPREFIX)
   options.mode32=!options.mode32;
}

/************************************************************************
* outcomment								*
* - prints a disassembly comment to the buffer				*
************************************************************************/
void disio::outcomment(dsmitem *inst) {
 LastPrintBuff(";%s",inst->tptr);
}

/************************************************************************
* dumpblocktofile							*
* - this is the text and asm output routine. It needs a lot more work	*
*   doing to it and is presently a fairly simple dump of a block of	*
*   code.								*
* - this routine was a quick hack and complete rip of the dumptofile	*
*   routine which follows. the two routines need the common workings	*
*   put together and both need rewriting				*
************************************************************************/
void disio::dumpblocktofile(char *fname,bool printaddrs) {
 HANDLE efile;
 dword num;
 dsmitem *tblock,fdsm;
 dsegitem *dblock,*nxtblock;
 lptr outhere,nptr;
 int i;
 char ehdr[300];
 if (!blk.checkblock()) return;
 efile=CreateFile(fname,GENERIC_WRITE,0,NULL,CREATE_NEW,0,NULL);
 if(efile==INVALID_HANDLE_VALUE) {
  MessageBox(MainWnd,"File Creation Failed","Borg Disassembler Alert",MB_OK);
  return;
 }
// scheduler.stopthread();
 WriteFile(efile,"; ",2,&num,NULL);
 WriteFile(efile,winname,lstrlen(winname),&num,NULL);
 WriteFile(efile,"\r\n;\r\n",5,&num,NULL);
 WriteFile(efile,hdr,lstrlen(hdr),&num,NULL);
 WriteFile(efile,hdr2,lstrlen(hdr2),&num,NULL);
 WriteFile(efile,";\r\n",3,&num,NULL);
 wsprintf(ehdr,"; block dump from: %04x:%08lxh to %04x:%08lxh",blk.top.segm,blk.top.offs,blk.bottom.segm,blk.bottom.offs);
 WriteFile(efile,ehdr,lstrlen(ehdr),&num,NULL);
 WriteFile(efile,"\r\n",2,&num,NULL);
 WriteFile(efile,"\r\n",2,&num,NULL);
  // find current position.
 fdsm.addr=blk.top;
 fdsm.type=dsmnull;
 tblock=dsm->find(&fdsm);
 while (tblock) {
  if (tblock->addr<blk.top) tblock=dsm->nextiterator();
  else break;
 }
  // now tblock= first position
 dblock=dta->findseg(blk.top);
 outhere=blk.top;
 while (dblock && (outhere<=blk.bottom)) {
  ClearBuff(); // clear buffer - ready to start
  for (i=0;i<size.y;i++) {
   if (tblock) {
    if(outhere==tblock->addr)	{
     switch(tblock->type) {
      case dsmcode:
      outinst(tblock,printaddrs);
      break;
      case dsmnameloc:
      printlineheader(tblock->addr,printaddrs);
      LastPrintBuff("%s:",tblock->data);
      break;
      case dsmxref:
      // temp measure - print first (build 17)
      printlineheader(tblock->addr,printaddrs);
      LastPrintBuff(";");
      LastPrintBuffEpos(COMMENTPOS);
      LastPrintBuff("XREFS First: ");
      xrefs->printfirst(tblock->addr);
      break;
      default:
      printlineheader(tblock->addr,printaddrs);
      outcomment(tblock);
     }
     outhere+=tblock->length;
     tblock=dsm->nextiterator();
    }else{
     outdb(&outhere,printaddrs);
     outhere++;
    }
   }else{
    outdb(&outhere,printaddrs);
    outhere++;
   }
     // check if gone beyond seg, get next seg.
     /*if(dta->beyondseg(outhere))*/	 // changed build 14
     // rewritten build 17. seeks dseg from start, and finds next now.
   if (outhere>=(dblock->addr+dblock->size)) {
    dta->resetiterator();
    nxtblock=dta->nextiterator();
    while (nxtblock) {
     if (nxtblock->addr==dblock->addr) {
      dblock=dta->nextiterator();
      break;
     }
     nxtblock=dta->nextiterator();
     if (!nxtblock) dblock=NULL;
    }
    if (!dblock) break;
    outhere=dblock->addr;
   }
   if (outhere>blk.bottom) break;
   if (!outhere.segm) break;
  }
  DumpBuff(efile);
 }
 DoneBuff();
 CloseHandle(efile);
// scheduler.continuethread();
 scheduler.addtask(windowupdate,priority_window,nlptr,NULL);
}

/************************************************************************
* dumptofile								*
* - this is the text and asm output routine. It needs a lot more work	*
*   doing to it and is presently a fairly simple dump of the code.	*
************************************************************************/
void disio::dumptofile(char *fname,bool printaddrs)
{ SECURITY_ATTRIBUTES securityatt;
  HANDLE efile;
  dword num;
  dsmitem *tblock;
  dsegitem *dblock,*nxtblock;
  lptr outhere,nptr;
  int i;
  securityatt.nLength=sizeof(SECURITY_ATTRIBUTES);
  securityatt.lpSecurityDescriptor=NULL;
  securityatt.bInheritHandle=false;
  efile=CreateFile(fname,GENERIC_WRITE,0,&securityatt,CREATE_NEW,0,NULL);
  if(efile==INVALID_HANDLE_VALUE)
  { MessageBox(MainWnd,"File Creation Failed","Borg Disassembler Alert",MB_OK);
    return;
  }
//  scheduler.stopthread();
  WriteFile(efile,"; ",2,&num,NULL);
  WriteFile(efile,winname,lstrlen(winname),&num,NULL);
  WriteFile(efile,"\r\n;\r\n",5,&num,NULL);
  WriteFile(efile,hdr,lstrlen(hdr),&num,NULL);
  WriteFile(efile,hdr2,lstrlen(hdr2),&num,NULL);
  WriteFile(efile,"\r\n",2,&num,NULL);
  // find current position.
  dsm->resetiterator();
  tblock=dsm->nextiterator();
  // now tblock= first position
  dta->resetiterator();
  dblock=dta->nextiterator();
  outhere=dblock->addr;
 while (dblock) {
  ClearBuff(); // clear buffer - ready to start
  for (i=0;i<size.y;i++) {
   if (tblock) {
    if (outhere==tblock->addr) {
     switch (tblock->type) {
      case dsmcode:
	      outinst(tblock,printaddrs);
	      break;
      case dsmnameloc:
	      printlineheader(tblock->addr,printaddrs);
	      LastPrintBuff("%s:",tblock->data);
	      break;
      case dsmxref:
	      // temp measure - print first (build 17)
	      printlineheader(tblock->addr,printaddrs);
	      LastPrintBuff(";");
	      LastPrintBuffEpos(COMMENTPOS);
	      LastPrintBuff("XREFS First: ");
	      xrefs->printfirst(tblock->addr);
	      break;
      default:
	      printlineheader(tblock->addr,printaddrs);
	      outcomment(tblock);
     }
     outhere+=tblock->length;
     tblock=dsm->nextiterator();
    }else{
     outdb(&outhere,printaddrs);
     outhere++;
    }
   }else{
    outdb(&outhere,printaddrs);
    outhere++;
   }
      // check if gone beyond seg, get next seg.
      /*if(dta->beyondseg(outhere))*/	 // changed build 14
      // rewritten build 17. seeks dseg from start, and finds next now.
   if (outhere>=(dblock->addr+dblock->size)) {
    dta->resetiterator();
    nxtblock=dta->nextiterator();
    while (nxtblock!=NULL) {
     if (nxtblock->addr==dblock->addr) {
      dblock=dta->nextiterator();
      break;
     }
     nxtblock=dta->nextiterator();
     if (!nxtblock) dblock=NULL;
    }
    if (!dblock) break;
    outhere=dblock->addr;
   }
   if (!outhere.segm) break;
  }
  DumpBuff(efile);
 }
 DoneBuff();
 CloseHandle(efile);
// scheduler.continuethread();
 scheduler.addtask(windowupdate,priority_window,nlptr,NULL);
}

/************************************************************************
*		  mainwind.cpp						*
* This is one of a few files which contain some old code, a lot of	*
* hacks, a lot of hastily stuffed lines and variables, etc, and is in	*
* in need of an overhaul. Its also why I left commenting it until the	*
* last four files (disasm,fileload,dasm and this one!). Whilst making	*
* these comments I have taken a small opportunity to tidy it a little.	*
* This file controls output to the main window. It keeps a small buffer *
* of disassembled lines, and it uses this buffer to repaint the main	*
* window when WM_PAINT requests are received.				*
* Whenever the disassembly is changed or the user scrolls there is a	*
* windowupdate request sent to the scheduler with a high priority.	*
* When the windowupdate is processed it is passed to the windowupdate	*
* or scroller function in disio. disio will regenerate the buffer as	*
* required and then invalidate the main window so that a repaint is	*
* done.									*
* The functions here include basic formatted output to the buffer as	*
* well as the window repaints. Repaints will not be done while the	*
* buffer is being updated, but will wait for the buffer update to	*
* finish and then do the repaint.					*
* The functions also use extensive critical sections which ensure that	*
* we do not start clearing the buffer during a window update, etc	*
************************************************************************/

/************************************************************************
* DoPaint								*
* - This is the main painting routine. If the program is quitting then	*
*   it returns (we dont want thread clashes due to critical sections	*
*   here and theres no point to repainting when we're exitting).	*
* - If the buffer is not ready then we wait, and go to sleep.		*
* - Otherwise the routine paints the screen from the buffer, using the	*
*   selected font and colours						*
************************************************************************/
void disio::DoPaint(void) {
 PAINTSTRUCT ps;    // holds PAINT information
 int i;

// while (bufferbusy) Sleep(0);	// wait if filling buffer
// if (KillThread) return;

// EnterCriticalSection(&cs);
 BeginPaint(wnd,&ps);
 SelectObject(ps.hdc,Font);
 SetTextColor(ps.hdc,options.textcolor);
 for (i=0;i<size.y;i++) {
  RECT R;
  SetBkColor(ps.hdc,
    i==sel ? options.highcolor : options.bgcolor);
  GetItemRect(i,&R);
  ExtTextOut(ps.hdc,-hpos*CharBox.x,R.top,ETO_OPAQUE,&R,
    MainBuff[i],i<lastline?lstrlen(MainBuff[i]):0,NULL);
 }
 EndPaint(wnd,&ps);
// LeaveCriticalSection(&cs);
}

/************************************************************************
* ClearBuff								*
* - This should be called before each reworking of the buffer. It	*
*   clears the buffer and stops any repainting from taking place.	*
* - It also resets the line pointer to the start of the buffer.		*
************************************************************************/
void disio::ClearBuff(void) {
// int i;
// EnterCriticalSection(&cs);
// for (i=0;i<size.y;i++) {
//  MainBuff[i][0]='\0';
// }
 lastline=0;
// bufferbusy=true;
// LeaveCriticalSection(&cs);
}

/************************************************************************
* DoneBuff								*
* - This should be called after a reworking of the buffer. It reenables *
*   window repainting.							*
************************************************************************/
void disio::DoneBuff(void) {
// EnterCriticalSection(&cs);
// bufferbusy=false;
// LeaveCriticalSection(&cs);
}

/************************************************************************
* PrintBuff								*
* - This is the printf of the buffer and is similar to wvsprintf but	*
*   output is to the main buffer. Note that the line pointer is moved	*
*   on after a call, so we move to the next line automatically.		*
************************************************************************/
// adds next line to the buffer.
void disio::PrintBuff(char *szFormat,...) {
// EnterCriticalSection(&cs);
 va_list vaArgs;
 va_start(vaArgs,szFormat);
 if (lastline<size.y) {
  wvsprintf(MainBuff[lastline++],szFormat,vaArgs);
 }
 va_end(vaArgs);
  // just zero end of buffer.....in case.
//  MainBuff[BUFFER_LINES*max_length]=0;
//  LeaveCriticalSection(&cs);
}

/************************************************************************
* LastPrintBuffEpos							*
* - Often we use PrintBuff followed by LastPrintBuff to construct a	*
*   line of output a piece at a time. This function provides basic	*
*   formatting by allowing us to set the cursor position on the last	*
*   line printed, by adding spaces until the position.			*
************************************************************************/
// adds blanks to previous line to set position
void disio::LastPrintBuffEpos(int xpos) {
 int i;
 int spos;
 if (!lastline) return;
// EnterCriticalSection(&cs);
 spos=lastline-1;
 i=lstrlen(MainBuff[spos]);
 while (i<xpos) {
  MainBuff[spos][i]=' ';
  i++;
  MainBuff[spos][i]='\0';
 }
// LeaveCriticalSection(&cs);
}

/************************************************************************
* LastPrintBuffHexValue							*
* - Same as LastPrintBuff, but prints num only, in hex. It prints a	*
*   leading zero where the leading char is alpha.			*
************************************************************************/
void disio::LastPrintBuffHexValue(byte num)
{ char tstr[20];
  wsprintf(tstr,"%02xh",num);
  if((tstr[0]>='a')&&(tstr[0]<='f'))
    LastPrintBuff("0");
  LastPrintBuff("%02xh",num);
}

/************************************************************************
* LastPrintBuffHexValue							*
* - Same as LastPrintBuff, but prints num only, in hex. It prints a	*
*   leading zero where the leading char is alpha.			*
************************************************************************/
void disio::LastPrintBuffLongHexValue(dword num) {
 char tstr[20];
 wsprintf(tstr,"%02lxh",num);
 if (tstr[0]>='a' && tstr[0]<='f') LastPrintBuff("0");
 LastPrintBuff(tstr);	// doesn't contain "%"
}

/************************************************************************
* LastPrintBuff								*
* - This is the same as PrintBuff except that instead of printing a new *
*   line it goes back to the last line and adds more to the end of it	*
* - So a set of calls tends to look like PrintBuff, LastPrintBuffEPos,	*
*   LastPrintBuff, LastPrintBuffEPos, LastPrintBuff, PrintBuff, etc	*
************************************************************************/
void disio::LastPrintBuff(char *szFormat,...) {
 int spos,len;
 char s[MAX_LENGTH];

 if (!lastline) return;
 spos=lastline-1;
// EnterCriticalSection(&cs);
 va_list vaArgs;
 va_start(vaArgs,szFormat);
 wvsprintf(s,szFormat,vaArgs);
 va_end(vaArgs);
 len=lstrlen(MainBuff[spos]);
 lstrcpyn(MainBuff[spos]+len,s,MAX_LENGTH-len);	// append and watch size
// LeaveCriticalSection(&cs);
}

/************************************************************************
* DumpBuff								*
* - Now this is a real hack. Instead of writing proper file IO routines *
*   I simply write a buffer full at a time, and dump each to a file.... *
*   Then when I've written the file out I regenerate the buffer for the *
*   display again....... to be rewritten......				*
************************************************************************/
void disio::DumpBuff(HANDLE efile) {
 DWORD num;
 int i;
 for (i=0;i<lastline;i++) {
  WriteFile(efile,MainBuff[i],lstrlen(MainBuff[i]),&num,NULL);
  WriteFile(efile,"\r\n",2,&num,NULL);
 }
}
Detected encoding: OEM (CP437)1
Wrong umlauts? -
Assume file is OEM (CP437) encoded