n386cnv.pas 54 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550
  1. {
  2. $Id$
  3. Copyright (c) 1998-2000 by Florian Klaempfl
  4. Generate i386 assembler for type converting nodes
  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 n386cnv;
  19. {$i defines.inc}
  20. interface
  21. uses
  22. node,ncnv,types;
  23. type
  24. ti386typeconvnode = class(ttypeconvnode)
  25. procedure second_int_to_int;virtual;
  26. procedure second_string_to_string;virtual;
  27. procedure second_cstring_to_pchar;virtual;
  28. procedure second_string_to_chararray;virtual;
  29. procedure second_array_to_pointer;virtual;
  30. procedure second_pointer_to_array;virtual;
  31. procedure second_chararray_to_string;virtual;
  32. procedure second_char_to_string;virtual;
  33. procedure second_int_to_real;virtual;
  34. procedure second_real_to_fix;virtual;
  35. procedure second_real_to_real;virtual;
  36. procedure second_fix_to_real;virtual;
  37. procedure second_cord_to_pointer;virtual;
  38. procedure second_int_to_fix;virtual;
  39. procedure second_proc_to_procvar;virtual;
  40. procedure second_bool_to_int;virtual;
  41. procedure second_int_to_bool;virtual;
  42. procedure second_load_smallset;virtual;
  43. procedure second_ansistring_to_pchar;virtual;
  44. procedure second_pchar_to_string;virtual;
  45. procedure second_class_to_intf;virtual;
  46. procedure second_nothing;virtual;
  47. procedure pass_2;override;
  48. procedure second_call_helper(c : tconverttype);
  49. end;
  50. ti386asnode = class(tasnode)
  51. procedure pass_2;override;
  52. end;
  53. ti386isnode = class(tisnode)
  54. procedure pass_2;override;
  55. end;
  56. implementation
  57. uses
  58. verbose,globals,systems,
  59. symconst,symdef,aasm,
  60. hcodegen,temp_gen,pass_2,
  61. ncon,ncal,
  62. cpubase,cpuasm,
  63. cgai386,tgcpu,n386util;
  64. {*****************************************************************************
  65. SecondTypeConv
  66. *****************************************************************************}
  67. procedure ti386typeconvnode.second_int_to_int;
  68. var
  69. op : tasmop;
  70. opsize : topsize;
  71. hregister,
  72. hregister2 : tregister;
  73. l : pasmlabel;
  74. begin
  75. { insert range check if not explicit conversion }
  76. if not(nf_explizit in flags) then
  77. emitrangecheck(left,resulttype);
  78. { is the result size smaller ? }
  79. if resulttype^.size<left.resulttype^.size then
  80. begin
  81. { only need to set the new size of a register }
  82. if (left.location.loc in [LOC_REGISTER,LOC_CREGISTER]) then
  83. begin
  84. case resulttype^.size of
  85. 1 : location.register:=makereg8(left.location.register);
  86. 2 : location.register:=makereg16(left.location.register);
  87. 4 : location.register:=makereg32(left.location.register);
  88. end;
  89. { we can release the upper register }
  90. if is_64bitint(left.resulttype) then
  91. ungetregister32(left.location.registerhigh);
  92. end;
  93. end
  94. { is the result size bigger ? }
  95. else if resulttype^.size>left.resulttype^.size then
  96. begin
  97. { remove reference }
  98. if not(left.location.loc in [LOC_REGISTER,LOC_CREGISTER]) then
  99. begin
  100. del_reference(left.location.reference);
  101. { we can do this here as we need no temp inside }
  102. ungetiftemp(left.location.reference);
  103. end;
  104. { get op and opsize, handle separate for constants, because
  105. movz doesn't support constant values }
  106. if (left.location.loc=LOC_MEM) and (left.location.reference.is_immediate) then
  107. begin
  108. if is_64bitint(resulttype) then
  109. opsize:=S_L
  110. else
  111. opsize:=def_opsize(resulttype);
  112. op:=A_MOV;
  113. end
  114. else
  115. begin
  116. opsize:=def2def_opsize(left.resulttype,resulttype);
  117. if opsize in [S_B,S_W,S_L] then
  118. op:=A_MOV
  119. else
  120. if is_signed(left.resulttype) then
  121. op:=A_MOVSX
  122. else
  123. op:=A_MOVZX;
  124. end;
  125. { load the register we need }
  126. if left.location.loc<>LOC_REGISTER then
  127. hregister:=getregister32
  128. else
  129. hregister:=left.location.register;
  130. { set the correct register size and location }
  131. clear_location(location);
  132. location.loc:=LOC_REGISTER;
  133. { do we need a second register for a 64 bit type ? }
  134. if is_64bitint(resulttype) then
  135. begin
  136. hregister2:=getregister32;
  137. location.registerhigh:=hregister2;
  138. end;
  139. case resulttype^.size of
  140. 1:
  141. location.register:=makereg8(hregister);
  142. 2:
  143. location.register:=makereg16(hregister);
  144. 4,8:
  145. location.register:=makereg32(hregister);
  146. end;
  147. { insert the assembler code }
  148. if left.location.loc in [LOC_CREGISTER,LOC_REGISTER] then
  149. emit_reg_reg(op,opsize,left.location.register,location.register)
  150. else
  151. emit_ref_reg(op,opsize,
  152. newreference(left.location.reference),location.register);
  153. { do we need a sign extension for int64? }
  154. if is_64bitint(resulttype) then
  155. { special case for constants (JM) }
  156. if is_constintnode(left) then
  157. begin
  158. if tordconstnode(left).value >= 0 then
  159. emit_reg_reg(A_XOR,S_L,
  160. hregister2,hregister2)
  161. else
  162. emit_const_reg(A_MOV,S_L,longint($ffffffff),hregister2);
  163. end
  164. else
  165. begin
  166. emit_reg_reg(A_XOR,S_L,
  167. hregister2,hregister2);
  168. if (porddef(resulttype)^.typ=s64bit) and
  169. is_signed(left.resulttype) then
  170. begin
  171. getlabel(l);
  172. emit_const_reg(A_TEST,S_L,longint($80000000),makereg32(hregister));
  173. emitjmp(C_Z,l);
  174. emit_reg(A_NOT,S_L,
  175. hregister2);
  176. emitlab(l);
  177. end;
  178. end;
  179. end;
  180. end;
  181. procedure ti386typeconvnode.second_string_to_string;
  182. var
  183. pushed : tpushed;
  184. regs_to_push: byte;
  185. begin
  186. { does anybody know a better solution than this big case statement ? }
  187. { ok, a proc table would do the job }
  188. case pstringdef(resulttype)^.string_typ of
  189. st_shortstring:
  190. case pstringdef(left.resulttype)^.string_typ of
  191. st_shortstring:
  192. begin
  193. gettempofsizereference(resulttype^.size,location.reference);
  194. copyshortstring(location.reference,left.location.reference,
  195. pstringdef(resulttype)^.len,false,true);
  196. { done by copyshortstring now (JM) }
  197. { del_reference(left.location.reference); }
  198. ungetiftemp(left.location.reference);
  199. end;
  200. st_longstring:
  201. begin
  202. {!!!!!!!}
  203. internalerror(8888);
  204. end;
  205. st_ansistring:
  206. begin
  207. gettempofsizereference(resulttype^.size,location.reference);
  208. loadansi2short(left,self);
  209. { this is done in secondtypeconv (FK)
  210. removetemps(exprasmlist,temptoremove);
  211. destroys:=true;
  212. }
  213. end;
  214. st_widestring:
  215. begin
  216. {!!!!!!!}
  217. internalerror(8888);
  218. end;
  219. end;
  220. st_longstring:
  221. case pstringdef(left.resulttype)^.string_typ of
  222. st_shortstring:
  223. begin
  224. {!!!!!!!}
  225. internalerror(8888);
  226. end;
  227. st_ansistring:
  228. begin
  229. {!!!!!!!}
  230. internalerror(8888);
  231. end;
  232. st_widestring:
  233. begin
  234. {!!!!!!!}
  235. internalerror(8888);
  236. end;
  237. end;
  238. st_ansistring:
  239. case pstringdef(left.resulttype)^.string_typ of
  240. st_shortstring:
  241. begin
  242. clear_location(location);
  243. location.loc:=LOC_REFERENCE;
  244. gettempansistringreference(location.reference);
  245. decrstringref(cansistringdef,location.reference);
  246. { We don't need the source regs anymore (JM) }
  247. regs_to_push := $ff;
  248. remove_non_regvars_from_loc(left.location,regs_to_push);
  249. pushusedregisters(pushed,regs_to_push);
  250. release_loc(left.location);
  251. emit_push_lea_loc(left.location,true);
  252. emit_push_lea_loc(location,false);
  253. saveregvars(regs_to_push);
  254. emitcall('FPC_SHORTSTR_TO_ANSISTR');
  255. maybe_loadesi;
  256. popusedregisters(pushed);
  257. end;
  258. st_longstring:
  259. begin
  260. {!!!!!!!}
  261. internalerror(8888);
  262. end;
  263. st_widestring:
  264. begin
  265. {!!!!!!!}
  266. internalerror(8888);
  267. end;
  268. end;
  269. st_widestring:
  270. case pstringdef(left.resulttype)^.string_typ of
  271. st_shortstring:
  272. begin
  273. {!!!!!!!}
  274. internalerror(8888);
  275. end;
  276. st_longstring:
  277. begin
  278. {!!!!!!!}
  279. internalerror(8888);
  280. end;
  281. st_ansistring:
  282. begin
  283. {!!!!!!!}
  284. internalerror(8888);
  285. end;
  286. st_widestring:
  287. begin
  288. {!!!!!!!}
  289. internalerror(8888);
  290. end;
  291. end;
  292. end;
  293. end;
  294. procedure ti386typeconvnode.second_cstring_to_pchar;
  295. var
  296. hr : preference;
  297. begin
  298. clear_location(location);
  299. location.loc:=LOC_REGISTER;
  300. location.register:=getregister32;
  301. case pstringdef(left.resulttype)^.string_typ of
  302. st_shortstring :
  303. begin
  304. inc(left.location.reference.offset);
  305. emit_ref_reg(A_LEA,S_L,newreference(left.location.reference),
  306. location.register);
  307. end;
  308. st_ansistring :
  309. begin
  310. if (left.nodetype=stringconstn) and
  311. (str_length(left)=0) then
  312. begin
  313. new(hr);
  314. reset_reference(hr^);
  315. hr^.symbol:=newasmsymbol('FPC_EMPTYCHAR');
  316. emit_ref_reg(A_LEA,S_L,hr,location.register);
  317. end
  318. else
  319. emit_ref_reg(A_MOV,S_L,newreference(left.location.reference),
  320. location.register);
  321. end;
  322. st_longstring:
  323. begin
  324. {!!!!!!!}
  325. internalerror(8888);
  326. end;
  327. st_widestring:
  328. begin
  329. {!!!!!!!}
  330. internalerror(8888);
  331. end;
  332. end;
  333. end;
  334. procedure ti386typeconvnode.second_string_to_chararray;
  335. var
  336. pushedregs: tpushed;
  337. //l1 : pasmlabel;
  338. //hr : preference;
  339. arrsize, strtype: longint;
  340. regstopush: byte;
  341. begin
  342. with parraydef(resulttype)^ do
  343. begin
  344. if highrange<lowrange then
  345. internalerror(75432653);
  346. arrsize := highrange-lowrange+1;
  347. end;
  348. if (left.nodetype = stringconstn) and
  349. { left.length+1 since there's always a terminating #0 character (JM) }
  350. (tstringconstnode(left).len+1 >= arrsize) and
  351. (pstringdef(left.resulttype)^.string_typ=st_shortstring) then
  352. begin
  353. inc(location.reference.offset);
  354. exit;
  355. end;
  356. clear_location(location);
  357. location.loc := LOC_REFERENCE;
  358. gettempofsizereference(arrsize,location.reference);
  359. regstopush := $ff;
  360. remove_non_regvars_from_loc(left.location,regstopush);
  361. pushusedregisters(pushedregs,regstopush);
  362. emit_push_lea_loc(location,false);
  363. case pstringdef(left.resulttype)^.string_typ of
  364. st_shortstring :
  365. begin
  366. { 0 means shortstring }
  367. strtype := 0;
  368. del_reference(left.location.reference);
  369. emit_push_lea_loc(left.location,true);
  370. ungetiftemp(left.location.reference);
  371. end;
  372. st_ansistring :
  373. begin
  374. { 1 means ansistring }
  375. strtype := 1;
  376. case left.location.loc of
  377. LOC_CREGISTER,LOC_REGISTER:
  378. begin
  379. ungetregister(left.location.register);
  380. emit_push_loc(left.location);
  381. end;
  382. LOC_MEM,LOC_REFERENCE:
  383. begin
  384. del_reference(left.location.reference);
  385. emit_push_loc(left.location);
  386. ungetiftemp(left.location.reference);
  387. end;
  388. end;
  389. end;
  390. st_longstring:
  391. begin
  392. {!!!!!!!}
  393. { 2 means longstring, but still needs support in FPC_STR_TO_CHARARRAY,
  394. which is in i386.inc and/or generic.inc (JM) }
  395. strtype := 2;
  396. internalerror(8888);
  397. end;
  398. st_widestring:
  399. begin
  400. {!!!!!!!}
  401. { 3 means widestring, but still needs support in FPC_STR_TO_CHARARRAY,
  402. which is in i386.inc and/or generic.inc (JM) }
  403. strtype := 3;
  404. internalerror(8888);
  405. end;
  406. end;
  407. push_int(arrsize);
  408. push_int(strtype);
  409. saveregvars(regstopush);
  410. emitcall('FPC_STR_TO_CHARARRAY');
  411. popusedregisters(pushedregs);
  412. end;
  413. procedure ti386typeconvnode.second_array_to_pointer;
  414. begin
  415. del_reference(left.location.reference);
  416. clear_location(location);
  417. location.loc:=LOC_REGISTER;
  418. location.register:=getregister32;
  419. emit_ref_reg(A_LEA,S_L,newreference(left.location.reference),
  420. location.register);
  421. end;
  422. procedure ti386typeconvnode.second_pointer_to_array;
  423. begin
  424. clear_location(location);
  425. location.loc:=LOC_REFERENCE;
  426. reset_reference(location.reference);
  427. case left.location.loc of
  428. LOC_REGISTER :
  429. location.reference.base:=left.location.register;
  430. LOC_CREGISTER :
  431. begin
  432. location.reference.base:=getregister32;
  433. emit_reg_reg(A_MOV,S_L,left.location.register,location.reference.base);
  434. end
  435. else
  436. begin
  437. del_reference(left.location.reference);
  438. location.reference.base:=getregister32;
  439. emit_ref_reg(A_MOV,S_L,newreference(left.location.reference),
  440. location.reference.base);
  441. end;
  442. end;
  443. end;
  444. { generates the code for the type conversion from an array of char }
  445. { to a string }
  446. procedure ti386typeconvnode.second_chararray_to_string;
  447. var
  448. pushed : tpushed;
  449. regstopush: byte;
  450. l : longint;
  451. begin
  452. { calc the length of the array }
  453. l:=parraydef(left.resulttype)^.highrange-parraydef(left.resulttype)^.lowrange+1;
  454. { this is a type conversion which copies the data, so we can't }
  455. { return a reference }
  456. clear_location(location);
  457. location.loc:=LOC_MEM;
  458. case pstringdef(resulttype)^.string_typ of
  459. st_shortstring :
  460. begin
  461. if l>255 then
  462. begin
  463. CGMessage(type_e_mismatch);
  464. l:=255;
  465. end;
  466. gettempofsizereference(resulttype^.size,location.reference);
  467. { we've also to release the registers ... }
  468. { Yes, but before pushusedregisters since that one resets unused! }
  469. { This caused web bug 1073 (JM) }
  470. regstopush := $ff;
  471. remove_non_regvars_from_loc(left.location,regstopush);
  472. pushusedregisters(pushed,regstopush);
  473. if l>=resulttype^.size then
  474. push_int(resulttype^.size-1)
  475. else
  476. push_int(l);
  477. { ... here only the temp. location is released }
  478. emit_push_lea_loc(left.location,true);
  479. del_reference(left.location.reference);
  480. emitpushreferenceaddr(location.reference);
  481. saveregvars(regstopush);
  482. emitcall('FPC_CHARARRAY_TO_SHORTSTR');
  483. maybe_loadesi;
  484. popusedregisters(pushed);
  485. end;
  486. st_ansistring :
  487. begin
  488. gettempansistringreference(location.reference);
  489. decrstringref(cansistringdef,location.reference);
  490. regstopush := $ff;
  491. remove_non_regvars_from_loc(left.location,regstopush);
  492. pushusedregisters(pushed,regstopush);
  493. push_int(l);
  494. emitpushreferenceaddr(left.location.reference);
  495. release_loc(left.location);
  496. emitpushreferenceaddr(location.reference);
  497. saveregvars(regstopush);
  498. emitcall('FPC_CHARARRAY_TO_ANSISTR');
  499. popusedregisters(pushed);
  500. maybe_loadesi;
  501. end;
  502. st_longstring:
  503. begin
  504. {!!!!!!!}
  505. internalerror(8888);
  506. end;
  507. st_widestring:
  508. begin
  509. {!!!!!!!}
  510. internalerror(8888);
  511. end;
  512. end;
  513. end;
  514. procedure ti386typeconvnode.second_char_to_string;
  515. var
  516. pushed : tpushed;
  517. begin
  518. clear_location(location);
  519. location.loc:=LOC_MEM;
  520. case pstringdef(resulttype)^.string_typ of
  521. st_shortstring :
  522. begin
  523. gettempofsizereference(256,location.reference);
  524. loadshortstring(left,self);
  525. end;
  526. st_ansistring :
  527. begin
  528. gettempansistringreference(location.reference);
  529. decrstringref(cansistringdef,location.reference);
  530. release_loc(left.location);
  531. pushusedregisters(pushed,$ff);
  532. emit_pushw_loc(left.location);
  533. emitpushreferenceaddr(location.reference);
  534. saveregvars($ff);
  535. emitcall('FPC_CHAR_TO_ANSISTR');
  536. popusedregisters(pushed);
  537. maybe_loadesi;
  538. end;
  539. else
  540. internalerror(4179);
  541. end;
  542. end;
  543. procedure ti386typeconvnode.second_int_to_real;
  544. var
  545. r : preference;
  546. hregister : tregister;
  547. l1,l2 : pasmlabel;
  548. begin
  549. { for u32bit a solution is to push $0 and to load a comp }
  550. { does this first, it destroys maybe EDI }
  551. hregister:=R_EDI;
  552. if porddef(left.resulttype)^.typ=u32bit then
  553. push_int(0);
  554. if (left.location.loc=LOC_REGISTER) or
  555. (left.location.loc=LOC_CREGISTER) then
  556. begin
  557. {$ifndef noAllocEdi}
  558. if not (porddef(left.resulttype)^.typ in [u32bit,s32bit,u64bit,s64bit]) then
  559. getexplicitregister32(R_EDI);
  560. {$endif noAllocEdi}
  561. case porddef(left.resulttype)^.typ of
  562. s8bit : emit_reg_reg(A_MOVSX,S_BL,left.location.register,R_EDI);
  563. u8bit : emit_reg_reg(A_MOVZX,S_BL,left.location.register,R_EDI);
  564. s16bit : emit_reg_reg(A_MOVSX,S_WL,left.location.register,R_EDI);
  565. u16bit : emit_reg_reg(A_MOVZX,S_WL,left.location.register,R_EDI);
  566. u32bit,s32bit:
  567. hregister:=left.location.register;
  568. u64bit,s64bit:
  569. begin
  570. emit_reg(A_PUSH,S_L,left.location.registerhigh);
  571. hregister:=left.location.registerlow;
  572. end;
  573. end;
  574. ungetregister(left.location.register);
  575. end
  576. else
  577. begin
  578. r:=newreference(left.location.reference);
  579. {$ifndef noAllocEdi}
  580. getexplicitregister32(R_EDI);
  581. {$endif noAllocEdi}
  582. case porddef(left.resulttype)^.typ of
  583. s8bit:
  584. emit_ref_reg(A_MOVSX,S_BL,r,R_EDI);
  585. u8bit:
  586. emit_ref_reg(A_MOVZX,S_BL,r,R_EDI);
  587. s16bit:
  588. emit_ref_reg(A_MOVSX,S_WL,r,R_EDI);
  589. u16bit:
  590. emit_ref_reg(A_MOVZX,S_WL,r,R_EDI);
  591. u32bit,s32bit:
  592. emit_ref_reg(A_MOV,S_L,r,R_EDI);
  593. u64bit,s64bit:
  594. begin
  595. inc(r^.offset,4);
  596. emit_ref_reg(A_MOV,S_L,r,R_EDI);
  597. emit_reg(A_PUSH,S_L,R_EDI);
  598. r:=newreference(left.location.reference);
  599. emit_ref_reg(A_MOV,S_L,r,R_EDI);
  600. end;
  601. end;
  602. del_reference(left.location.reference);
  603. ungetiftemp(left.location.reference);
  604. end;
  605. { for 64 bit integers, the high dword is already pushed }
  606. emit_reg(A_PUSH,S_L,hregister);
  607. {$ifndef noAllocEdi}
  608. if hregister = R_EDI then
  609. ungetregister32(R_EDI);
  610. {$endif noAllocEdi}
  611. r:=new_reference(R_ESP,0);
  612. case porddef(left.resulttype)^.typ of
  613. u32bit:
  614. begin
  615. emit_ref(A_FILD,S_IQ,r);
  616. emit_const_reg(A_ADD,S_L,8,R_ESP);
  617. end;
  618. s64bit:
  619. begin
  620. emit_ref(A_FILD,S_IQ,r);
  621. emit_const_reg(A_ADD,S_L,8,R_ESP);
  622. end;
  623. u64bit:
  624. begin
  625. { unsigned 64 bit ints are harder to handle: }
  626. { we load bits 0..62 and then check bit 63: }
  627. { if it is 1 then we add $80000000 000000000 }
  628. { as double }
  629. inc(r^.offset,4);
  630. {$ifndef noAllocEdi}
  631. getexplicitregister32(R_EDI);
  632. {$endif noAllocEdi}
  633. emit_ref_reg(A_MOV,S_L,r,R_EDI);
  634. r:=new_reference(R_ESP,4);
  635. emit_const_ref(A_AND,S_L,$7fffffff,r);
  636. emit_const_reg(A_TEST,S_L,longint($80000000),R_EDI);
  637. {$ifndef noAllocEdi}
  638. ungetregister32(R_EDI);
  639. {$endif noAllocEdi}
  640. r:=new_reference(R_ESP,0);
  641. emit_ref(A_FILD,S_IQ,r);
  642. getdatalabel(l1);
  643. getlabel(l2);
  644. emitjmp(C_Z,l2);
  645. Consts.concat(Tai_label.Create(l1));
  646. { I got this constant from a test progtram (FK) }
  647. Consts.concat(Tai_const.Create_32bit(0));
  648. Consts.concat(Tai_const.Create_32bit(1138753536));
  649. r:=new_reference(R_NO,0);
  650. r^.symbol:=l1;
  651. emit_ref(A_FADD,S_FL,r);
  652. emitlab(l2);
  653. emit_const_reg(A_ADD,S_L,8,R_ESP);
  654. end
  655. else
  656. begin
  657. emit_ref(A_FILD,S_IL,r);
  658. {$ifndef noAllocEdi}
  659. getexplicitregister32(R_EDI);
  660. {$endif noAllocEdi}
  661. emit_reg(A_POP,S_L,R_EDI);
  662. {$ifndef noAllocEdi}
  663. ungetregister32(R_EDI);
  664. {$endif noAllocEdi}
  665. end;
  666. end;
  667. inc(fpuvaroffset);
  668. clear_location(location);
  669. location.loc:=LOC_FPU;
  670. end;
  671. procedure ti386typeconvnode.second_real_to_fix;
  672. var
  673. rreg : tregister;
  674. ref : treference;
  675. begin
  676. { real must be on fpu stack }
  677. if (left.location.loc<>LOC_FPU) then
  678. emit_ref(A_FLD,S_FL,newreference(left.location.reference));
  679. push_int($1f3f);
  680. push_int(65536);
  681. reset_reference(ref);
  682. ref.base:=R_ESP;
  683. emit_ref(A_FIMUL,S_IL,newreference(ref));
  684. ref.offset:=4;
  685. emit_ref(A_FSTCW,S_NO,newreference(ref));
  686. ref.offset:=6;
  687. emit_ref(A_FLDCW,S_NO,newreference(ref));
  688. ref.offset:=0;
  689. emit_ref(A_FISTP,S_IL,newreference(ref));
  690. ref.offset:=4;
  691. emit_ref(A_FLDCW,S_NO,newreference(ref));
  692. rreg:=getregister32;
  693. emit_reg(A_POP,S_L,rreg);
  694. { better than an add on all processors }
  695. {$ifndef noAllocEdi}
  696. getexplicitregister32(R_EDI);
  697. {$endif noAllocEdi}
  698. emit_reg(A_POP,S_L,R_EDI);
  699. {$ifndef noAllocEdi}
  700. ungetregister32(R_EDI);
  701. {$endif noAllocEdi}
  702. clear_location(location);
  703. location.loc:=LOC_REGISTER;
  704. location.register:=rreg;
  705. inc(fpuvaroffset);
  706. end;
  707. procedure ti386typeconvnode.second_real_to_real;
  708. begin
  709. case left.location.loc of
  710. LOC_FPU : ;
  711. LOC_CFPUREGISTER:
  712. begin
  713. location:=left.location;
  714. exit;
  715. end;
  716. LOC_MEM,
  717. LOC_REFERENCE:
  718. begin
  719. floatload(pfloatdef(left.resulttype)^.typ,
  720. left.location.reference);
  721. { we have to free the reference }
  722. del_reference(left.location.reference);
  723. end;
  724. end;
  725. clear_location(location);
  726. location.loc:=LOC_FPU;
  727. end;
  728. procedure ti386typeconvnode.second_fix_to_real;
  729. var
  730. popeax,popebx,popecx,popedx : boolean;
  731. startreg : tregister;
  732. hl : pasmlabel;
  733. r : treference;
  734. begin
  735. if (left.location.loc=LOC_REGISTER) or
  736. (left.location.loc=LOC_CREGISTER) then
  737. begin
  738. startreg:=left.location.register;
  739. ungetregister(startreg);
  740. popeax:=(startreg<>R_EAX) and not (R_EAX in unused);
  741. if popeax then
  742. emit_reg(A_PUSH,S_L,R_EAX);
  743. { mov eax,eax is removed by emit_reg_reg }
  744. emit_reg_reg(A_MOV,S_L,startreg,R_EAX);
  745. end
  746. else
  747. begin
  748. emit_ref_reg(A_MOV,S_L,newreference(
  749. left.location.reference),R_EAX);
  750. del_reference(left.location.reference);
  751. startreg:=R_NO;
  752. end;
  753. popebx:=(startreg<>R_EBX) and not (R_EBX in unused);
  754. if popebx then
  755. emit_reg(A_PUSH,S_L,R_EBX);
  756. popecx:=(startreg<>R_ECX) and not (R_ECX in unused);
  757. if popecx then
  758. emit_reg(A_PUSH,S_L,R_ECX);
  759. popedx:=(startreg<>R_EDX) and not (R_EDX in unused);
  760. if popedx then
  761. emit_reg(A_PUSH,S_L,R_EDX);
  762. emit_none(A_CDQ,S_NO);
  763. emit_reg_reg(A_XOR,S_L,R_EDX,R_EAX);
  764. emit_reg_reg(A_MOV,S_L,R_EAX,R_EBX);
  765. emit_reg_reg(A_SUB,S_L,R_EDX,R_EAX);
  766. getlabel(hl);
  767. emitjmp(C_Z,hl);
  768. emit_const_reg(A_RCL,S_L,1,R_EBX);
  769. emit_reg_reg(A_BSR,S_L,R_EAX,R_EDX);
  770. emit_const_reg(A_MOV,S_B,32,R_CL);
  771. emit_reg_reg(A_SUB,S_B,R_DL,R_CL);
  772. emit_reg_reg(A_SHL,S_L,R_CL,R_EAX);
  773. emit_const_reg(A_ADD,S_W,1007,R_DX);
  774. emit_const_reg(A_SHL,S_W,5,R_DX);
  775. emit_const_reg_reg(A_SHLD,S_W,11,R_DX,R_BX);
  776. emit_const_reg_reg(A_SHLD,S_L,20,R_EAX,R_EBX);
  777. emit_const_reg(A_SHL,S_L,20,R_EAX);
  778. emitlab(hl);
  779. { better than an add on all processors }
  780. emit_reg(A_PUSH,S_L,R_EBX);
  781. emit_reg(A_PUSH,S_L,R_EAX);
  782. reset_reference(r);
  783. r.base:=R_ESP;
  784. emit_ref(A_FLD,S_FL,newreference(r));
  785. emit_const_reg(A_ADD,S_L,8,R_ESP);
  786. if popedx then
  787. emit_reg(A_POP,S_L,R_EDX);
  788. if popecx then
  789. emit_reg(A_POP,S_L,R_ECX);
  790. if popebx then
  791. emit_reg(A_POP,S_L,R_EBX);
  792. if popeax then
  793. emit_reg(A_POP,S_L,R_EAX);
  794. clear_location(location);
  795. location.loc:=LOC_FPU;
  796. end;
  797. procedure ti386typeconvnode.second_cord_to_pointer;
  798. begin
  799. { this can't happend, because constants are already processed in
  800. pass 1 }
  801. internalerror(47423985);
  802. end;
  803. procedure ti386typeconvnode.second_int_to_fix;
  804. var
  805. hregister : tregister;
  806. begin
  807. if (left.location.loc=LOC_REGISTER) then
  808. hregister:=left.location.register
  809. else if (left.location.loc=LOC_CREGISTER) then
  810. hregister:=getregister32
  811. else
  812. begin
  813. del_reference(left.location.reference);
  814. hregister:=getregister32;
  815. case porddef(left.resulttype)^.typ of
  816. s8bit : emit_ref_reg(A_MOVSX,S_BL,newreference(left.location.reference),
  817. hregister);
  818. u8bit : emit_ref_reg(A_MOVZX,S_BL,newreference(left.location.reference),
  819. hregister);
  820. s16bit : emit_ref_reg(A_MOVSX,S_WL,newreference(left.location.reference),
  821. hregister);
  822. u16bit : emit_ref_reg(A_MOVZX,S_WL,newreference(left.location.reference),
  823. hregister);
  824. u32bit,s32bit : emit_ref_reg(A_MOV,S_L,newreference(left.location.reference),
  825. hregister);
  826. {!!!! u32bit }
  827. end;
  828. end;
  829. emit_const_reg(A_SHL,S_L,16,hregister);
  830. clear_location(location);
  831. location.loc:=LOC_REGISTER;
  832. location.register:=hregister;
  833. end;
  834. procedure ti386typeconvnode.second_proc_to_procvar;
  835. begin
  836. { method pointer ? }
  837. if assigned(tcallnode(left).left) then
  838. begin
  839. set_location(location,left.location);
  840. end
  841. else
  842. begin
  843. clear_location(location);
  844. location.loc:=LOC_REGISTER;
  845. location.register:=getregister32;
  846. del_reference(left.location.reference);
  847. emit_ref_reg(A_LEA,S_L,
  848. newreference(left.location.reference),location.register);
  849. end;
  850. end;
  851. procedure ti386typeconvnode.second_bool_to_int;
  852. var
  853. oldtruelabel,oldfalselabel,hlabel : pasmlabel;
  854. hregister : tregister;
  855. newsize,
  856. opsize : topsize;
  857. op : tasmop;
  858. begin
  859. oldtruelabel:=truelabel;
  860. oldfalselabel:=falselabel;
  861. getlabel(truelabel);
  862. getlabel(falselabel);
  863. secondpass(left);
  864. { byte(boolean) or word(wordbool) or longint(longbool) must
  865. be accepted for var parameters }
  866. if (nf_explizit in flags) and
  867. (left.resulttype^.size=resulttype^.size) and
  868. (left.location.loc in [LOC_REFERENCE,LOC_MEM,LOC_CREGISTER]) then
  869. begin
  870. set_location(location,left.location);
  871. truelabel:=oldtruelabel;
  872. falselabel:=oldfalselabel;
  873. exit;
  874. end;
  875. clear_location(location);
  876. location.loc:=LOC_REGISTER;
  877. del_reference(left.location.reference);
  878. case left.resulttype^.size of
  879. 1 : begin
  880. case resulttype^.size of
  881. 1 : opsize:=S_B;
  882. 2 : opsize:=S_BW;
  883. 4 : opsize:=S_BL;
  884. end;
  885. end;
  886. 2 : begin
  887. case resulttype^.size of
  888. 1 : begin
  889. if left.location.loc in [LOC_REGISTER,LOC_CREGISTER] then
  890. left.location.register:=reg16toreg8(left.location.register);
  891. opsize:=S_B;
  892. end;
  893. 2 : opsize:=S_W;
  894. 4 : opsize:=S_WL;
  895. end;
  896. end;
  897. 4 : begin
  898. case resulttype^.size of
  899. 1 : begin
  900. if left.location.loc in [LOC_REGISTER,LOC_CREGISTER] then
  901. left.location.register:=reg32toreg8(left.location.register);
  902. opsize:=S_B;
  903. end;
  904. 2 : begin
  905. if left.location.loc in [LOC_REGISTER,LOC_CREGISTER] then
  906. left.location.register:=reg32toreg16(left.location.register);
  907. opsize:=S_W;
  908. end;
  909. 4 : opsize:=S_L;
  910. end;
  911. end;
  912. end;
  913. if opsize in [S_B,S_W,S_L] then
  914. op:=A_MOV
  915. else
  916. if is_signed(resulttype) then
  917. op:=A_MOVSX
  918. else
  919. op:=A_MOVZX;
  920. hregister:=getregister32;
  921. case resulttype^.size of
  922. 1 : begin
  923. location.register:=reg32toreg8(hregister);
  924. newsize:=S_B;
  925. end;
  926. 2 : begin
  927. location.register:=reg32toreg16(hregister);
  928. newsize:=S_W;
  929. end;
  930. 4 : begin
  931. location.register:=hregister;
  932. newsize:=S_L;
  933. end;
  934. else
  935. internalerror(10060);
  936. end;
  937. case left.location.loc of
  938. LOC_MEM,
  939. LOC_REFERENCE : emit_ref_reg(op,opsize,
  940. newreference(left.location.reference),location.register);
  941. LOC_REGISTER,
  942. LOC_CREGISTER : begin
  943. { remove things like movb %al,%al }
  944. if left.location.register<>location.register then
  945. emit_reg_reg(op,opsize,
  946. left.location.register,location.register);
  947. end;
  948. LOC_FLAGS : begin
  949. emit_flag2reg(left.location.resflags,location.register);
  950. end;
  951. LOC_JUMP : begin
  952. getlabel(hlabel);
  953. emitlab(truelabel);
  954. emit_const_reg(A_MOV,newsize,1,location.register);
  955. emitjmp(C_None,hlabel);
  956. emitlab(falselabel);
  957. emit_reg_reg(A_XOR,newsize,location.register,
  958. location.register);
  959. emitlab(hlabel);
  960. end;
  961. else
  962. internalerror(10061);
  963. end;
  964. truelabel:=oldtruelabel;
  965. falselabel:=oldfalselabel;
  966. end;
  967. procedure ti386typeconvnode.second_int_to_bool;
  968. var
  969. hregister : tregister;
  970. resflags : tresflags;
  971. opsize : topsize;
  972. begin
  973. clear_location(location);
  974. { byte(boolean) or word(wordbool) or longint(longbool) must
  975. be accepted for var parameters }
  976. if (nf_explizit in flags) and
  977. (left.resulttype^.size=resulttype^.size) and
  978. (left.location.loc in [LOC_REFERENCE,LOC_MEM,LOC_CREGISTER]) then
  979. begin
  980. set_location(location,left.location);
  981. exit;
  982. end;
  983. location.loc:=LOC_REGISTER;
  984. del_reference(left.location.reference);
  985. opsize:=def_opsize(left.resulttype);
  986. case left.location.loc of
  987. LOC_MEM,LOC_REFERENCE :
  988. begin
  989. hregister:=def_getreg(left.resulttype);
  990. emit_ref_reg(A_MOV,opsize,
  991. newreference(left.location.reference),hregister);
  992. emit_reg_reg(A_OR,opsize,hregister,hregister);
  993. resflags:=F_NE;
  994. end;
  995. LOC_FLAGS :
  996. begin
  997. hregister:=getregister32;
  998. resflags:=left.location.resflags;
  999. end;
  1000. LOC_REGISTER,LOC_CREGISTER :
  1001. begin
  1002. hregister:=left.location.register;
  1003. emit_reg_reg(A_OR,opsize,hregister,hregister);
  1004. resflags:=F_NE;
  1005. end;
  1006. else
  1007. internalerror(10062);
  1008. end;
  1009. case resulttype^.size of
  1010. 1 : location.register:=makereg8(hregister);
  1011. 2 : location.register:=makereg16(hregister);
  1012. 4 : location.register:=makereg32(hregister);
  1013. else
  1014. internalerror(10064);
  1015. end;
  1016. emit_flag2reg(resflags,location.register);
  1017. end;
  1018. procedure ti386typeconvnode.second_load_smallset;
  1019. var
  1020. href : treference;
  1021. pushedregs : tpushed;
  1022. begin
  1023. href.symbol:=nil;
  1024. pushusedregisters(pushedregs,$ff);
  1025. gettempofsizereference(32,href);
  1026. emit_push_mem_size(left.location.reference,4);
  1027. emitpushreferenceaddr(href);
  1028. saveregvars($ff);
  1029. emitcall('FPC_SET_LOAD_SMALL');
  1030. maybe_loadesi;
  1031. popusedregisters(pushedregs);
  1032. clear_location(location);
  1033. location.loc:=LOC_MEM;
  1034. location.reference:=href;
  1035. end;
  1036. procedure ti386typeconvnode.second_ansistring_to_pchar;
  1037. var
  1038. l1 : pasmlabel;
  1039. hr : preference;
  1040. begin
  1041. clear_location(location);
  1042. location.loc:=LOC_REGISTER;
  1043. getlabel(l1);
  1044. case left.location.loc of
  1045. LOC_CREGISTER,LOC_REGISTER:
  1046. location.register:=left.location.register;
  1047. LOC_MEM,LOC_REFERENCE:
  1048. begin
  1049. location.register:=getregister32;
  1050. emit_ref_reg(A_MOV,S_L,newreference(left.location.reference),
  1051. location.register);
  1052. del_reference(left.location.reference);
  1053. end;
  1054. end;
  1055. emit_const_reg(A_CMP,S_L,0,location.register);
  1056. emitjmp(C_NZ,l1);
  1057. new(hr);
  1058. reset_reference(hr^);
  1059. hr^.symbol:=newasmsymbol('FPC_EMPTYCHAR');
  1060. emit_ref_reg(A_LEA,S_L,hr,location.register);
  1061. emitlab(l1);
  1062. end;
  1063. procedure ti386typeconvnode.second_pchar_to_string;
  1064. var
  1065. pushed : tpushed;
  1066. regs_to_push: byte;
  1067. begin
  1068. case pstringdef(resulttype)^.string_typ of
  1069. st_shortstring:
  1070. begin
  1071. location.loc:=LOC_REFERENCE;
  1072. gettempofsizereference(resulttype^.size,location.reference);
  1073. pushusedregisters(pushed,$ff);
  1074. case left.location.loc of
  1075. LOC_REGISTER,LOC_CREGISTER:
  1076. begin
  1077. emit_reg(A_PUSH,S_L,left.location.register);
  1078. ungetregister32(left.location.register);
  1079. end;
  1080. LOC_REFERENCE,LOC_MEM:
  1081. begin
  1082. { Now release the registers (see cgai386.pas: }
  1083. { loadansistring for more info on the order) (JM) }
  1084. del_reference(left.location.reference);
  1085. emit_push_mem(left.location.reference);
  1086. end;
  1087. end;
  1088. emitpushreferenceaddr(location.reference);
  1089. saveregvars($ff);
  1090. emitcall('FPC_PCHAR_TO_SHORTSTR');
  1091. maybe_loadesi;
  1092. popusedregisters(pushed);
  1093. end;
  1094. st_ansistring:
  1095. begin
  1096. location.loc:=LOC_REFERENCE;
  1097. gettempansistringreference(location.reference);
  1098. decrstringref(cansistringdef,location.reference);
  1099. { Find out which regs have to be pushed (JM) }
  1100. regs_to_push := $ff;
  1101. remove_non_regvars_from_loc(left.location,regs_to_push);
  1102. pushusedregisters(pushed,regs_to_push);
  1103. case left.location.loc of
  1104. LOC_REFERENCE,LOC_MEM:
  1105. begin
  1106. { Now release the registers (see cgai386.pas: }
  1107. { loadansistring for more info on the order) (JM) }
  1108. del_reference(left.location.reference);
  1109. emit_push_mem(left.location.reference);
  1110. end;
  1111. LOC_REGISTER,LOC_CREGISTER:
  1112. begin
  1113. { Now release the registers (see cgai386.pas: }
  1114. { loadansistring for more info on the order) (JM) }
  1115. emit_reg(A_PUSH,S_L,left.location.register);
  1116. ungetregister32(left.location.register);
  1117. end;
  1118. end;
  1119. emitpushreferenceaddr(location.reference);
  1120. saveregvars(regs_to_push);
  1121. emitcall('FPC_PCHAR_TO_ANSISTR');
  1122. maybe_loadesi;
  1123. popusedregisters(pushed);
  1124. end;
  1125. else
  1126. begin
  1127. internalerror(12121);
  1128. end;
  1129. end;
  1130. end;
  1131. procedure ti386typeconvnode.second_class_to_intf;
  1132. var
  1133. hreg : tregister;
  1134. l1 : pasmlabel;
  1135. begin
  1136. case left.location.loc of
  1137. LOC_MEM,
  1138. LOC_REFERENCE:
  1139. begin
  1140. del_reference(left.location.reference);
  1141. hreg:=getregister32;
  1142. exprasmList.concat(Taicpu.Op_ref_reg(
  1143. A_MOV,S_L,newreference(left.location.reference),hreg));
  1144. end;
  1145. LOC_CREGISTER:
  1146. begin
  1147. hreg:=getregister32;
  1148. exprasmList.concat(Taicpu.Op_reg_reg(
  1149. A_MOV,S_L,left.location.register,hreg));
  1150. end;
  1151. LOC_REGISTER:
  1152. hreg:=left.location.register;
  1153. else internalerror(121120001);
  1154. end;
  1155. emit_reg_reg(A_TEST,S_L,hreg,hreg);
  1156. getlabel(l1);
  1157. emitjmp(C_Z,l1);
  1158. emit_const_reg(A_ADD,S_L,pobjectdef(left.resulttype)^.implementedinterfaces^.ioffsets(
  1159. pobjectdef(left.resulttype)^.implementedinterfaces^.searchintf(resulttype))^,hreg);
  1160. emitlab(l1);
  1161. location.loc:=LOC_REGISTER;
  1162. location.register:=hreg;
  1163. end;
  1164. procedure ti386typeconvnode.second_nothing;
  1165. begin
  1166. end;
  1167. {****************************************************************************
  1168. TI386TYPECONVNODE
  1169. ****************************************************************************}
  1170. procedure ti386typeconvnode.second_call_helper(c : tconverttype);
  1171. const
  1172. secondconvert : array[tconverttype] of pointer = (
  1173. @ti386typeconvnode.second_nothing, {equal}
  1174. @ti386typeconvnode.second_nothing, {not_possible}
  1175. @ti386typeconvnode.second_string_to_string,
  1176. @ti386typeconvnode.second_char_to_string,
  1177. @ti386typeconvnode.second_pchar_to_string,
  1178. @ti386typeconvnode.second_nothing, {cchar_to_pchar}
  1179. @ti386typeconvnode.second_cstring_to_pchar,
  1180. @ti386typeconvnode.second_ansistring_to_pchar,
  1181. @ti386typeconvnode.second_string_to_chararray,
  1182. @ti386typeconvnode.second_chararray_to_string,
  1183. @ti386typeconvnode.second_array_to_pointer,
  1184. @ti386typeconvnode.second_pointer_to_array,
  1185. @ti386typeconvnode.second_int_to_int,
  1186. @ti386typeconvnode.second_int_to_bool,
  1187. @ti386typeconvnode.second_bool_to_int, { bool_to_bool }
  1188. @ti386typeconvnode.second_bool_to_int,
  1189. @ti386typeconvnode.second_real_to_real,
  1190. @ti386typeconvnode.second_int_to_real,
  1191. @ti386typeconvnode.second_int_to_fix,
  1192. @ti386typeconvnode.second_real_to_fix,
  1193. @ti386typeconvnode.second_fix_to_real,
  1194. @ti386typeconvnode.second_proc_to_procvar,
  1195. @ti386typeconvnode.second_nothing, {arrayconstructor_to_set}
  1196. @ti386typeconvnode.second_load_smallset,
  1197. @ti386typeconvnode.second_cord_to_pointer,
  1198. @ti386typeconvnode.second_nothing, { interface 2 string }
  1199. @ti386typeconvnode.second_nothing, { interface 2 guid }
  1200. @ti386typeconvnode.second_class_to_intf
  1201. );
  1202. type
  1203. tprocedureofobject = procedure of object;
  1204. var
  1205. r : packed record
  1206. proc : pointer;
  1207. obj : pointer;
  1208. end;
  1209. begin
  1210. { this is a little bit dirty but it works }
  1211. { and should be quite portable too }
  1212. r.proc:=secondconvert[c];
  1213. r.obj:=self;
  1214. tprocedureofobject(r){$ifdef FPC}();{$endif FPC}
  1215. end;
  1216. procedure ti386typeconvnode.pass_2;
  1217. {$ifdef TESTOBJEXT2}
  1218. var
  1219. r : preference;
  1220. nillabel : plabel;
  1221. {$endif TESTOBJEXT2}
  1222. begin
  1223. { this isn't good coding, I think tc_bool_2_int, shouldn't be }
  1224. { type conversion (FK) }
  1225. if not(convtype in [tc_bool_2_int,tc_bool_2_bool]) then
  1226. begin
  1227. secondpass(left);
  1228. set_location(location,left.location);
  1229. if codegenerror then
  1230. exit;
  1231. end;
  1232. second_call_helper(convtype);
  1233. {$ifdef TESTOBJEXT2}
  1234. { Check explicit conversions to objects pointers !! }
  1235. if p^.explizit and
  1236. (p^.resulttype^.deftype=pointerdef) and
  1237. (ppointerdef(p^.resulttype)^.definition^.deftype=objectdef) and not
  1238. (pobjectdef(ppointerdef(p^.resulttype)^.definition)^.isclass) and
  1239. ((pobjectdef(ppointerdef(p^.resulttype)^.definition)^.options and oo_hasvmt)<>0) and
  1240. (cs_check_range in aktlocalswitches) then
  1241. begin
  1242. new(r);
  1243. reset_reference(r^);
  1244. if p^.location.loc in [LOC_REGISTER,LOC_CREGISTER] then
  1245. r^.base:=p^.location.register
  1246. else
  1247. begin
  1248. {$ifndef noAllocEdi}
  1249. getexplicitregister32(R_EDI);
  1250. {$endif noAllocEdi}
  1251. emit_mov_loc_reg(p^.location,R_EDI);
  1252. r^.base:=R_EDI;
  1253. end;
  1254. { NIL must be accepted !! }
  1255. emit_reg_reg(A_OR,S_L,r^.base,r^.base);
  1256. {$ifndef noAllocEdi}
  1257. ungetregister32(R_EDI);
  1258. {$endif noAllocEdi}
  1259. getlabel(nillabel);
  1260. emitjmp(C_E,nillabel);
  1261. { this is one point where we need vmt_offset (PM) }
  1262. r^.offset:= pobjectdef(ppointerdef(p^.resulttype)^.definition)^.vmt_offset;
  1263. {$ifndef noAllocEdi}
  1264. getexplicitregister32(R_EDI);
  1265. {$endif noAllocEdi}
  1266. emit_ref_reg(A_MOV,S_L,r,R_EDI);
  1267. emit_sym(A_PUSH,S_L,
  1268. newasmsymbol(pobjectdef(ppointerdef(p^.resulttype)^.definition)^.vmt_mangledname));
  1269. emit_reg(A_PUSH,S_L,R_EDI);
  1270. {$ifndef noAllocEdi}
  1271. ungetregister32(R_EDI);
  1272. {$endif noAllocEdi}
  1273. emitcall('FPC_CHECK_OBJECT_EXT');
  1274. emitlab(nillabel);
  1275. end;
  1276. {$endif TESTOBJEXT2}
  1277. end;
  1278. {*****************************************************************************
  1279. TI386ISNODE
  1280. *****************************************************************************}
  1281. procedure ti386isnode.pass_2;
  1282. var
  1283. pushed : tpushed;
  1284. begin
  1285. { save all used registers }
  1286. pushusedregisters(pushed,$ff);
  1287. secondpass(left);
  1288. clear_location(location);
  1289. location.loc:=LOC_FLAGS;
  1290. location.resflags:=F_NE;
  1291. { push instance to check: }
  1292. case left.location.loc of
  1293. LOC_REGISTER,LOC_CREGISTER:
  1294. begin
  1295. emit_reg(A_PUSH,
  1296. S_L,left.location.register);
  1297. ungetregister32(left.location.register);
  1298. end;
  1299. LOC_MEM,LOC_REFERENCE:
  1300. begin
  1301. emit_ref(A_PUSH,
  1302. S_L,newreference(left.location.reference));
  1303. del_reference(left.location.reference);
  1304. end;
  1305. else internalerror(100);
  1306. end;
  1307. { generate type checking }
  1308. secondpass(right);
  1309. case right.location.loc of
  1310. LOC_REGISTER,LOC_CREGISTER:
  1311. begin
  1312. emit_reg(A_PUSH,
  1313. S_L,right.location.register);
  1314. ungetregister32(right.location.register);
  1315. end;
  1316. LOC_MEM,LOC_REFERENCE:
  1317. begin
  1318. emit_ref(A_PUSH,
  1319. S_L,newreference(right.location.reference));
  1320. del_reference(right.location.reference);
  1321. end;
  1322. else internalerror(100);
  1323. end;
  1324. saveregvars($ff);
  1325. emitcall('FPC_DO_IS');
  1326. emit_reg_reg(A_OR,S_B,R_AL,R_AL);
  1327. popusedregisters(pushed);
  1328. maybe_loadesi;
  1329. end;
  1330. {*****************************************************************************
  1331. TI386ASNODE
  1332. *****************************************************************************}
  1333. procedure ti386asnode.pass_2;
  1334. var
  1335. pushed : tpushed;
  1336. begin
  1337. secondpass(left);
  1338. { save all used registers }
  1339. pushusedregisters(pushed,$ff);
  1340. { push instance to check: }
  1341. case left.location.loc of
  1342. LOC_REGISTER,LOC_CREGISTER:
  1343. emit_reg(A_PUSH,
  1344. S_L,left.location.register);
  1345. LOC_MEM,LOC_REFERENCE:
  1346. emit_ref(A_PUSH,
  1347. S_L,newreference(left.location.reference));
  1348. else internalerror(100);
  1349. end;
  1350. { we doesn't modifiy the left side, we check only the type }
  1351. set_location(location,left.location);
  1352. { generate type checking }
  1353. secondpass(right);
  1354. case right.location.loc of
  1355. LOC_REGISTER,LOC_CREGISTER:
  1356. begin
  1357. emit_reg(A_PUSH,
  1358. S_L,right.location.register);
  1359. ungetregister32(right.location.register);
  1360. end;
  1361. LOC_MEM,LOC_REFERENCE:
  1362. begin
  1363. emit_ref(A_PUSH,
  1364. S_L,newreference(right.location.reference));
  1365. del_reference(right.location.reference);
  1366. end;
  1367. else internalerror(100);
  1368. end;
  1369. saveregvars($ff);
  1370. emitcall('FPC_DO_AS');
  1371. { restore register, this restores automatically the }
  1372. { result }
  1373. popusedregisters(pushed);
  1374. maybe_loadesi;
  1375. end;
  1376. begin
  1377. ctypeconvnode:=ti386typeconvnode;
  1378. cisnode:=ti386isnode;
  1379. casnode:=ti386asnode;
  1380. end.
  1381. {
  1382. $Log$
  1383. Revision 1.12 2001-01-08 21:45:11 peter
  1384. * internalerror for string to chararray
  1385. Revision 1.11 2000/12/25 00:07:32 peter
  1386. + new tlinkedlist class (merge of old tstringqueue,tcontainer and
  1387. tlinkedlist objects)
  1388. Revision 1.10 2000/12/07 17:19:46 jonas
  1389. * new constant handling: from now on, hex constants >$7fffffff are
  1390. parsed as unsigned constants (otherwise, $80000000 got sign extended
  1391. and became $ffffffff80000000), all constants in the longint range
  1392. become longints, all constants >$7fffffff and <=cardinal($ffffffff)
  1393. are cardinals and the rest are int64's.
  1394. * added lots of longint typecast to prevent range check errors in the
  1395. compiler and rtl
  1396. * type casts of symbolic ordinal constants are now preserved
  1397. * fixed bug where the original resulttype wasn't restored correctly
  1398. after doing a 64bit rangecheck
  1399. Revision 1.9 2000/12/05 11:44:33 jonas
  1400. + new integer regvar handling, should be much more efficient
  1401. Revision 1.8 2000/11/29 00:30:46 florian
  1402. * unused units removed from uses clause
  1403. * some changes for widestrings
  1404. Revision 1.7 2000/11/16 15:27:48 jonas
  1405. * fixed web bug 1242
  1406. Revision 1.6 2000/11/13 11:30:56 florian
  1407. * some bugs with interfaces and NIL fixed
  1408. Revision 1.5 2000/11/12 23:24:14 florian
  1409. * interfaces are basically running
  1410. Revision 1.4 2000/11/11 16:00:10 jonas
  1411. * optimize converting of 8/16/32 bit constants to 64bit ones
  1412. Revision 1.3 2000/11/04 14:25:23 florian
  1413. + merged Attila's changes for interfaces, not tested yet
  1414. Revision 1.2 2000/10/31 22:02:56 peter
  1415. * symtable splitted, no real code changes
  1416. Revision 1.1 2000/10/15 09:33:31 peter
  1417. * moved n386*.pas to i386/ cpu_target dir
  1418. Revision 1.1 2000/10/14 10:14:48 peter
  1419. * moehrendorf oct 2000 rewrite
  1420. }