I64 InstEntriesCompare(CInst *tmpins1,CInst *tmpins2)
{
  I64 i1,i2,j=0,res=0,oc_cnt1=tmpins1->opcode_cnt,oc_cnt2=tmpins2->opcode_cnt;
  if (tmpins1->flags&IEF_STI_LIKE)
    oc_cnt1--;
  if (tmpins2->flags&IEF_STI_LIKE)
    oc_cnt2--;
  while (TRUE) {
    if (j<oc_cnt1 && j<oc_cnt2) {
      if (res=tmpins1->opcode[j]-tmpins2->opcode[j])
        return res;
      j++;
    } else {
      if (res=oc_cnt1-oc_cnt2)
        return res;

      if (tmpins1->flags&IEF_STI_LIKE && tmpins2->flags&IEF_STI_LIKE)
        return tmpins1->opcode[j]-tmpins2->opcode[j];

      if (res=tmpins1->flags&IEF_STI_LIKE - tmpins2->flags&IEF_STI_LIKE)
        return res;

      if (res=tmpins1->slash_val-tmpins2->slash_val)
        return res;

      if (res=tmpins1->flags&IEF_OP_SIZE32 - tmpins2->flags&IEF_OP_SIZE32)
        return res;

      i1=Bt(&uasm.ins64_arg_mask,tmpins1->arg1) ||
            Bt(&uasm.ins64_arg_mask,tmpins1->arg2);
      i2=Bt(&uasm.ins64_arg_mask,tmpins2->arg1) ||
            Bt(&uasm.ins64_arg_mask,tmpins2->arg2);
      if (res=i1-i2)
        return res;

      if (res=tmpins1->flags&IEF_48_REX - tmpins2->flags&IEF_48_REX)
        return res;

      i1=tmpins1->arg2==ARGT_IMM64 || tmpins1->arg2==ARGT_UIMM64;
      i2=tmpins2->arg2==ARGT_IMM64 || tmpins2->arg2==ARGT_UIMM64;
      return i1-i2;
    }
  }
}

/*
U0 DumpUAsmIns(CInst *tmpins)
{
  CHashOpcode *tmpo=tmpins(U8 *)-tmpins->ins_entry_num*sizeof(CInst)
  -offset(CHashOpcode.ins);
  "%10s:%02d,%02d SV:%02d\n",tmpo->str,
        tmpins->arg1,tmpins->arg2,tmpins->slash_val;
}
U0 DumpUAsmTables()
{
  I64 k;
  "16/32 Bit Table\n";
  for (k=0;k<uasm.table_16_32_entries;k++)
    DumpUAsmIns(uasm.table_16_32[k]);
  "\n\n\n\n64 Bit Table\n";
  for (k=0;k<uasm.table_64_entries;k++)
    DumpUAsmIns(uasm.table_64[k]);
}
*/

