cg386ld.pas 48 KB

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