cg386ld.pas 54 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312
  1. {
  2. $Id$
  3. Copyright (c) 1993-98 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)^.consttype=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. if gettempansistringreference(hr) then
  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)^.definition) or
  233. is_array_of_const(pvarsym(p^.symtableentry)^.definition) or
  234. ((pvarsym(p^.symtableentry)^.varspez=vs_const) and
  235. push_addr_param(pvarsym(p^.symtableentry)^.definition)) 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. hregister:=R_EDI;
  278. if pobjectdef(p^.left^.resulttype)^.is_class then
  279. emit_ref_reg(A_MOV,S_L,
  280. newreference(p^.left^.location.reference),R_EDI)
  281. else
  282. emit_ref_reg(A_LEA,S_L,
  283. newreference(p^.left^.location.reference),R_EDI);
  284. del_reference(p^.left^.location.reference);
  285. ungetiftemp(p^.left^.location.reference);
  286. end;
  287. else internalerror(26019);
  288. end;
  289. { store the class instance address }
  290. new(hp);
  291. hp^:=p^.location.reference;
  292. inc(hp^.offset,4);
  293. emit_reg_ref(A_MOV,S_L,
  294. hregister,hp);
  295. { virtual method ? }
  296. if (po_virtualmethod in pprocsym(p^.symtableentry)^.definition^.procoptions) then
  297. begin
  298. new(hp);
  299. reset_reference(hp^);
  300. hp^.base:=hregister;
  301. { load vmt pointer }
  302. emit_ref_reg(A_MOV,S_L,
  303. hp,R_EDI);
  304. {$IfDef regallocfix}
  305. del_reference(hp^);
  306. {$EndIf regallocfix}
  307. { load method address }
  308. new(hp);
  309. reset_reference(hp^);
  310. hp^.base:=R_EDI;
  311. hp^.offset:=pprocsym(p^.symtableentry)^.definition^._class^.vmtmethodoffset(
  312. pprocsym(p^.symtableentry)^.definition^.extnumber);
  313. emit_ref_reg(A_MOV,S_L,
  314. hp,R_EDI);
  315. { ... and store it }
  316. emit_reg_ref(A_MOV,S_L,
  317. R_EDI,newreference(p^.location.reference));
  318. end
  319. else
  320. begin
  321. s:=newasmsymbol(pprocsym(p^.symtableentry)^.definition^.mangledname);
  322. emit_sym_ofs_ref(A_MOV,S_L,s,0,
  323. newreference(p^.location.reference));
  324. end;
  325. end
  326. else
  327. begin
  328. {!!!!! Be aware, work on virtual methods too }
  329. p^.location.reference.symbol:=newasmsymbol(pprocsym(p^.symtableentry)^.definition^.mangledname);
  330. end;
  331. end;
  332. typedconstsym :
  333. begin
  334. p^.location.reference.symbol:=newasmsymbol(p^.symtableentry^.mangledname);
  335. end;
  336. else internalerror(4);
  337. end;
  338. end;
  339. {*****************************************************************************
  340. SecondAssignment
  341. *****************************************************************************}
  342. procedure secondassignment(var p : ptree);
  343. var
  344. opsize : topsize;
  345. otlabel,hlabel,oflabel : pasmlabel;
  346. fputyp : tfloattype;
  347. loc : tloc;
  348. r : preference;
  349. ai : paicpu;
  350. op : tasmop;
  351. pushed : boolean;
  352. begin
  353. otlabel:=truelabel;
  354. oflabel:=falselabel;
  355. getlabel(truelabel);
  356. getlabel(falselabel);
  357. { calculate left sides }
  358. if not(p^.concat_string) then
  359. secondpass(p^.left);
  360. if codegenerror then
  361. exit;
  362. {$ifdef dummy}
  363. { we use now the standard mechanism via maybe_push/restore
  364. to do that (FK)
  365. }
  366. case p^.left^.location.loc of
  367. LOC_REFERENCE : begin
  368. { in case left operator uses to register }
  369. { but to few are free then LEA }
  370. if (p^.left^.location.reference.base<>R_NO) and
  371. (p^.left^.location.reference.index<>R_NO) and
  372. (usablereg32<p^.right^.registers32) then
  373. begin
  374. del_reference(p^.left^.location.reference);
  375. hregister:=getregister32;
  376. emit_ref_reg(A_LEA,S_L,newreference(
  377. p^.left^.location.reference),
  378. hregister);
  379. reset_reference(p^.left^.location.reference);
  380. p^.left^.location.reference.base:=hregister;
  381. p^.left^.location.reference.index:=R_NO;
  382. end;
  383. loc:=LOC_REFERENCE;
  384. end;
  385. LOC_CFPUREGISTER:
  386. loc:=LOC_CFPUREGISTER;
  387. LOC_CREGISTER:
  388. loc:=LOC_CREGISTER;
  389. LOC_MMXREGISTER:
  390. loc:=LOC_MMXREGISTER;
  391. LOC_CMMXREGISTER:
  392. loc:=LOC_CMMXREGISTER;
  393. else
  394. begin
  395. CGMessage(cg_e_illegal_expression);
  396. exit;
  397. end;
  398. end;
  399. {$endif dummy}
  400. loc:=p^.left^.location.loc;
  401. { lets try to optimize this (PM) }
  402. { define a dest_loc that is the location }
  403. { and a ptree to verify that it is the right }
  404. { place to insert it }
  405. {$ifdef test_dest_loc}
  406. if (aktexprlevel<4) then
  407. begin
  408. dest_loc_known:=true;
  409. dest_loc:=p^.left^.location;
  410. dest_loc_tree:=p^.right;
  411. end;
  412. {$endif test_dest_loc}
  413. { left can't be never a 64 bit LOC_REGISTER, so the 3. arg }
  414. { can be false }
  415. pushed:=maybe_push(p^.right^.registers32,p^.left,false);
  416. secondpass(p^.right);
  417. if pushed then restore(p^.left,false);
  418. if codegenerror then
  419. exit;
  420. {$ifdef test_dest_loc}
  421. dest_loc_known:=false;
  422. if in_dest_loc then
  423. begin
  424. truelabel:=otlabel;
  425. falselabel:=oflabel;
  426. in_dest_loc:=false;
  427. exit;
  428. end;
  429. {$endif test_dest_loc}
  430. if p^.left^.resulttype^.deftype=stringdef then
  431. begin
  432. if is_ansistring(p^.left^.resulttype) then
  433. begin
  434. { the source and destinations are released
  435. in loadansistring, because an ansi string can
  436. also be in a register
  437. }
  438. loadansistring(p);
  439. end
  440. else
  441. if is_shortstring(p^.left^.resulttype) and
  442. not (p^.concat_string) then
  443. begin
  444. if is_ansistring(p^.right^.resulttype) then
  445. begin
  446. if (p^.right^.treetype=stringconstn) and
  447. (p^.right^.length=0) then
  448. begin
  449. emit_const_ref(A_MOV,S_B,
  450. 0,newreference(p^.left^.location.reference));
  451. {$IfDef regallocfix}
  452. del_reference(p^.left^.location.reference);
  453. {$EndIf regallocfix}
  454. end
  455. else
  456. loadansi2short(p^.right,p^.left);
  457. end
  458. else
  459. begin
  460. { we do not need destination anymore }
  461. del_reference(p^.left^.location.reference);
  462. del_reference(p^.right^.location.reference);
  463. loadshortstring(p);
  464. ungetiftemp(p^.right^.location.reference);
  465. end;
  466. end
  467. else if is_longstring(p^.left^.resulttype) then
  468. begin
  469. end
  470. else
  471. begin
  472. { its the only thing we have to do }
  473. del_reference(p^.right^.location.reference);
  474. end
  475. end
  476. else case p^.right^.location.loc of
  477. LOC_REFERENCE,
  478. LOC_MEM : begin
  479. { extra handling for ordinal constants }
  480. if (p^.right^.treetype in [ordconstn,fixconstn]) or
  481. (loc=LOC_CREGISTER) then
  482. begin
  483. case p^.left^.resulttype^.size of
  484. 1 : opsize:=S_B;
  485. 2 : opsize:=S_W;
  486. 4 : opsize:=S_L;
  487. { S_L is correct, the copy is done }
  488. { with two moves }
  489. 8 : opsize:=S_L;
  490. end;
  491. if loc=LOC_CREGISTER then
  492. begin
  493. emit_ref_reg(A_MOV,opsize,
  494. newreference(p^.right^.location.reference),
  495. p^.left^.location.register);
  496. if is_64bitint(p^.right^.resulttype) then
  497. begin
  498. r:=newreference(p^.right^.location.reference);
  499. inc(r^.offset,4);
  500. emit_ref_reg(A_MOV,opsize,r,
  501. p^.left^.location.registerhigh);
  502. end;
  503. {$IfDef regallocfix}
  504. del_reference(p^.right^.location.reference);
  505. {$EndIf regallocfix}
  506. end
  507. else
  508. begin
  509. emit_const_ref(A_MOV,opsize,
  510. p^.right^.location.reference.offset,
  511. newreference(p^.left^.location.reference));
  512. if is_64bitint(p^.right^.resulttype) then
  513. begin
  514. r:=newreference(p^.left^.location.reference);
  515. inc(r^.offset,4);
  516. emit_const_ref(A_MOV,opsize,
  517. 0,r);
  518. end;
  519. {$IfDef regallocfix}
  520. del_reference(p^.left^.location.reference);
  521. {$EndIf regallocfix}
  522. {emit_const_loc(A_MOV,opsize,
  523. p^.right^.location.reference.offset,
  524. p^.left^.location);}
  525. end;
  526. end
  527. else if loc=LOC_CFPUREGISTER then
  528. begin
  529. floatloadops(pfloatdef(p^.right^.resulttype)^.typ,op,opsize);
  530. emit_ref(op,opsize,
  531. newreference(p^.right^.location.reference));
  532. emit_reg(A_FSTP,S_NO,
  533. correct_fpuregister(p^.left^.location.register,fpuvaroffset+1));
  534. end
  535. else
  536. begin
  537. if (p^.right^.resulttype^.needs_inittable) and
  538. ( (p^.right^.resulttype^.deftype<>objectdef) or
  539. not(pobjectdef(p^.right^.resulttype)^.is_class)) then
  540. begin
  541. { this would be a problem }
  542. if not(p^.left^.resulttype^.needs_inittable) then
  543. internalerror(3457);
  544. { increment source reference counter }
  545. new(r);
  546. reset_reference(r^);
  547. r^.symbol:=p^.right^.resulttype^.get_inittable_label;
  548. emitpushreferenceaddr(r^);
  549. emitpushreferenceaddr(p^.right^.location.reference);
  550. emitcall('FPC_ADDREF');
  551. { decrement destination reference counter }
  552. new(r);
  553. reset_reference(r^);
  554. r^.symbol:=p^.left^.resulttype^.get_inittable_label;
  555. emitpushreferenceaddr(r^);
  556. emitpushreferenceaddr(p^.left^.location.reference);
  557. emitcall('FPC_DECREF');
  558. end;
  559. {$ifdef regallocfix}
  560. concatcopy(p^.right^.location.reference,
  561. p^.left^.location.reference,p^.left^.resulttype^.size,true,false);
  562. ungetiftemp(p^.right^.location.reference);
  563. {$Else regallocfix}
  564. concatcopy(p^.right^.location.reference,
  565. p^.left^.location.reference,p^.left^.resulttype^.size,false,false);
  566. ungetiftemp(p^.right^.location.reference);
  567. {$endif regallocfix}
  568. end;
  569. end;
  570. {$ifdef SUPPORT_MMX}
  571. LOC_CMMXREGISTER,
  572. LOC_MMXREGISTER:
  573. begin
  574. if loc=LOC_CMMXREGISTER then
  575. emit_reg_reg(A_MOVQ,S_NO,
  576. p^.right^.location.register,p^.left^.location.register)
  577. else
  578. emit_reg_ref(A_MOVQ,S_NO,
  579. p^.right^.location.register,newreference(p^.left^.location.reference));
  580. end;
  581. {$endif SUPPORT_MMX}
  582. LOC_REGISTER,
  583. LOC_CREGISTER : begin
  584. case p^.right^.resulttype^.size of
  585. 1 : opsize:=S_B;
  586. 2 : opsize:=S_W;
  587. 4 : opsize:=S_L;
  588. 8 : opsize:=S_L;
  589. end;
  590. { simplified with op_reg_loc }
  591. if loc=LOC_CREGISTER then
  592. begin
  593. emit_reg_reg(A_MOV,opsize,
  594. p^.right^.location.register,
  595. p^.left^.location.register);
  596. {$IfDef regallocfix}
  597. ungetregister(p^.right^.location.register);
  598. {$EndIf regallocfix}
  599. end
  600. else
  601. Begin
  602. emit_reg_ref(A_MOV,opsize,
  603. p^.right^.location.register,
  604. newreference(p^.left^.location.reference));
  605. {$IfDef regallocfix}
  606. ungetregister(p^.right^.location.register);
  607. del_reference(p^.left^.location.reference);
  608. {$EndIf regallocfix}
  609. end;
  610. if is_64bitint(p^.right^.resulttype) then
  611. begin
  612. { simplified with op_reg_loc }
  613. if loc=LOC_CREGISTER then
  614. emit_reg_reg(A_MOV,opsize,
  615. p^.right^.location.registerhigh,
  616. p^.left^.location.registerhigh)
  617. else
  618. begin
  619. r:=newreference(p^.left^.location.reference);
  620. inc(r^.offset,4);
  621. emit_reg_ref(A_MOV,opsize,
  622. p^.right^.location.registerhigh,r);
  623. end;
  624. end;
  625. {emit_reg_loc(A_MOV,opsize,
  626. p^.right^.location.register,
  627. p^.left^.location); }
  628. end;
  629. LOC_FPU : begin
  630. if (p^.left^.resulttype^.deftype=floatdef) then
  631. fputyp:=pfloatdef(p^.left^.resulttype)^.typ
  632. else
  633. if (p^.right^.resulttype^.deftype=floatdef) then
  634. fputyp:=pfloatdef(p^.right^.resulttype)^.typ
  635. else
  636. if (p^.right^.treetype=typeconvn) and
  637. (p^.right^.left^.resulttype^.deftype=floatdef) then
  638. fputyp:=pfloatdef(p^.right^.left^.resulttype)^.typ
  639. else
  640. fputyp:=s32real;
  641. case loc of
  642. LOC_CFPUREGISTER:
  643. begin
  644. emit_reg(A_FSTP,S_NO,
  645. correct_fpuregister(p^.left^.location.register,fpuvaroffset));
  646. dec(fpuvaroffset);
  647. end;
  648. LOC_REFERENCE:
  649. floatstore(fputyp,p^.left^.location.reference);
  650. else
  651. internalerror(48991);
  652. end;
  653. end;
  654. LOC_CFPUREGISTER: begin
  655. if (p^.left^.resulttype^.deftype=floatdef) then
  656. fputyp:=pfloatdef(p^.left^.resulttype)^.typ
  657. else
  658. if (p^.right^.resulttype^.deftype=floatdef) then
  659. fputyp:=pfloatdef(p^.right^.resulttype)^.typ
  660. else
  661. if (p^.right^.treetype=typeconvn) and
  662. (p^.right^.left^.resulttype^.deftype=floatdef) then
  663. fputyp:=pfloatdef(p^.right^.left^.resulttype)^.typ
  664. else
  665. fputyp:=s32real;
  666. emit_reg(A_FLD,S_NO,
  667. correct_fpuregister(p^.right^.location.register,fpuvaroffset));
  668. inc(fpuvaroffset);
  669. case loc of
  670. LOC_CFPUREGISTER:
  671. begin
  672. emit_reg(A_FSTP,S_NO,
  673. correct_fpuregister(p^.right^.location.register,fpuvaroffset));
  674. dec(fpuvaroffset);
  675. end;
  676. LOC_REFERENCE:
  677. floatstore(fputyp,p^.left^.location.reference);
  678. else
  679. internalerror(48992);
  680. end;
  681. end;
  682. LOC_JUMP : begin
  683. getlabel(hlabel);
  684. emitlab(truelabel);
  685. if loc=LOC_CREGISTER then
  686. emit_const_reg(A_MOV,S_B,
  687. 1,p^.left^.location.register)
  688. else
  689. emit_const_ref(A_MOV,S_B,
  690. 1,newreference(p^.left^.location.reference));
  691. {emit_const_loc(A_MOV,S_B,
  692. 1,p^.left^.location);}
  693. emitjmp(C_None,hlabel);
  694. emitlab(falselabel);
  695. if loc=LOC_CREGISTER then
  696. emit_reg_reg(A_XOR,S_B,
  697. p^.left^.location.register,
  698. p^.left^.location.register)
  699. else
  700. begin
  701. emit_const_ref(A_MOV,S_B,
  702. 0,newreference(p^.left^.location.reference));
  703. {$IfDef regallocfix}
  704. del_reference(p^.left^.location.reference);
  705. {$EndIf regallocfix}
  706. end;
  707. emitlab(hlabel);
  708. end;
  709. LOC_FLAGS : begin
  710. if loc=LOC_CREGISTER then
  711. emit_flag2reg(p^.right^.location.resflags,p^.left^.location.register)
  712. else
  713. begin
  714. ai:=new(paicpu,op_ref(A_Setcc,S_B,newreference(p^.left^.location.reference)));
  715. ai^.SetCondition(flag_2_cond[p^.right^.location.resflags]);
  716. exprasmlist^.concat(ai);
  717. end;
  718. {$IfDef regallocfix}
  719. del_reference(p^.left^.location.reference);
  720. {$EndIf regallocfix}
  721. end;
  722. end;
  723. freelabel(truelabel);
  724. freelabel(falselabel);
  725. truelabel:=otlabel;
  726. falselabel:=oflabel;
  727. end;
  728. {*****************************************************************************
  729. SecondFuncRet
  730. *****************************************************************************}
  731. procedure secondfuncret(var p : ptree);
  732. var
  733. hr : tregister;
  734. hp : preference;
  735. pp : pprocinfo;
  736. hr_valid : boolean;
  737. begin
  738. reset_reference(p^.location.reference);
  739. hr_valid:=false;
  740. if @procinfo<>pprocinfo(p^.funcretprocinfo) then
  741. begin
  742. hr:=getregister32;
  743. hr_valid:=true;
  744. hp:=new_reference(procinfo.framepointer,
  745. procinfo.framepointer_offset);
  746. emit_ref_reg(A_MOV,S_L,hp,hr);
  747. pp:=procinfo.parent;
  748. { walk up the stack frame }
  749. while pp<>pprocinfo(p^.funcretprocinfo) do
  750. begin
  751. hp:=new_reference(hr,
  752. pp^.framepointer_offset);
  753. emit_ref_reg(A_MOV,S_L,hp,hr);
  754. pp:=pp^.parent;
  755. end;
  756. p^.location.reference.base:=hr;
  757. end
  758. else
  759. p^.location.reference.base:=procinfo.framepointer;
  760. p^.location.reference.offset:=procinfo.retoffset;
  761. if ret_in_param(p^.retdef) then
  762. begin
  763. if not hr_valid then
  764. hr:=getregister32;
  765. emit_ref_reg(A_MOV,S_L,newreference(p^.location.reference),hr);
  766. p^.location.reference.base:=hr;
  767. p^.location.reference.offset:=0;
  768. end;
  769. end;
  770. {*****************************************************************************
  771. SecondArrayConstruct
  772. *****************************************************************************}
  773. const
  774. vtInteger = 0;
  775. vtBoolean = 1;
  776. vtChar = 2;
  777. vtExtended = 3;
  778. vtString = 4;
  779. vtPointer = 5;
  780. vtPChar = 6;
  781. vtObject = 7;
  782. vtClass = 8;
  783. vtWideChar = 9;
  784. vtPWideChar = 10;
  785. vtAnsiString = 11;
  786. vtCurrency = 12;
  787. vtVariant = 13;
  788. vtInterface = 14;
  789. vtWideString = 15;
  790. vtInt64 = 16;
  791. procedure secondarrayconstruct(var p : ptree);
  792. var
  793. hp : ptree;
  794. href : treference;
  795. lt : pdef;
  796. vaddr : boolean;
  797. vtype : longint;
  798. freetemp,
  799. dovariant : boolean;
  800. elesize : longint;
  801. begin
  802. dovariant:=p^.forcevaria or parraydef(p^.resulttype)^.isvariant;
  803. if dovariant then
  804. elesize:=8
  805. else
  806. begin
  807. elesize:=parraydef(p^.resulttype)^.elesize;
  808. if elesize>4 then
  809. internalerror(8765678);
  810. end;
  811. if not p^.cargs then
  812. begin
  813. reset_reference(p^.location.reference);
  814. { Allocate always a temp, also if no elements are required, to
  815. be sure that location is valid (PFV) }
  816. if parraydef(p^.resulttype)^.highrange=-1 then
  817. gettempofsizereference(elesize,p^.location.reference)
  818. else
  819. gettempofsizereference((parraydef(p^.resulttype)^.highrange+1)*elesize,p^.location.reference);
  820. href:=p^.location.reference;
  821. end;
  822. hp:=p;
  823. while assigned(hp) do
  824. begin
  825. if assigned(hp^.left) then
  826. begin
  827. freetemp:=true;
  828. secondpass(hp^.left);
  829. if codegenerror then
  830. exit;
  831. if dovariant then
  832. begin
  833. { find the correct vtype value }
  834. vtype:=$ff;
  835. vaddr:=false;
  836. lt:=hp^.left^.resulttype;
  837. case lt^.deftype of
  838. enumdef,
  839. orddef :
  840. begin
  841. if (lt^.deftype=enumdef) or
  842. is_integer(lt) then
  843. vtype:=vtInteger
  844. else
  845. if is_boolean(lt) then
  846. vtype:=vtBoolean
  847. else
  848. if (lt^.deftype=orddef) and (porddef(lt)^.typ=uchar) then
  849. vtype:=vtChar;
  850. end;
  851. floatdef :
  852. begin
  853. vtype:=vtExtended;
  854. vaddr:=true;
  855. end;
  856. procvardef,
  857. pointerdef :
  858. begin
  859. if is_pchar(lt) then
  860. vtype:=vtPChar
  861. else
  862. vtype:=vtPointer;
  863. end;
  864. classrefdef :
  865. vtype:=vtClass;
  866. objectdef :
  867. begin
  868. vtype:=vtObject;
  869. end;
  870. stringdef :
  871. begin
  872. if is_shortstring(lt) then
  873. begin
  874. vtype:=vtString;
  875. vaddr:=true;
  876. freetemp:=false;
  877. end
  878. else
  879. if is_ansistring(lt) then
  880. vtype:=vtAnsiString;
  881. end;
  882. end;
  883. if vtype=$ff then
  884. internalerror(14357);
  885. { write C style pushes or an pascal array }
  886. if p^.cargs then
  887. begin
  888. if vaddr then
  889. begin
  890. emit_to_reference(hp^.left);
  891. emit_push_lea_loc(hp^.left^.location,freetemp);
  892. end
  893. else
  894. emit_push_loc(hp^.left^.location);
  895. end
  896. else
  897. begin
  898. { update href to the vtype field and write it }
  899. emit_const_ref(A_MOV,S_L,
  900. vtype,newreference(href));
  901. inc(href.offset,4);
  902. { write changing field update href to the next element }
  903. if vaddr then
  904. begin
  905. emit_to_reference(hp^.left);
  906. emit_lea_loc_ref(hp^.left^.location,href,freetemp);
  907. end
  908. else
  909. emit_mov_loc_ref(hp^.left^.location,href,S_L);
  910. inc(href.offset,4);
  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);
  919. 2 :
  920. emit_mov_loc_ref(hp^.left^.location,href,S_W);
  921. 4 :
  922. emit_mov_loc_ref(hp^.left^.location,href,S_L);
  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.84 1999-09-11 09:08:31 florian
  937. * fixed bug 596
  938. * fixed some problems with procedure variables and procedures of object,
  939. especially in TP mode. Procedure of object doesn't apply only to classes,
  940. it is also allowed for objects !!
  941. Revision 1.83 1999/09/01 09:37:14 peter
  942. * removed warning
  943. Revision 1.82 1999/09/01 09:26:21 peter
  944. * fixed temp allocation for arrayconstructor
  945. Revision 1.81 1999/08/28 15:34:17 florian
  946. * bug 519 fixed
  947. Revision 1.80 1999/08/26 20:24:37 michael
  948. + Hopefuly last fixes for resourcestrings
  949. Revision 1.79 1999/08/25 16:41:05 peter
  950. * resources are working again
  951. Revision 1.78 1999/08/25 11:59:43 jonas
  952. * changed pai386, paippc and paiapha (same for tai*) to paicpu (taicpu)
  953. Revision 1.77 1999/08/24 22:38:51 michael
  954. * more resourcestring changes
  955. Revision 1.76 1999/08/23 11:45:39 michael
  956. * Hopefully final attempt at resourcestrings
  957. Revision 1.75 1999/08/19 13:08:49 pierre
  958. * emit_??? used
  959. Revision 1.74 1999/08/17 13:26:06 peter
  960. * arrayconstructor -> arrayofconst fixed when arraycosntructor was not
  961. variant.
  962. Revision 1.73 1999/08/13 21:33:09 peter
  963. * support for array constructors extended and more error checking
  964. Revision 1.72 1999/08/09 22:19:50 peter
  965. * classes vmt changed to only positive addresses
  966. * sharedlib creation is working
  967. Revision 1.71 1999/08/07 14:20:55 florian
  968. * some small problems fixed
  969. Revision 1.70 1999/08/04 13:45:22 florian
  970. + floating point register variables !!
  971. * pairegalloc is now generated for register variables
  972. Revision 1.69 1999/08/04 00:22:50 florian
  973. * renamed i386asm and i386base to cpuasm and cpubase
  974. Revision 1.68 1999/08/03 22:02:43 peter
  975. * moved bitmask constants to sets
  976. * some other type/const renamings
  977. Revision 1.67 1999/07/27 23:36:36 peter
  978. * try to determine the fpu type needed in assignment
  979. Revision 1.66 1999/07/24 15:12:56 michael
  980. changes for resourcestrings
  981. Revision 1.65 1999/07/23 23:09:06 peter
  982. * resourcestring fix
  983. Revision 1.64 1999/07/22 09:37:37 florian
  984. + resourcestring implemented
  985. + start of longstring support
  986. Revision 1.63 1999/07/05 20:13:12 peter
  987. * removed temp defines
  988. Revision 1.62 1999/06/30 15:43:18 florian
  989. * two bugs regarding method variables fixed
  990. - if you take in a method the address of another method
  991. don't need self anymore
  992. - if the class pointer was in a register, wrong code for a method
  993. variable load was generated
  994. Revision 1.61 1999/06/28 22:29:11 florian
  995. * qword division fixed
  996. + code for qword/int64 type casting added:
  997. range checking isn't implemented yet
  998. Revision 1.60 1999/05/31 12:42:43 peter
  999. * fixed crash with empty array constructor
  1000. Revision 1.59 1999/05/27 19:44:14 peter
  1001. * removed oldasm
  1002. * plabel -> pasmlabel
  1003. * -a switches to source writing automaticly
  1004. * assembler readers OOPed
  1005. * asmsymbol automaticly external
  1006. * jumptables and other label fixes for asm readers
  1007. Revision 1.58 1999/05/23 18:42:02 florian
  1008. * better error recovering in typed constants
  1009. * some problems with arrays of const fixed, some problems
  1010. due my previous
  1011. - the location type of array constructor is now LOC_MEM
  1012. - the pushing of high fixed
  1013. - parameter copying fixed
  1014. - zero temp. allocation removed
  1015. * small problem in the assembler writers fixed:
  1016. ref to nil wasn't written correctly
  1017. Revision 1.57 1999/05/21 13:54:51 peter
  1018. * NEWLAB for label as symbol
  1019. Revision 1.56 1999/05/17 23:51:38 peter
  1020. * with temp vars now use a reference with a persistant temp instead
  1021. of setting datasize
  1022. Revision 1.55 1999/05/17 21:57:04 florian
  1023. * new temporary ansistring handling
  1024. Revision 1.54 1999/05/12 00:19:43 peter
  1025. * removed R_DEFAULT_SEG
  1026. * uniform float names
  1027. Revision 1.53 1999/05/06 09:05:16 peter
  1028. * generic write_float and str_float
  1029. * fixed constant float conversions
  1030. Revision 1.52 1999/05/01 13:24:10 peter
  1031. * merged nasm compiler
  1032. * old asm moved to oldasm/
  1033. Revision 1.51 1999/04/28 06:01:55 florian
  1034. * changes of Bruessel:
  1035. + message handler can now take an explicit self
  1036. * typinfo fixed: sometimes the type names weren't written
  1037. * the type checking for pointer comparisations and subtraction
  1038. and are now more strict (was also buggy)
  1039. * small bug fix to link.pas to support compiling on another
  1040. drive
  1041. * probable bug in popt386 fixed: call/jmp => push/jmp
  1042. transformation didn't count correctly the jmp references
  1043. + threadvar support
  1044. * warning if ln/sqrt gets an invalid constant argument
  1045. Revision 1.50 1999/04/16 13:42:26 jonas
  1046. * more regalloc fixes (still not complete)
  1047. Revision 1.49 1999/04/13 18:57:48 florian
  1048. * classes which contain ansistring get unnecessary calls
  1049. to addref/decref when they are assigned, fixed
  1050. Revision 1.48 1999/04/09 15:48:47 jonas
  1051. * added fix for missing register deallocation (-dregallocfix)
  1052. Revision 1.47 1999/03/31 13:55:07 peter
  1053. * assembler inlining working for ag386bin
  1054. Revision 1.46 1999/03/24 23:16:52 peter
  1055. * fixed bugs 212,222,225,227,229,231,233
  1056. Revision 1.45 1999/02/25 21:02:28 peter
  1057. * ag386bin updates
  1058. + coff writer
  1059. Revision 1.44 1999/02/22 02:15:12 peter
  1060. * updates for ag386bin
  1061. Revision 1.43 1999/01/27 00:13:54 florian
  1062. * "procedure of object"-stuff fixed
  1063. Revision 1.42 1999/01/21 22:10:40 peter
  1064. * fixed array of const
  1065. * generic platform independent high() support
  1066. Revision 1.41 1999/01/20 10:20:18 peter
  1067. * don't make localvar copies for assembler procedures
  1068. Revision 1.40 1998/12/30 13:41:07 peter
  1069. * released valuepara
  1070. Revision 1.39 1998/12/19 00:23:45 florian
  1071. * ansistring memory leaks fixed
  1072. Revision 1.38 1998/12/11 00:02:51 peter
  1073. + globtype,tokens,version unit splitted from globals
  1074. Revision 1.37 1998/12/10 09:47:17 florian
  1075. + basic operations with int64/qord (compiler with -dint64)
  1076. + rtti of enumerations extended: names are now written
  1077. Revision 1.36 1998/12/04 10:18:06 florian
  1078. * some stuff for procedures of object added
  1079. * bug with overridden virtual constructors fixed (reported by Italo Gomes)
  1080. Revision 1.35 1998/11/30 09:43:04 pierre
  1081. * some range check bugs fixed (still not working !)
  1082. + added DLL writing support for win32 (also accepts variables)
  1083. + TempAnsi for code that could be used for Temporary ansi strings
  1084. handling
  1085. Revision 1.34 1998/11/28 16:20:48 peter
  1086. + support for dll variables
  1087. Revision 1.33 1998/11/27 14:50:33 peter
  1088. + open strings, $P switch support
  1089. Revision 1.32 1998/11/26 09:53:36 florian
  1090. * for classes no init/final. code is necessary, fixed
  1091. Revision 1.31 1998/11/20 15:35:54 florian
  1092. * problems with rtti fixed, hope it works
  1093. Revision 1.30 1998/11/18 17:45:24 peter
  1094. * fixes for VALUEPARA
  1095. Revision 1.29 1998/11/18 15:44:11 peter
  1096. * VALUEPARA for tp7 compatible value parameters
  1097. Revision 1.28 1998/11/17 11:32:44 peter
  1098. * optimize str:='' in H+ mode
  1099. + -! to test ansistrings
  1100. Revision 1.27 1998/11/16 15:35:39 peter
  1101. * rename laod/copystring -> load/copyshortstring
  1102. * fixed int-bool cnv bug
  1103. + char-ansistring conversion
  1104. Revision 1.26 1998/11/10 10:09:10 peter
  1105. * va_list -> array of const
  1106. Revision 1.25 1998/11/05 12:02:35 peter
  1107. * released useansistring
  1108. * removed -Sv, its now available in fpc modes
  1109. Revision 1.24 1998/10/14 08:47:14 pierre
  1110. * bugs in secondfuncret for result in subprocedures removed
  1111. Revision 1.23 1998/10/06 17:16:44 pierre
  1112. * some memory leaks fixed (thanks to Peter for heaptrc !)
  1113. Revision 1.22 1998/10/01 09:22:53 peter
  1114. * fixed value openarray
  1115. * ungettemp of arrayconstruct
  1116. Revision 1.21 1998/09/28 11:07:39 peter
  1117. + floatdef support for array of const
  1118. Revision 1.20 1998/09/24 14:26:03 peter
  1119. * updated for new tvarrec
  1120. Revision 1.19 1998/09/23 17:49:59 peter
  1121. * high(arrayconstructor) is now correct
  1122. * procvardef support for variant record
  1123. Revision 1.18 1998/09/23 09:58:48 peter
  1124. * first working array of const things
  1125. Revision 1.17 1998/09/20 18:00:19 florian
  1126. * small compiling problems fixed
  1127. Revision 1.16 1998/09/20 17:46:48 florian
  1128. * some things regarding ansistrings fixed
  1129. Revision 1.15 1998/09/17 09:42:16 peter
  1130. + pass_2 for cg386
  1131. * Message() -> CGMessage() for pass_1/pass_2
  1132. Revision 1.14 1998/09/14 10:43:50 peter
  1133. * all internal RTL functions start with FPC_
  1134. Revision 1.13 1998/09/04 12:24:24 florian
  1135. * bug0159 fixed
  1136. Revision 1.12 1998/09/04 11:55:17 florian
  1137. * problem with -Or fixed
  1138. Revision 1.11 1998/09/03 16:03:14 florian
  1139. + rtti generation
  1140. * init table generation changed
  1141. Revision 1.10 1998/08/21 14:08:40 pierre
  1142. + TEST_FUNCRET now default (old code removed)
  1143. works also for m68k (at least compiles)
  1144. Revision 1.9 1998/08/20 09:26:37 pierre
  1145. + funcret setting in underproc testing
  1146. compile with _dTEST_FUNCRET
  1147. Revision 1.8 1998/08/10 14:49:48 peter
  1148. + localswitches, moduleswitches, globalswitches splitting
  1149. Revision 1.7 1998/07/30 13:30:33 florian
  1150. * final implemenation of exception support, maybe it needs
  1151. some fixes :)
  1152. Revision 1.6 1998/07/26 21:58:57 florian
  1153. + better support for switch $H
  1154. + index access to ansi strings added
  1155. + assigment of data (records/arrays) containing ansi strings
  1156. Revision 1.5 1998/07/24 22:16:54 florian
  1157. * internal error 10 together with array access fixed. I hope
  1158. that's the final fix.
  1159. Revision 1.4 1998/06/11 13:58:45 peter
  1160. * fixed too long line
  1161. Revision 1.3 1998/06/09 16:01:35 pierre
  1162. + added procedure directive parsing for procvars
  1163. (accepted are popstack cdecl and pascal)
  1164. + added C vars with the following syntax
  1165. var C calias 'true_c_name';(can be followed by external)
  1166. reason is that you must add the Cprefix
  1167. which is target dependent
  1168. Revision 1.2 1998/06/08 13:13:34 pierre
  1169. + temporary variables now in temp_gen.pas unit
  1170. because it is processor independent
  1171. * mppc68k.bat modified to undefine i386 and support_mmx
  1172. (which are defaults for i386)
  1173. Revision 1.1 1998/06/05 17:44:12 peter
  1174. * splitted cgi386
  1175. }