cg386ld.pas 47 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094
  1. {
  2. $Id$
  3. Copyright (c) 1998-2000 by Florian Klaempfl
  4. Generate i386 assembler for load/assignment 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 cg386ld;
  19. interface
  20. uses
  21. tree;
  22. procedure secondload(var p : ptree);
  23. procedure secondassignment(var p : ptree);
  24. procedure secondfuncret(var p : ptree);
  25. procedure secondarrayconstruct(var p : ptree);
  26. implementation
  27. uses
  28. globtype,systems,
  29. cobjects,verbose,globals,files,
  30. symconst,symtable,aasm,types,
  31. hcodegen,temp_gen,pass_2,
  32. cpubase,cpuasm,
  33. cgai386,tgeni386,cg386cnv,cresstr;
  34. {*****************************************************************************
  35. SecondLoad
  36. *****************************************************************************}
  37. procedure secondload(var p : ptree);
  38. var
  39. hregister : tregister;
  40. symtabletype : tsymtabletype;
  41. i : longint;
  42. hp : preference;
  43. s : pasmsymbol;
  44. popeax : boolean;
  45. pushed : tpushed;
  46. hr : treference;
  47. begin
  48. simple_loadn:=true;
  49. reset_reference(p^.location.reference);
  50. case p^.symtableentry^.typ of
  51. { this is only for toasm and toaddr }
  52. absolutesym :
  53. begin
  54. p^.location.reference.symbol:=nil;
  55. if (pabsolutesym(p^.symtableentry)^.abstyp=toaddr) then
  56. begin
  57. if pabsolutesym(p^.symtableentry)^.absseg then
  58. p^.location.reference.segment:=R_FS;
  59. p^.location.reference.offset:=pabsolutesym(p^.symtableentry)^.address;
  60. end
  61. else
  62. p^.location.reference.symbol:=newasmsymbol(p^.symtableentry^.mangledname);
  63. end;
  64. constsym:
  65. begin
  66. if pconstsym(p^.symtableentry)^.consttyp=constresourcestring then
  67. begin
  68. pushusedregisters(pushed,$ff);
  69. emit_const(A_PUSH,S_L,
  70. pconstsym(p^.symtableentry)^.resstrindex);
  71. emit_sym(A_PUSH,S_L,newasmsymbol(pconstsym(p^.symtableentry)^.owner^.name^+'_RESOURCESTRINGLIST'));
  72. emitcall('FPC_GETRESOURCESTRING');
  73. hregister:=getexplicitregister32(R_EAX);
  74. emit_reg_reg(A_MOV,S_L,R_EAX,hregister);
  75. gettempansistringreference(hr);
  76. decrstringref(p^.resulttype,hr);
  77. emit_reg_ref(A_MOV,S_L,hregister,
  78. newreference(hr));
  79. ungetregister32(hregister);
  80. popusedregisters(pushed);
  81. p^.location.loc:=LOC_MEM;
  82. p^.location.reference:=hr;
  83. end
  84. else
  85. internalerror(22798);
  86. end;
  87. varsym :
  88. begin
  89. hregister:=R_NO;
  90. { C variable }
  91. if (vo_is_C_var in pvarsym(p^.symtableentry)^.varoptions) then
  92. begin
  93. p^.location.reference.symbol:=newasmsymbol(p^.symtableentry^.mangledname);
  94. end
  95. { DLL variable }
  96. else if (vo_is_dll_var in pvarsym(p^.symtableentry)^.varoptions) then
  97. begin
  98. hregister:=getregister32;
  99. p^.location.reference.symbol:=newasmsymbol(p^.symtableentry^.mangledname);
  100. emit_ref_reg(A_MOV,S_L,newreference(p^.location.reference),hregister);
  101. p^.location.reference.symbol:=nil;
  102. p^.location.reference.base:=hregister;
  103. end
  104. { external variable }
  105. else if (vo_is_external in pvarsym(p^.symtableentry)^.varoptions) then
  106. begin
  107. p^.location.reference.symbol:=newasmsymbol(p^.symtableentry^.mangledname);
  108. end
  109. { thread variable }
  110. else if (vo_is_thread_var in pvarsym(p^.symtableentry)^.varoptions) then
  111. begin
  112. popeax:=not(R_EAX in unused);
  113. if popeax then
  114. emit_reg(A_PUSH,S_L,R_EAX);
  115. p^.location.reference.symbol:=newasmsymbol(p^.symtableentry^.mangledname);
  116. emit_ref(A_PUSH,S_L,newreference(p^.location.reference));
  117. { the called procedure isn't allowed to change }
  118. { any register except EAX }
  119. emitcall('FPC_RELOCATE_THREADVAR');
  120. reset_reference(p^.location.reference);
  121. p^.location.reference.base:=getregister32;
  122. emit_reg_reg(A_MOV,S_L,R_EAX,p^.location.reference.base);
  123. if popeax then
  124. emit_reg(A_POP,S_L,R_EAX);
  125. end
  126. { normal variable }
  127. else
  128. begin
  129. symtabletype:=p^.symtable^.symtabletype;
  130. { in case it is a register variable: }
  131. if pvarsym(p^.symtableentry)^.reg<>R_NO then
  132. begin
  133. if pvarsym(p^.symtableentry)^.reg in [R_ST0..R_ST7] then
  134. begin
  135. p^.location.loc:=LOC_CFPUREGISTER;
  136. p^.location.register:=pvarsym(p^.symtableentry)^.reg;
  137. end
  138. else
  139. begin
  140. p^.location.loc:=LOC_CREGISTER;
  141. p^.location.register:=pvarsym(p^.symtableentry)^.reg;
  142. unused:=unused-[pvarsym(p^.symtableentry)^.reg];
  143. end;
  144. end
  145. else
  146. begin
  147. { first handle local and temporary variables }
  148. if (symtabletype in [parasymtable,inlinelocalsymtable,
  149. inlineparasymtable,localsymtable]) then
  150. begin
  151. p^.location.reference.base:=procinfo^.framepointer;
  152. if (symtabletype in [inlinelocalsymtable,
  153. localsymtable]) then
  154. p^.location.reference.offset:=
  155. pvarsym(p^.symtableentry)^.address-p^.symtable^.address_fixup
  156. else
  157. p^.location.reference.offset:=
  158. pvarsym(p^.symtableentry)^.address+p^.symtable^.address_fixup;
  159. if (symtabletype in [localsymtable,inlinelocalsymtable]) then
  160. begin
  161. if use_esp_stackframe then
  162. dec(p^.location.reference.offset,
  163. pvarsym(p^.symtableentry)^.getsize)
  164. else
  165. p^.location.reference.offset:=-p^.location.reference.offset;
  166. end;
  167. if (lexlevel>(p^.symtable^.symtablelevel)) then
  168. begin
  169. hregister:=getregister32;
  170. { make a reference }
  171. hp:=new_reference(procinfo^.framepointer,
  172. procinfo^.framepointer_offset);
  173. emit_ref_reg(A_MOV,S_L,hp,hregister);
  174. simple_loadn:=false;
  175. i:=lexlevel-1;
  176. while i>(p^.symtable^.symtablelevel) do
  177. begin
  178. { make a reference }
  179. hp:=new_reference(hregister,8);
  180. emit_ref_reg(A_MOV,S_L,hp,hregister);
  181. dec(i);
  182. end;
  183. p^.location.reference.base:=hregister;
  184. end;
  185. end
  186. else
  187. case symtabletype of
  188. unitsymtable,globalsymtable,
  189. staticsymtable :
  190. begin
  191. p^.location.reference.symbol:=newasmsymbol(p^.symtableentry^.mangledname);
  192. end;
  193. stt_exceptsymtable:
  194. begin
  195. p^.location.reference.base:=procinfo^.framepointer;
  196. p^.location.reference.offset:=pvarsym(p^.symtableentry)^.address;
  197. end;
  198. objectsymtable:
  199. begin
  200. getexplicitregister32(R_ESI);
  201. if (sp_static in pvarsym(p^.symtableentry)^.symoptions) then
  202. begin
  203. p^.location.reference.symbol:=newasmsymbol(p^.symtableentry^.mangledname);
  204. end
  205. else
  206. begin
  207. p^.location.reference.base:=R_ESI;
  208. p^.location.reference.offset:=pvarsym(p^.symtableentry)^.address;
  209. end;
  210. end;
  211. withsymtable:
  212. begin
  213. { make a reference }
  214. { symtable datasize field
  215. contains the offset of the temp
  216. stored }
  217. { hp:=new_reference(procinfo^.framepointer,
  218. p^.symtable^.datasize);
  219. emit_ref_reg(A_MOV,S_L,hp,hregister);}
  220. if ptree(pwithsymtable(p^.symtable)^.withnode)^.islocal then
  221. begin
  222. p^.location.reference:=ptree(pwithsymtable(p^.symtable)^.withnode)^.withreference^;
  223. end
  224. else
  225. begin
  226. hregister:=getregister32;
  227. p^.location.reference.base:=hregister;
  228. emit_ref_reg(A_MOV,S_L,
  229. newreference(ptree(pwithsymtable(p^.symtable)^.withnode)^.withreference^),
  230. hregister);
  231. end;
  232. inc(p^.location.reference.offset,pvarsym(p^.symtableentry)^.address);
  233. end;
  234. end;
  235. end;
  236. { in case call by reference, then calculate. Open array
  237. is always an reference! }
  238. if (pvarsym(p^.symtableentry)^.varspez=vs_var) or
  239. is_open_array(pvarsym(p^.symtableentry)^.vartype.def) or
  240. is_array_of_const(pvarsym(p^.symtableentry)^.vartype.def) or
  241. ((pvarsym(p^.symtableentry)^.varspez=vs_const) and
  242. push_addr_param(pvarsym(p^.symtableentry)^.vartype.def)) then
  243. begin
  244. simple_loadn:=false;
  245. if hregister=R_NO then
  246. hregister:=getregister32;
  247. if p^.location.loc=LOC_CREGISTER then
  248. begin
  249. emit_reg_reg(A_MOV,S_L,
  250. p^.location.register,hregister);
  251. p^.location.loc:=LOC_REFERENCE;
  252. end
  253. else
  254. begin
  255. emit_ref_reg(A_MOV,S_L,
  256. newreference(p^.location.reference),
  257. hregister);
  258. end;
  259. reset_reference(p^.location.reference);
  260. p^.location.reference.base:=hregister;
  261. end;
  262. end;
  263. end;
  264. procsym:
  265. begin
  266. if assigned(p^.left) then
  267. begin
  268. secondpass(p^.left);
  269. p^.location.loc:=LOC_MEM;
  270. gettempofsizereference(8,p^.location.reference);
  271. { load class instance address }
  272. case p^.left^.location.loc of
  273. LOC_CREGISTER,
  274. LOC_REGISTER:
  275. begin
  276. hregister:=p^.left^.location.register;
  277. ungetregister32(p^.left^.location.register);
  278. if not(pobjectdef(p^.left^.resulttype)^.is_class) then
  279. CGMessage(cg_e_illegal_expression);
  280. end;
  281. LOC_MEM,
  282. LOC_REFERENCE:
  283. begin
  284. {$ifndef noAllocEdi}
  285. getexplicitregister32(R_EDI);
  286. {$endif noAllocEdi}
  287. hregister:=R_EDI;
  288. if pobjectdef(p^.left^.resulttype)^.is_class then
  289. emit_ref_reg(A_MOV,S_L,
  290. newreference(p^.left^.location.reference),R_EDI)
  291. else
  292. emit_ref_reg(A_LEA,S_L,
  293. newreference(p^.left^.location.reference),R_EDI);
  294. del_reference(p^.left^.location.reference);
  295. ungetiftemp(p^.left^.location.reference);
  296. end;
  297. else internalerror(26019);
  298. end;
  299. { store the class instance address }
  300. new(hp);
  301. hp^:=p^.location.reference;
  302. inc(hp^.offset,4);
  303. emit_reg_ref(A_MOV,S_L,
  304. hregister,hp);
  305. { virtual method ? }
  306. if (po_virtualmethod in pprocsym(p^.symtableentry)^.definition^.procoptions) then
  307. begin
  308. new(hp);
  309. reset_reference(hp^);
  310. hp^.base:=hregister;
  311. { load vmt pointer }
  312. emit_ref_reg(A_MOV,S_L,
  313. hp,R_EDI);
  314. {$IfDef regallocfix}
  315. del_reference(hp^);
  316. {$EndIf regallocfix}
  317. { load method address }
  318. new(hp);
  319. reset_reference(hp^);
  320. hp^.base:=R_EDI;
  321. hp^.offset:=pprocsym(p^.symtableentry)^.definition^._class^.vmtmethodoffset(
  322. pprocsym(p^.symtableentry)^.definition^.extnumber);
  323. emit_ref_reg(A_MOV,S_L,
  324. hp,R_EDI);
  325. { ... and store it }
  326. emit_reg_ref(A_MOV,S_L,
  327. R_EDI,newreference(p^.location.reference));
  328. {$ifndef noAllocEdi}
  329. ungetregister32(R_EDI);
  330. {$endif noAllocEdi}
  331. end
  332. else
  333. begin
  334. {$ifndef noAllocEdi}
  335. ungetregister32(R_EDI);
  336. {$endif noAllocEdi}
  337. s:=newasmsymbol(pprocsym(p^.symtableentry)^.definition^.mangledname);
  338. emit_sym_ofs_ref(A_MOV,S_L,s,0,
  339. newreference(p^.location.reference));
  340. end;
  341. end
  342. else
  343. begin
  344. {!!!!! Be aware, work on virtual methods too }
  345. p^.location.reference.symbol:=newasmsymbol(pprocsym(p^.symtableentry)^.definition^.mangledname);
  346. end;
  347. end;
  348. typedconstsym :
  349. begin
  350. p^.location.reference.symbol:=newasmsymbol(p^.symtableentry^.mangledname);
  351. end;
  352. else internalerror(4);
  353. end;
  354. end;
  355. {*****************************************************************************
  356. SecondAssignment
  357. *****************************************************************************}
  358. procedure secondassignment(var p : ptree);
  359. var
  360. opsize : topsize;
  361. otlabel,hlabel,oflabel : pasmlabel;
  362. fputyp : tfloattype;
  363. loc : tloc;
  364. r : preference;
  365. ai : paicpu;
  366. op : tasmop;
  367. pushed : boolean;
  368. begin
  369. otlabel:=truelabel;
  370. oflabel:=falselabel;
  371. getlabel(truelabel);
  372. getlabel(falselabel);
  373. { calculate left sides }
  374. if not(p^.concat_string) then
  375. secondpass(p^.left);
  376. if codegenerror then
  377. exit;
  378. if not(p^.left^.location.loc in [LOC_REFERENCE,LOC_CFPUREGISTER,
  379. LOC_CREGISTER,LOC_CMMXREGISTER]) then
  380. begin
  381. CGMessage(cg_e_illegal_expression);
  382. exit;
  383. end;
  384. loc:=p^.left^.location.loc;
  385. { lets try to optimize this (PM) }
  386. { define a dest_loc that is the location }
  387. { and a ptree to verify that it is the right }
  388. { place to insert it }
  389. {$ifdef test_dest_loc}
  390. if (aktexprlevel<4) then
  391. begin
  392. dest_loc_known:=true;
  393. dest_loc:=p^.left^.location;
  394. dest_loc_tree:=p^.right;
  395. end;
  396. {$endif test_dest_loc}
  397. { left can't be never a 64 bit LOC_REGISTER, so the 3. arg }
  398. { can be false }
  399. pushed:=maybe_push(p^.right^.registers32,p^.left,false);
  400. secondpass(p^.right);
  401. { restoring here is nonsense for LOC_JMP !! }
  402. { This generated code that was after a jmp and before any
  403. label => unreachable !!
  404. Could this be tested somehow ?? PM }
  405. if pushed and (p^.right^.location.loc <>LOC_JUMP) then
  406. restore(p^.left,false);
  407. if codegenerror then
  408. exit;
  409. {$ifdef test_dest_loc}
  410. dest_loc_known:=false;
  411. if in_dest_loc then
  412. begin
  413. truelabel:=otlabel;
  414. falselabel:=oflabel;
  415. in_dest_loc:=false;
  416. exit;
  417. end;
  418. {$endif test_dest_loc}
  419. if p^.left^.resulttype^.deftype=stringdef then
  420. begin
  421. if is_ansistring(p^.left^.resulttype) then
  422. begin
  423. { the source and destinations are released
  424. in loadansistring, because an ansi string can
  425. also be in a register
  426. }
  427. loadansistring(p);
  428. end
  429. else
  430. if is_shortstring(p^.left^.resulttype) and
  431. not (p^.concat_string) then
  432. begin
  433. if is_ansistring(p^.right^.resulttype) then
  434. begin
  435. if (p^.right^.treetype=stringconstn) and
  436. (p^.right^.length=0) then
  437. begin
  438. emit_const_ref(A_MOV,S_B,
  439. 0,newreference(p^.left^.location.reference));
  440. del_reference(p^.left^.location.reference);
  441. end
  442. else
  443. loadansi2short(p^.right,p^.left);
  444. end
  445. else
  446. begin
  447. { we do not need destination anymore }
  448. del_reference(p^.left^.location.reference);
  449. {del_reference(p^.right^.location.reference);
  450. done in loadshortstring }
  451. loadshortstring(p);
  452. ungetiftemp(p^.right^.location.reference);
  453. end;
  454. end
  455. else if is_longstring(p^.left^.resulttype) then
  456. begin
  457. end
  458. else
  459. begin
  460. { its the only thing we have to do }
  461. del_reference(p^.right^.location.reference);
  462. end
  463. end
  464. else case p^.right^.location.loc of
  465. LOC_REFERENCE,
  466. LOC_MEM : begin
  467. { extra handling for ordinal constants }
  468. if (p^.right^.treetype in [ordconstn,fixconstn]) or
  469. (loc=LOC_CREGISTER) then
  470. begin
  471. case p^.left^.resulttype^.size of
  472. 1 : opsize:=S_B;
  473. 2 : opsize:=S_W;
  474. 4 : opsize:=S_L;
  475. { S_L is correct, the copy is done }
  476. { with two moves }
  477. 8 : opsize:=S_L;
  478. end;
  479. if loc=LOC_CREGISTER then
  480. begin
  481. emit_ref_reg(A_MOV,opsize,
  482. newreference(p^.right^.location.reference),
  483. p^.left^.location.register);
  484. if is_64bitint(p^.right^.resulttype) then
  485. begin
  486. r:=newreference(p^.right^.location.reference);
  487. inc(r^.offset,4);
  488. emit_ref_reg(A_MOV,opsize,r,
  489. p^.left^.location.registerhigh);
  490. end;
  491. {$IfDef regallocfix}
  492. del_reference(p^.right^.location.reference);
  493. {$EndIf regallocfix}
  494. end
  495. else
  496. begin
  497. emit_const_ref(A_MOV,opsize,
  498. p^.right^.location.reference.offset,
  499. newreference(p^.left^.location.reference));
  500. if is_64bitint(p^.right^.resulttype) then
  501. begin
  502. r:=newreference(p^.left^.location.reference);
  503. inc(r^.offset,4);
  504. emit_const_ref(A_MOV,opsize,
  505. 0,r);
  506. end;
  507. {$IfDef regallocfix}
  508. del_reference(p^.left^.location.reference);
  509. {$EndIf regallocfix}
  510. {emit_const_loc(A_MOV,opsize,
  511. p^.right^.location.reference.offset,
  512. p^.left^.location);}
  513. end;
  514. end
  515. else if loc=LOC_CFPUREGISTER then
  516. begin
  517. floatloadops(pfloatdef(p^.right^.resulttype)^.typ,op,opsize);
  518. emit_ref(op,opsize,
  519. newreference(p^.right^.location.reference));
  520. emit_reg(A_FSTP,S_NO,
  521. correct_fpuregister(p^.left^.location.register,fpuvaroffset+1));
  522. end
  523. else
  524. begin
  525. if (p^.right^.resulttype^.needs_inittable) and
  526. ( (p^.right^.resulttype^.deftype<>objectdef) or
  527. not(pobjectdef(p^.right^.resulttype)^.is_class)) then
  528. begin
  529. { this would be a problem }
  530. if not(p^.left^.resulttype^.needs_inittable) then
  531. internalerror(3457);
  532. { increment source reference counter }
  533. new(r);
  534. reset_reference(r^);
  535. r^.symbol:=p^.right^.resulttype^.get_inittable_label;
  536. emitpushreferenceaddr(r^);
  537. emitpushreferenceaddr(p^.right^.location.reference);
  538. emitcall('FPC_ADDREF');
  539. { decrement destination reference counter }
  540. new(r);
  541. reset_reference(r^);
  542. r^.symbol:=p^.left^.resulttype^.get_inittable_label;
  543. emitpushreferenceaddr(r^);
  544. emitpushreferenceaddr(p^.left^.location.reference);
  545. emitcall('FPC_DECREF');
  546. end;
  547. {$ifdef regallocfix}
  548. concatcopy(p^.right^.location.reference,
  549. p^.left^.location.reference,p^.left^.resulttype^.size,true,false);
  550. ungetiftemp(p^.right^.location.reference);
  551. {$Else regallocfix}
  552. concatcopy(p^.right^.location.reference,
  553. p^.left^.location.reference,p^.left^.resulttype^.size,false,false);
  554. ungetiftemp(p^.right^.location.reference);
  555. {$endif regallocfix}
  556. end;
  557. end;
  558. {$ifdef SUPPORT_MMX}
  559. LOC_CMMXREGISTER,
  560. LOC_MMXREGISTER:
  561. begin
  562. if loc=LOC_CMMXREGISTER then
  563. emit_reg_reg(A_MOVQ,S_NO,
  564. p^.right^.location.register,p^.left^.location.register)
  565. else
  566. emit_reg_ref(A_MOVQ,S_NO,
  567. p^.right^.location.register,newreference(p^.left^.location.reference));
  568. end;
  569. {$endif SUPPORT_MMX}
  570. LOC_REGISTER,
  571. LOC_CREGISTER : begin
  572. case p^.right^.resulttype^.size of
  573. 1 : opsize:=S_B;
  574. 2 : opsize:=S_W;
  575. 4 : opsize:=S_L;
  576. 8 : opsize:=S_L;
  577. end;
  578. { simplified with op_reg_loc }
  579. if loc=LOC_CREGISTER then
  580. begin
  581. emit_reg_reg(A_MOV,opsize,
  582. p^.right^.location.register,
  583. p^.left^.location.register);
  584. ungetregister(p^.right^.location.register);
  585. end
  586. else
  587. Begin
  588. emit_reg_ref(A_MOV,opsize,
  589. p^.right^.location.register,
  590. newreference(p^.left^.location.reference));
  591. ungetregister(p^.right^.location.register);
  592. {$IfDef regallocfix}
  593. del_reference(p^.left^.location.reference);
  594. {$EndIf regallocfix}
  595. end;
  596. if is_64bitint(p^.right^.resulttype) then
  597. begin
  598. { simplified with op_reg_loc }
  599. if loc=LOC_CREGISTER then
  600. emit_reg_reg(A_MOV,opsize,
  601. p^.right^.location.registerhigh,
  602. p^.left^.location.registerhigh)
  603. else
  604. begin
  605. r:=newreference(p^.left^.location.reference);
  606. inc(r^.offset,4);
  607. emit_reg_ref(A_MOV,opsize,
  608. p^.right^.location.registerhigh,r);
  609. end;
  610. end;
  611. {emit_reg_loc(A_MOV,opsize,
  612. p^.right^.location.register,
  613. p^.left^.location); }
  614. end;
  615. LOC_FPU : begin
  616. if (p^.left^.resulttype^.deftype=floatdef) then
  617. fputyp:=pfloatdef(p^.left^.resulttype)^.typ
  618. else
  619. if (p^.right^.resulttype^.deftype=floatdef) then
  620. fputyp:=pfloatdef(p^.right^.resulttype)^.typ
  621. else
  622. if (p^.right^.treetype=typeconvn) and
  623. (p^.right^.left^.resulttype^.deftype=floatdef) then
  624. fputyp:=pfloatdef(p^.right^.left^.resulttype)^.typ
  625. else
  626. fputyp:=s32real;
  627. case loc of
  628. LOC_CFPUREGISTER:
  629. begin
  630. emit_reg(A_FSTP,S_NO,
  631. correct_fpuregister(p^.left^.location.register,fpuvaroffset));
  632. dec(fpuvaroffset);
  633. end;
  634. LOC_REFERENCE:
  635. floatstore(fputyp,p^.left^.location.reference);
  636. else
  637. internalerror(48991);
  638. end;
  639. end;
  640. LOC_CFPUREGISTER: begin
  641. if (p^.left^.resulttype^.deftype=floatdef) then
  642. fputyp:=pfloatdef(p^.left^.resulttype)^.typ
  643. else
  644. if (p^.right^.resulttype^.deftype=floatdef) then
  645. fputyp:=pfloatdef(p^.right^.resulttype)^.typ
  646. else
  647. if (p^.right^.treetype=typeconvn) and
  648. (p^.right^.left^.resulttype^.deftype=floatdef) then
  649. fputyp:=pfloatdef(p^.right^.left^.resulttype)^.typ
  650. else
  651. fputyp:=s32real;
  652. emit_reg(A_FLD,S_NO,
  653. correct_fpuregister(p^.right^.location.register,fpuvaroffset));
  654. inc(fpuvaroffset);
  655. case loc of
  656. LOC_CFPUREGISTER:
  657. begin
  658. emit_reg(A_FSTP,S_NO,
  659. correct_fpuregister(p^.right^.location.register,fpuvaroffset));
  660. dec(fpuvaroffset);
  661. end;
  662. LOC_REFERENCE:
  663. floatstore(fputyp,p^.left^.location.reference);
  664. else
  665. internalerror(48992);
  666. end;
  667. end;
  668. LOC_JUMP : begin
  669. getlabel(hlabel);
  670. emitlab(truelabel);
  671. if pushed then
  672. restore(p^.left,false);
  673. if loc=LOC_CREGISTER then
  674. emit_const_reg(A_MOV,S_B,
  675. 1,p^.left^.location.register)
  676. else
  677. emit_const_ref(A_MOV,S_B,
  678. 1,newreference(p^.left^.location.reference));
  679. {emit_const_loc(A_MOV,S_B,
  680. 1,p^.left^.location);}
  681. emitjmp(C_None,hlabel);
  682. emitlab(falselabel);
  683. if pushed then
  684. restore(p^.left,false);
  685. if loc=LOC_CREGISTER then
  686. emit_reg_reg(A_XOR,S_B,
  687. p^.left^.location.register,
  688. p^.left^.location.register)
  689. else
  690. begin
  691. emit_const_ref(A_MOV,S_B,
  692. 0,newreference(p^.left^.location.reference));
  693. {$IfDef regallocfix}
  694. del_reference(p^.left^.location.reference);
  695. {$EndIf regallocfix}
  696. end;
  697. emitlab(hlabel);
  698. end;
  699. LOC_FLAGS : begin
  700. if loc=LOC_CREGISTER then
  701. emit_flag2reg(p^.right^.location.resflags,p^.left^.location.register)
  702. else
  703. begin
  704. ai:=new(paicpu,op_ref(A_Setcc,S_B,newreference(p^.left^.location.reference)));
  705. ai^.SetCondition(flag_2_cond[p^.right^.location.resflags]);
  706. exprasmlist^.concat(ai);
  707. end;
  708. {$IfDef regallocfix}
  709. del_reference(p^.left^.location.reference);
  710. {$EndIf regallocfix}
  711. end;
  712. end;
  713. truelabel:=otlabel;
  714. falselabel:=oflabel;
  715. end;
  716. {*****************************************************************************
  717. SecondFuncRet
  718. *****************************************************************************}
  719. procedure secondfuncret(var p : ptree);
  720. var
  721. hr : tregister;
  722. hp : preference;
  723. pp : pprocinfo;
  724. hr_valid : boolean;
  725. begin
  726. reset_reference(p^.location.reference);
  727. hr_valid:=false;
  728. if (not inlining_procedure) and
  729. (procinfo<>pprocinfo(p^.funcretprocinfo)) then
  730. begin
  731. hr:=getregister32;
  732. hr_valid:=true;
  733. hp:=new_reference(procinfo^.framepointer,
  734. procinfo^.framepointer_offset);
  735. emit_ref_reg(A_MOV,S_L,hp,hr);
  736. pp:=procinfo^.parent;
  737. { walk up the stack frame }
  738. while pp<>pprocinfo(p^.funcretprocinfo) do
  739. begin
  740. hp:=new_reference(hr,
  741. pp^.framepointer_offset);
  742. emit_ref_reg(A_MOV,S_L,hp,hr);
  743. pp:=pp^.parent;
  744. end;
  745. p^.location.reference.base:=hr;
  746. p^.location.reference.offset:=pp^.return_offset;
  747. end
  748. else
  749. begin
  750. p^.location.reference.base:=procinfo^.framepointer;
  751. p^.location.reference.offset:=procinfo^.return_offset;
  752. end;
  753. if ret_in_param(p^.rettype.def) then
  754. begin
  755. if not hr_valid then
  756. hr:=getregister32;
  757. emit_ref_reg(A_MOV,S_L,newreference(p^.location.reference),hr);
  758. p^.location.reference.base:=hr;
  759. p^.location.reference.offset:=0;
  760. end;
  761. end;
  762. {*****************************************************************************
  763. SecondArrayConstruct
  764. *****************************************************************************}
  765. const
  766. vtInteger = 0;
  767. vtBoolean = 1;
  768. vtChar = 2;
  769. vtExtended = 3;
  770. vtString = 4;
  771. vtPointer = 5;
  772. vtPChar = 6;
  773. vtObject = 7;
  774. vtClass = 8;
  775. vtWideChar = 9;
  776. vtPWideChar = 10;
  777. vtAnsiString = 11;
  778. vtCurrency = 12;
  779. vtVariant = 13;
  780. vtInterface = 14;
  781. vtWideString = 15;
  782. vtInt64 = 16;
  783. procedure secondarrayconstruct(var p : ptree);
  784. var
  785. hp : ptree;
  786. href : treference;
  787. lt : pdef;
  788. vaddr : boolean;
  789. vtype : longint;
  790. freetemp,
  791. dovariant : boolean;
  792. elesize : longint;
  793. begin
  794. dovariant:=p^.forcevaria or parraydef(p^.resulttype)^.isvariant;
  795. if dovariant then
  796. elesize:=8
  797. else
  798. begin
  799. elesize:=parraydef(p^.resulttype)^.elesize;
  800. if elesize>4 then
  801. internalerror(8765678);
  802. end;
  803. if not p^.cargs then
  804. begin
  805. reset_reference(p^.location.reference);
  806. { Allocate always a temp, also if no elements are required, to
  807. be sure that location is valid (PFV) }
  808. if parraydef(p^.resulttype)^.highrange=-1 then
  809. gettempofsizereference(elesize,p^.location.reference)
  810. else
  811. gettempofsizereference((parraydef(p^.resulttype)^.highrange+1)*elesize,p^.location.reference);
  812. href:=p^.location.reference;
  813. end;
  814. hp:=p;
  815. while assigned(hp) do
  816. begin
  817. if assigned(hp^.left) then
  818. begin
  819. freetemp:=true;
  820. secondpass(hp^.left);
  821. if codegenerror then
  822. exit;
  823. if dovariant then
  824. begin
  825. { find the correct vtype value }
  826. vtype:=$ff;
  827. vaddr:=false;
  828. lt:=hp^.left^.resulttype;
  829. case lt^.deftype of
  830. enumdef,
  831. orddef :
  832. begin
  833. if (lt^.deftype=enumdef) or
  834. is_integer(lt) then
  835. vtype:=vtInteger
  836. else
  837. if is_boolean(lt) then
  838. vtype:=vtBoolean
  839. else
  840. if (lt^.deftype=orddef) and (porddef(lt)^.typ=uchar) then
  841. vtype:=vtChar;
  842. end;
  843. floatdef :
  844. begin
  845. vtype:=vtExtended;
  846. vaddr:=true;
  847. end;
  848. procvardef,
  849. pointerdef :
  850. begin
  851. if is_pchar(lt) then
  852. vtype:=vtPChar
  853. else
  854. vtype:=vtPointer;
  855. end;
  856. classrefdef :
  857. vtype:=vtClass;
  858. objectdef :
  859. begin
  860. vtype:=vtObject;
  861. end;
  862. stringdef :
  863. begin
  864. if is_shortstring(lt) then
  865. begin
  866. vtype:=vtString;
  867. vaddr:=true;
  868. freetemp:=false;
  869. end
  870. else
  871. if is_ansistring(lt) then
  872. begin
  873. vtype:=vtAnsiString;
  874. freetemp:=false;
  875. end;
  876. end;
  877. end;
  878. if vtype=$ff then
  879. internalerror(14357);
  880. { write C style pushes or an pascal array }
  881. if p^.cargs then
  882. begin
  883. if vaddr then
  884. begin
  885. emit_to_mem(hp^.left);
  886. emit_push_lea_loc(hp^.left^.location,freetemp);
  887. del_reference(hp^.left^.location.reference);
  888. end
  889. else
  890. emit_push_loc(hp^.left^.location);
  891. inc(pushedparasize);
  892. end
  893. else
  894. begin
  895. { write changing field update href to the next element }
  896. inc(href.offset,4);
  897. if vaddr then
  898. begin
  899. emit_to_mem(hp^.left);
  900. emit_lea_loc_ref(hp^.left^.location,href,freetemp);
  901. end
  902. else
  903. begin
  904. emit_mov_loc_ref(hp^.left^.location,href,S_L,freetemp);
  905. end;
  906. { update href to the vtype field and write it }
  907. dec(href.offset,4);
  908. emit_const_ref(A_MOV,S_L,vtype,newreference(href));
  909. { goto next array element }
  910. inc(href.offset,8);
  911. end;
  912. end
  913. else
  914. { normal array constructor of the same type }
  915. begin
  916. case elesize of
  917. 1 :
  918. emit_mov_loc_ref(hp^.left^.location,href,S_B,freetemp);
  919. 2 :
  920. emit_mov_loc_ref(hp^.left^.location,href,S_W,freetemp);
  921. 4 :
  922. emit_mov_loc_ref(hp^.left^.location,href,S_L,freetemp);
  923. else
  924. internalerror(87656781);
  925. end;
  926. inc(href.offset,elesize);
  927. end;
  928. end;
  929. { load next entry }
  930. hp:=hp^.right;
  931. end;
  932. end;
  933. end.
  934. {
  935. $Log$
  936. Revision 1.105 2000-03-19 11:55:08 peter
  937. * fixed temp ansi handling within array constructor
  938. Revision 1.104 2000/03/19 08:14:17 peter
  939. * small order change for array of const which allows better optimization
  940. Revision 1.103 2000/03/01 15:36:11 florian
  941. * some new stuff for the new cg
  942. Revision 1.102 2000/03/01 13:20:33 pierre
  943. * fix for bug 859
  944. Revision 1.101 2000/03/01 00:03:11 pierre
  945. * fixes for locals in inlined procedures
  946. fix for bug797
  947. + stabs generation for inlined paras and locals
  948. Revision 1.100 2000/02/09 18:08:33 jonas
  949. * added regallocs for esi
  950. Revision 1.99 2000/02/09 13:22:47 peter
  951. * log truncated
  952. Revision 1.98 2000/02/01 12:54:20 peter
  953. * cargs must also increase pushedparasize else it won't be 'popped'
  954. Revision 1.97 2000/01/21 12:17:42 jonas
  955. * regallocation fixes
  956. Revision 1.96 2000/01/09 12:35:01 jonas
  957. * changed edi allocation to use getexplicitregister32/ungetregister
  958. (adapted tgeni386 a bit for this) and enabled it by default
  959. * fixed very big and stupid bug of mine in cg386mat that broke the
  960. include() code (and make cycle :( ) if you compiled without
  961. -dnewoptimizations
  962. Revision 1.95 2000/01/09 01:44:20 jonas
  963. + (de)allocation info for EDI to fix reported bug on mailinglist.
  964. Also some (de)allocation info for ESI added. Between -dallocEDI
  965. because at this time of the night bugs could easily slip in ;)
  966. Revision 1.94 2000/01/07 01:14:21 peter
  967. * updated copyright to 2000
  968. Revision 1.93 1999/12/30 15:04:31 peter
  969. * fixed funcret within inlined procedure
  970. Revision 1.92 1999/12/22 01:01:47 peter
  971. - removed freelabel()
  972. * added undefined label detection in internal assembler, this prevents
  973. a lot of ld crashes and wrong .o files
  974. * .o files aren't written anymore if errors have occured
  975. * inlining of assembler labels is now correct
  976. Revision 1.91 1999/11/30 10:40:43 peter
  977. + ttype, tsymlist
  978. Revision 1.90 1999/11/06 14:34:18 peter
  979. * truncated log to 20 revs
  980. Revision 1.89 1999/10/12 22:35:48 florian
  981. * compiler didn't complain about l1+l2:=l1+l2; it gave only an assembler
  982. error, fixed
  983. Revision 1.88 1999/09/27 23:44:47 peter
  984. * procinfo is now a pointer
  985. * support for result setting in sub procedure
  986. Revision 1.87 1999/09/26 13:26:06 florian
  987. * exception patch of Romio nevertheless the excpetion handling
  988. needs some corections regarding register saving
  989. * gettempansistring is again a procedure
  990. Revision 1.86 1999/09/16 07:56:46 pierre
  991. * double del_reference removed
  992. Revision 1.85 1999/09/12 08:48:03 florian
  993. * bugs 593 and 607 fixed
  994. * some other potential bugs with array constructors fixed
  995. * for classes compiled in $M+ and it's childs, the default access method
  996. is now published
  997. * fixed copyright message (it is now 1998-2000)
  998. Revision 1.84 1999/09/11 09:08:31 florian
  999. * fixed bug 596
  1000. * fixed some problems with procedure variables and procedures of object,
  1001. especially in TP mode. Procedure of object doesn't apply only to classes,
  1002. it is also allowed for objects !!
  1003. Revision 1.83 1999/09/01 09:37:14 peter
  1004. * removed warning
  1005. Revision 1.82 1999/09/01 09:26:21 peter
  1006. * fixed temp allocation for arrayconstructor
  1007. Revision 1.81 1999/08/28 15:34:17 florian
  1008. * bug 519 fixed
  1009. Revision 1.80 1999/08/26 20:24:37 michael
  1010. + Hopefuly last fixes for resourcestrings
  1011. Revision 1.79 1999/08/25 16:41:05 peter
  1012. * resources are working again
  1013. }