| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521 | {    Copyright (c) 1998-2002 by Peter Vreman and Florian Klaempfl    Convert i386ins.dat from Nasm to a .inc file for usage with    the Free pascal compiler    See the file COPYING.FPC, included in this distribution,    for details about the copyright.    This program is distributed in the hope that it will be useful,    but WITHOUT ANY WARRANTY; without even the implied warranty of    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. **********************************************************************}{$mode objfpc}program mkx86ins;const  Version = '1.6.2';  max_operands = 4;var   s : string;   i : longint;   i8086  : boolean;   x86_64 : boolean;    function lower(const s : string) : string;    {      return lowercased string of s    }      var         i : longint;      begin         for i:=1 to length(s) do          if s[i] in ['A'..'Z'] then           lower[i]:=char(byte(s[i])+32)          else           lower[i]:=s[i];         lower[0]:=s[0];      end;      function Replace(var s:string;const s1,s2:string):boolean;      var        i  : longint;      begin        i:=pos(s1,s);        if i>0 then         begin           Delete(s,i,length(s1));           Insert(s2,s,i);           Replace:=true;         end        else         Replace:=false;      end;function formatop(s:string;allowsizeonly:boolean):string;   const     replaces=29;     replacetab : array[1..replaces,1..2] of string[32]=(       (':',' or ot_colon'),       ('reg','regnorm'),       ('regmem','rm_gpr'),       ('rm8','rm_gpr or ot_bits8'),       ('rm16','rm_gpr or ot_bits16'),       ('rm32','rm_gpr or ot_bits32'),       ('rm64','rm_gpr or ot_bits64'),       ('rm80','rm_gpr or ot_bits80'),       ('mem8','memory or ot_bits8'),       ('mem16','memory or ot_bits16'),       ('mem32','memory or ot_bits32'),       ('mem64','memory or ot_bits64'),       ('mem128','memory or ot_bits128'),       ('mem256','memory or ot_bits256'),       ('mem512','memory or ot_bits512'),       ('mem80','memory or ot_bits80'),       ('mem','memory'),       ('memory_offs','mem_offs'),       ('imm8','immediate or ot_bits8'),       ('imm16','immediate or ot_bits16'),       ('imm32','immediate or ot_bits32'),       ('imm64','immediate or ot_bits64'),       ('imm80','immediate or ot_bits80'),       ('imm','immediate'),       ('8','bits8'),       ('16','bits16'),       ('32','bits32'),       ('64','bits64'),       ('80','bits80')     );  var    i : longint;  begin    for i:=1to replaces do      begin        if s=replacetab[i,1] then          begin            s:=replacetab[i,2];            break;          end;      end;    formatop:=s;  end;function readnumber : longint;  var     base : longint;  begin     result:=0;     if s[i]='\' then       begin          base:=8;          inc(i);          if s[i]='x' then            begin               base:=16;               inc(i);            end;       end     else       base:=10;     s[i]:=upcase(s[i]);     while s[i] in ['0'..'9','A'..'F'] do       begin          case s[i] of             '0'..'9':               result:=result*base+ord(s[i])-ord('0');             'A'..'F':               result:=result*base+ord(s[i])-ord('A')+10;          end;          inc(i);       end;  end;function tostr(l : longint) : string;  var     hs : string;  begin     str(l,hs);     tostr:=hs;  end;function readstr : string;  begin     result:='';     while (s[i] in ['0'..'9','A'..'Z','a'..'z','_']) and (i<=length(s)) do       begin          result:=result+s[i];          inc(i);       end;  end;procedure skipspace;  begin     while (s[i] in [' ',#9]) do       inc(i);  end;procedure openinc(out f:text;const fn:string);begin  writeln('creating ',fn);  assign(f,fn);  rewrite(f);  writeln(f,'{ don''t edit, this file is generated from x86ins.dat }');  writeln(f,'(');end;procedure closeinc(var f:text);begin  writeln(f);  writeln(f,');');  close(f);end;var   attsuffix,   hs : string;   j : longint;   firstopcode,   first : boolean;   maxinfolen,   code : byte;   insns : longint;   attsuffile,propfile,opfile,   nopfile,attfile,intfile,   infile,insfile : text;   { instruction fields }   skip : boolean;   literalcount,   ops    : longint;   intopcode,   attopcode,   opcode,   codes,   flags   : string;   optypes : array[1..max_operands] of string;   inschanges: string;   instrwritten: boolean;   SignCheck: Cardinal;   StrSearch: Integer;const  SIGNED_INT = ' or ot_signed';procedure DoWriteInstr;begin  if firstopcode then    firstopcode:=false  else    begin      writeln(opfile,',');      writeln(attfile,',');      writeln(attsuffile,',');      writeln(intfile,',');      writeln(propfile,',');    end;  write(opfile,opcode);  write(intfile,'''',intopcode,'''');  write(attfile,'''',attopcode,'''');  write(attsuffile,attsuffix);  write(propfile,'(Ch: ',inschanges,')');end;begin   writeln('Nasm Instruction Table Converter Version ',Version);   i8086:=paramstr(1)='i8086';   x86_64:=paramstr(1)='x86_64';   insns:=0;   maxinfolen:=0;   { open dat file }   assign(infile,'../x86/x86ins.dat');   if x86_64 then     begin       { create inc files }       openinc(insfile,'x8664tab.inc');       openinc(opfile,'x8664op.inc');       assign(nopfile,'x8664nop.inc');       openinc(attfile,'x8664att.inc');       openinc(attsuffile,'x8664ats.inc');       openinc(intfile,'x8664int.inc');       openinc(propfile,'x8664pro.inc');     end   else if i8086 then     begin       { create inc files }       openinc(insfile,'i8086tab.inc');       openinc(opfile,'i8086op.inc');       assign(nopfile,'i8086nop.inc');       openinc(attfile,'i8086att.inc');       openinc(attsuffile,'i8086atts.inc');       openinc(intfile,'i8086int.inc');       openinc(propfile,'i8086prop.inc');     end   else     begin       { create inc files }       openinc(insfile,'i386tab.inc');       openinc(opfile,'i386op.inc');       assign(nopfile,'i386nop.inc');       openinc(attfile,'i386att.inc');       openinc(attsuffile,'i386atts.inc');       openinc(intfile,'i386int.inc');       openinc(propfile,'i386prop.inc');     end;   rewrite(nopfile);   writeln(nopfile,'{ don''t edit, this file is generated from x86ins.dat }');   reset(infile);   first:=true;   opcode:='';   firstopcode:=true;   while not(eof(infile)) do     begin        { handle comment }        readln(infile,s);        while (s[1]=' ') do         delete(s,1,1);        if (s='') or (s[1]=';') then          continue;        if (s[1]='[') then         begin           i:=pos(',',s);           j:=pos(']',s);           if i=0 then            begin              opcode:='A_'+Copy(s,2,j-2);              intopcode:=Copy(s,2,j-2);              { Conditional }              if (intopcode[length(intopcode)]='c') and                 (intopcode[length(intopcode)-1]='c') then                dec(byte(intopcode[0]),2);              attopcode:=intopcode;              attsuffix:='attsufNONE';            end           else            begin              SignCheck := 0;              opcode:='A_'+Copy(s,2,i-2);              intopcode:=Copy(s,2,i-2);              { intel conditional }              if (intopcode[length(intopcode)]='c') and                 (intopcode[length(intopcode)-1]='c') then                dec(byte(intopcode[0]),2);              attopcode:=Copy(s,i+1,j-i-1);              { att Suffix }              case attopcode[length(attopcode)] of                'M' :                  begin                    dec(attopcode[0]);                    attsuffix:='attsufMM';                  end;                'N' :                  begin                    dec(attopcode[0]);                    attsuffix:='attsufMMX';                  end;                'S' :		  begin                    dec(attopcode[0]);                    attsuffix:='attsufMMS';		  end;	                  'X' :                  begin                    dec(attopcode[0]);                    attsuffix:='attsufINT';                  end;                'Y' :                  begin                    dec(attopcode[0]);                    attsuffix:='attsufINTdual';                  end;                'F' :                  begin                    dec(attopcode[0]);                    attsuffix:='attsufFPU';                  end;                'R' :                  begin                    dec(attopcode[0]);                    attsuffix:='attsufFPUint';                  end;                else                  attsuffix:='attsufNONE';              end;              { att Conditional }              if (attopcode[length(attopcode)]='C') and                 (attopcode[length(attopcode)-1]='C') then                dec(byte(attopcode[0]),2);            end;           intopcode:=Lower(intopcode);           attopcode:=Lower(attopcode);           instrwritten:=false;           { read the next line which contains the Change options }           repeat             readln(infile,inschanges);           until eof(infile) or ((inschanges<>'') and (inschanges[1]<>';'));           inschanges[1]:='[';           inschanges[length(inschanges)]:=']';           continue;         end;        { we must have an opcode }        if opcode='' then         runerror(234);        { clear }        ops:=0;        for i:=low(optypes) to high(optypes) do          optypes[i]:='';        codes:='';        flags:='';        skip:=false;        { ops and optypes }        i:=1;        repeat          hs:=readstr;          if (hs='void') or (hs='ignore') then            break;          inc(ops);          optypes[ops]:=optypes[ops]+'ot_'+formatop(hs,false);          while s[i]='|' do            begin               inc(i);               optypes[ops]:=optypes[ops]+' or ot_'+formatop(readstr,true);            end;          if s[i] in [',',':'] then            inc(i)          else            break;        until false;        for j:=1 to max_operands-ops do          optypes[max_operands-j+1]:='ot_none';        { codes }        skipspace;        j:=0;        literalcount:=0;        if s[i] in ['\','0'..'9'] then          begin             while not(s[i] in [' ',#9]) do               begin                 code:=readnumber;                 { for some codes we want also to change the optypes, but not                   if the code belongs to a literal sequence }                 if (literalcount=0) and (code>=1) and (code<=3) then                   literalcount:=code                 else                   begin                     if literalcount>0 then                       dec(literalcount)                     else                       begin                         case code of                           12,13,14:    {signed byte}                             optypes[code-11]:=optypes[code-11]+SIGNED_INT;                           172,173,174: {signed long}                           begin                             { Addition by J. Gareth "Kit" Moreton }                             { See below for workaround for routines that take a 64-bit destination but a 32-bit operand in a non-decorated opcode }                             SignCheck := code-171;                             optypes[SignCheck]:=optypes[SignCheck]+SIGNED_INT;                           end;                         end;                       end;                   end;                 codes:=codes+'#'+tostr(code);                 inc(j);               end;          end        else          begin            readstr;            codes:='#0';          end;        if j>maxinfolen then         maxinfolen:=j;        { flags }        skipspace;        while not(s[i] in [' ',#9,#13,#10]) and (i<=length(s)) do          begin             hs:=readstr;             if hs='none' then               break;             if x86_64 then               begin                 { x86_64 }                 if (upcase(hs)='NOX86_64') or (upcase(hs)='16BITONLY') then                   skip:=true;               end             else if not i8086 then               begin                 { i386 }                 if (upcase(hs)='X86_64') or (upcase(hs)='16BITONLY') then                   skip:=true;               end             else               begin                 { i8086 }                 if (upcase(hs)='X86_64') then                   skip:=true;               end;             if hs<>'ND' then              begin                { Addition by J. Gareth "Kit" Moreton }                { Workaround for routines that take a 64-bit destination but a 32-bit operand in a non-decorated opcode }                if (lower(hs)='sm') and (SignCheck > 0) then                 begin                   { Remove signed flag }                   StrSearch := Pos(SIGNED_INT, optypes[SignCheck]);                   Delete(optypes[SignCheck], StrSearch, Length(SIGNED_INT));                 end;                  if flags<>'' then                   flags:=flags+',';                  flags:=flags+'if_'+lower(hs);              end;             if (s[i]=',') and (i<=length(s)) then              inc(i)             else              break;          end;        { write instruction }        if not skip then          begin            if not instrwritten then              DoWriteInstr;            instrwritten:=true;            if not(first) then              writeln(insfile,',')            else              first:=false;            writeln(insfile,'  (');            writeln(insfile,'    opcode  : ',opcode,';');            writeln(insfile,'    ops     : ',ops,';');            writeln(insfile,'    optypes : (',optypes[1],',',optypes[2],',',optypes[3],',',optypes[4],');');            writeln(insfile,'    code    : ',codes,';');            writeln(insfile,'    flags   : [',flags,']');            write(insfile,'  )');            inc(insns);          end;     end;   close(infile);   closeinc(insfile);   closeinc(intfile);   closeinc(attfile);   closeinc(attsuffile);   closeinc(opfile);   writeln(nopfile,insns,';');   close(nopfile);   closeinc(propfile);   writeln(insns,' nodes processed (maxinfolen=',maxinfolen,')');end.
 |