CInst *InstEntryFind(U8 *rip,I64 opsize,I64 seg_size)
{//Binary Search
  I64 i,j,n,m,k,arg1,arg2,o1,o2,oc_cnt;
  CInst *tmpins,**table;
  i=0;
  if (seg_size==64) {
    table=uasm.table_64;
    j=uasm.table_64_entries-1;
  } else {
    table=uasm.table_16_32;
    j=uasm.table_16_32_entries-1;
  }
  while (TRUE) {
    k=(i+j)>>1;  //binary search
    tmpins=table[k];
//DumpUAsmIns(tmpins);
    m=0;
    n=0;
    while (TRUE) { //ief_compare_start
      arg1=tmpins->arg1;
      arg2=tmpins->arg2;
      oc_cnt=tmpins->opcode_cnt;
      if (tmpins->flags&IEF_STI_LIKE)
        oc_cnt--;
      if (n<oc_cnt) {
        o1=rip[n];
        if (n==tmpins->opcode_cnt-1 && tmpins->flags & IEF_PLUS_OPCODE)
          o1&=-8;
        o2=tmpins->opcode[n++];
        if (m=o1-o2)
          goto ief_compare_done;
      } else
        switch [tmpins->uasm_slash_val] {
          case 0...7:
            if (!(m=rip[n]>>3&7-tmpins->slash_val)) {
              if ((Bt(&uasm.mem_arg_mask,arg1) ||
                    Bt(&uasm.mem_arg_mask,arg2)) &&
                    rip[n]&0xC0==0xC0) {
                m=1;
                goto ief_compare_done;
              }
              if (opsize==16) {
                if (tmpins->flags & IEF_OP_SIZE32) {
                  m=-1;
                  goto ief_compare_done;
                }
              } else {
                if (tmpins->flags & IEF_OP_SIZE16) {
                  m=1;
                  goto ief_compare_done;
                }
              }
              if (opsize==64||arg1==ARGT_M64||arg2==ARGT_M64) {
                if (!Bt(&uasm.ins64_arg_mask,arg1)&&
                      !Bt(&uasm.ins64_arg_mask,arg2)&&
                      !(tmpins->flags&IEF_48_REX))
                  m=1;
              } else {
                if (Bt(&uasm.ins64_arg_mask,arg1)||
                      Bt(&uasm.ins64_arg_mask,arg2) ||
                      tmpins->flags&IEF_48_REX)
                  m=-1;
              }
            } else if ((Bt(&uasm.mem_arg_mask,arg1)||
                  Bt(&uasm.mem_arg_mask,arg2)) &&
                  rip[n]&0xC0==0xC0)
              m=1;
            goto ief_compare_done;
          case SV_I_REG:
            m=rip[n]>>3-tmpins->opcode[tmpins->opcode_cnt-1]>>3;
            goto ief_compare_done;
          case SV_STI_LIKE:
            if (!(m=rip[n]>>3-tmpins->opcode[tmpins->opcode_cnt-1]>>3))
              m=rip[n]-tmpins->opcode[tmpins->opcode_cnt-1];
            goto ief_compare_done;
          case SV_R_REG:
          case SV_NONE:
            m=0;
            if (opsize==16) {
              if (tmpins->flags & IEF_OP_SIZE32) {
                m=-1;
                goto ief_compare_done;
              }
            } else {
              if (tmpins->flags & IEF_OP_SIZE16) {
                m=1;
                goto ief_compare_done;
              }
            }
            if (opsize==64 || arg1==ARGT_M64 || arg2==ARGT_M64) {
              if (!Bt(&uasm.ins64_arg_mask,arg1) &&
                    !Bt(&uasm.ins64_arg_mask,arg2) &&
                    !(tmpins->flags&IEF_48_REX)&& !(arg2==ARGT_NONE &&
                    (ARGT_UIMM8<=arg1<=ARGT_UIMM64 ||
                    ARGT_IMM8<=arg1<=ARGT_IMM64)))
                m=1;
              else if (tmpins->arg2==ARGT_IMM64 || tmpins->arg2==ARGT_UIMM64) {
                if (arg2!=ARGT_IMM64&&arg2!=ARGT_UIMM64)
                  m=1;
              } else if (arg2==ARGT_IMM64||arg2==ARGT_UIMM64)
                m=-1;
            } else {
              if (Bt(&uasm.ins64_arg_mask,arg1) ||
                    Bt(&uasm.ins64_arg_mask,arg2) ||
                    tmpins->flags&IEF_48_REX)
                m=-1;
            }
            goto ief_compare_done;
        }
    }
ief_compare_done:
    if (m>0) {
      if (k==i) {
        k=j;
        break;
      } else
        i=k;
    } else if (m<0) {
      if (k-i<=1) {
        k=i;
        break;
      } else
        j=k;
    } else
      break;
  }
  return table[k];
}

U0 UAsmHashLoad()
{
  CHashOpcode *tmph;
  CInst *tmpins;
  I64 i,j1,j2,k;

  uasm.ins64_arg_mask=0x0880888880+1<<ARGT_ST0+1<<ARGT_STI;
  uasm.signed_arg_mask=1<<ARGT_REL8+1<<ARGT_REL16+1<<ARGT_REL32+
        1<<ARGT_IMM8+1<<ARGT_IMM16+1<<ARGT_IMM32+1<<ARGT_IMM64;
  uasm.mem_arg_mask=1<<ARGT_M8+1<<ARGT_M16+1<<ARGT_M32+1<<ARGT_M64;

  uasm.table_16_32_entries=uasm.table_64_entries=0;
  for (i=0;i<=cmp.asm_hash->mask;i++) {
    tmph=cmp.asm_hash->body[i];
    while (tmph) {
      if (tmph->type==HTT_OPCODE && !(tmph->oc_flags&OCF_ALIAS)) {
        tmpins=&tmph->ins;
        for (k=0;k<tmph->inst_entry_cnt;k++) {
          uasm.table_16_32_entries++;
          if (!(tmpins->flags&IEF_NOT_IN_64_BIT))
            uasm.table_64_entries++;
          tmpins++;
        }
      }
      tmph=tmph->next;
    }
  }

  j1=j2=0;
  uasm.table_16_32=MAlloc(uasm.table_16_32_entries*sizeof(U8 *));
  uasm.table_64   =MAlloc(uasm.table_64_entries   *sizeof(U8 *));
  for (i=0;i<=cmp.asm_hash->mask;i++) {
    tmph=cmp.asm_hash->body[i];
    while (tmph) {
      if (tmph->type==HTT_OPCODE && !(tmph->oc_flags&OCF_ALIAS)) {
        tmpins=&tmph->ins;
        for (k=0;k<tmph->inst_entry_cnt;k++) {
          uasm.table_16_32[j1++]=tmpins;
          if (!(tmpins->flags&IEF_NOT_IN_64_BIT))
            uasm.table_64[j2++]=tmpins;
          tmpins++;
        }
      }
      tmph=tmph->next;
    }
  }
  QSortI64(uasm.table_16_32,uasm.table_16_32_entries,&InstEntriesCompare);
  QSortI64(uasm.table_64   ,uasm.table_64_entries   ,&InstEntriesCompare);
}

