cpupara.pas 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666
  1. {
  2. Copyright (c) 2002 by Florian Klaempfl
  3. Generates the argument location information for i386
  4. This program is free software; you can redistribute it and/or modify
  5. it under the terms of the GNU General Public License as published bymethodpointer
  6. the Free Software Foundation; either version 2 of the License, or
  7. (at your option) any later version.
  8. This program is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. GNU General Public License for more details.
  12. You should have received a copy of the GNU General Public License
  13. along with this program; if not, write to the Free Software
  14. Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  15. ****************************************************************************
  16. }
  17. unit cpupara;
  18. {$i fpcdefs.inc}
  19. interface
  20. uses
  21. globtype,
  22. aasmtai,aasmdata,cpubase,cgbase,
  23. symconst,symtype,symsym,symdef,
  24. parabase,paramgr;
  25. type
  26. ti386paramanager = class(tparamanager)
  27. function param_use_paraloc(const cgpara:tcgpara):boolean;override;
  28. function ret_in_param(def : tdef;calloption : tproccalloption) : boolean;override;
  29. function push_addr_param(varspez:tvarspez;def : tdef;calloption : tproccalloption) : boolean;override;
  30. function get_para_align(calloption : tproccalloption):byte;override;
  31. function get_volatile_registers_int(calloption : tproccalloption):tcpuregisterset;override;
  32. function get_volatile_registers_fpu(calloption : tproccalloption):tcpuregisterset;override;
  33. function get_volatile_registers_mm(calloption : tproccalloption):tcpuregisterset;override;
  34. { Returns the location for the nr-st 32 Bit int parameter
  35. if every parameter before is an 32 Bit int parameter as well
  36. and if the calling conventions for the helper routines of the
  37. rtl are used.
  38. }
  39. procedure getintparaloc(calloption : tproccalloption; nr : longint;var cgpara:TCGPara);override;
  40. function create_paraloc_info(p : tabstractprocdef; side: tcallercallee):longint;override;
  41. function create_varargs_paraloc_info(p : tabstractprocdef; varargspara:tvarargsparalist):longint;override;
  42. procedure createtempparaloc(list: TAsmList;calloption : tproccalloption;parasym : tparavarsym;var cgpara:TCGPara);override;
  43. private
  44. procedure create_funcretloc_info(p : tabstractprocdef; side: tcallercallee);
  45. procedure create_stdcall_paraloc_info(p : tabstractprocdef; side: tcallercallee;paras:tparalist;var parasize:longint);
  46. procedure create_register_paraloc_info(p : tabstractprocdef; side: tcallercallee;paras:tparalist;var parareg,parasize:longint);
  47. end;
  48. implementation
  49. uses
  50. cutils,
  51. systems,verbose,
  52. defutil,
  53. cgutils;
  54. const
  55. parasupregs : array[0..2] of tsuperregister = (RS_EAX,RS_EDX,RS_ECX);
  56. {****************************************************************************
  57. TI386PARAMANAGER
  58. ****************************************************************************}
  59. function ti386paramanager.param_use_paraloc(const cgpara:tcgpara):boolean;
  60. var
  61. paraloc : pcgparalocation;
  62. begin
  63. if not assigned(cgpara.location) then
  64. internalerror(200410102);
  65. result:=true;
  66. { All locations are LOC_REFERENCE }
  67. paraloc:=cgpara.location;
  68. while assigned(paraloc) do
  69. begin
  70. if (paraloc^.loc<>LOC_REFERENCE) then
  71. begin
  72. result:=false;
  73. exit;
  74. end;
  75. paraloc:=paraloc^.next;
  76. end;
  77. end;
  78. function ti386paramanager.ret_in_param(def : tdef;calloption : tproccalloption) : boolean;
  79. var
  80. size: longint;
  81. begin
  82. case target_info.system of
  83. system_i386_win32 :
  84. begin
  85. if calloption=pocall_safecall then
  86. begin
  87. result:=true;
  88. exit;
  89. end
  90. else
  91. case def.typ of
  92. recorddef :
  93. begin
  94. { Win32 GCC returns small records in the FUNCTION_RETURN_REG.
  95. For stdcall we follow delphi instead of GCC }
  96. if (calloption in [pocall_cdecl,pocall_cppdecl]) and
  97. (def.size>0) and
  98. (def.size<=8) then
  99. begin
  100. result:=false;
  101. exit;
  102. end;
  103. end;
  104. end;
  105. end;
  106. system_i386_darwin :
  107. begin
  108. case def.typ of
  109. recorddef :
  110. begin
  111. size := def.size;
  112. if (size > 0) and
  113. (size <= 8) and
  114. { only if size is a power of 2 }
  115. ((size and (size-1)) = 0) then
  116. begin
  117. result := false;
  118. exit;
  119. end;
  120. end;
  121. end;
  122. end;
  123. end;
  124. result:=inherited ret_in_param(def,calloption);
  125. end;
  126. function ti386paramanager.push_addr_param(varspez:tvarspez;def : tdef;calloption : tproccalloption) : boolean;
  127. begin
  128. result:=false;
  129. { var,out always require address }
  130. if varspez in [vs_var,vs_out] then
  131. begin
  132. result:=true;
  133. exit;
  134. end;
  135. { Only vs_const, vs_value here }
  136. case def.typ of
  137. variantdef :
  138. begin
  139. { variants are small enough to be passed by value except if
  140. required by the windows api
  141. variants are somethings very delphi/windows specific so do it like
  142. windows/delphi (FK)
  143. }
  144. if ((target_info.system=system_i386_win32) and
  145. (calloption=pocall_stdcall) and
  146. (varspez=vs_const)) or
  147. (calloption=pocall_register) then
  148. result:=true
  149. else
  150. result:=false;
  151. end;
  152. formaldef :
  153. result:=true;
  154. recorddef :
  155. begin
  156. { Win32 stdcall passes small records on the stack for call by
  157. value }
  158. if (target_info.system=system_i386_win32) and
  159. (calloption=pocall_stdcall) and
  160. (varspez=vs_value) and
  161. (def.size<=16) then
  162. result:=false
  163. else
  164. result:=
  165. (not(calloption in [pocall_cdecl,pocall_cppdecl,pocall_mwpascal]) and
  166. (def.size>sizeof(aint))) or
  167. ((calloption = pocall_mwpascal) and
  168. (varspez=vs_const));
  169. end;
  170. arraydef :
  171. begin
  172. { Win32 stdcall passes arrays on the stack for call by
  173. value }
  174. if (target_info.system=system_i386_win32) and
  175. (calloption=pocall_stdcall) and
  176. (varspez=vs_value) and
  177. (tarraydef(def).highrange>=tarraydef(def).lowrange) then
  178. result:=false
  179. else
  180. { array of const values are pushed on the stack as
  181. well as dyn. arrays }
  182. if (calloption in [pocall_cdecl,pocall_cppdecl]) then
  183. result:=not(is_array_of_const(def) or
  184. is_dynamic_array(def))
  185. else
  186. begin
  187. result:=(
  188. (tarraydef(def).highrange>=tarraydef(def).lowrange) and
  189. (def.size>sizeof(aint))
  190. ) or
  191. is_open_array(def) or
  192. is_array_of_const(def) or
  193. is_array_constructor(def);
  194. end;
  195. end;
  196. objectdef :
  197. result:=is_object(def);
  198. stringdef :
  199. result:= (tstringdef(def).stringtype in [st_shortstring,st_longstring]);
  200. procvardef :
  201. result:=not(calloption in [pocall_cdecl,pocall_cppdecl]) and (po_methodpointer in tprocvardef(def).procoptions);
  202. setdef :
  203. result:=not(calloption in [pocall_cdecl,pocall_cppdecl]) and (tsetdef(def).settype<>smallset);
  204. end;
  205. end;
  206. function ti386paramanager.get_para_align(calloption : tproccalloption):byte;
  207. begin
  208. if calloption=pocall_oldfpccall then
  209. begin
  210. if target_info.system in [system_i386_go32v2,system_i386_watcom] then
  211. result:=2
  212. else
  213. result:=4;
  214. end
  215. else
  216. result:=std_param_align;
  217. end;
  218. function ti386paramanager.get_volatile_registers_int(calloption : tproccalloption):tcpuregisterset;
  219. begin
  220. case calloption of
  221. pocall_internproc :
  222. result:=[];
  223. pocall_register,
  224. pocall_safecall,
  225. pocall_stdcall,
  226. pocall_cdecl,
  227. pocall_cppdecl,
  228. pocall_mwpascal :
  229. result:=[RS_EAX,RS_EDX,RS_ECX];
  230. pocall_far16,
  231. pocall_pascal,
  232. pocall_oldfpccall :
  233. result:=[RS_EAX,RS_EDX,RS_ECX,RS_ESI,RS_EDI,RS_EBX];
  234. else
  235. internalerror(200309071);
  236. end;
  237. end;
  238. function ti386paramanager.get_volatile_registers_fpu(calloption : tproccalloption):tcpuregisterset;
  239. begin
  240. result:=[0..first_fpu_imreg-1];
  241. end;
  242. function ti386paramanager.get_volatile_registers_mm(calloption : tproccalloption):tcpuregisterset;
  243. begin
  244. result:=[0..first_mm_imreg-1];
  245. end;
  246. procedure ti386paramanager.getintparaloc(calloption : tproccalloption; nr : longint;var cgpara:TCGPara);
  247. var
  248. paraloc : pcgparalocation;
  249. begin
  250. cgpara.reset;
  251. cgpara.size:=OS_INT;
  252. cgpara.intsize:=tcgsize2size[OS_INT];
  253. cgpara.alignment:=get_para_align(calloption);
  254. paraloc:=cgpara.add_location;
  255. with paraloc^ do
  256. begin
  257. size:=OS_INT;
  258. if calloption=pocall_register then
  259. begin
  260. if (nr<=high(parasupregs)+1) then
  261. begin
  262. if nr=0 then
  263. internalerror(200309271);
  264. loc:=LOC_REGISTER;
  265. register:=newreg(R_INTREGISTER,parasupregs[nr-1],R_SUBWHOLE);
  266. end
  267. else
  268. begin
  269. loc:=LOC_REFERENCE;
  270. reference.index:=NR_STACK_POINTER_REG;
  271. reference.offset:=sizeof(aint)*nr;
  272. end;
  273. end
  274. else
  275. begin
  276. loc:=LOC_REFERENCE;
  277. reference.index:=NR_STACK_POINTER_REG;
  278. reference.offset:=sizeof(aint)*nr;
  279. end;
  280. end;
  281. end;
  282. procedure ti386paramanager.create_funcretloc_info(p : tabstractprocdef; side: tcallercallee);
  283. var
  284. retcgsize : tcgsize;
  285. begin
  286. { Constructors return self instead of a boolean }
  287. if (p.proctypeoption=potype_constructor) then
  288. retcgsize:=OS_ADDR
  289. else
  290. retcgsize:=def_cgsize(p.returndef);
  291. location_reset(p.funcretloc[side],LOC_INVALID,OS_NO);
  292. { void has no location }
  293. if is_void(p.returndef) then
  294. begin
  295. location_reset(p.funcretloc[side],LOC_VOID,OS_NO);
  296. exit;
  297. end;
  298. { Return in FPU register? }
  299. if p.returndef.typ=floatdef then
  300. begin
  301. p.funcretloc[side].loc:=LOC_FPUREGISTER;
  302. p.funcretloc[side].register:=NR_FPU_RESULT_REG;
  303. p.funcretloc[side].size:=retcgsize;
  304. end
  305. else
  306. { Return in register? }
  307. if not ret_in_param(p.returndef,p.proccalloption) then
  308. begin
  309. if retcgsize in [OS_64,OS_S64] then
  310. begin
  311. { low 32bits }
  312. p.funcretloc[side].loc:=LOC_REGISTER;
  313. p.funcretloc[side].size:=OS_64;
  314. if side=callerside then
  315. p.funcretloc[side].register64.reglo:=NR_FUNCTION_RESULT64_LOW_REG
  316. else
  317. p.funcretloc[side].register64.reglo:=NR_FUNCTION_RETURN64_LOW_REG;
  318. { high 32bits }
  319. if side=callerside then
  320. p.funcretloc[side].register64.reghi:=NR_FUNCTION_RESULT64_HIGH_REG
  321. else
  322. p.funcretloc[side].register64.reghi:=NR_FUNCTION_RETURN64_HIGH_REG;
  323. end
  324. else
  325. begin
  326. p.funcretloc[side].loc:=LOC_REGISTER;
  327. p.funcretloc[side].size:=retcgsize;
  328. if side=callerside then
  329. p.funcretloc[side].register:=newreg(R_INTREGISTER,RS_FUNCTION_RESULT_REG,cgsize2subreg(retcgsize))
  330. else
  331. p.funcretloc[side].register:=newreg(R_INTREGISTER,RS_FUNCTION_RETURN_REG,cgsize2subreg(retcgsize));
  332. end;
  333. end
  334. else
  335. begin
  336. p.funcretloc[side].loc:=LOC_REFERENCE;
  337. p.funcretloc[side].size:=retcgsize;
  338. end;
  339. end;
  340. procedure ti386paramanager.create_stdcall_paraloc_info(p : tabstractprocdef; side: tcallercallee;paras:tparalist;var parasize:longint);
  341. var
  342. i : integer;
  343. hp : tparavarsym;
  344. paraloc : pcgparalocation;
  345. l,
  346. paralen,
  347. varalign : longint;
  348. paraalign : shortint;
  349. pushaddr : boolean;
  350. paracgsize : tcgsize;
  351. begin
  352. paraalign:=get_para_align(p.proccalloption);
  353. { we push Flags and CS as long
  354. to cope with the IRETD
  355. and we save 6 register + 4 selectors }
  356. if po_interrupt in p.procoptions then
  357. inc(parasize,8+6*4+4*2);
  358. { Offset is calculated like:
  359. sub esp,12
  360. mov [esp+8],para3
  361. mov [esp+4],para2
  362. mov [esp],para1
  363. call function
  364. That means for pushes the para with the
  365. highest offset (see para3) needs to be pushed first
  366. }
  367. for i:=0 to paras.count-1 do
  368. begin
  369. hp:=tparavarsym(paras[i]);
  370. pushaddr:=push_addr_param(hp.varspez,hp.vardef,p.proccalloption);
  371. if pushaddr then
  372. begin
  373. paralen:=sizeof(aint);
  374. paracgsize:=OS_ADDR;
  375. end
  376. else
  377. begin
  378. paralen:=push_size(hp.varspez,hp.vardef,p.proccalloption);
  379. { darwin/x86 requires that parameters < sizeof(aint) are sign/ }
  380. { zero extended to sizeof(aint) }
  381. if (target_info.system = system_i386_darwin) and
  382. (side = callerside) and
  383. (paralen > 0) and
  384. (paralen < sizeof(aint)) then
  385. begin
  386. paralen := sizeof(aint);
  387. paracgsize:=OS_INT;
  388. end
  389. else
  390. paracgsize:=def_cgsize(hp.vardef);
  391. end;
  392. hp.paraloc[side].reset;
  393. hp.paraloc[side].size:=paracgsize;
  394. hp.paraloc[side].intsize:=paralen;
  395. hp.paraloc[side].Alignment:=paraalign;
  396. { Copy to stack? }
  397. if (paracgsize=OS_NO) or
  398. (use_fixed_stack) then
  399. begin
  400. paraloc:=hp.paraloc[side].add_location;
  401. paraloc^.loc:=LOC_REFERENCE;
  402. paraloc^.size:=paracgsize;
  403. if side=callerside then
  404. paraloc^.reference.index:=NR_STACK_POINTER_REG
  405. else
  406. paraloc^.reference.index:=NR_FRAME_POINTER_REG;
  407. varalign:=used_align(size_2_align(paralen),paraalign,paraalign);
  408. { don't let push_size return 16, because then we can }
  409. { read past the end of the heap since the value is only }
  410. { 10 bytes long (JM) }
  411. if (paracgsize = OS_F80) and
  412. (target_info.system = system_i386_darwin) then
  413. paralen:=16;
  414. paraloc^.reference.offset:=parasize;
  415. if side=calleeside then
  416. inc(paraloc^.reference.offset,target_info.first_parm_offset);
  417. parasize:=align(parasize+paralen,varalign);
  418. end
  419. else
  420. begin
  421. if paralen=0 then
  422. internalerror(200501163);
  423. while (paralen>0) do
  424. begin
  425. paraloc:=hp.paraloc[side].add_location;
  426. paraloc^.loc:=LOC_REFERENCE;
  427. { single and double need a single location }
  428. if (paracgsize in [OS_F64,OS_F32]) then
  429. begin
  430. paraloc^.size:=paracgsize;
  431. l:=paralen;
  432. end
  433. else
  434. begin
  435. { We can allocate at maximum 32 bits per location }
  436. if paralen>sizeof(aint) then
  437. l:=sizeof(aint)
  438. else
  439. l:=paralen;
  440. paraloc^.size:=int_cgsize(l);
  441. end;
  442. if side=callerside then
  443. paraloc^.reference.index:=NR_STACK_POINTER_REG
  444. else
  445. paraloc^.reference.index:=NR_FRAME_POINTER_REG;
  446. varalign:=used_align(size_2_align(l),paraalign,paraalign);
  447. paraloc^.reference.offset:=parasize;
  448. if side=calleeside then
  449. inc(paraloc^.reference.offset,target_info.first_parm_offset);
  450. parasize:=align(parasize+l,varalign);
  451. dec(paralen,l);
  452. end;
  453. end;
  454. end;
  455. end;
  456. procedure ti386paramanager.create_register_paraloc_info(p : tabstractprocdef; side: tcallercallee;paras:tparalist;
  457. var parareg,parasize:longint);
  458. var
  459. hp : tparavarsym;
  460. paraloc : pcgparalocation;
  461. paracgsize : tcgsize;
  462. i : integer;
  463. l,
  464. paralen,
  465. varalign : longint;
  466. pushaddr : boolean;
  467. paraalign : shortint;
  468. begin
  469. paraalign:=get_para_align(p.proccalloption);
  470. { Register parameters are assigned from left to right }
  471. for i:=0 to paras.count-1 do
  472. begin
  473. hp:=tparavarsym(paras[i]);
  474. pushaddr:=push_addr_param(hp.varspez,hp.vardef,p.proccalloption);
  475. if pushaddr then
  476. begin
  477. paralen:=sizeof(aint);
  478. paracgsize:=OS_ADDR;
  479. end
  480. else
  481. begin
  482. paralen:=push_size(hp.varspez,hp.vardef,p.proccalloption);
  483. paracgsize:=def_cgsize(hp.vardef);
  484. end;
  485. hp.paraloc[side].reset;
  486. hp.paraloc[side].size:=paracgsize;
  487. hp.paraloc[side].intsize:=paralen;
  488. hp.paraloc[side].Alignment:=paraalign;
  489. {
  490. EAX
  491. EDX
  492. ECX
  493. Stack
  494. Stack
  495. 64bit values,floats,arrays and records are always
  496. on the stack.
  497. }
  498. if (parareg<=high(parasupregs)) and
  499. (paralen<=sizeof(aint)) and
  500. (
  501. not(hp.vardef.typ in [floatdef,recorddef,arraydef]) or
  502. pushaddr
  503. ) then
  504. begin
  505. paraloc:=hp.paraloc[side].add_location;
  506. paraloc^.size:=paracgsize;
  507. paraloc^.loc:=LOC_REGISTER;
  508. paraloc^.register:=newreg(R_INTREGISTER,parasupregs[parareg],cgsize2subreg(paracgsize));
  509. inc(parareg);
  510. end
  511. else
  512. begin
  513. { Copy to stack? }
  514. if (use_fixed_stack) or
  515. (paracgsize=OS_NO) then
  516. begin
  517. paraloc:=hp.paraloc[side].add_location;
  518. paraloc^.loc:=LOC_REFERENCE;
  519. paraloc^.size:=paracgsize;
  520. if side=callerside then
  521. paraloc^.reference.index:=NR_STACK_POINTER_REG
  522. else
  523. paraloc^.reference.index:=NR_FRAME_POINTER_REG;
  524. varalign:=used_align(size_2_align(paralen),paraalign,paraalign);
  525. paraloc^.reference.offset:=parasize;
  526. if side=calleeside then
  527. inc(paraloc^.reference.offset,target_info.first_parm_offset);
  528. parasize:=align(parasize+paralen,varalign);
  529. end
  530. else
  531. begin
  532. if paralen=0 then
  533. internalerror(200501163);
  534. while (paralen>0) do
  535. begin
  536. paraloc:=hp.paraloc[side].add_location;
  537. paraloc^.loc:=LOC_REFERENCE;
  538. { Extended and double need a single location }
  539. if (paracgsize in [OS_F64,OS_F32]) then
  540. begin
  541. paraloc^.size:=paracgsize;
  542. l:=paralen;
  543. end
  544. else
  545. begin
  546. { We can allocate at maximum 32 bits per location }
  547. if paralen>sizeof(aint) then
  548. l:=sizeof(aint)
  549. else
  550. l:=paralen;
  551. paraloc^.size:=int_cgsize(l);
  552. end;
  553. if side=callerside then
  554. paraloc^.reference.index:=NR_STACK_POINTER_REG
  555. else
  556. paraloc^.reference.index:=NR_FRAME_POINTER_REG;
  557. varalign:=used_align(size_2_align(l),paraalign,paraalign);
  558. paraloc^.reference.offset:=parasize;
  559. if side=calleeside then
  560. inc(paraloc^.reference.offset,target_info.first_parm_offset);
  561. parasize:=align(parasize+l,varalign);
  562. dec(paralen,l);
  563. end;
  564. end;
  565. end;
  566. end;
  567. end;
  568. function ti386paramanager.create_paraloc_info(p : tabstractprocdef; side: tcallercallee):longint;
  569. var
  570. parasize,
  571. parareg : longint;
  572. begin
  573. parasize:=0;
  574. parareg:=0;
  575. case p.proccalloption of
  576. pocall_register :
  577. create_register_paraloc_info(p,side,p.paras,parareg,parasize);
  578. pocall_internproc :
  579. begin
  580. { Use default calling }
  581. if (pocall_default=pocall_register) then
  582. create_register_paraloc_info(p,side,p.paras,parareg,parasize)
  583. else
  584. create_stdcall_paraloc_info(p,side,p.paras,parasize);
  585. end;
  586. else
  587. create_stdcall_paraloc_info(p,side,p.paras,parasize);
  588. end;
  589. create_funcretloc_info(p,side);
  590. result:=parasize;
  591. end;
  592. function ti386paramanager.create_varargs_paraloc_info(p : tabstractprocdef; varargspara:tvarargsparalist):longint;
  593. var
  594. parasize : longint;
  595. begin
  596. parasize:=0;
  597. { calculate the registers for the normal parameters }
  598. create_stdcall_paraloc_info(p,callerside,p.paras,parasize);
  599. { append the varargs }
  600. create_stdcall_paraloc_info(p,callerside,varargspara,parasize);
  601. result:=parasize;
  602. end;
  603. procedure ti386paramanager.createtempparaloc(list: TAsmList;calloption : tproccalloption;parasym : tparavarsym;var cgpara:TCGPara);
  604. var
  605. paraloc : pcgparalocation;
  606. begin
  607. paraloc:=parasym.paraloc[callerside].location;
  608. { No need for temps when value is pushed }
  609. if not(use_fixed_stack) and
  610. assigned(paraloc) and
  611. (paraloc^.loc=LOC_REFERENCE) and
  612. (paraloc^.reference.index=NR_STACK_POINTER_REG) then
  613. duplicateparaloc(list,calloption,parasym,cgpara)
  614. else
  615. inherited createtempparaloc(list,calloption,parasym,cgpara);
  616. end;
  617. begin
  618. paramanager:=ti386paramanager.create;
  619. end.