nllvmcnv.pas 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  1. {
  2. Copyright (c) 2014 by Jonas Maebe
  3. Generate LLVM IR for type converting 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 nllvmcnv;
  18. {$i fpcdefs.inc}
  19. interface
  20. uses
  21. node,ncnv,ncgcnv,defcmp;
  22. type
  23. tllvmtypeconvnode = class(tcgtypeconvnode)
  24. protected
  25. function first_int_to_real: tnode; override;
  26. function first_int_to_bool: tnode; override;
  27. procedure second_int_to_int;override;
  28. { procedure second_string_to_string;override; }
  29. { procedure second_cstring_to_pchar;override; }
  30. { procedure second_string_to_chararray;override; }
  31. { procedure second_array_to_pointer;override; }
  32. procedure second_pointer_to_array;override;
  33. { procedure second_chararray_to_string;override; }
  34. { procedure second_char_to_string;override; }
  35. procedure second_int_to_real;override;
  36. { procedure second_real_to_real;override; }
  37. { procedure second_cord_to_pointer;override; }
  38. { procedure second_proc_to_procvar;override; }
  39. procedure second_bool_to_int;override;
  40. procedure second_int_to_bool;override;
  41. { procedure second_load_smallset;override; }
  42. { procedure second_ansistring_to_pchar;override; }
  43. { procedure second_pchar_to_string;override; }
  44. { procedure second_class_to_intf;override; }
  45. { procedure second_char_to_char;override; }
  46. procedure second_nothing; override;
  47. end;
  48. implementation
  49. uses
  50. globtype,globals,verbose,
  51. aasmbase,aasmdata,
  52. llvmbase,aasmllvm,
  53. procinfo,
  54. symconst,symtype,symdef,defutil,
  55. cgbase,cgutils,hlcgobj,pass_2;
  56. { tllvmtypeconvnode }
  57. function tllvmtypeconvnode.first_int_to_real: tnode;
  58. begin
  59. expectloc:=LOC_FPUREGISTER;
  60. result:=nil;
  61. end;
  62. function tllvmtypeconvnode.first_int_to_bool: tnode;
  63. begin
  64. result:=inherited;
  65. if not assigned(result) then
  66. expectloc:=LOC_JUMP;
  67. end;
  68. procedure tllvmtypeconvnode.second_int_to_int;
  69. var
  70. fromsize, tosize: tcgint;
  71. hreg: tregister;
  72. begin
  73. if not(nf_explicit in flags) then
  74. hlcg.g_rangecheck(current_asmdata.CurrAsmList,left.location,left.resultdef,resultdef);
  75. fromsize:=left.resultdef.size;
  76. tosize:=resultdef.size;
  77. location_copy(location,left.location);
  78. if not(left.location.loc in [LOC_REFERENCE,LOC_CREFERENCE]) or
  79. (fromsize<>tosize) then
  80. begin
  81. hlcg.location_force_reg(current_asmdata.CurrAsmList,location,left.resultdef,resultdef,left.location.loc=LOC_CREGISTER);
  82. end
  83. else if left.resultdef<>resultdef then
  84. begin
  85. { just typecast the pointer type }
  86. hreg:=hlcg.getaddressregister(current_asmdata.CurrAsmList,cpointerdef.getreusable(resultdef));
  87. hlcg.a_loadaddr_ref_reg(current_asmdata.CurrAsmList,left.resultdef,cpointerdef.getreusable(resultdef),left.location.reference,hreg);
  88. hlcg.reference_reset_base(location.reference,cpointerdef.getreusable(resultdef),hreg,0,location.reference.alignment);
  89. end;
  90. end;
  91. procedure tllvmtypeconvnode.second_pointer_to_array;
  92. var
  93. hreg: tregister;
  94. begin
  95. inherited;
  96. { insert type conversion }
  97. hreg:=hlcg.getaddressregister(current_asmdata.CurrAsmList,cpointerdef.getreusable(resultdef));
  98. hlcg.a_loadaddr_ref_reg(current_asmdata.CurrAsmList,tpointerdef(left.resultdef).pointeddef,cpointerdef.getreusable(resultdef),location.reference,hreg);
  99. reference_reset_base(location.reference,hreg,0,location.reference.alignment);
  100. end;
  101. procedure tllvmtypeconvnode.second_int_to_real;
  102. var
  103. op: tllvmop;
  104. begin
  105. if is_signed(left.resultdef) then
  106. op:=la_sitofp
  107. else
  108. op:=la_uitofp;
  109. location_reset(location,LOC_FPUREGISTER,def_cgsize(resultdef));
  110. location.register:=hlcg.getfpuregister(current_asmdata.CurrAsmList,resultdef);
  111. hlcg.location_force_reg(current_asmdata.CurrAsmList,left.location,left.resultdef,left.resultdef,true);
  112. current_asmdata.CurrAsmList.concat(taillvm.op_reg_size_reg_size(op,location.register,left.resultdef,left.location.register,resultdef));
  113. end;
  114. procedure tllvmtypeconvnode.second_bool_to_int;
  115. var
  116. pdef: tdef;
  117. hreg: tregister;
  118. begin
  119. inherited;
  120. { all boolean/integer of the same size are represented using the same type
  121. by FPC in LLVM, except for Pascal booleans, which are i1 -> convert
  122. the type if necessary. This never has to be done for registers on the
  123. assignment side, because we make everything that's explicitly typecasted
  124. on the assignment side non regable for llvm }
  125. if is_pasbool(left.resultdef) and
  126. (nf_explicit in flags) and
  127. not(left.location.loc in [LOC_FLAGS,LOC_JUMP]) and
  128. (resultdef.size=1) then
  129. case location.loc of
  130. LOC_REFERENCE,LOC_CREFERENCE:
  131. begin
  132. pdef:=cpointerdef.getreusable(resultdef);
  133. hreg:=hlcg.getaddressregister(current_asmdata.CurrAsmList,pdef);
  134. hlcg.a_loadaddr_ref_reg(current_asmdata.CurrAsmList,left.resultdef,pdef,location.reference,hreg);
  135. hlcg.reference_reset_base(location.reference,pdef,hreg,0,location.reference.alignment);
  136. end;
  137. LOC_REGISTER,LOC_CREGISTER:
  138. begin
  139. hreg:=hlcg.getintregister(current_asmdata.CurrAsmList,resultdef);
  140. hlcg.a_load_reg_reg(current_asmdata.CurrAsmList,left.resultdef,resultdef,location.register,hreg);
  141. location.register:=hreg;
  142. end;
  143. end;
  144. end;
  145. procedure tllvmtypeconvnode.second_int_to_bool;
  146. var
  147. truelabel,
  148. falselabel: tasmlabel;
  149. newsize : tcgsize;
  150. begin
  151. secondpass(left);
  152. if codegenerror then
  153. exit;
  154. { Explicit typecasts from any ordinal type to a boolean type }
  155. { must not change the ordinal value }
  156. if (nf_explicit in flags) and
  157. not(left.location.loc in [LOC_FLAGS,LOC_JUMP]) then
  158. begin
  159. location_copy(location,left.location);
  160. newsize:=def_cgsize(resultdef);
  161. { change of size? change sign only if location is LOC_(C)REGISTER? Then we have to sign/zero-extend }
  162. if (tcgsize2size[newsize]<>tcgsize2size[left.location.size]) or
  163. ((newsize<>left.location.size) and (location.loc in [LOC_REGISTER,LOC_CREGISTER])) then
  164. hlcg.location_force_reg(current_asmdata.CurrAsmList,location,left.resultdef,resultdef,true)
  165. else
  166. location.size:=newsize;
  167. exit;
  168. end;
  169. case left.location.loc of
  170. LOC_SUBSETREG,LOC_CSUBSETREG,LOC_SUBSETREF,LOC_CSUBSETREF,
  171. LOC_CREFERENCE,LOC_REFERENCE,LOC_REGISTER,LOC_CREGISTER:
  172. begin
  173. current_asmdata.getjumplabel(truelabel);
  174. current_asmdata.getjumplabel(falselabel);
  175. location_reset_jump(location,truelabel,falselabel);
  176. hlcg.a_cmp_const_loc_label(current_asmdata.CurrAsmList,left.resultdef,OC_EQ,0,left.location,location.falselabel);
  177. hlcg.a_jmp_always(current_asmdata.CurrAsmList,location.truelabel);
  178. end;
  179. LOC_JUMP :
  180. begin
  181. location:=left.location;
  182. end;
  183. else
  184. internalerror(10062);
  185. end;
  186. end;
  187. procedure tllvmtypeconvnode.second_nothing;
  188. var
  189. hreg: tregister;
  190. begin
  191. if left.resultdef<>resultdef then
  192. begin
  193. { handle sometype(voidptr^) }
  194. if not is_void(left.resultdef) and
  195. (left.resultdef.typ<>formaldef) and
  196. (left.resultdef.size<>resultdef.size) then
  197. internalerror(2014012216);
  198. hlcg.location_force_mem(current_asmdata.CurrAsmList,left.location,left.resultdef);
  199. hreg:=hlcg.getaddressregister(current_asmdata.CurrAsmList,cpointerdef.getreusable(resultdef));
  200. hlcg.a_loadaddr_ref_reg(current_asmdata.CurrAsmList,left.resultdef,cpointerdef.getreusable(resultdef),left.location.reference,hreg);
  201. location_reset_ref(location,left.location.loc,left.location.size,left.location.reference.alignment);
  202. reference_reset_base(location.reference,hreg,0,location.reference.alignment);
  203. end
  204. else
  205. location_copy(location,left.location);
  206. end;
  207. begin
  208. ctypeconvnode:=tllvmtypeconvnode;
  209. end.