cpubase.pas 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742
  1. {******************************************************************************
  2. $Id$
  3. Copyright (c) 1998-2000 by Florian Klaempfl and Peter Vreman
  4. Contains the base types for the Scalable Processor ARChitecture (SPARC)
  5. This program is free software; you can redistribute it and/or modify
  6. it under the terms of the GNU General Public License as published by
  7. the Free Software Foundation; either version 2 of the License, or
  8. (at your option) any later version.
  9. This program is distributed in the hope that it will be useful,
  10. but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. GNU General Public License for more details.
  13. You should have received a copy of the GNU General Public License
  14. along with this program; if not, write to the Free Software
  15. Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  16. ****************************************************************************}
  17. unit cpuBase;
  18. {$INCLUDE fpcdefs.inc}
  19. interface
  20. uses globals,cutils,cclasses,aasmbase,cpuinfo,cginfo;
  21. const
  22. {Size of the instruction table converted by nasmconv.pas}
  23. maxinfolen=8;
  24. {Defines the default address size for a processor}
  25. OS_ADDR=OS_32;
  26. {the natural int size for a processor}
  27. OS_INT=OS_32;
  28. {the maximum float size for a processor}
  29. OS_FLOAT=OS_F64;
  30. {the size of a vector register for a processor}
  31. OS_VECTOR=OS_M64;
  32. {Operand types}
  33. OT_NONE = $00000000;
  34. OT_BITS8 = $00000001; { size, and other attributes, of the operand }
  35. OT_BITS16 = $00000002;
  36. OT_BITS32 = $00000004;
  37. OT_BITS64 = $00000008; { FPU only }
  38. OT_BITS80 = $00000010;
  39. OT_FAR = $00000020; { this means 16:16 or 16:32, like in CALL/JMP }
  40. OT_NEAR = $00000040;
  41. OT_SHORT = $00000080;
  42. OT_SIZE_MASK = $000000FF; { all the size attributes }
  43. OT_NON_SIZE = LongInt(not OT_SIZE_MASK);
  44. OT_SIGNED = $00000100; { the operand need to be signed -128-127 }
  45. OT_TO = $00000200; { operand is followed by a colon }
  46. { reverse effect in FADD, FSUB &c }
  47. OT_COLON = $00000400;
  48. OT_REGISTER = $00001000;
  49. OT_IMMEDIATE = $00002000;
  50. OT_IMM8 = $00002001;
  51. OT_IMM16 = $00002002;
  52. OT_IMM32 = $00002004;
  53. OT_IMM64 = $00002008;
  54. OT_IMM80 = $00002010;
  55. OT_REGMEM = $00200000; { for r/m, ie EA, operands }
  56. OT_REGNORM = $00201000; { 'normal' reg, qualifies as EA }
  57. OT_REG8 = $00201001;
  58. OT_REG16 = $00201002;
  59. OT_REG32 = $00201004;
  60. OT_MMXREG = $00201008; { MMX registers }
  61. OT_XMMREG = $00201010; { Katmai registers }
  62. OT_MEMORY = $00204000; { register number in 'basereg' }
  63. OT_MEM8 = $00204001;
  64. OT_MEM16 = $00204002;
  65. OT_MEM32 = $00204004;
  66. OT_MEM64 = $00204008;
  67. OT_MEM80 = $00204010;
  68. OT_FPUREG = $01000000; { floating point stack registers }
  69. OT_FPU0 = $01000800; { FPU stack register zero }
  70. OT_REG_SMASK = $00070000; { special register operands: these may be treated differently }
  71. { a mask for the following }
  72. OT_REG_ACCUM = $00211000; { accumulator: AL, AX or EAX }
  73. OT_REG_AL = $00211001; { REG_ACCUM | BITSxx }
  74. OT_REG_AX = $00211002; { ditto }
  75. OT_REG_EAX = $00211004; { and again }
  76. OT_REG_COUNT = $00221000; { counter: CL, CX or ECX }
  77. OT_REG_CL = $00221001; { REG_COUNT | BITSxx }
  78. OT_REG_CX = $00221002; { ditto }
  79. OT_REG_ECX = $00221004; { another one }
  80. OT_REG_DX = $00241002;
  81. OT_REG_SREG = $00081002; { any segment register }
  82. OT_REG_CS = $01081002; { CS }
  83. OT_REG_DESS = $02081002; { DS, ES, SS (non-CS 86 registers) }
  84. OT_REG_FSGS = $04081002; { FS, GS (386 extENDed registers) }
  85. OT_REG_CDT = $00101004; { CRn, DRn and TRn }
  86. OT_REG_CREG = $08101004; { CRn }
  87. OT_REG_CR4 = $08101404; { CR4 (Pentium only) }
  88. OT_REG_DREG = $10101004; { DRn }
  89. OT_REG_TREG = $20101004; { TRn }
  90. OT_MEM_OFFS = $00604000; { special type of EA }
  91. { simple [address] offset }
  92. OT_ONENESS = $00800000; { special type of immediate operand }
  93. { so UNITY == IMMEDIATE | ONENESS }
  94. OT_UNITY = $00802000; { for shift/rotate instructions }
  95. {Instruction flags }
  96. IF_NONE = $00000000;
  97. IF_SM = $00000001; { size match first two operands }
  98. IF_SM2 = $00000002;
  99. IF_SB = $00000004; { unsized operands can't be non-byte }
  100. IF_SW = $00000008; { unsized operands can't be non-word }
  101. IF_SD = $00000010; { unsized operands can't be nondword }
  102. IF_AR0 = $00000020; { SB, SW, SD applies to argument 0 }
  103. IF_AR1 = $00000040; { SB, SW, SD applies to argument 1 }
  104. IF_AR2 = $00000060; { SB, SW, SD applies to argument 2 }
  105. IF_ARMASK = $00000060; { mask for unsized argument spec }
  106. IF_PRIV = $00000100; { it's a privileged instruction }
  107. IF_SMM = $00000200; { it's only valid in SMM }
  108. IF_PROT = $00000400; { it's protected mode only }
  109. IF_UNDOC = $00001000; { it's an undocumented instruction }
  110. IF_FPU = $00002000; { it's an FPU instruction }
  111. IF_MMX = $00004000; { it's an MMX instruction }
  112. IF_3DNOW = $00008000; { it's a 3DNow! instruction }
  113. IF_SSE = $00010000; { it's a SSE (KNI, MMX2) instruction }
  114. IF_PMASK = LongInt($FF000000); { the mask for processor types }
  115. IF_PFMASK = LongInt($F001FF00); { the mask for disassembly "prefer" }
  116. IF_V7 = $00000000; { SPARC V7 instruction only (not supported)}
  117. IF_V8 = $01000000; { SPARC V8 instruction (the default)}
  118. IF_V9 = $02000000; { SPARC V9 instruction (not yet supported)}
  119. { added flags }
  120. IF_PRE = $40000000; { it's a prefix instruction }
  121. IF_PASS2 = LongInt($80000000);{instruction can change in a second pass?}
  122. TYPE
  123. {$WARNING CPU32 opcodes do not fully include the Ultra SPRAC instruction set.}
  124. { don't change the order of these opcodes! }
  125. TAsmOp=({$INCLUDE opcode.inc});
  126. op2strtable=ARRAY[TAsmOp]OF STRING[11];
  127. CONST
  128. FirstOp=Low(TAsmOp);
  129. LastOp=High(TAsmOp);
  130. std_op2str:op2strtable=({$INCLUDE strinst.inc});
  131. {*****************************************************************************
  132. Operand Sizes
  133. *****************************************************************************}
  134. TYPE
  135. TOpSize=(S_NO,
  136. S_B,{Byte}
  137. S_H,{Half word}
  138. S_W,{Word}
  139. S_L:=S_W,
  140. S_D,{Double Word}
  141. S_Q,{Quad word}
  142. S_IQ:=S_Q,
  143. S_SB,{Signed byte}
  144. S_SH,{Signed half word}
  145. S_SW,{Signed word}
  146. S_SD,{Signed double word}
  147. S_SQ,{Signed quad word}
  148. S_FS,{Float single word}
  149. S_FX:=S_FS,
  150. S_FD,{Float double word}
  151. S_FQ,{Float quad word}
  152. S_NEAR,
  153. S_FAR,
  154. S_SHORT);
  155. {*****************************************************************************}
  156. { Conditions }
  157. {*****************************************************************************}
  158. TYPE
  159. TAsmCond=(C_None,
  160. C_A,C_AE,C_B,C_BE,C_C,C_E,C_G,C_GE,C_L,C_LE,C_NA,C_NAE,
  161. C_NB,C_NBE,C_NC,C_NE,C_NG,C_NGE,C_NL,C_NLE,C_NO,C_NP,
  162. C_NS,C_NZ,C_O,C_P,C_PE,C_PO,C_S,C_Z
  163. );
  164. CONST
  165. cond2str:ARRAY[TAsmCond] of string[3]=('',
  166. 'a','ae','b','be','c','e','g','ge','l','le','na','nae',
  167. 'nb','nbe','nc','ne','ng','nge','nl','nle','no','np',
  168. 'ns','nz','o','p','pe','po','s','z'
  169. );
  170. inverse_cond:ARRAY[TAsmCond] of TAsmCond=(C_None,
  171. C_NA,C_NAE,C_NB,C_NBE,C_NC,C_NE,C_NG,C_NGE,C_NL,C_NLE,C_A,C_AE,
  172. C_B,C_BE,C_C,C_E,C_G,C_GE,C_L,C_LE,C_O,C_P,
  173. C_S,C_Z,C_NO,C_NP,C_NP,C_P,C_NS,C_NZ
  174. );
  175. CONST
  176. CondAsmOps=3;
  177. CondAsmOp:ARRAY[0..CondAsmOps-1] of TAsmOp=(A_FCMPd, A_JMPL, A_FCMPs);
  178. CondAsmOpStr:ARRAY[0..CondAsmOps-1] of string[7]=('FCMPd','JMPL','FCMPs');
  179. {*****************************************************************************}
  180. { Registers }
  181. {*****************************************************************************}
  182. TYPE
  183. { enumeration for registers, don't change the order }
  184. { it's used by the register size conversions }
  185. ToldRegister=({$INCLUDE registers.inc});
  186. Tnewregister=word;
  187. Tsuperregister=byte;
  188. Tsubregister=byte;
  189. Tregister=record
  190. enum:Toldregister;
  191. number:Tnewregister;
  192. end;
  193. TRegister64=PACKED RECORD
  194. {A type to store register locations for 64 Bit values.}
  195. RegLo,RegHi:TRegister;
  196. END;
  197. treg64=tregister64;{alias for compact code}
  198. TRegisterSet=SET OF ToldRegister;
  199. Tsupregset=set of Tsuperregister;
  200. CONST
  201. R_NO=R_NONE;
  202. firstreg = Succ(R_NONE);
  203. lastreg = Pred(R_INTREGISTER);
  204. {General registers.}
  205. const
  206. NR_NO=$0000;
  207. NR_G0=$0001;
  208. NR_G1=$0002;
  209. NR_G2=$0003;
  210. NR_G3=$0004;
  211. NR_G4=$0005;
  212. NR_G5=$0006;
  213. NR_G6=$0007;
  214. NR_G7=$0008;
  215. NR_O0=$0100;
  216. NR_O1=$0200;
  217. NR_O2=$0300;
  218. NR_O3=$0400;
  219. NR_O4=$0500;
  220. NR_O5=$0600;
  221. NR_O6=$0700;
  222. NR_O7=$0800;
  223. NR_L0=$0900;
  224. NR_L1=$0A00;
  225. NR_L2=$0B00;
  226. NR_L3=$0C00;
  227. NR_L4=$0D00;
  228. NR_L5=$0E00;
  229. NR_L6=$0F00;
  230. NR_L7=$1000;
  231. NR_I0=$1100;
  232. NR_I1=$1200;
  233. NR_I2=$1300;
  234. NR_I3=$1400;
  235. NR_I4=$1500;
  236. NR_I5=$1600;
  237. NR_I6=$1700;
  238. NR_I7=$1800;
  239. {Superregisters.}
  240. const
  241. RS_O0=$01;
  242. RS_O1=$02;
  243. RS_O2=$03;
  244. RS_O3=$04;
  245. RS_O4=$05;
  246. RS_O5=$06;
  247. RS_O6=$07;
  248. RS_O7=$08;
  249. RS_L0=$09;
  250. RS_L1=$0A;
  251. RS_L2=$0B;
  252. RS_L3=$0C;
  253. RS_L4=$0D;
  254. RS_L5=$0E;
  255. RS_L6=$0F;
  256. RS_L7=$10;
  257. RS_I0=$11;
  258. RS_I1=$12;
  259. RS_I2=$13;
  260. RS_I3=$14;
  261. RS_I4=$15;
  262. RS_I5=$16;
  263. RS_I6=$17;
  264. RS_I7=$18;
  265. first_supreg = $01;
  266. last_supreg = $18;
  267. first_imreg = $19;
  268. last_imreg = $ff;
  269. {Subregisters; nothing known about.}
  270. R_SUBWHOLE=$00;
  271. R_SUBL=$00;
  272. type
  273. reg2strtable=ARRAY[TOldRegister] OF STRING[7];
  274. const
  275. std_reg2str:reg2strtable=({$INCLUDE strregs.inc});
  276. {*****************************************************************************
  277. Flags
  278. *****************************************************************************}
  279. TYPE
  280. TResFlags=(
  281. F_E, {Equal}
  282. F_NE, {Not Equal}
  283. F_G, {Greater}
  284. F_L, {Less}
  285. F_GE, {Greater or Equal}
  286. F_LE, {Less or Equal}
  287. F_C, {Carry}
  288. F_NC, {Not Carry}
  289. F_A, {Above}
  290. F_AE, {Above or Equal}
  291. F_B, {Below}
  292. F_BE {Below or Equal}
  293. );
  294. {*****************************************************************************
  295. Reference
  296. *****************************************************************************}
  297. TYPE
  298. trefoptions=(ref_none,ref_parafixup,ref_localfixup,ref_selffixup);
  299. { immediate/reference record }
  300. poperreference = ^treference;
  301. Preference=^Treference;
  302. treference = packed record
  303. segment,
  304. base,
  305. index : tregister;
  306. scalefactor : byte;
  307. offset : LongInt;
  308. symbol : tasmsymbol;
  309. offsetfixup : LongInt;
  310. options : trefoptions;
  311. alignment : byte;
  312. END;
  313. { reference record }
  314. PParaReference=^TParaReference;
  315. TParaReference=PACKED RECORD
  316. Index:TRegister;
  317. Offset:longint;
  318. END;
  319. {*****************************************************************************
  320. Operands
  321. *****************************************************************************}
  322. { Types of operand }
  323. toptype=(top_none,top_reg,top_ref,top_const,top_symbol,top_raddr,top_caddr);
  324. toper=record
  325. ot:LongInt;
  326. case typ:toptype of
  327. top_none:();
  328. top_reg:(reg:tregister);
  329. top_ref:(ref:poperreference);
  330. top_const:(val:aword);
  331. top_symbol:(sym:tasmsymbol;symofs:LongInt);
  332. top_raddr:(reg1,reg2:TRegister);
  333. top_caddr:(regb:TRegister;const13:Integer);
  334. end;
  335. {*****************************************************************************
  336. Argument Classification
  337. *****************************************************************************}
  338. TYPE
  339. TArgClass = (
  340. { the following classes should be defined by all processor implemnations }
  341. AC_NOCLASS,
  342. AC_MEMORY,
  343. AC_INTEGER,
  344. AC_FPU,
  345. { the following argument classes are i386 specific }
  346. AC_FPUUP,
  347. AC_SSE,
  348. AC_SSEUP);
  349. {*****************************************************************************
  350. Generic Location
  351. *****************************************************************************}
  352. TYPE
  353. {tparamlocation describes where a parameter for a procedure is stored.
  354. References are given from the caller's point of view. The usual TLocation isn't
  355. used, because contains a lot of unnessary fields.}
  356. TParaLocation=PACKED RECORD
  357. Size:TCGSize;
  358. Loc:TCGLoc;
  359. sp_fixup:LongInt;
  360. CASE TCGLoc OF
  361. LOC_REFERENCE:(reference:tparareference);
  362. { segment in reference at the same place as in loc_register }
  363. LOC_REGISTER,LOC_CREGISTER : (
  364. CASE LongInt OF
  365. 1 : (register,registerhigh : tregister);
  366. { overlay a registerlow }
  367. 2 : (registerlow : tregister);
  368. { overlay a 64 Bit register type }
  369. 3 : (reg64 : tregister64);
  370. 4 : (register64 : tregister64);
  371. );
  372. { it's only for better handling }
  373. LOC_MMXREGISTER,LOC_CMMXREGISTER : (mmxreg : tregister);
  374. END;
  375. TLocation=PACKED RECORD
  376. loc : TCGLoc;
  377. size : TCGSize;
  378. case TCGLoc of
  379. LOC_FLAGS : (resflags : tresflags);
  380. LOC_CONSTANT : (
  381. case longint of
  382. 1 : (value : AWord);
  383. 2 : (valuelow, valuehigh:AWord);
  384. { overlay a complete 64 Bit value }
  385. 3 : (valueqword : qword);
  386. );
  387. LOC_CREFERENCE,
  388. LOC_REFERENCE : (reference : treference);
  389. { segment in reference at the same place as in loc_register }
  390. LOC_REGISTER,LOC_CREGISTER : (
  391. case longint of
  392. 1 : (register,registerhigh,segment : tregister);
  393. { overlay a registerlow }
  394. 2 : (registerlow : tregister);
  395. { overlay a 64 Bit register type }
  396. 3 : (reg64 : tregister64);
  397. 4 : (register64 : tregister64);
  398. );
  399. { it's only for better handling }
  400. LOC_MMXREGISTER,LOC_CMMXREGISTER : (mmxreg : tregister);
  401. end;
  402. {*****************************************************************************
  403. Constants
  404. *****************************************************************************}
  405. const
  406. general_registers = [R_G0..R_I7];
  407. general_superregisters = [RS_O0..RS_I7];
  408. { legend: }
  409. { xxxregs = set of all possibly used registers of that type in the code }
  410. { generator }
  411. { usableregsxxx = set of all 32bit components of registers that can be }
  412. { possible allocated to a regvar or using getregisterxxx (this }
  413. { excludes registers which can be only used for parameter }
  414. { passing on ABI's that define this) }
  415. { c_countusableregsxxx = amount of registers in the usableregsxxx set }
  416. IntRegs=[R_G0..R_I7];
  417. usableregsint=[RS_O0..RS_I7];
  418. c_countusableregsint = 24;
  419. fpuregs=[R_F0..R_F31];
  420. usableregsfpu=[R_F0..R_F31];
  421. c_countusableregsfpu=32;
  422. mmregs=[];
  423. usableregsmm=[];
  424. c_countusableregsmm=0;
  425. { no distinction on this platform }
  426. maxaddrregs = 0;
  427. addrregs = [];
  428. usableregsaddr = [];
  429. c_countusableregsaddr = 0;
  430. firstsaveintreg = RS_O0;
  431. lastsaveintreg = RS_I7;
  432. firstsavefpureg = R_F0;
  433. lastsavefpureg = R_F31;
  434. firstsavemmreg = R_NONE;
  435. lastsavemmreg = R_NONE;
  436. lowsavereg = R_G0;
  437. highsavereg = R_I7;
  438. ALL_REGISTERS = [lowsavereg..highsavereg];
  439. ALL_INTREGISTERS = [1..255];
  440. lvaluelocations = [LOC_REFERENCE,LOC_CFPUREGISTER,
  441. LOC_CREGISTER,LOC_MMXREGISTER,LOC_CMMXREGISTER];
  442. {*****************************************************************************
  443. GDB Information
  444. *****************************************************************************}
  445. {# Register indexes for stabs information, when some parameters or variables
  446. are stored in registers.
  447. Taken from rs6000.h (DBX_REGISTER_NUMBER) from GCC 3.x source code.}
  448. stab_regindex:ARRAY[TOldRegister]OF ShortInt=({$INCLUDE stabregi.inc});
  449. {*************************** generic register names **************************}
  450. stack_pointer_reg = R_O6;
  451. NR_STACK_POINTER_REG = NR_O6;
  452. RS_STACK_POINTER_REG = RS_O6;
  453. frame_pointer_reg = R_I6;
  454. NR_FRAME_POINTER_REG = NR_I6;
  455. RS_FRAME_POINTER_REG = RS_I6;
  456. {the return_result_reg, is used inside the called function to store its return
  457. value when that is a scalar value otherwise a pointer to the address of the
  458. result is placed inside it}
  459. return_result_reg = R_I0;
  460. NR_RETURN_RESULT_REG = NR_I0;
  461. RS_RETURN_RESULT_REG = RS_I0;
  462. {the function_result_reg contains the function result after a call to a scalar
  463. function othewise it contains a pointer to the returned result}
  464. function_result_reg = R_O0;
  465. NR_FUNCTION_RESULT_REG = NR_O0;
  466. RS_FUNCTION_RESULT_REG = RS_O0;
  467. self_pointer_reg =R_G5;
  468. NR_SELF_POINTER_REG = NR_G5;
  469. { RS_SELF_POINTER_REG = RS_G5;}
  470. {There is no accumulator in the SPARC architecture. There are just families
  471. of registers. All registers belonging to the same family are identical except
  472. in the "global registers" family where GO is different from the others :
  473. G0 gives always 0 when it is red and thows away any value written to it.
  474. Nevertheless, scalar routine results are returned onto R_O0.}
  475. accumulator = R_O0;
  476. NR_ACCUMULATOR = NR_O0;
  477. RS_ACCUMULATOR = RS_O1;
  478. accumulatorhigh = R_O1;
  479. NR_ACCUMULATORHIGH = NR_O1;
  480. RS_ACCUMULATORHIGH = RS_O1;
  481. fpu_result_reg =R_F0;
  482. mmresultreg =R_G0;
  483. {*****************************************************************************}
  484. { GCC /ABI linking information }
  485. {*****************************************************************************}
  486. {# Registers which must be saved when calling a routine declared as cppdecl,
  487. cdecl, stdcall, safecall, palmossyscall. The registers saved should be the ones
  488. as defined in the target ABI and / or GCC.
  489. This value can be deduced from the CALLED_USED_REGISTERS array in the GCC
  490. source.}
  491. std_saved_registers=[RS_O6];
  492. {# Required parameter alignment when calling a routine declared as stdcall and
  493. cdecl. The alignment value should be the one defined by GCC or the target ABI.
  494. The value of this constant is equal to the constant
  495. PARM_BOUNDARY / BITS_PER_UNIT in the GCC source.}
  496. std_param_align=4;
  497. {# Registers which are defined as scratch and no need to save across routine
  498. calls or in assembler blocks.}
  499. ScratchRegsCount=8;
  500. scratch_regs:ARRAY[1..ScratchRegsCount] OF Tsuperregister=(RS_L0,RS_L1,RS_L2,RS_L3,RS_L4,RS_L5,RS_L6,RS_L7);
  501. { low and high of the available maximum width integer general purpose }
  502. { registers }
  503. LoGPReg = R_G0;
  504. HiGPReg = R_I7;
  505. { low and high of every possible width general purpose register (same as }
  506. { above on most architctures apart from the 80x86) }
  507. LoReg = R_G0;
  508. HiReg = R_I7;
  509. cpuflags = [];
  510. { sizes }
  511. pointersize = 4;
  512. extENDed_size = 8;{SPARC architecture uses IEEE floating point numbers}
  513. mmreg_size = 8;
  514. SizePostfix_pointer = S_SW;
  515. {*****************************************************************************
  516. Instruction table
  517. *****************************************************************************}
  518. {$ifndef NOAG386BIN}
  519. TYPE
  520. tinsentry=packed record
  521. opcode : tasmop;
  522. ops : byte;
  523. optypes : ARRAY[0..2] of LongInt;
  524. code : ARRAY[0..maxinfolen] of char;
  525. flags : LongInt;
  526. END;
  527. pinsentry=^tinsentry;
  528. TInsTabCache=ARRAY[TasmOp] of LongInt;
  529. PInsTabCache=^TInsTabCache;
  530. VAR
  531. InsTabCache : PInsTabCache;
  532. {$ENDif NOAG386BIN}
  533. {*****************************************************************************
  534. Helpers
  535. *****************************************************************************}
  536. const
  537. maxvarregs=30;
  538. VarRegs:ARRAY[1..maxvarregs]OF ToldRegister=(
  539. R_G0,R_G1,R_G2,R_G3,R_G4,R_G5,R_G6,R_G7,
  540. R_O0,R_O1,R_O2,R_O3,R_O4,R_O5,{R_R14=R_SP}R_O7,
  541. R_L0,R_L1,R_L2,R_L3,R_L4,R_L5,R_L6,R_L7,
  542. R_I0,R_I1,R_I2,R_I3,R_I4,R_I5,{R_R30=R_FP}R_I7
  543. );
  544. maxfpuvarregs = 8;
  545. max_operands = 3;
  546. maxintregs = maxvarregs;
  547. maxfpuregs = maxfpuvarregs;
  548. max_scratch_regs=8;
  549. FUNCTION is_calljmp(o:tasmop):boolean;
  550. FUNCTION flags_to_cond(CONST f:TResFlags):TAsmCond;
  551. procedure convert_register_to_enum(var r:Tregister);
  552. function cgsize2subreg(s:Tcgsize):Tsubregister;
  553. IMPLEMENTATION
  554. uses verbose;
  555. const
  556. CallJmpOp=[A_JMPL..A_CBccc];
  557. function is_calljmp(o:tasmop):boolean;
  558. begin
  559. if o in CallJmpOp
  560. then
  561. is_calljmp:=true
  562. else
  563. is_calljmp:=false;
  564. end;
  565. function flags_to_cond(const f:TResFlags):TAsmCond;
  566. CONST
  567. flags_2_cond:ARRAY[TResFlags]OF TAsmCond=
  568. (C_E,C_NE,C_G,C_L,C_GE,C_LE,C_C,C_NC,C_A,C_AE,C_B,C_BE);
  569. BEGIN
  570. result:=flags_2_cond[f];
  571. END;
  572. procedure convert_register_to_enum(var r:Tregister);
  573. begin
  574. if r.enum=R_INTREGISTER
  575. then
  576. case r.number of
  577. NR_NO: r.enum:= R_NO;
  578. NR_G0: r.enum:= R_G0;
  579. NR_G1: r.enum:= R_G1;
  580. NR_G2: r.enum:= R_G2;
  581. NR_G3: r.enum:= R_G3;
  582. NR_G4: r.enum:= R_G4;
  583. NR_G5: r.enum:= R_G5;
  584. NR_G6: r.enum:= R_G6;
  585. NR_G7: r.enum:= R_G7;
  586. NR_O0: r.enum:= R_O0;
  587. NR_O1: r.enum:= R_O1;
  588. NR_O2: r.enum:= R_O2;
  589. NR_O3: r.enum:= R_O3;
  590. NR_O4: r.enum:= R_O4;
  591. NR_O5: r.enum:= R_O5;
  592. NR_O6: r.enum:= R_O6;
  593. NR_O7: r.enum:= R_O7;
  594. NR_L0: r.enum:= R_L0;
  595. NR_L1: r.enum:= R_L1;
  596. NR_L2: r.enum:= R_L2;
  597. NR_L3: r.enum:= R_L3;
  598. NR_L4: r.enum:= R_L4;
  599. NR_L5: r.enum:= R_L5;
  600. NR_L6: r.enum:= R_L6;
  601. NR_L7: r.enum:= R_L7;
  602. NR_I0: r.enum:= R_I0;
  603. NR_I1: r.enum:= R_I1;
  604. NR_I2: r.enum:= R_I2;
  605. NR_I3: r.enum:= R_I3;
  606. NR_I4: r.enum:= R_I4;
  607. NR_I5: r.enum:= R_I5;
  608. NR_I6: r.enum:= R_I6;
  609. NR_I7: r.enum:= R_I7;
  610. else
  611. internalerror(200301082);
  612. end;
  613. end;
  614. function cgsize2subreg(s:Tcgsize):Tsubregister;
  615. begin
  616. cgsize2subreg:=R_SUBWHOLE;
  617. end;
  618. END.
  619. {
  620. $Log$
  621. Revision 1.28 2003-04-28 09:46:30 mazen
  622. + max_scratch_regs variable added because requested by common compiler code
  623. Revision 1.27 2003/04/23 13:35:39 peter
  624. * fix sparc compile
  625. Revision 1.26 2003/04/23 12:35:35 florian
  626. * fixed several issues with powerpc
  627. + applied a patch from Jonas for nested function calls (PowerPC only)
  628. * ...
  629. Revision 1.25 2003/03/10 21:59:54 mazen
  630. * fixing index overflow in handling new registers arrays.
  631. Revision 1.24 2003/02/26 22:06:27 mazen
  632. * FirstReg <-- R_G0 instead of Low(TOldRegister)=R_NONE
  633. * LastReg <-- R_L7 instead of High(R_ASR31)=High(TOldRegister)
  634. * FirstReg..LastReg rplaced by TOldRegister in several arrays declarions
  635. Revision 1.23 2003/02/19 22:00:17 daniel
  636. * Code generator converted to new register notation
  637. - Horribily outdated todo.txt removed
  638. Revision 1.22 2003/02/02 19:25:54 carl
  639. * Several bugfixes for m68k target (register alloc., opcode emission)
  640. + VIS target
  641. + Generic add more complete (still not verified)
  642. Revision 1.21 2003/01/20 22:21:36 mazen
  643. * many stuff related to RTL fixed
  644. Revision 1.20 2003/01/09 20:41:00 daniel
  645. * Converted some code in cgx86.pas to new register numbering
  646. Revision 1.19 2003/01/09 15:49:56 daniel
  647. * Added register conversion
  648. Revision 1.18 2003/01/08 18:43:58 daniel
  649. * Tregister changed into a record
  650. Revision 1.17 2003/01/05 20:39:53 mazen
  651. * warnings about FreeTemp already free fixed with appropriate registers handling
  652. Revision 1.16 2002/10/28 20:59:17 mazen
  653. * TOpSize values changed S_L --> S_SW
  654. Revision 1.15 2002/10/28 20:37:44 mazen
  655. * TOpSize values changed S_L --> S_SW
  656. Revision 1.14 2002/10/20 19:01:38 mazen
  657. + op_raddr_reg and op_caddr_reg added to fix functions prologue
  658. Revision 1.13 2002/10/19 20:35:07 mazen
  659. * carl's patch applied
  660. Revision 1.12 2002/10/11 13:35:14 mazen
  661. *** empty log message ***
  662. Revision 1.11 2002/10/10 19:57:51 mazen
  663. * Just to update repsitory
  664. Revision 1.10 2002/10/02 22:20:28 mazen
  665. + out registers allocator for the first 6 scalar parameters which must be passed into %o0..%o5
  666. Revision 1.9 2002/10/01 21:06:29 mazen
  667. attinst.inc --> strinst.inc
  668. Revision 1.8 2002/09/30 19:12:14 mazen
  669. * function prologue fixed
  670. Revision 1.7 2002/09/27 04:30:53 mazen
  671. * cleanup made
  672. Revision 1.6 2002/09/24 03:57:53 mazen
  673. * some cleanup was made
  674. }