mirror of
				https://gitlab.com/freepascal.org/fpc/source.git
				synced 2025-10-26 23:51:34 +01:00 
			
		
		
		
	 45bcc5b07a
			
		
	
	
		45bcc5b07a
		
	
	
	
	
		
			
			a user that is writing an operating system. * Fix mkx86ins bug to interpret imm16:imm32 as a single operand git-svn-id: trunk@606 -
		
			
				
	
	
		
			455 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
			
		
		
	
	
			455 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
| {
 | |
|     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.
 | |
| 
 | |
|  **********************************************************************}
 | |
| program mkx86ins;
 | |
| 
 | |
| const
 | |
|   Version = '1.5.0';
 | |
| 
 | |
| var
 | |
|    s : string;
 | |
|    i : longint;
 | |
|    x86_64 : boolean;
 | |
| 
 | |
| {$ifndef FPC}
 | |
|   procedure readln(var t:text;var s:string);
 | |
|   var
 | |
|     c : char;
 | |
|     i : longint;
 | |
|   begin
 | |
|     c:=#0;
 | |
|     i:=0;
 | |
|     while (not eof(t)) and (c<>#10) do
 | |
|      begin
 | |
|        read(t,c);
 | |
|        if c<>#10 then
 | |
|         begin
 | |
|           inc(i);
 | |
|           s[i]:=c;
 | |
|         end;
 | |
|      end;
 | |
|     if (i>0) and (s[i]=#13) then
 | |
|      dec(i);
 | |
|     s[0]:=chr(i);
 | |
|   end;
 | |
| {$endif}
 | |
| 
 | |
|     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):string;
 | |
|    const
 | |
|      replaces=19;
 | |
|      replacetab : array[1..replaces,1..2] of string[32]=(
 | |
|        (':',' or ot_colon'),
 | |
|        ('mem8','mem or ot_bits8'),
 | |
|        ('mem16','mem or ot_bits16'),
 | |
|        ('mem32','mem or ot_bits32'),
 | |
|        ('mem64','mem or ot_bits64'),
 | |
|        ('mem80','mem or ot_bits80'),
 | |
|        ('mem','memory'),
 | |
|        ('memory_offs','mem_offs'),
 | |
|        ('imm8','imm or ot_bits8'),
 | |
|        ('imm16','imm or ot_bits16'),
 | |
|        ('imm32','imm or ot_bits32'),
 | |
|        ('imm64','imm or ot_bits64'),
 | |
|        ('imm80','imm or ot_bits80'),
 | |
|        ('imm','immediate'),
 | |
|        ('rm8','regmem or ot_bits8'),
 | |
|        ('rm16','regmem or ot_bits16'),
 | |
|        ('rm32','regmem or ot_bits32'),
 | |
|        ('rm64','regmem or ot_bits64'),
 | |
|        ('rm80','regmem or ot_bits80')
 | |
|      );
 | |
|   var
 | |
|     i : longint;
 | |
|   begin
 | |
|     for i:=1to replaces do
 | |
|      replace(s,replacetab[i,1],replacetab[i,2]);
 | |
|     formatop:=s;
 | |
|   end;
 | |
| 
 | |
| 
 | |
| function readnumber : longint;
 | |
| 
 | |
|   var
 | |
|      base : longint;
 | |
|      result : 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;
 | |
|      readnumber:=result;
 | |
|   end;
 | |
| 
 | |
| function tostr(l : longint) : string;
 | |
| 
 | |
|   var
 | |
|      hs : string;
 | |
| 
 | |
|   begin
 | |
|      str(l,hs);
 | |
|      tostr:=hs;
 | |
|   end;
 | |
| 
 | |
| function readstr : string;
 | |
| 
 | |
|   var
 | |
|      result : 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;
 | |
|      readstr:=result;
 | |
|   end;
 | |
| 
 | |
| procedure skipspace;
 | |
| 
 | |
|   begin
 | |
|      while (s[i] in [' ',#9]) do
 | |
|        inc(i);
 | |
|   end;
 | |
| 
 | |
| procedure openinc(var 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;
 | |
|    last,
 | |
|    ops    : longint;
 | |
|    intopcode,
 | |
|    attopcode,
 | |
|    opcode,
 | |
|    codes,
 | |
|    flags   : string;
 | |
|    optypes : array[1..3] of string;
 | |
| begin
 | |
|    writeln('Nasm Instruction Table Converter Version ',Version);
 | |
|    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
 | |
|      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
 | |
|               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
 | |
|                 'X' :
 | |
|                   begin
 | |
|                     dec(attopcode[0]);
 | |
|                     attsuffix:='attsufINT';
 | |
|                   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);
 | |
|            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);
 | |
|            { read the next line which contains the Change options }
 | |
|            repeat
 | |
|              readln(infile,s);
 | |
|            until eof(infile) or ((s<>'') and (s[1]<>';'));
 | |
|            write(propfile,'(Ch: ',s,')');
 | |
|            continue;
 | |
|          end;
 | |
|         { we must have an opcode }
 | |
|         if opcode='' then
 | |
|          runerror(234);
 | |
|         { clear }
 | |
|         ops:=0;
 | |
|         optypes[1]:='';
 | |
|         optypes[2]:='';
 | |
|         optypes[3]:='';
 | |
|         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);
 | |
| {          if s[i]=':' then
 | |
|             begin
 | |
|                inc(i);
 | |
|                optypes[ops]:=optypes[ops]+' or ot_'+formatop(readstr);
 | |
|             end;}
 | |
|           while s[i]='|' do
 | |
|             begin
 | |
|                inc(i);
 | |
|                optypes[ops]:=optypes[ops]+' or ot_'+formatop(readstr);
 | |
|             end;
 | |
|           if s[i] in [',',':'] then
 | |
|             inc(i)
 | |
|           else
 | |
|             break;
 | |
|         until false;
 | |
|         for j:=1 to 3-ops do
 | |
|           optypes[3-j+1]:='ot_none';
 | |
|         { codes }
 | |
|         skipspace;
 | |
|         j:=0;
 | |
|         last:=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 last byte was a 1 then this byte belongs to a direct
 | |
|                    copy }
 | |
|                  if last<>1 then
 | |
|                   begin
 | |
|                     case code of
 | |
|                       12,13,14 :
 | |
|                         optypes[code-11]:=optypes[code-11]+' or ot_signed';
 | |
|                     end;
 | |
|                   end;
 | |
|                  codes:=codes+'#'+tostr(code);
 | |
|                  last:=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 x86_64 then
 | |
|                begin
 | |
|                  if (upcase(hs)='NOX86_64') then
 | |
|                    skip:=true;
 | |
|                end
 | |
|              else
 | |
|                begin
 | |
|                  if (upcase(hs)='X86_64') then
 | |
|                    skip:=true;
 | |
|                end;
 | |
|              if hs<>'ND' then
 | |
|               begin
 | |
|                 if flags<>'' then
 | |
|                  flags:=flags+' or ';
 | |
|                 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(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],');');
 | |
|             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 procesed (maxinfolen=',maxinfolen,')');
 | |
| end.
 |