cpupara.pas 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697
  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. { Delphi stdcall passes records on the stack for call by value }
  157. if (target_info.system=system_i386_win32) and
  158. (calloption=pocall_stdcall) and
  159. (varspez=vs_value) then
  160. result:=false
  161. else
  162. result:=
  163. (not(calloption in [pocall_cdecl,pocall_cppdecl,pocall_mwpascal]) and
  164. (def.size>sizeof(aint))) or
  165. (((calloption = pocall_mwpascal) or (target_info.system=system_i386_wince)) and
  166. (varspez=vs_const));
  167. end;
  168. arraydef :
  169. begin
  170. { array of const values are pushed on the stack as
  171. well as dyn. arrays }
  172. if (calloption in [pocall_cdecl,pocall_cppdecl]) then
  173. result:=not(is_array_of_const(def) or
  174. is_dynamic_array(def))
  175. else
  176. begin
  177. result:=(
  178. (tarraydef(def).highrange>=tarraydef(def).lowrange) and
  179. (def.size>sizeof(aint))
  180. ) or
  181. is_open_array(def) or
  182. is_array_of_const(def) or
  183. is_array_constructor(def);
  184. end;
  185. end;
  186. objectdef :
  187. result:=is_object(def);
  188. stringdef :
  189. result:= (tstringdef(def).stringtype in [st_shortstring,st_longstring]);
  190. procvardef :
  191. result:=not(calloption in [pocall_cdecl,pocall_cppdecl]) and (po_methodpointer in tprocvardef(def).procoptions);
  192. setdef :
  193. result:=not(calloption in [pocall_cdecl,pocall_cppdecl]) and (tsetdef(def).settype<>smallset);
  194. end;
  195. end;
  196. function ti386paramanager.get_para_align(calloption : tproccalloption):byte;
  197. begin
  198. if calloption=pocall_oldfpccall then
  199. begin
  200. if target_info.system in [system_i386_go32v2,system_i386_watcom] then
  201. result:=2
  202. else
  203. result:=4;
  204. end
  205. else
  206. result:=std_param_align;
  207. end;
  208. function ti386paramanager.get_volatile_registers_int(calloption : tproccalloption):tcpuregisterset;
  209. begin
  210. case calloption of
  211. pocall_internproc :
  212. result:=[];
  213. pocall_register,
  214. pocall_safecall,
  215. pocall_stdcall,
  216. pocall_cdecl,
  217. pocall_cppdecl,
  218. pocall_mwpascal :
  219. result:=[RS_EAX,RS_EDX,RS_ECX];
  220. pocall_far16,
  221. pocall_pascal,
  222. pocall_oldfpccall :
  223. result:=[RS_EAX,RS_EDX,RS_ECX,RS_ESI,RS_EDI,RS_EBX];
  224. else
  225. internalerror(200309071);
  226. end;
  227. end;
  228. function ti386paramanager.get_volatile_registers_fpu(calloption : tproccalloption):tcpuregisterset;
  229. begin
  230. result:=[0..first_fpu_imreg-1];
  231. end;
  232. function ti386paramanager.get_volatile_registers_mm(calloption : tproccalloption):tcpuregisterset;
  233. begin
  234. result:=[0..first_mm_imreg-1];
  235. end;
  236. procedure ti386paramanager.getintparaloc(calloption : tproccalloption; nr : longint;var cgpara:TCGPara);
  237. var
  238. paraloc : pcgparalocation;
  239. begin
  240. cgpara.reset;
  241. cgpara.size:=OS_ADDR;
  242. cgpara.intsize:=sizeof(pint);
  243. cgpara.alignment:=get_para_align(calloption);
  244. paraloc:=cgpara.add_location;
  245. with paraloc^ do
  246. begin
  247. size:=OS_INT;
  248. if calloption=pocall_register then
  249. begin
  250. if (nr<=high(parasupregs)+1) then
  251. begin
  252. if nr=0 then
  253. internalerror(200309271);
  254. loc:=LOC_REGISTER;
  255. register:=newreg(R_INTREGISTER,parasupregs[nr-1],R_SUBWHOLE);
  256. end
  257. else
  258. begin
  259. loc:=LOC_REFERENCE;
  260. reference.index:=NR_STACK_POINTER_REG;
  261. reference.offset:=sizeof(aint)*nr;
  262. end;
  263. end
  264. else
  265. begin
  266. loc:=LOC_REFERENCE;
  267. reference.index:=NR_STACK_POINTER_REG;
  268. reference.offset:=sizeof(aint)*nr;
  269. end;
  270. end;
  271. end;
  272. procedure ti386paramanager.create_funcretloc_info(p : tabstractprocdef; side: tcallercallee);
  273. var
  274. retcgsize : tcgsize;
  275. begin
  276. { Constructors return self instead of a boolean }
  277. if (p.proctypeoption=potype_constructor) then
  278. retcgsize:=OS_ADDR
  279. else
  280. retcgsize:=def_cgsize(p.returndef);
  281. location_reset(p.funcretloc[side],LOC_INVALID,OS_NO);
  282. { void has no location }
  283. if is_void(p.returndef) then
  284. begin
  285. location_reset(p.funcretloc[side],LOC_VOID,OS_NO);
  286. exit;
  287. end;
  288. { Return is passed as var parameter }
  289. if ret_in_param(p.returndef,p.proccalloption) then
  290. begin
  291. p.funcretloc[side].loc:=LOC_REFERENCE;
  292. p.funcretloc[side].size:=retcgsize;
  293. exit;
  294. end;
  295. { Return in FPU register? }
  296. if p.returndef.typ=floatdef then
  297. begin
  298. p.funcretloc[side].loc:=LOC_FPUREGISTER;
  299. p.funcretloc[side].register:=NR_FPU_RESULT_REG;
  300. p.funcretloc[side].size:=retcgsize;
  301. end
  302. else
  303. { Return in register }
  304. begin
  305. if retcgsize in [OS_64,OS_S64] then
  306. begin
  307. { low 32bits }
  308. p.funcretloc[side].loc:=LOC_REGISTER;
  309. p.funcretloc[side].size:=OS_64;
  310. if side=callerside then
  311. p.funcretloc[side].register64.reglo:=NR_FUNCTION_RESULT64_LOW_REG
  312. else
  313. p.funcretloc[side].register64.reglo:=NR_FUNCTION_RETURN64_LOW_REG;
  314. { high 32bits }
  315. if side=callerside then
  316. p.funcretloc[side].register64.reghi:=NR_FUNCTION_RESULT64_HIGH_REG
  317. else
  318. p.funcretloc[side].register64.reghi:=NR_FUNCTION_RETURN64_HIGH_REG;
  319. end
  320. else
  321. begin
  322. p.funcretloc[side].loc:=LOC_REGISTER;
  323. p.funcretloc[side].size:=retcgsize;
  324. if side=callerside then
  325. p.funcretloc[side].register:=newreg(R_INTREGISTER,RS_FUNCTION_RESULT_REG,cgsize2subreg(retcgsize))
  326. else
  327. p.funcretloc[side].register:=newreg(R_INTREGISTER,RS_FUNCTION_RETURN_REG,cgsize2subreg(retcgsize));
  328. end;
  329. end;
  330. end;
  331. procedure ti386paramanager.create_stdcall_paraloc_info(p : tabstractprocdef; side: tcallercallee;paras:tparalist;var parasize:longint);
  332. var
  333. i : integer;
  334. hp : tparavarsym;
  335. paraloc : pcgparalocation;
  336. l,
  337. paralen,
  338. varalign : longint;
  339. paraalign : shortint;
  340. pushaddr : boolean;
  341. paracgsize : tcgsize;
  342. begin
  343. paraalign:=get_para_align(p.proccalloption);
  344. { we push Flags and CS as long
  345. to cope with the IRETD
  346. and we save 6 register + 4 selectors }
  347. if po_interrupt in p.procoptions then
  348. inc(parasize,8+6*4+4*2);
  349. { Offset is calculated like:
  350. sub esp,12
  351. mov [esp+8],para3
  352. mov [esp+4],para2
  353. mov [esp],para1
  354. call function
  355. That means for pushes the para with the
  356. highest offset (see para3) needs to be pushed first
  357. }
  358. for i:=0 to paras.count-1 do
  359. begin
  360. hp:=tparavarsym(paras[i]);
  361. pushaddr:=push_addr_param(hp.varspez,hp.vardef,p.proccalloption);
  362. if pushaddr then
  363. begin
  364. paralen:=sizeof(aint);
  365. paracgsize:=OS_ADDR;
  366. end
  367. else
  368. begin
  369. paralen:=push_size(hp.varspez,hp.vardef,p.proccalloption);
  370. { darwin/x86 requires that parameters < sizeof(aint) are sign/ }
  371. { zero extended to sizeof(aint) }
  372. if (target_info.system = system_i386_darwin) and
  373. (side = callerside) and
  374. (paralen > 0) and
  375. (paralen < sizeof(aint)) then
  376. begin
  377. paralen := sizeof(aint);
  378. paracgsize:=OS_INT;
  379. end
  380. else
  381. paracgsize:=def_cgsize(hp.vardef);
  382. end;
  383. hp.paraloc[side].reset;
  384. hp.paraloc[side].size:=paracgsize;
  385. hp.paraloc[side].intsize:=paralen;
  386. hp.paraloc[side].Alignment:=paraalign;
  387. { Copy to stack? }
  388. if (paracgsize=OS_NO) or
  389. (use_fixed_stack) then
  390. begin
  391. paraloc:=hp.paraloc[side].add_location;
  392. paraloc^.loc:=LOC_REFERENCE;
  393. paraloc^.size:=paracgsize;
  394. if side=callerside then
  395. paraloc^.reference.index:=NR_STACK_POINTER_REG
  396. else
  397. paraloc^.reference.index:=NR_FRAME_POINTER_REG;
  398. varalign:=used_align(size_2_align(paralen),paraalign,paraalign);
  399. { don't let push_size return 16, because then we can }
  400. { read past the end of the heap since the value is only }
  401. { 10 bytes long (JM) }
  402. if (paracgsize = OS_F80) and
  403. (target_info.system = system_i386_darwin) then
  404. paralen:=16;
  405. paraloc^.reference.offset:=parasize;
  406. if side=calleeside then
  407. inc(paraloc^.reference.offset,target_info.first_parm_offset);
  408. parasize:=align(parasize+paralen,varalign);
  409. end
  410. else
  411. begin
  412. if paralen=0 then
  413. internalerror(200501163);
  414. while (paralen>0) do
  415. begin
  416. paraloc:=hp.paraloc[side].add_location;
  417. paraloc^.loc:=LOC_REFERENCE;
  418. { single and double need a single location }
  419. if (paracgsize in [OS_F64,OS_F32]) then
  420. begin
  421. paraloc^.size:=paracgsize;
  422. l:=paralen;
  423. end
  424. else
  425. begin
  426. { We can allocate at maximum 32 bits per location }
  427. if paralen>sizeof(aint) then
  428. l:=sizeof(aint)
  429. else
  430. l:=paralen;
  431. paraloc^.size:=int_cgsize(l);
  432. end;
  433. if side=callerside then
  434. paraloc^.reference.index:=NR_STACK_POINTER_REG
  435. else
  436. paraloc^.reference.index:=NR_FRAME_POINTER_REG;
  437. varalign:=used_align(size_2_align(l),paraalign,paraalign);
  438. paraloc^.reference.offset:=parasize;
  439. if side=calleeside then
  440. inc(paraloc^.reference.offset,target_info.first_parm_offset);
  441. parasize:=align(parasize+l,varalign);
  442. dec(paralen,l);
  443. end;
  444. end;
  445. end;
  446. end;
  447. procedure ti386paramanager.create_register_paraloc_info(p : tabstractprocdef; side: tcallercallee;paras:tparalist;
  448. var parareg,parasize:longint);
  449. var
  450. hp : tparavarsym;
  451. paraloc : pcgparalocation;
  452. paracgsize : tcgsize;
  453. i : integer;
  454. l,
  455. paralen,
  456. varalign : longint;
  457. pushaddr : boolean;
  458. paraalign : shortint;
  459. pass : byte;
  460. begin
  461. if paras.count=0 then
  462. exit;
  463. paraalign:=get_para_align(p.proccalloption);
  464. { clean up here so we can later detect properly if a parameter has been
  465. assigned or not
  466. }
  467. for i:=0 to paras.count-1 do
  468. tparavarsym(paras[i]).paraloc[side].reset;
  469. { Register parameters are assigned from left to right,
  470. stack parameters from right to left so assign first the
  471. register parameters in a first pass, in the second
  472. pass all unhandled parameters are done }
  473. for pass:=1 to 2 do
  474. begin
  475. if pass=1 then
  476. i:=0
  477. else
  478. i:=paras.count-1;
  479. while true do
  480. begin
  481. hp:=tparavarsym(paras[i]);
  482. if not(assigned(hp.paraloc[side].location)) then
  483. begin
  484. pushaddr:=push_addr_param(hp.varspez,hp.vardef,p.proccalloption);
  485. if pushaddr then
  486. begin
  487. paralen:=sizeof(aint);
  488. paracgsize:=OS_ADDR;
  489. end
  490. else
  491. begin
  492. paralen:=push_size(hp.varspez,hp.vardef,p.proccalloption);
  493. paracgsize:=def_cgsize(hp.vardef);
  494. end;
  495. hp.paraloc[side].size:=paracgsize;
  496. hp.paraloc[side].intsize:=paralen;
  497. hp.paraloc[side].Alignment:=paraalign;
  498. {
  499. EAX
  500. EDX
  501. ECX
  502. Stack
  503. Stack
  504. 64bit values,floats,arrays and records are always
  505. on the stack.
  506. }
  507. if (parareg<=high(parasupregs)) and
  508. (paralen<=sizeof(aint)) and
  509. (
  510. not(hp.vardef.typ in [floatdef,recorddef,arraydef]) or
  511. pushaddr
  512. ) then
  513. begin
  514. if pass=1 then
  515. begin
  516. paraloc:=hp.paraloc[side].add_location;
  517. paraloc^.size:=paracgsize;
  518. paraloc^.loc:=LOC_REGISTER;
  519. paraloc^.register:=newreg(R_INTREGISTER,parasupregs[parareg],cgsize2subreg(paracgsize));
  520. inc(parareg);
  521. end;
  522. end
  523. else
  524. if pass=2 then
  525. begin
  526. { Copy to stack? }
  527. if (use_fixed_stack) or
  528. (paracgsize=OS_NO) then
  529. begin
  530. paraloc:=hp.paraloc[side].add_location;
  531. paraloc^.loc:=LOC_REFERENCE;
  532. paraloc^.size:=paracgsize;
  533. if side=callerside then
  534. paraloc^.reference.index:=NR_STACK_POINTER_REG
  535. else
  536. paraloc^.reference.index:=NR_FRAME_POINTER_REG;
  537. varalign:=used_align(size_2_align(paralen),paraalign,paraalign);
  538. paraloc^.reference.offset:=parasize;
  539. if side=calleeside then
  540. inc(paraloc^.reference.offset,target_info.first_parm_offset);
  541. parasize:=align(parasize+paralen,varalign);
  542. end
  543. else
  544. begin
  545. if paralen=0 then
  546. internalerror(200501163);
  547. while (paralen>0) do
  548. begin
  549. paraloc:=hp.paraloc[side].add_location;
  550. paraloc^.loc:=LOC_REFERENCE;
  551. { Extended and double need a single location }
  552. if (paracgsize in [OS_F64,OS_F32]) then
  553. begin
  554. paraloc^.size:=paracgsize;
  555. l:=paralen;
  556. end
  557. else
  558. begin
  559. { We can allocate at maximum 32 bits per location }
  560. if paralen>sizeof(aint) then
  561. l:=sizeof(aint)
  562. else
  563. l:=paralen;
  564. paraloc^.size:=int_cgsize(l);
  565. end;
  566. if side=callerside then
  567. paraloc^.reference.index:=NR_STACK_POINTER_REG
  568. else
  569. paraloc^.reference.index:=NR_FRAME_POINTER_REG;
  570. varalign:=used_align(size_2_align(l),paraalign,paraalign);
  571. paraloc^.reference.offset:=parasize;
  572. if side=calleeside then
  573. inc(paraloc^.reference.offset,target_info.first_parm_offset);
  574. parasize:=align(parasize+l,varalign);
  575. dec(paralen,l);
  576. end;
  577. end;
  578. end;
  579. end;
  580. case pass of
  581. 1:
  582. begin
  583. if i=paras.count-1 then
  584. break;
  585. inc(i);
  586. end;
  587. 2:
  588. begin
  589. if i=0 then
  590. break;
  591. dec(i);
  592. end;
  593. end;
  594. end;
  595. end;
  596. end;
  597. function ti386paramanager.create_paraloc_info(p : tabstractprocdef; side: tcallercallee):longint;
  598. var
  599. parasize,
  600. parareg : longint;
  601. begin
  602. parasize:=0;
  603. parareg:=0;
  604. case p.proccalloption of
  605. pocall_register :
  606. create_register_paraloc_info(p,side,p.paras,parareg,parasize);
  607. pocall_internproc :
  608. begin
  609. { Use default calling }
  610. if (pocall_default=pocall_register) then
  611. create_register_paraloc_info(p,side,p.paras,parareg,parasize)
  612. else
  613. create_stdcall_paraloc_info(p,side,p.paras,parasize);
  614. end;
  615. else
  616. create_stdcall_paraloc_info(p,side,p.paras,parasize);
  617. end;
  618. create_funcretloc_info(p,side);
  619. result:=parasize;
  620. end;
  621. function ti386paramanager.create_varargs_paraloc_info(p : tabstractprocdef; varargspara:tvarargsparalist):longint;
  622. var
  623. parasize : longint;
  624. begin
  625. parasize:=0;
  626. { calculate the registers for the normal parameters }
  627. create_stdcall_paraloc_info(p,callerside,p.paras,parasize);
  628. { append the varargs }
  629. create_stdcall_paraloc_info(p,callerside,varargspara,parasize);
  630. result:=parasize;
  631. end;
  632. procedure ti386paramanager.createtempparaloc(list: TAsmList;calloption : tproccalloption;parasym : tparavarsym;var cgpara:TCGPara);
  633. var
  634. paraloc : pcgparalocation;
  635. begin
  636. paraloc:=parasym.paraloc[callerside].location;
  637. { No need for temps when value is pushed }
  638. if not(use_fixed_stack) and
  639. assigned(paraloc) and
  640. (paraloc^.loc=LOC_REFERENCE) and
  641. (paraloc^.reference.index=NR_STACK_POINTER_REG) then
  642. duplicateparaloc(list,calloption,parasym,cgpara)
  643. else
  644. inherited createtempparaloc(list,calloption,parasym,cgpara);
  645. end;
  646. begin
  647. paramanager:=ti386paramanager.create;
  648. end.