aoptcpu.pas 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539
  1. {
  2. Copyright (c) 1998-2002 by Jonas Maebe, member of the Free Pascal
  3. Development Team
  4. This unit implements the PowerPC optimizer object
  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. }
  18. Unit aoptcpu;
  19. Interface
  20. {$i fpcdefs.inc}
  21. uses cpubase, aoptobj, aoptcpub, aopt, aasmtai,aasmdata, aasmcpu;
  22. Type
  23. TCpuAsmOptimizer = class(TAsmOptimizer)
  24. { uses the same constructor as TAopObj }
  25. function PeepHoleOptPass1Cpu(var p: tai): boolean; override;
  26. function PostPeepHoleOptsCpu(var p: tai): boolean; override;
  27. private
  28. function cmpi_mfcr_opt(p, next1, next2: taicpu): boolean;
  29. End;
  30. Implementation
  31. uses
  32. cutils, verbose, cgbase, cgcpu, cgobj;
  33. const
  34. calculation_target_op0: array[tasmop] of tasmop = (a_none,
  35. a_add, a_add_, a_addo, a_addo_, a_addc, a_addc_, a_addco, a_addco_,
  36. a_adde, a_adde_, a_addeo, a_addeo_, a_addi, a_addic, a_addic_, a_addis,
  37. a_addme, a_addme_, a_addmeo, a_addmeo_, a_addze, a_addze_, a_addzeo,
  38. a_addzeo_, a_and, a_and_, a_andc, a_andc_, a_andi_, a_andis_, a_none,
  39. a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  40. a_none, a_none, a_none, a_none, a_none, a_cntlzw, a_cntlzw_, a_none,
  41. a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  42. a_none, a_none, a_none, a_none, a_none, a_none, a_divw, a_divw_, a_divwo, a_divwo_,
  43. a_divwu, a_divwu_, a_divwuo, a_divwuo_, a_none, a_none, a_none, a_eqv,
  44. a_eqv_, a_extsb, a_extsb_, a_extsh, a_extsh_, a_fabs, a_fabs_, a_fadd,
  45. a_fadd_, a_fadds, a_fadds_, a_none, a_none,
  46. a_fctid, a_fctid_, a_fctidz, a_fctidz_,
  47. a_fctiw, a_fctiw_, a_fctiwz, a_fctiwz_,
  48. a_fdiv, a_fdiv_, a_fdivs, a_fdivs_, a_fmadd, a_fmadd_, a_fmadds,
  49. a_fmadds_, a_none, a_fmsub, a_fmsub_, a_fmsubs, a_fmsubs_, a_fmul, a_fmul_,
  50. a_fmuls, a_fmuls_, a_fnabs, a_fnabs_, a_fneg, a_fneg_, a_fnmadd,
  51. a_fnmadd_, a_fnmadds, a_fnmadds_, a_fnmsub, a_fnmsub_, a_fnmsubs,
  52. a_fnmsubs_, a_fres, a_fres_, a_frsp, a_frsp_, a_frsqrte, a_frsqrte_,
  53. a_none, a_none, a_fsqrt, a_fsqrt_, a_fsqrts, a_fsqrts_, a_fsub, a_fsub_,
  54. a_fsubs, a_fsubs_, a_none, a_none, a_none, a_none, a_none, a_none,
  55. a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  56. a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  57. a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  58. a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  59. a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  60. a_none, a_none, a_none, a_none, a_none, a_none, a_mulhw,
  61. a_mulhw_, a_mulhwu, a_mulhwu_, a_mulli, a_mullw, a_mullw_, a_mullwo,
  62. a_mullwo_, a_nand, a_nand_, a_neg, a_neg_, a_nego, a_nego_, a_nor, a_nor_,
  63. a_or, a_or_, a_orc, a_orc_, a_ori, a_oris, a_rfi, a_rlwimi, a_rlwimi_,
  64. a_rlwinm, a_rlwinm_, a_rlwnm, a_rlwnm_, a_none, a_slw, a_slw_, a_sraw, a_sraw_,
  65. a_srawi, a_srawi_,a_srw, a_srw_, a_none, a_none, a_none, a_none, a_none,
  66. a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  67. a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  68. a_none, a_none, a_none, a_none, a_none, a_subf, a_subf_, a_subfo,
  69. a_subfo_, a_subfc, a_subfc_, a_subfco, a_subfco_, a_subfe, a_subfe_,
  70. a_subfeo, a_subfeo_, a_subfic, a_subfme, a_subfme_, a_subfmeo, a_subfmeo_,
  71. a_subfze, a_subfze_, a_subfzeo, a_subfzeo_, a_none, a_none, a_none,
  72. a_none, a_none, a_none, a_xor, a_xor_, a_xori, a_xoris,
  73. { simplified mnemonics }
  74. a_subi, a_subis, a_subic, a_subic_, a_sub, a_sub_, a_subo, a_subo_,
  75. a_subc, a_subc_, a_subco, a_subco_, a_none, a_none, a_none, a_none,
  76. a_extlwi, a_extlwi_, a_extrwi, a_extrwi_, a_inslwi, a_inslwi_, a_insrwi,
  77. a_insrwi_, a_rotlwi, a_rotlwi_, a_rotlw, a_rotlw_, a_slwi, a_slwi_,
  78. a_srwi, a_srwi_, a_clrlwi, a_clrlwi_, a_clrrwi, a_clrrwi_, a_clrslwi,
  79. a_clrslwi_, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  80. a_none, a_none {move to special purpose reg}, a_none {move from special purpose reg},
  81. a_none, a_none, a_none, a_none, a_none, a_none, a_not, a_not_, a_none, a_none, a_none,
  82. a_none, a_none, a_none, a_none,
  83. a_none, a_none, a_none, a_none, a_none);
  84. function TCpuAsmOptimizer.cmpi_mfcr_opt(p, next1, next2: taicpu): boolean;
  85. var
  86. next3, prev: tai;
  87. inverse, prevrlwinm: boolean;
  88. begin
  89. result := true;
  90. inverse :=
  91. getnextinstruction(next2,next3) and
  92. (next3.typ = ait_instruction) and
  93. (taicpu(next3).opcode = A_XORI) and
  94. (taicpu(next3).oper[0]^.reg = taicpu(next3).oper[1]^.reg) and
  95. (taicpu(next3).oper[0]^.reg = taicpu(next2).oper[0]^.reg) and
  96. (taicpu(next3).oper[2]^.val = 1);
  97. case taicpu(next2).oper[2]^.val of
  98. 1:
  99. begin
  100. // less than zero or greater/equal than zero (the xori remains in
  101. // in the latter case). Doesn't make sense for unsigned comparisons.
  102. if (p.opcode = A_CMPWI) then
  103. begin
  104. p.opcode := A_SRWI;
  105. p.ops := 3;
  106. p.loadreg(1,p.oper[0]^.reg);
  107. p.loadreg(0,next1.oper[0]^.reg);
  108. p.loadconst(2,31);
  109. asml.remove(next1);
  110. next1.free;
  111. asml.remove(next2);
  112. next2.free;
  113. end
  114. else
  115. result := false;
  116. end;
  117. {
  118. needs two registers to work with
  119. 2:
  120. begin
  121. // greater or less/equal to zero
  122. end;
  123. }
  124. 3:
  125. begin
  126. prevrlwinm :=
  127. getlastinstruction(p,prev) and
  128. (prev.typ = ait_instruction) and
  129. ((taicpu(prev).opcode = A_RLWINM) or
  130. (taicpu(prev).opcode = A_RLWINM_)) and
  131. (taicpu(prev).oper[0]^.reg = p.oper[0]^.reg) and
  132. (taicpu(prev).oper[3]^.val = taicpu(prev).oper[4]^.val);
  133. if (prevrlwinm) then
  134. begin
  135. // isolate the bit we need
  136. if (taicpu(prev).oper[3]^.val <> 31) then
  137. begin
  138. p.opcode := A_RLWINM;
  139. p.ops := 5;
  140. p.loadreg(1,p.oper[0]^.reg);
  141. p.loadreg(0,next1.oper[0]^.reg);
  142. p.loadconst(2,taicpu(prev).oper[3]^.val + 1);
  143. p.loadconst(3,31);
  144. p.loadconst(4,31);
  145. end
  146. else { if (taicpu(prev).oper[0]^.reg <> next1.oper[0]^.reg) then }
  147. begin
  148. p.opcode := A_MR;
  149. p.loadreg(1,p.oper[0]^.reg);
  150. p.loadreg(0,next1.oper[0]^.reg);
  151. end;
  152. if not inverse then
  153. begin
  154. next1.ops := 3;
  155. next1.opcode := A_XORI;
  156. next1.loadreg(1,next1.oper[0]^.reg);
  157. next1.loadconst(2,1);
  158. end
  159. else
  160. begin
  161. asml.remove(next1);
  162. next1.free;
  163. asml.remove(next3);
  164. next3.free;
  165. end;
  166. asml.remove(next2);
  167. next2.free;
  168. end
  169. else
  170. begin
  171. // equal/not equal to zero (the xori remains in the latter case;
  172. // there's a more optimal sequence without it, but needs extra
  173. // register)
  174. p.opcode := A_CNTLZW;
  175. p.loadreg(1,p.oper[0]^.reg);
  176. p.loadreg(0,next1.oper[0]^.reg);
  177. next1.ops := 3;
  178. next1.opcode := A_SRWI;
  179. next1.loadreg(1,next1.oper[0]^.reg);
  180. next1.loadconst(2,5);
  181. asml.remove(next2);
  182. next2.free;
  183. end;
  184. end;
  185. else
  186. result := false;
  187. end;
  188. end;
  189. function rlwinm2mask(l1,l2: longint): longint;
  190. begin
  191. // 1 shl 32 = 1 instead of 0 on x86
  192. if (l1 <> 0) then
  193. result := longint(cardinal(1) shl (32 - l1) - 1) xor (cardinal(1) shl (31 - l2) - 1)
  194. else
  195. result := longint(not(cardinal(1) shl (31 - l2) - 1));
  196. if (l1 > l2) then
  197. result := not(result);
  198. end;
  199. function TCpuAsmOptimizer.PeepHoleOptPass1Cpu(var p: tai): boolean;
  200. var
  201. next1, next2: tai;
  202. l1, l2, shlcount: longint;
  203. begin
  204. result := false;
  205. case p.typ of
  206. ait_instruction:
  207. begin
  208. case taicpu(p).opcode of
  209. A_CMPWI,
  210. A_CMPLWI:
  211. begin
  212. if (taicpu(p).oper[1]^.typ = top_const) and
  213. (taicpu(p).oper[1]^.val = 0) and
  214. getnextinstruction(p,next1) and
  215. (next1.typ = ait_instruction) and
  216. (taicpu(next1).opcode = A_MFCR) and
  217. getnextinstruction(next1,next2) and
  218. (taicpu(next2).opcode = A_RLWINM) and
  219. (taicpu(next2).oper[0]^.reg = taicpu(next2).oper[1]^.reg) and
  220. (taicpu(next2).oper[0]^.reg = taicpu(next1).oper[0]^.reg) and
  221. (taicpu(next2).oper[3]^.val = 31) and
  222. (taicpu(next2).oper[4]^.val = 31) and
  223. cmpi_mfcr_opt(taicpu(p),taicpu(next1),taicpu(next2)) then
  224. result := true;
  225. end;
  226. { seems the register allocator doesn't generate superfluous fmr's }
  227. { A_FMR, }
  228. A_MR:
  229. begin
  230. if getnextinstruction(p,next1) and
  231. (next1.typ = ait_instruction) and
  232. (calculation_target_op0[taicpu(next1).opcode] <> a_none) and
  233. (taicpu(next1).oper[0]^.reg = taicpu(p).oper[0]^.reg) then
  234. begin
  235. for l1 := 1 to taicpu(next1).ops - 1 do
  236. if (taicpu(next1).oper[l1]^.typ = top_reg) and
  237. (taicpu(next1).oper[l1]^.reg = taicpu(p).oper[0]^.reg) then
  238. taicpu(next1).loadreg(l1,taicpu(p).oper[1]^.reg);
  239. asml.remove(p);
  240. p.free;
  241. p := next1;
  242. result := true;
  243. end;
  244. end;
  245. A_SLWI:
  246. begin
  247. if getnextinstruction(p,next1) and
  248. (next1.typ = ait_instruction) and
  249. ((taicpu(next1).opcode = A_RLWINM) or
  250. (taicpu(next1).opcode = A_SLWI) or
  251. (taicpu(next1).opcode = A_SRWI)) and
  252. (taicpu(next1).oper[0]^.reg = taicpu(p).oper[0]^.reg) and
  253. (taicpu(next1).oper[1]^.reg = taicpu(p).oper[0]^.reg) then
  254. begin
  255. { convert slwi to rlwinm and see if the rlwinm }
  256. { optimization can do something with it }
  257. taicpu(p).opcode := A_RLWINM;
  258. taicpu(p).ops := 5;
  259. taicpu(p).loadconst(2,taicpu(p).oper[2]^.val);
  260. taicpu(p).loadconst(3,0);
  261. taicpu(p).loadconst(4,31-taicpu(p).oper[2]^.val);
  262. result := true;
  263. end;
  264. end;
  265. A_SRWI:
  266. begin
  267. if getnextinstruction(p,next1) and
  268. (next1.typ = ait_instruction) and
  269. ((taicpu(next1).opcode = A_SLWI) or
  270. (taicpu(next1).opcode = A_RLWINM) or
  271. (taicpu(next1).opcode = A_SRWI)) and
  272. (taicpu(next1).oper[0]^.reg = taicpu(p).oper[0]^.reg) and
  273. (taicpu(next1).oper[1]^.reg = taicpu(p).oper[0]^.reg) then
  274. case taicpu(next1).opcode of
  275. A_SLWI:
  276. begin
  277. taicpu(p).opcode := A_RLWINM;
  278. taicpu(p).ops := 5;
  279. taicpu(p).loadconst(2,taicpu(next1).oper[2]^.val-taicpu(p).oper[2]^.val);
  280. if (taicpu(p).oper[2]^.val < 0) then
  281. begin
  282. taicpu(p).loadconst(3,-taicpu(p).oper[2]^.val);
  283. taicpu(p).loadconst(4,31-taicpu(next1).oper[2]^.val);
  284. inc(taicpu(p).oper[2]^.val,32);
  285. end
  286. else
  287. begin
  288. taicpu(p).loadconst(3,0);
  289. taicpu(p).loadconst(4,31-taicpu(next1).oper[2]^.val);
  290. end;
  291. asml.remove(next1);
  292. next1.free;
  293. result := true;
  294. end;
  295. A_RLWINM:
  296. begin
  297. { convert srwi to rlwinm and see if the rlwinm }
  298. { optimization can do something with it }
  299. taicpu(p).opcode := A_RLWINM;
  300. taicpu(p).ops := 5;
  301. taicpu(p).loadconst(3,taicpu(p).oper[2]^.val);
  302. taicpu(p).loadconst(4,31);
  303. taicpu(p).loadconst(2,(32-taicpu(p).oper[2]^.val) and 31);
  304. result := true;
  305. end;
  306. end;
  307. end;
  308. A_RLWINM:
  309. begin
  310. if getnextinstruction(p,next1) and
  311. (next1.typ = ait_instruction) and
  312. ((taicpu(next1).opcode = A_RLWINM) or
  313. (taicpu(next1).opcode = A_SRWI) or
  314. (taicpu(next1).opcode = A_SLWI)) and
  315. (taicpu(next1).oper[0]^.reg = taicpu(p).oper[0]^.reg) and
  316. // both source and target of next1 must equal target of p
  317. (taicpu(next1).oper[1]^.reg = taicpu(p).oper[0]^.reg) then
  318. begin
  319. case taicpu(next1).opcode of
  320. A_RLWINM:
  321. begin
  322. shlcount := taicpu(next1).oper[2]^.val;
  323. l2 := rlwinm2mask(taicpu(next1).oper[3]^.val,taicpu(next1).oper[4]^.val);
  324. end;
  325. A_SLWI:
  326. begin
  327. shlcount := taicpu(next1).oper[2]^.val;
  328. l2 := (-1) shl shlcount;
  329. end;
  330. A_SRWI:
  331. begin
  332. shlcount := 32-taicpu(next1).oper[2]^.val;
  333. l2 := (-1) shr taicpu(next1).oper[2]^.val;
  334. end;
  335. else
  336. internalerror(2013113008);
  337. end;
  338. l1 := rlwinm2mask((taicpu(p).oper[3]^.val-shlcount) and 31,(taicpu(p).oper[4]^.val-shlcount) and 31);
  339. l1 := l1 and l2;
  340. case l1 of
  341. -1:
  342. begin
  343. taicpu(p).oper[2]^.val := (taicpu(p).oper[2]^.val + shlcount) and 31;
  344. asml.remove(next1);
  345. next1.free;
  346. if (taicpu(p).oper[2]^.val = 0) then
  347. begin
  348. next1 := tai(p.next);
  349. asml.remove(p);
  350. p.free;
  351. p := next1;
  352. result := true;
  353. end;
  354. end;
  355. 0:
  356. begin
  357. // masks have no bits in common
  358. taicpu(p).opcode := A_LI;
  359. taicpu(p).loadconst(1,0);
  360. taicpu(p).freeop(2);
  361. taicpu(p).freeop(3);
  362. taicpu(p).freeop(4);
  363. taicpu(p).ops := 2;
  364. taicpu(p).opercnt := 2;
  365. asml.remove(next1);
  366. next1.free;
  367. result := true;
  368. end
  369. else if tcgppc(cg).get_rlwi_const(l1,l1,l2) then
  370. begin
  371. taicpu(p).oper[2]^.val := (taicpu(p).oper[2]^.val + shlcount) and 31;
  372. taicpu(p).oper[3]^.val := l1;
  373. taicpu(p).oper[4]^.val := l2;
  374. asml.remove(next1);
  375. next1.free;
  376. result := true;
  377. end;
  378. end;
  379. end;
  380. end;
  381. end;
  382. end;
  383. end;
  384. end;
  385. const
  386. modifyflags: array[tasmop] of tasmop =
  387. (a_none, a_add_, a_add_, a_addo_, a_addo_, a_addc_, a_addc_, a_addco_, a_addco_,
  388. a_adde_, a_adde_, a_addeo_, a_addeo_, {a_addi could be addic_ if sure doesn't disturb carry} a_none, a_addic_, a_addic_, a_none,
  389. a_addme_, a_addme_, a_addmeo_, a_addmeo_, a_addze_, a_addze_, a_addzeo_,
  390. a_addzeo_, a_and_, a_and_, a_andc_, a_andc_, a_andi_, a_andis_, a_none,
  391. a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  392. a_none, a_none, a_none, a_none, a_none, a_cntlzw_, a_cntlzw_, a_none,
  393. a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  394. a_none, a_none, a_none, a_none, a_none, a_none, a_divw_, a_divw_, a_divwo_, a_divwo_,
  395. a_divwu_, a_divwu_, a_divwuo_, a_divwuo_, a_none, a_none, a_none, a_eqv_,
  396. a_eqv_, a_extsb_, a_extsb_, a_extsh_, a_extsh_, a_none, a_none, a_none,
  397. a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  398. a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  399. a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  400. a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  401. a_none, a_none, a_none, a_none, a_none, a_none,
  402. a_none, a_none, a_none, a_none,
  403. a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  404. a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  405. a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  406. a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  407. a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  408. a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  409. a_none, a_none, a_none, a_mffs, a_mffs_, a_mfmsr, a_mfspr, a_mfsr,
  410. a_mfsrin, a_mftb, a_mtcrf, a_none, a_none, a_none, a_none,
  411. a_none, a_none, a_none, a_none, a_none, a_none, a_mulhw_,
  412. a_mulhw_, a_mulhwu_, a_mulhwu_, a_none, a_mullw_, a_mullw_, a_mullwo_,
  413. a_mullwo_, a_nand_, a_nand_, a_neg_, a_neg_, a_nego_, a_nego_, a_nor_, a_nor_,
  414. a_or_, a_or_, a_orc_, a_orc_, a_none, a_none, a_none, a_rlwimi_, a_rlwimi_,
  415. a_rlwinm_, a_rlwinm_, a_rlwnm_, a_rlwnm_, a_none, a_slw_, a_slw_, a_sraw_, a_sraw_,
  416. a_srawi_, a_srawi_,a_srw_, a_srw_, a_none, a_none, a_none, a_none, a_none,
  417. a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  418. a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  419. a_none, a_none, a_none, a_none, a_none, a_subf_, a_subf_, a_subfo_,
  420. a_subfo_, a_subfc_, a_subfc_, a_subfco_, a_subfco_, a_subfe_, a_subfe_,
  421. a_subfeo_, a_subfeo_, a_none, a_subfme_, a_subfme_, a_subfmeo_, a_subfmeo_,
  422. a_subfze_, a_subfze_, a_subfzeo_, a_subfzeo_, a_none, a_none, a_none,
  423. a_none, a_none, a_none, a_xor_, a_xor_, a_none, a_none,
  424. { simplified mnemonics }
  425. a_none, a_none, a_subic_, a_subic_, a_sub_, a_sub_, a_subo_, a_subo_,
  426. a_subc_, a_subc_, a_subco_, a_subco_, a_none, a_none, a_none, a_none,
  427. a_extlwi_, a_extlwi_, a_extrwi_, a_extrwi_, a_inslwi_, a_inslwi_, a_insrwi_,
  428. a_insrwi_, a_rotlwi_, a_rotlwi_, a_rotlw_, a_rotlw_, a_slwi_, a_slwi_,
  429. a_srwi_, a_srwi_, a_clrlwi_, a_clrlwi_, a_clrrwi_, a_clrrwi_, a_clrslwi_,
  430. a_clrslwi_, a_none, a_none, a_none, a_none, a_none, a_none, a_none,
  431. a_none, a_none {move to special purpose reg}, a_none {move from special purpose reg},
  432. a_none, a_none, a_none, a_none, a_mr_, a_mr_, a_not_, a_not_, a_none, a_none, a_none,
  433. a_none, a_none, a_none, a_none,
  434. a_none, a_none, a_none, a_none, a_none);
  435. function changetomodifyflags(p: taicpu): boolean;
  436. begin
  437. result := false;
  438. if (modifyflags[p.opcode] <> a_none) then
  439. begin
  440. p.opcode := modifyflags[p.opcode];
  441. result := true;
  442. end;
  443. end;
  444. function TCpuAsmOptimizer.PostPeepHoleOptsCpu(var p: tai): boolean;
  445. var
  446. next1: tai;
  447. begin
  448. result := false;
  449. case p.typ of
  450. ait_instruction:
  451. begin
  452. case taicpu(p).opcode of
  453. A_RLWINM_:
  454. begin
  455. // rlwinm_ is cracked on the G5, andi_/andis_ aren't
  456. if (taicpu(p).oper[2]^.val = 0) then
  457. if (taicpu(p).oper[3]^.val < 16) and
  458. (taicpu(p).oper[4]^.val < 16) then
  459. begin
  460. taicpu(p).opcode := A_ANDIS_;
  461. taicpu(p).oper[2]^.val := word(
  462. ((1 shl (16-taicpu(p).oper[3]^.val)) - 1) xor
  463. ((1 shl (15-taicpu(p).oper[4]^.val)) - 1));
  464. taicpu(p).freeop(3);
  465. taicpu(p).freeop(4);
  466. taicpu(p).ops := 3;
  467. taicpu(p).opercnt := 3;
  468. end
  469. else if (taicpu(p).oper[3]^.val >= 16) and
  470. (taicpu(p).oper[4]^.val >= 16) then
  471. begin
  472. taicpu(p).opcode := A_ANDI_;
  473. taicpu(p).oper[2]^.val := word(rlwinm2mask(taicpu(p).oper[3]^.val,taicpu(p).oper[4]^.val));
  474. taicpu(p).freeop(3);
  475. taicpu(p).freeop(4);
  476. taicpu(p).ops := 3;
  477. taicpu(p).opercnt := 3;
  478. end;
  479. end;
  480. end;
  481. // change "integer operation with destination reg" followed by a
  482. // comparison to zero of that reg, with a variant of that integer
  483. // operation which sets the flags (if it exists)
  484. if not(result) and
  485. (taicpu(p).ops >= 2) and
  486. (taicpu(p).oper[0]^.typ = top_reg) and
  487. (taicpu(p).oper[1]^.typ = top_reg) and
  488. getnextinstruction(p,next1) and
  489. (next1.typ = ait_instruction) and
  490. (taicpu(next1).opcode = A_CMPWI) and
  491. // make sure it the result goes to cr0
  492. (((taicpu(next1).ops = 2) and
  493. (taicpu(next1).oper[1]^.val = 0) and
  494. (taicpu(next1).oper[0]^.reg = taicpu(p).oper[0]^.reg)) or
  495. ((taicpu(next1).ops = 3) and
  496. (taicpu(next1).oper[2]^.val = 0) and
  497. (taicpu(next1).oper[0]^.typ = top_reg) and
  498. (getsupreg(taicpu(next1).oper[0]^.reg) = RS_CR0) and
  499. (taicpu(next1).oper[1]^.reg = taicpu(p).oper[0]^.reg))) and
  500. changetomodifyflags(taicpu(p)) then
  501. begin
  502. asml.remove(next1);
  503. next1.free;
  504. result := true;
  505. end;
  506. end;
  507. end;
  508. end;
  509. begin
  510. casmoptimizer:=TCpuAsmOptimizer;
  511. End.