cpubase.pas 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566
  1. {
  2. Copyright (c) 1998-2002 by Florian Klaempfl and Peter Vreman
  3. Contains the base types for the i8086, i386 and x86-64 architecture
  4. * This code was inspired by the NASM sources
  5. The Netwide Assembler is Copyright (c) 1996 Simon Tatham and
  6. Julian Hall. All rights reserved.
  7. This program is free software; you can redistribute it and/or modify
  8. it under the terms of the GNU General Public License as published by
  9. the Free Software Foundation; either version 2 of the License, or
  10. (at your option) any later version.
  11. This program is distributed in the hope that it will be useful,
  12. but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. GNU General Public License for more details.
  15. You should have received a copy of the GNU General Public License
  16. along with this program; if not, write to the Free Software
  17. Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  18. ****************************************************************************
  19. }
  20. {# Base unit for processor information. This unit contains
  21. enumerations of registers, opcodes, sizes, and other
  22. such things which are processor specific.
  23. }
  24. unit cpubase;
  25. {$i fpcdefs.inc}
  26. interface
  27. uses
  28. cutils,cclasses,
  29. globtype,
  30. cgbase
  31. ;
  32. {*****************************************************************************
  33. Assembler Opcodes
  34. *****************************************************************************}
  35. type
  36. {$if defined(x86_64)}
  37. TAsmOp={$i x8664op.inc}
  38. {$elseif defined(i386)}
  39. TAsmOp={$i i386op.inc}
  40. {$elseif defined(i8086)}
  41. TAsmOp={$i i8086op.inc}
  42. {$endif}
  43. { This should define the array of instructions as string }
  44. op2strtable=array[tasmop] of string[16];
  45. const
  46. { First value of opcode enumeration }
  47. firstop = low(tasmop);
  48. { Last value of opcode enumeration }
  49. lastop = high(tasmop);
  50. {*****************************************************************************
  51. Registers
  52. *****************************************************************************}
  53. const
  54. { Integer Super registers }
  55. RS_RAX = $00; {EAX}
  56. RS_RCX = $01; {ECX}
  57. RS_RDX = $02; {EDX}
  58. RS_RBX = $03; {EBX}
  59. RS_RSI = $04; {ESI}
  60. RS_RDI = $05; {EDI}
  61. RS_RBP = $06; {EBP}
  62. RS_RSP = $07; {ESP}
  63. RS_R8 = $08; {R8}
  64. RS_R9 = $09; {R9}
  65. RS_R10 = $0a; {R10}
  66. RS_R11 = $0b; {R11}
  67. RS_R12 = $0c; {R12}
  68. RS_R13 = $0d; {R13}
  69. RS_R14 = $0e; {R14}
  70. RS_R15 = $0f; {R15}
  71. { create aliases to allow code sharing between x86-64 and i386 }
  72. RS_EAX = RS_RAX;
  73. RS_EBX = RS_RBX;
  74. RS_ECX = RS_RCX;
  75. RS_EDX = RS_RDX;
  76. RS_ESI = RS_RSI;
  77. RS_EDI = RS_RDI;
  78. RS_EBP = RS_RBP;
  79. RS_ESP = RS_RSP;
  80. { create aliases to allow code sharing between i386 and i8086 }
  81. RS_AX = RS_RAX;
  82. RS_BX = RS_RBX;
  83. RS_CX = RS_RCX;
  84. RS_DX = RS_RDX;
  85. RS_SI = RS_RSI;
  86. RS_DI = RS_RDI;
  87. RS_BP = RS_RBP;
  88. RS_SP = RS_RSP;
  89. { Number of first imaginary register }
  90. first_int_imreg = $10;
  91. { Float Super registers }
  92. RS_ST0 = $00;
  93. RS_ST1 = $01;
  94. RS_ST2 = $02;
  95. RS_ST3 = $03;
  96. RS_ST4 = $04;
  97. RS_ST5 = $05;
  98. RS_ST6 = $06;
  99. RS_ST7 = $07;
  100. { Number of first imaginary register }
  101. first_fpu_imreg = $08;
  102. { MM Super registers }
  103. RS_XMM0 = $00;
  104. RS_XMM1 = $01;
  105. RS_XMM2 = $02;
  106. RS_XMM3 = $03;
  107. RS_XMM4 = $04;
  108. RS_XMM5 = $05;
  109. RS_XMM6 = $06;
  110. RS_XMM7 = $07;
  111. RS_XMM8 = $08;
  112. RS_XMM9 = $09;
  113. RS_XMM10 = $0a;
  114. RS_XMM11 = $0b;
  115. RS_XMM12 = $0c;
  116. RS_XMM13 = $0d;
  117. RS_XMM14 = $0e;
  118. RS_XMM15 = $0f;
  119. RS_FLAGS = $07;
  120. { Number of first imaginary register }
  121. {$ifdef x86_64}
  122. first_mm_imreg = $10;
  123. {$else x86_64}
  124. first_mm_imreg = $08;
  125. {$endif x86_64}
  126. { The subregister that specifies the entire register and an address }
  127. {$if defined(x86_64)}
  128. { Hammer }
  129. R_SUBWHOLE = R_SUBQ;
  130. R_SUBADDR = R_SUBQ;
  131. {$elseif defined(i386)}
  132. { i386 }
  133. R_SUBWHOLE = R_SUBD;
  134. R_SUBADDR = R_SUBD;
  135. {$elseif defined(i8086)}
  136. { i8086 }
  137. R_SUBWHOLE = R_SUBW;
  138. R_SUBADDR = R_SUBW;
  139. {$endif}
  140. { Available Registers }
  141. {$if defined(x86_64)}
  142. {$i r8664con.inc}
  143. {$elseif defined(i386)}
  144. {$i r386con.inc}
  145. {$elseif defined(i8086)}
  146. {$i r8086con.inc}
  147. {$endif}
  148. type
  149. { Number of registers used for indexing in tables }
  150. {$if defined(x86_64)}
  151. tregisterindex=0..{$i r8664nor.inc}-1;
  152. {$elseif defined(i386)}
  153. tregisterindex=0..{$i r386nor.inc}-1;
  154. {$elseif defined(i8086)}
  155. tregisterindex=0..{$i r8086nor.inc}-1;
  156. {$endif}
  157. const
  158. { TODO: Calculate bsstart}
  159. regnumber_count_bsstart = 64;
  160. regnumber_table : array[tregisterindex] of tregister = (
  161. {$if defined(x86_64)}
  162. {$i r8664num.inc}
  163. {$elseif defined(i386)}
  164. {$i r386num.inc}
  165. {$elseif defined(i8086)}
  166. {$i r8086num.inc}
  167. {$endif}
  168. );
  169. regstabs_table : array[tregisterindex] of shortint = (
  170. {$if defined(x86_64)}
  171. {$i r8664stab.inc}
  172. {$elseif defined(i386)}
  173. {$i r386stab.inc}
  174. {$elseif defined(i8086)}
  175. {$i r8086stab.inc}
  176. {$endif}
  177. );
  178. regdwarf_table : array[tregisterindex] of shortint = (
  179. {$if defined(x86_64)}
  180. {$i r8664dwrf.inc}
  181. {$elseif defined(i386)}
  182. {$i r386dwrf.inc}
  183. {$elseif defined(i8086)}
  184. {$i r8086dwrf.inc}
  185. {$endif}
  186. );
  187. RS_DEFAULTFLAGS = RS_FLAGS;
  188. NR_DEFAULTFLAGS = NR_FLAGS;
  189. type
  190. totherregisterset = set of tregisterindex;
  191. {*****************************************************************************
  192. Conditions
  193. *****************************************************************************}
  194. type
  195. TAsmCond=(C_None,
  196. C_A,C_AE,C_B,C_BE,C_C,C_E,C_G,C_GE,C_L,C_LE,C_NA,C_NAE,
  197. C_NB,C_NBE,C_NC,C_NE,C_NG,C_NGE,C_NL,C_NLE,C_NO,C_NP,
  198. C_NS,C_NZ,C_O,C_P,C_PE,C_PO,C_S,C_Z
  199. );
  200. const
  201. cond2str:array[TAsmCond] of string[3]=('',
  202. 'a','ae','b','be','c','e','g','ge','l','le','na','nae',
  203. 'nb','nbe','nc','ne','ng','nge','nl','nle','no','np',
  204. 'ns','nz','o','p','pe','po','s','z'
  205. );
  206. {*****************************************************************************
  207. Flags
  208. *****************************************************************************}
  209. type
  210. TResFlags = (F_E,F_NE,F_G,F_L,F_GE,F_LE,F_C,F_NC,
  211. F_A,F_AE,F_B,F_BE,
  212. F_S,F_NS,F_O,F_NO);
  213. {*****************************************************************************
  214. Constants
  215. *****************************************************************************}
  216. const
  217. { declare aliases }
  218. LOC_SSEREGISTER = LOC_MMREGISTER;
  219. LOC_CSSEREGISTER = LOC_CMMREGISTER;
  220. max_operands = 4;
  221. maxfpuregs = 8;
  222. {*****************************************************************************
  223. CPU Dependent Constants
  224. *****************************************************************************}
  225. {$i cpubase.inc}
  226. {*****************************************************************************
  227. Helpers
  228. *****************************************************************************}
  229. function cgsize2subreg(regtype: tregistertype; s:Tcgsize):Tsubregister;
  230. function reg2opsize(r:Tregister):topsize;
  231. function reg_cgsize(const reg: tregister): tcgsize;
  232. function is_calljmp(o:tasmop):boolean;
  233. procedure inverse_flags(var f: TResFlags);
  234. function flags_to_cond(const f: TResFlags) : TAsmCond;
  235. function is_segment_reg(r:tregister):boolean;
  236. function findreg_by_number(r:Tregister):tregisterindex;
  237. function std_regnum_search(const s:string):Tregister;
  238. function std_regname(r:Tregister):string;
  239. function dwarf_reg(r:tregister):shortint;
  240. function inverse_cond(const c: TAsmCond): TAsmCond; {$ifdef USEINLINE}inline;{$endif USEINLINE}
  241. function conditions_equal(const c1, c2: TAsmCond): boolean; {$ifdef USEINLINE}inline;{$endif USEINLINE}
  242. {$ifdef i8086}
  243. { returns the next virtual register }
  244. function GetNextReg(const r : TRegister) : TRegister;
  245. {$endif i8086}
  246. implementation
  247. uses
  248. rgbase,verbose;
  249. const
  250. {$if defined(x86_64)}
  251. std_regname_table : TRegNameTable = (
  252. {$i r8664std.inc}
  253. );
  254. regnumber_index : array[tregisterindex] of tregisterindex = (
  255. {$i r8664rni.inc}
  256. );
  257. std_regname_index : array[tregisterindex] of tregisterindex = (
  258. {$i r8664sri.inc}
  259. );
  260. {$elseif defined(i386)}
  261. std_regname_table : TRegNameTable = (
  262. {$i r386std.inc}
  263. );
  264. regnumber_index : array[tregisterindex] of tregisterindex = (
  265. {$i r386rni.inc}
  266. );
  267. std_regname_index : array[tregisterindex] of tregisterindex = (
  268. {$i r386sri.inc}
  269. );
  270. {$elseif defined(i8086)}
  271. std_regname_table : TRegNameTable = (
  272. {$i r8086std.inc}
  273. );
  274. regnumber_index : array[tregisterindex] of tregisterindex = (
  275. {$i r8086rni.inc}
  276. );
  277. std_regname_index : array[tregisterindex] of tregisterindex = (
  278. {$i r8086sri.inc}
  279. );
  280. {$endif}
  281. {*****************************************************************************
  282. Helpers
  283. *****************************************************************************}
  284. function cgsize2subreg(regtype: tregistertype; s:Tcgsize):Tsubregister;
  285. begin
  286. case s of
  287. OS_8,OS_S8:
  288. cgsize2subreg:=R_SUBL;
  289. OS_16,OS_S16:
  290. cgsize2subreg:=R_SUBW;
  291. OS_32,OS_S32:
  292. cgsize2subreg:=R_SUBD;
  293. OS_64,OS_S64:
  294. cgsize2subreg:=R_SUBQ;
  295. OS_M64:
  296. cgsize2subreg:=R_SUBNONE;
  297. OS_F32,OS_F64,OS_C64:
  298. case regtype of
  299. R_FPUREGISTER:
  300. cgsize2subreg:=R_SUBWHOLE;
  301. R_MMREGISTER:
  302. case s of
  303. OS_F32:
  304. cgsize2subreg:=R_SUBMMS;
  305. OS_F64:
  306. cgsize2subreg:=R_SUBMMD;
  307. else
  308. internalerror(2009071901);
  309. end;
  310. else
  311. internalerror(2009071902);
  312. end;
  313. OS_M128,OS_MS128:
  314. cgsize2subreg:=R_SUBMMX;
  315. OS_M256,OS_MS256:
  316. cgsize2subreg:=R_SUBMMY;
  317. else
  318. internalerror(200301231);
  319. end;
  320. end;
  321. function reg_cgsize(const reg: tregister): tcgsize;
  322. const subreg2cgsize:array[Tsubregister] of Tcgsize =
  323. (OS_NO,OS_8,OS_8,OS_16,OS_32,OS_64,OS_NO,OS_NO,OS_NO,OS_F32,OS_F64,OS_NO,OS_M128,OS_M256);
  324. begin
  325. case getregtype(reg) of
  326. R_INTREGISTER :
  327. reg_cgsize:=subreg2cgsize[getsubreg(reg)];
  328. R_FPUREGISTER :
  329. reg_cgsize:=OS_F80;
  330. R_MMXREGISTER:
  331. reg_cgsize:=OS_M64;
  332. R_MMREGISTER:
  333. reg_cgsize:=subreg2cgsize[getsubreg(reg)];
  334. R_SPECIALREGISTER :
  335. case reg of
  336. NR_CS,NR_DS,NR_ES,NR_SS,NR_FS,NR_GS:
  337. reg_cgsize:=OS_16;
  338. {$ifdef x86_64}
  339. NR_DR0..NR_TR7:
  340. reg_cgsize:=OS_64;
  341. {$endif x86_64}
  342. else
  343. reg_cgsize:=OS_32
  344. end
  345. else
  346. internalerror(2003031801);
  347. end;
  348. end;
  349. function reg2opsize(r:Tregister):topsize;
  350. const
  351. subreg2opsize : array[tsubregister] of topsize =
  352. (S_NO,S_B,S_B,S_W,S_L,S_Q,S_NO,S_NO,S_NO,S_NO,S_NO,S_NO,S_NO,S_NO);
  353. begin
  354. reg2opsize:=S_L;
  355. case getregtype(r) of
  356. R_INTREGISTER :
  357. reg2opsize:=subreg2opsize[getsubreg(r)];
  358. R_FPUREGISTER :
  359. reg2opsize:=S_FL;
  360. R_MMXREGISTER,
  361. R_MMREGISTER :
  362. reg2opsize:=S_MD;
  363. R_SPECIALREGISTER :
  364. begin
  365. case r of
  366. NR_CS,NR_DS,NR_ES,
  367. NR_SS,NR_FS,NR_GS :
  368. reg2opsize:=S_W;
  369. end;
  370. end;
  371. else
  372. internalerror(200303181);
  373. end;
  374. end;
  375. function is_calljmp(o:tasmop):boolean;
  376. begin
  377. case o of
  378. A_CALL,
  379. {$ifdef i386}
  380. A_JCXZ,
  381. {$endif i386}
  382. A_JECXZ,
  383. {$ifdef x86_64}
  384. A_JRCXZ,
  385. {$endif x86_64}
  386. A_JMP,
  387. A_LOOP,
  388. A_LOOPE,
  389. A_LOOPNE,
  390. A_LOOPNZ,
  391. A_LOOPZ,
  392. A_LCALL,
  393. A_LJMP,
  394. A_Jcc :
  395. is_calljmp:=true;
  396. else
  397. is_calljmp:=false;
  398. end;
  399. end;
  400. procedure inverse_flags(var f: TResFlags);
  401. const
  402. inv_flags: array[TResFlags] of TResFlags =
  403. (F_NE,F_E,F_LE,F_GE,F_L,F_G,F_NC,F_C,
  404. F_BE,F_B,F_AE,F_A,
  405. F_NS,F_S,F_NO,F_O);
  406. begin
  407. f:=inv_flags[f];
  408. end;
  409. function flags_to_cond(const f: TResFlags) : TAsmCond;
  410. const
  411. flags_2_cond : array[TResFlags] of TAsmCond =
  412. (C_E,C_NE,C_G,C_L,C_GE,C_LE,C_C,C_NC,C_A,C_AE,C_B,C_BE,C_S,C_NS,C_O,C_NO);
  413. begin
  414. result := flags_2_cond[f];
  415. end;
  416. function is_segment_reg(r:tregister):boolean;
  417. begin
  418. result:=false;
  419. case r of
  420. NR_CS,NR_DS,NR_ES,
  421. NR_SS,NR_FS,NR_GS :
  422. result:=true;
  423. end;
  424. end;
  425. function findreg_by_number(r:Tregister):tregisterindex;
  426. var
  427. hr : tregister;
  428. begin
  429. { for the name the sub reg doesn't matter }
  430. hr:=r;
  431. if (getregtype(hr)=R_MMREGISTER) and
  432. (getsubreg(hr)<>R_SUBMMY) then
  433. setsubreg(hr,R_SUBMMX);
  434. result:=findreg_by_number_table(hr,regnumber_index);
  435. end;
  436. function std_regnum_search(const s:string):Tregister;
  437. begin
  438. result:=regnumber_table[findreg_by_name_table(s,std_regname_table,std_regname_index)];
  439. end;
  440. function std_regname(r:Tregister):string;
  441. var
  442. p : tregisterindex;
  443. begin
  444. if getregtype(r) in [R_MMREGISTER,R_MMXREGISTER] then
  445. r:=newreg(getregtype(r),getsupreg(r),R_SUBNONE);
  446. p:=findreg_by_number(r);
  447. if p<>0 then
  448. result:=std_regname_table[p]
  449. else
  450. result:=generic_regname(r);
  451. end;
  452. function inverse_cond(const c: TAsmCond): TAsmCond; {$ifdef USEINLINE}inline;{$endif USEINLINE}
  453. const
  454. inverse: array[TAsmCond] of TAsmCond=(C_None,
  455. C_NA,C_NAE,C_NB,C_NBE,C_NC,C_NE,C_NG,C_NGE,C_NL,C_NLE,C_A,C_AE,
  456. C_B,C_BE,C_C,C_E,C_G,C_GE,C_L,C_LE,C_O,C_P,
  457. C_S,C_Z,C_NO,C_NP,C_NP,C_P,C_NS,C_NZ
  458. );
  459. begin
  460. result := inverse[c];
  461. end;
  462. function conditions_equal(const c1, c2: TAsmCond): boolean; {$ifdef USEINLINE}inline;{$endif USEINLINE}
  463. begin
  464. result := c1 = c2;
  465. end;
  466. function dwarf_reg(r:tregister):shortint;
  467. begin
  468. result:=regdwarf_table[findreg_by_number(r)];
  469. if result=-1 then
  470. internalerror(200603251);
  471. end;
  472. {$ifdef i8086}
  473. function GetNextReg(const r: TRegister): TRegister;
  474. begin
  475. if getsupreg(r)<first_int_imreg then
  476. internalerror(2013051401);
  477. result:=TRegister(longint(r)+1);
  478. end;
  479. {$endif i8086}
  480. end.