U0 Ui(U8 *buf,U8 **_rip,I64 seg_size=64,I64 *_jmp_dst=NULL,Bool just_ins=FALSE)
{//Unassembles one inst
  I64 i,disp,imm,opsize,opadd,
        arg1,arg2,reloced_arg1,reloced_arg2,
        arg1_size=0,arg2_size=0,reloced_arg1_size,reloced_arg2_size,
        ModrM=-1,SIB=-1,scale,r1,r2,
        Mod=-1,RM1=-1,RM2=-1,REX=-1,REX_r=0,REX_x=0,REX_b=0;
  Bool cont;
  CInst *tmpins,*tmpins2;
  CHashOpcode *tmpo;
  U8 *rip=*_rip,*ptr,*reloced_arg1_st,*reloced_arg2_st,
        *bin_data_area1,*bin_data_area2,
        line1[512],line2[512],buf2[512],arg1_st[512],
        arg2_st[512],seg_overrides[32];

  if (_jmp_dst) *_jmp_dst=-1;
  if (seg_size==16) {
    opsize=16;
    opadd=16;
  } else if (seg_size==32) {
    opsize=32;
    opadd=32;
  } else {
    opsize=32;
    opadd=64;
  }
  *arg1_st=0;
  *arg2_st=0;
  if (!IsRaw && PutSrcLink(rip,1,line1))
    CatPrint(line1,"\n");
  else
    *line1=0;

  StrPrint(line1+StrLen(line1),"%24tp ",rip);
  bin_data_area1=line1+StrLen(line1);
  for (i=0;i<6;i++)
    CatPrint(line1,"%02X",rip[i]);
  CatPrint(line1," ");

  StrPrint(line2,"%24tp ",rip+6);
  bin_data_area2=line2+StrLen(line2);
  for (i=6;i<12;i++)
    CatPrint(line2,"%02X",rip[i]);

  *seg_overrides=0;
  cont=TRUE;
  while (TRUE) {
    switch (*rip) {
      case 0x2E: if (StrLen(seg_overrides)<24)
          CatPrint(seg_overrides,"CS:"); break;
      case 0x36: if (StrLen(seg_overrides)<24)
          CatPrint(seg_overrides,"SS:"); break;
      case 0x3E: if (StrLen(seg_overrides)<24)
          CatPrint(seg_overrides,"DS:"); break;
      case 0x26: if (StrLen(seg_overrides)<24)
          CatPrint(seg_overrides,"ES:"); break;
      case 0x64: if (StrLen(seg_overrides)<24)
          CatPrint(seg_overrides,"FS:"); break;
      case 0x65: if (StrLen(seg_overrides)<24)
          CatPrint(seg_overrides,"GS:"); break;
      case OC_OP_SIZE_PREFIX:
        if (opsize==16)
          opsize=32;
        else
          opsize=16;
        break;
      case OC_ADDR_SIZE_PREFIX:
        if (opadd==16)
          opadd=32;
        else
          opadd=16;
        break;
      case 0x40...0x4F:
        if (seg_size==64) {
          REX=*rip;
          if (REX>=0x48)
            opsize=64;
          REX_b=Bt(&REX,0)<<3;
          REX_x=Bt(&REX,1)<<3;
          REX_r=Bt(&REX,2)<<3;
          break;
        } //Fall thru if !64
      default:
        cont=FALSE;
    }
    if (cont)
      rip++;
    else
      break;
  }

  tmpins=InstEntryFind(rip,opsize,seg_size);
  if (opsize==32 && seg_size==64) {
    tmpins2=InstEntryFind(rip,64,seg_size);
    if (tmpins2!=tmpins && tmpins2->flags&IEF_REX_ONLY_R8_R15 ||
          tmpins2->flags&IEF_REX_XOR_LIKE&& rip[1]>>3&7==rip[1]&7)
      tmpins=tmpins2;
  }

  rip+=tmpins->opcode_cnt;
  tmpo=tmpins(U8 *)-tmpins->ins_entry_num*sizeof(CInst)
  -offset(CHashOpcode.ins);
  if (just_ins)
    *line1=0;
  CatPrint(line1,tmpo->str);

  arg1=tmpins->arg1;
  arg2=tmpins->arg2;

  if (arg1_size=tmpins->size1) {
    if (Bt(&uasm.signed_arg_mask,arg1))
      CatPrint(arg1_st,"I%d ",arg1_size);
    else
      CatPrint(arg1_st,"U%d ",arg1_size);
  }

  if (arg2_size=tmpins->size2) {
    if (Bt(&uasm.signed_arg_mask,arg2))
      CatPrint(arg2_st,"I%d ",arg2_size);
    else
      CatPrint(arg2_st,"U%d ",arg2_size);
  }

  if (tmpins->flags & IEF_PLUS_OPCODE) {
    rip--;
    RM1=*rip++ - tmpins->opcode[tmpins->opcode_cnt-1]+REX_b;
    ptr=NULL;
    if (ARGT_R8<=arg1<=ARGT_R64) {
      if (arg1_size==8) {
        if (REX!=-1)
          ptr="ST_U8_REX_REGS";
        else
          ptr="ST_U8_REGS";
      } else if (arg1_size==16)
        ptr="ST_U16_REGS";
      else if (arg1_size==32)
        ptr="ST_U32_REGS";
      else if (arg1_size==64)
        ptr="ST_U64_REGS";
      if (ptr)
        CatPrint(arg1_st,"%Z",RM1,ptr);
    } else {
      if (arg2_size==8) {
        if (REX!=-1)
          ptr="ST_U8_REX_REGS";
        else
          ptr="ST_U8_REGS";
      } else if (arg2_size==16)
        ptr="ST_U16_REGS";
      else if (arg2_size==32)
        ptr="ST_U32_REGS";
      else if (arg2_size==64)
        ptr="ST_U64_REGS";
      if (ptr)
        CatPrint(arg2_st,"%Z",RM1,ptr);
    }
  }

  if (ARGT_RM8<=arg1<=ARGT_RM64 || ARGT_M8<=arg1<=ARGT_M64 ||
        ARGT_RM8<=arg2<=ARGT_RM64 || ARGT_M8<=arg2<=ARGT_M64) {
    if (ARGT_RM8<=arg2<=ARGT_RM64 || ARGT_M8<=arg2<=ARGT_M64) {
      reloced_arg1=arg2;
      reloced_arg2=arg1;
      reloced_arg1_size=arg2_size;
      reloced_arg2_size=arg1_size;
      reloced_arg1_st=arg2_st;
      reloced_arg2_st=arg1_st;
    } else {
      reloced_arg1=arg1;
      reloced_arg2=arg2;
      reloced_arg1_size=arg1_size;
      reloced_arg2_size=arg2_size;
      reloced_arg1_st=arg1_st;
      reloced_arg2_st=arg2_st;
    }

    CatPrint(reloced_arg1_st,seg_overrides);
    ModrM=*rip++;
    Mod=ModrM>>6 & 3;
    RM1=ModrM & 7+REX_b;
    RM2=ModrM>>3 & 7+REX_r;
    if (Mod<3 && RM1&7==4)
      SIB=*rip++;
    if (Mod==1) {
      disp=*rip(U8 *)++;
      CatPrint(reloced_arg1_st,"%02X",disp);
    } else if (Mod==2) {
      disp=*rip(U32 *)++;
      CatPrint(reloced_arg1_st,"%08X",disp);
    }
    if (tmpins->slash_val<8)
      RM2=-1;
    else {
      ptr=NULL;
      if (reloced_arg2==ARGT_SREG) {
        if (RM2<=5)
          ptr="ST_SEG_REGS";
      } else if (!(ARGT_IMM8<=reloced_arg2<=ARGT_IMM64) &&
            !(ARGT_UIMM8<=reloced_arg2<=ARGT_UIMM64)) {
        if (reloced_arg2_size==8) {
          if (REX!=-1)
            ptr="ST_U8_REX_REGS";
          else
            ptr="ST_U8_REGS";
        } else if (reloced_arg2_size==16)
          ptr="ST_U16_REGS";
        else if (reloced_arg2_size==32)
          ptr="ST_U32_REGS";
        else if (reloced_arg2_size==64)
          ptr="ST_U64_REGS";
      }
      if (ptr)
        CatPrint(reloced_arg2_st,"%Z",RM2,ptr);
    }
    if (RM1&7==5 && !Mod) {
      disp=*rip(I32 *)++;
      if (seg_size==64) {
        disp+=rip;
        if (reloced_arg2==ARGT_IMM8 || reloced_arg2==ARGT_UIMM8)
          disp++;
        else if (reloced_arg2==ARGT_IMM16 || reloced_arg2==ARGT_UIMM16)
          disp+=2;
        else if (reloced_arg2==ARGT_IMM32 || reloced_arg2==ARGT_UIMM32)
          disp+=4;
        else if (reloced_arg2==ARGT_IMM64 || reloced_arg2==ARGT_UIMM64)
          disp+=8;
      }
      CatPrint(reloced_arg1_st,"[%X]",disp);
      RM1=-1;
    } else {
      if (Mod<3) {
        if (RM1&7==4) {
          RM1=-1;
          r1=SIB & 7+REX_b;
          r2=SIB>>3 & 7+REX_x;
          scale=SIB>>6 &3;
          if (scale==3)
            scale=8;
          else if (scale==2)
            scale=4;
          else if (scale==1)
            scale=2;
          else
            scale=1;
          if (seg_size==64)
            ptr="ST_U64_REGS";
          else
            ptr="ST_U32_REGS";
          if (r1==REG_RBP && !Mod) {
            disp=*rip(U32 *)++;
            CatPrint(reloced_arg1_st,"%08X[%Z*%d]",disp,r2,ptr,scale);
          } else if (r2==4)
            CatPrint(reloced_arg1_st,"[%Z]",r1,ptr);
          else
            CatPrint(reloced_arg1_st,"[%Z+%Z*%d]",r1,ptr,r2,ptr,scale);
        } else {
          if (opadd==16)
            ptr="ST_U16_REGS";
          else if (opadd==32)
            ptr="ST_U32_REGS";
          else
            ptr="ST_U64_REGS";
          CatPrint(reloced_arg1_st,"[%Z]",RM1,ptr);
        }
      } else {
        ptr=NULL;
        if (reloced_arg1_size==8) {
          if (REX!=-1)
            ptr="ST_U8_REX_REGS";
          else
            ptr="ST_U8_REGS";
        } else if (reloced_arg1_size==16)
          ptr="ST_U16_REGS";
        else if (reloced_arg1_size==32)
          ptr="ST_U32_REGS";
        else if (reloced_arg1_size==64)
          ptr="ST_U64_REGS";
        if (ptr)
          CatPrint(reloced_arg1_st,DefineSub(RM1,ptr));
      }
    }
  }

  switch (arg1) {
    case ARGT_IMM8:
    case ARGT_UIMM8:
      imm=*rip(U8 *)++;
      CatPrint(arg1_st,"%02X",imm);
      if (tmpins->opcode[0]==0xCD && (ptr=DefineSub(imm,"ST_INT_NAMES")))
        CatPrint(arg1_st," %s",ptr);
      break;
    case ARGT_IMM16:
    case ARGT_UIMM16:
      CatPrint(arg1_st,"%04X",*rip(U16 *)++);
      break;
    case ARGT_IMM32:
    case ARGT_UIMM32:
      CatPrint(arg1_st,"%08X",*rip(U32 *)++);
      break;
    case ARGT_IMM64:
    case ARGT_UIMM64:
      CatPrint(arg1_st,"%016X",*rip(I64 *)++);
      break;
    start:
      case ARGT_REL8:
        disp=*rip(I8 *)++;
        break;
      case ARGT_REL16:
        disp=*rip(I16 *)++;
        break;
      case ARGT_REL32:
        disp=*rip(I32 *)++;
        break;
    end:
      disp+=rip;
      if (IsDbgMode)
        CatPrint(arg1_st,"%p ",disp);
      else if (PutSrcLink(disp,512,buf2))
        CatPrint(arg1_st,"%s ",buf2);
      else
        CatPrint(arg1_st,"%P ",disp);
      if (_jmp_dst) *_jmp_dst=disp;
      break;
    case ARGT_MOFFS8...ARGT_MOFFS64:
      CatPrint(arg1_st,seg_overrides);
      if (arg1_size==8)
        disp=*rip(U8 *)++;
      else if (opadd==16)
        disp=*rip(U16 *)++;
      else
        disp=*rip(U32 *)++;
      CatPrint(arg1_st,"[%X]",disp);
      break;
    case ARGT_AL ... ARGT_DX:
    case ARGT_SS ... ARGT_ST0:
      CatPrint(arg1_st,"%z",arg1-ARGT_AL,
            "AL\0AX\0EAX\0RAX\0CL\0DX\0 \0 \0SS\0DS\0ES\0FS\0GS\0CS\0ST0\0");
      break;
    case ARGT_STI:
      rip--;
      CatPrint(arg1_st,"%Z",*rip++ - tmpins->opcode[tmpins->opcode_cnt-1],
            "ST_FSTK_REGS");
      break;
  }

  switch (arg2) {
    case ARGT_IMM8:
    case ARGT_UIMM8:
      CatPrint(arg2_st,"%02X",*rip(U8 *)++);
      break;
    case ARGT_IMM16:
    case ARGT_UIMM16:
      CatPrint(arg2_st,"%04X",*rip(U16 *)++);
      break;
    case ARGT_IMM32:
    case ARGT_UIMM32:
      CatPrint(arg2_st,"%08X",*rip(U32 *)++);
      break;
    case ARGT_IMM64:
    case ARGT_UIMM64:
      CatPrint(arg2_st,"%016X",*rip(I64 *)++);
      break;
    case ARGT_MOFFS8...ARGT_MOFFS64:
      CatPrint(arg2_st,seg_overrides);
      if (arg2_size==8)
        disp=*rip(U8 *)++;
      else if (opadd==16)
        disp=*rip(U16 *)++;
      else
        disp=*rip(U32 *)++;
      CatPrint(arg2_st,"[%X]",disp);
      break;
    case ARGT_AL ... ARGT_DX:
    case ARGT_SS ... ARGT_ST0:
      CatPrint(arg2_st,"%z",arg2-ARGT_AL,
            "AL\0AX\0EAX\0RAX\0CL\0DX\0 \0 \0SS\0DS\0ES\0FS\0GS\0CS\0ST0\0");
      break;
    case ARGT_STI:
      rip--;
      CatPrint(arg2_st,"%Z",*rip++ -tmpins->opcode[tmpins->opcode_cnt-1],
            "ST_FSTK_REGS");
      break;
  }
  if (tmpins->flags&IEF_ENDING_ZERO)
    rip++;

  if (*arg1_st)
    CatPrint(line1,"\t%s",arg1_st);
  if (*arg2_st)
    CatPrint(line1,",%s",arg2_st);
  CatPrint(line1,"\n");
  CatPrint(line2,"\n");
  if (!just_ins) {
    for (i=rip-(*_rip)(I64);i<6;i++) {
      bin_data_area1[i<<1]=CH_SPACE;
      bin_data_area1[i<<1+1]=CH_SPACE;
    }
    for (i=rip-(*_rip)(I64);i<12;i++) {
      bin_data_area2[(i-6)<<1]=CH_SPACE;
      bin_data_area2[(i-6)<<1+1]=CH_SPACE;
    }
  }
  StrCpy(buf,line1);
  if (!just_ins && rip-(*_rip)(I64)>6)
    CatPrint(buf,line2);
  *_rip=rip;
}

U8 *U(U8 *rip,I64 cnt=20,I64 seg_size=64)
{//Unassembles a num of insts.
  I64 i;
  U8 buf[1024];
  if (seg_size==16)
    PrintWarn("16-bit unassembly is not well supported.\n");
  "$HL,1$";
  for (i=0;i<cnt;i++) {
    Ui(buf,&rip,seg_size);
    "%s",buf;
  }
  "$HL,0$";
  return rip;
}

I64 Un(U8 *rip,I64 cnt=0x80,I64 seg_size=64)
{//Unassembles a num of bytes
  I64 i=0;
  U8 buf[1024],*end_rip=rip(I64)+cnt;
  if (seg_size==16)
    PrintWarn("16-bit unassembly is not well supported.\n");
  "$HL,1$";
  while (rip<end_rip) {
    Ui(buf,&rip,seg_size);
    "%s",buf;
    i++;
  }
  "$HL,0$";
  return i;
}