n386mem.pas 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158
  1. {
  2. Copyright (c) 1998-2002 by Florian Klaempfl
  3. Generate i386 assembler for in memory related nodes
  4. This program is free software; you can redistribute it and/or modify
  5. it under the terms of the GNU General Public License as published by
  6. the Free Software Foundation; either version 2 of the License, or
  7. (at your option) any later version.
  8. This program is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. GNU General Public License for more details.
  12. You should have received a copy of the GNU General Public License
  13. along with this program; if not, write to the Free Software
  14. Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  15. ****************************************************************************
  16. }
  17. unit n386mem;
  18. {$i fpcdefs.inc}
  19. interface
  20. uses
  21. globtype,
  22. cgbase,cpuinfo,cpubase,
  23. node,nmem,ncgmem;
  24. type
  25. ti386addrnode = class(tcgaddrnode)
  26. procedure pass_generate_code;override;
  27. end;
  28. ti386derefnode = class(tcgderefnode)
  29. procedure pass_generate_code;override;
  30. end;
  31. ti386vecnode = class(tcgvecnode)
  32. procedure update_reference_reg_mul(maybe_const_reg:tregister;l:aint);override;
  33. procedure pass_generate_code;override;
  34. end;
  35. implementation
  36. uses
  37. systems,
  38. cutils,verbose,
  39. symdef,paramgr,
  40. aasmtai,aasmdata,
  41. nld,ncon,nadd,
  42. cgutils,cgobj;
  43. {*****************************************************************************
  44. TI386ADDRNODE
  45. *****************************************************************************}
  46. procedure ti386addrnode.pass_generate_code;
  47. begin
  48. inherited pass_generate_code;
  49. { for use of other segments, not used }
  50. {if left.location.reference.segment<>NR_NO then
  51. location.segment:=left.location.reference.segment;}
  52. end;
  53. {*****************************************************************************
  54. TI386DEREFNODE
  55. *****************************************************************************}
  56. procedure ti386derefnode.pass_generate_code;
  57. begin
  58. inherited pass_generate_code;
  59. if tpointerdef(left.resultdef).is_far then
  60. location.reference.segment:=NR_FS;
  61. end;
  62. {*****************************************************************************
  63. TI386VECNODE
  64. *****************************************************************************}
  65. { this routine must, like any other routine, not change the contents }
  66. { of base/index registers of references, as these may be regvars. }
  67. { The register allocator can coalesce one LOC_REGISTER being moved }
  68. { into another (as their live ranges won't overlap), but not a }
  69. { LOC_CREGISTER moved into a LOC_(C)REGISTER most of the time (as }
  70. { the live range of the LOC_CREGISTER will most likely overlap the }
  71. { the live range of the target LOC_(C)REGISTER) }
  72. { The passed register may be a LOC_CREGISTER as well. }
  73. procedure ti386vecnode.update_reference_reg_mul(maybe_const_reg:tregister;l:aint);
  74. var
  75. l2 : integer;
  76. hreg : tregister;
  77. begin
  78. { Optimized for x86 to use the index register and scalefactor }
  79. if location.reference.index=NR_NO then
  80. begin
  81. { no preparations needed }
  82. end
  83. else if location.reference.base=NR_NO then
  84. begin
  85. if (location.reference.scalefactor > 1) then
  86. hreg:=cg.getaddressregister(current_asmdata.CurrAsmList);
  87. case location.reference.scalefactor of
  88. 0,1 : hreg:=location.reference.index;
  89. 2 : cg.a_op_const_reg_reg(current_asmdata.CurrAsmList,OP_SHL,OS_ADDR,1,location.reference.index,hreg);
  90. 4 : cg.a_op_const_reg_reg(current_asmdata.CurrAsmList,OP_SHL,OS_ADDR,2,location.reference.index,hreg);
  91. 8 : cg.a_op_const_reg_reg(current_asmdata.CurrAsmList,OP_SHL,OS_ADDR,3,location.reference.index,hreg);
  92. else
  93. internalerror(2008091401);
  94. end;
  95. location.reference.base:=hreg;
  96. end
  97. else
  98. begin
  99. hreg:=cg.getaddressregister(current_asmdata.CurrAsmList);
  100. cg.a_loadaddr_ref_reg(current_asmdata.CurrAsmList,location.reference,hreg);
  101. reference_reset_base(location.reference,hreg,0,location.reference.alignment);
  102. end;
  103. { insert the new index register and scalefactor or
  104. do the multiplication manual }
  105. case l of
  106. 1,2,4,8 :
  107. begin
  108. location.reference.scalefactor:=l;
  109. hreg:=maybe_const_reg;
  110. end;
  111. else
  112. begin
  113. hreg:=cg.getaddressregister(current_asmdata.CurrAsmList);
  114. if ispowerof2(l,l2) then
  115. cg.a_op_const_reg_reg(current_asmdata.CurrAsmList,OP_SHL,OS_ADDR,l2,maybe_const_reg,hreg)
  116. else
  117. cg.a_op_const_reg_reg(current_asmdata.CurrAsmList,OP_IMUL,OS_ADDR,l,maybe_const_reg,hreg);
  118. end;
  119. end;
  120. location.reference.index:=hreg;
  121. end;
  122. procedure ti386vecnode.pass_generate_code;
  123. begin
  124. inherited pass_generate_code;
  125. if nf_memseg in flags then
  126. location.reference.segment:=NR_FS;
  127. end;
  128. begin
  129. caddrnode:=ti386addrnode;
  130. cderefnode:=ti386derefnode;
  131. cvecnode:=ti386vecnode;
  132. end.