symcpu.pas 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678
  1. {
  2. Copyright (c) 2014 by Florian Klaempfl
  3. Symbol table overrides for i8086
  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 by
  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 symcpu;
  18. {$i fpcdefs.inc}
  19. interface
  20. uses
  21. globtype,
  22. symconst,symtype,symdef,symsym,symx86,symi86;
  23. type
  24. { defs }
  25. tcpufiledef = class(tfiledef)
  26. end;
  27. tcpufiledefclass = class of tcpufiledef;
  28. tcpuvariantdef = class(tvariantdef)
  29. end;
  30. tcpuvariantdefclass = class of tcpuvariantdef;
  31. tcpuformaldef = class(tformaldef)
  32. end;
  33. tcpuformaldefclass = class of tcpuformaldef;
  34. tcpuforwarddef = class(tforwarddef)
  35. end;
  36. tcpuforwarddefclass = class of tcpuforwarddef;
  37. tcpuundefineddef = class(tundefineddef)
  38. end;
  39. tcpuundefineddefclass = class of tcpuundefineddef;
  40. tcpuerrordef = class(terrordef)
  41. end;
  42. tcpuerrordefclass = class of tcpuerrordef;
  43. tcpupointerdef = class(tx86pointerdef)
  44. class function default_x86_data_pointer_type: tx86pointertyp; override;
  45. function alignment:shortint;override;
  46. function pointer_arithmetic_int_type:tdef; override;
  47. function pointer_arithmetic_uint_type:tdef; override;
  48. function pointer_subtraction_result_type:tdef; override;
  49. function converted_pointer_to_array_range_type: tdef; override;
  50. end;
  51. tcpupointerdefclass = class of tcpupointerdef;
  52. tcpurecorddef = class(trecorddef)
  53. end;
  54. tcpurecorddefclass = class of tcpurecorddef;
  55. tcpuimplementedinterface = class(timplementedinterface)
  56. end;
  57. tcpuimplementedinterfaceclass = class of tcpuimplementedinterface;
  58. tcpuobjectdef = class(tobjectdef)
  59. end;
  60. tcpuobjectdefclass = class of tcpuobjectdef;
  61. tcpuclassrefdef = class(tclassrefdef)
  62. function alignment:shortint;override;
  63. end;
  64. tcpuclassrefdefclass = class of tcpuclassrefdef;
  65. { tcpuarraydef }
  66. tcpuarraydef = class(tarraydef)
  67. private
  68. huge: Boolean;
  69. protected
  70. procedure ppuload_platform(ppufile: tcompilerppufile); override;
  71. procedure ppuwrite_platform(ppufile: tcompilerppufile); override;
  72. public
  73. constructor create_from_pointer(def:tpointerdef);override;
  74. function getcopy: tstoreddef; override;
  75. function GetTypeName:string;override;
  76. property is_huge: Boolean read huge write huge;
  77. end;
  78. tcpuarraydefclass = class of tcpuarraydef;
  79. tcpuorddef = class(torddef)
  80. end;
  81. tcpuorddefclass = class of tcpuorddef;
  82. tcpufloatdef = class(tfloatdef)
  83. end;
  84. tcpufloatdefclass = class of tcpufloatdef;
  85. { tcpuprocvardef }
  86. tcpuprocvardef = class(ti86procvardef)
  87. constructor create(level:byte);override;
  88. function getcopyas(newtyp:tdeftyp;copytyp:tproccopytyp;const paraprefix:string):tstoreddef;override;
  89. function address_type:tdef;override;
  90. function ofs_address_type:tdef;override;
  91. function size:asizeint;override;
  92. procedure declared_far;override;
  93. procedure declared_near;override;
  94. function is_far:boolean;
  95. end;
  96. tcpuprocvardefclass = class of tcpuprocvardef;
  97. { tcpuprocdef }
  98. tcpuprocdef = class(ti86procdef)
  99. private
  100. { returns whether the function is far by default, i.e. whether it would be
  101. far if _all_ of the following conditions are true:
  102. - we're in a far code memory model
  103. - it has no 'near' or 'far' specifiers
  104. - it is compiled in a $F- state }
  105. function default_far:boolean;
  106. protected
  107. procedure Setinterfacedef(AValue: boolean);override;
  108. public
  109. constructor create(level:byte;doregister:boolean);override;
  110. function getcopyas(newtyp:tdeftyp;copytyp:tproccopytyp;const paraprefix:string):tstoreddef;override;
  111. function address_type:tdef;override;
  112. function ofs_address_type:tdef;override;
  113. function size:asizeint;override;
  114. procedure declared_far;override;
  115. procedure declared_near;override;
  116. function is_far:boolean;
  117. end;
  118. tcpuprocdefclass = class of tcpuprocdef;
  119. tcpustringdef = class(tstringdef)
  120. end;
  121. tcpustringdefclass = class of tcpustringdef;
  122. tcpuenumdef = class(tenumdef)
  123. end;
  124. tcpuenumdefclass = class of tcpuenumdef;
  125. tcpusetdef = class(tsetdef)
  126. end;
  127. tcpusetdefclass = class of tcpusetdef;
  128. { syms }
  129. tcpulabelsym = class(tlabelsym)
  130. end;
  131. tcpulabelsymclass = class of tcpulabelsym;
  132. tcpuunitsym = class(tunitsym)
  133. end;
  134. tcpuunitsymclass = class of tcpuunitsym;
  135. tcpuprogramparasym = class(tprogramparasym)
  136. end;
  137. tcpuprogramparasymclass = class(tprogramparasym);
  138. tcpunamespacesym = class(tnamespacesym)
  139. end;
  140. tcpunamespacesymclass = class of tcpunamespacesym;
  141. tcpuprocsym = class(tprocsym)
  142. end;
  143. tcpuprocsymclass = class of tcpuprocsym;
  144. tcputypesym = class(ttypesym)
  145. end;
  146. tcpuypesymclass = class of tcputypesym;
  147. tcpufieldvarsym = class(tfieldvarsym)
  148. end;
  149. tcpufieldvarsymclass = class of tcpufieldvarsym;
  150. tcpulocalvarsym = class(tlocalvarsym)
  151. end;
  152. tcpulocalvarsymclass = class of tcpulocalvarsym;
  153. tcpuparavarsym = class(tparavarsym)
  154. end;
  155. tcpuparavarsymclass = class of tcpuparavarsym;
  156. tcpustaticvarsym = class(tstaticvarsym)
  157. end;
  158. tcpustaticvarsymclass = class of tcpustaticvarsym;
  159. tcpuabsolutevarsym = class(ti86absolutevarsym)
  160. protected
  161. procedure ppuload_platform(ppufile: tcompilerppufile); override;
  162. procedure ppuwrite_platform(ppufile: tcompilerppufile); override;
  163. public
  164. addrsegment : aword;
  165. end;
  166. tcpuabsolutevarsymclass = class of tcpuabsolutevarsym;
  167. tcpupropertysym = class(tpropertysym)
  168. end;
  169. tcpupropertysymclass = class of tcpupropertysym;
  170. tcpuconstsym = class(tconstsym)
  171. end;
  172. tcpuconstsymclass = class of tcpuconstsym;
  173. tcpuenumsym = class(tenumsym)
  174. end;
  175. tcpuenumsymclass = class of tcpuenumsym;
  176. tcpusyssym = class(tsyssym)
  177. end;
  178. tcpusyssymclass = class of tcpusyssym;
  179. const
  180. pbestrealtype : ^tdef = @s80floattype;
  181. function is_proc_far(p: tabstractprocdef): boolean;
  182. {# Returns true if p is a far proc var }
  183. function is_farprocvar(p : tdef): boolean;
  184. {# Returns true if p is a far pointer def }
  185. function is_farpointer(p : tdef) : boolean;
  186. {# Returns true if p is a huge pointer def }
  187. function is_hugepointer(p : tdef) : boolean;
  188. implementation
  189. uses
  190. globals, cpuinfo, verbose, fmodule;
  191. function is_proc_far(p: tabstractprocdef): boolean;
  192. begin
  193. if p is tcpuprocdef then
  194. result:=tcpuprocdef(p).is_far
  195. else if p is tcpuprocvardef then
  196. result:=tcpuprocvardef(p).is_far
  197. else
  198. internalerror(2014041303);
  199. end;
  200. { true if p is a far proc var }
  201. function is_farprocvar(p : tdef): boolean;
  202. begin
  203. result:=(p.typ=procvardef) and tcpuprocvardef(p).is_far;
  204. end;
  205. { true if p is a far pointer def }
  206. function is_farpointer(p : tdef) : boolean;
  207. begin
  208. result:=(p.typ=pointerdef) and (tcpupointerdef(p).x86pointertyp=x86pt_far);
  209. end;
  210. { true if p is a huge pointer def }
  211. function is_hugepointer(p : tdef) : boolean;
  212. begin
  213. result:=(p.typ=pointerdef) and (tcpupointerdef(p).x86pointertyp=x86pt_huge);
  214. end;
  215. procedure handle_procdef_copyas(src: tabstractprocdef; is_far: boolean; copytyp:tproccopytyp; var result: tabstractprocdef);
  216. begin
  217. if is_far then
  218. include(result.procoptions,po_far)
  219. else
  220. exclude(result.procoptions,po_far);
  221. case copytyp of
  222. pc_far_address:
  223. begin
  224. include(result.procoptions,po_addressonly);
  225. include(result.procoptions,po_far);
  226. end;
  227. pc_offset:
  228. begin
  229. include(result.procoptions,po_addressonly);
  230. exclude(result.procoptions,po_far);
  231. end;
  232. else
  233. ; {none}
  234. end;
  235. end;
  236. {****************************************************************************
  237. tcpuclassrefdef
  238. ****************************************************************************}
  239. function tcpuclassrefdef.alignment:shortint;
  240. begin
  241. Result:=2;
  242. end;
  243. {****************************************************************************
  244. tcpuarraydef
  245. ****************************************************************************}
  246. constructor tcpuarraydef.create_from_pointer(def: tpointerdef);
  247. begin
  248. huge:=tcpupointerdef(def).x86pointertyp=x86pt_huge;
  249. inherited create_from_pointer(def);
  250. end;
  251. function tcpuarraydef.getcopy: tstoreddef;
  252. begin
  253. result:=inherited;
  254. tcpuarraydef(result).huge:=huge;
  255. end;
  256. function tcpuarraydef.GetTypeName: string;
  257. begin
  258. Result:=inherited;
  259. if is_huge then
  260. Result:='Huge '+Result;
  261. end;
  262. procedure tcpuarraydef.ppuload_platform(ppufile: tcompilerppufile);
  263. begin
  264. inherited;
  265. huge:=(ppufile.getbyte<>0);
  266. end;
  267. procedure tcpuarraydef.ppuwrite_platform(ppufile: tcompilerppufile);
  268. begin
  269. inherited;
  270. ppufile.putbyte(byte(huge));
  271. end;
  272. {****************************************************************************
  273. tcpuprocdef
  274. ****************************************************************************}
  275. constructor tcpuprocdef.create(level: byte;doregister:boolean);
  276. begin
  277. inherited create(level,doregister);
  278. if (current_settings.x86memorymodel in x86_far_code_models) and
  279. ((cs_huge_code in current_settings.moduleswitches) or
  280. (cs_force_far_calls in current_settings.localswitches)) then
  281. procoptions:=procoptions+[po_far];
  282. end;
  283. function tcpuprocdef.getcopyas(newtyp:tdeftyp;copytyp:tproccopytyp;const paraprefix:string):tstoreddef;
  284. begin
  285. result:=inherited;
  286. handle_procdef_copyas(self,is_far,copytyp,tabstractprocdef(result));
  287. end;
  288. function tcpuprocdef.address_type: tdef;
  289. begin
  290. if is_far then
  291. result:=voidfarpointertype
  292. else
  293. result:=voidnearpointertype;
  294. end;
  295. function tcpuprocdef.ofs_address_type:tdef;
  296. begin
  297. result:=voidnearpointertype;
  298. end;
  299. function tcpuprocdef.size: asizeint;
  300. begin
  301. result:=address_type.size;
  302. end;
  303. procedure tcpuprocdef.declared_far;
  304. begin
  305. include(procoptions,po_far);
  306. include(procoptions,po_hasnearfarcallmodel);
  307. end;
  308. procedure tcpuprocdef.declared_near;
  309. begin
  310. if not (cs_huge_code in current_settings.moduleswitches) then
  311. begin
  312. exclude(procoptions,po_far);
  313. include(procoptions,po_hasnearfarcallmodel);
  314. end
  315. else
  316. inherited declared_near;
  317. end;
  318. function tcpuprocdef.default_far: boolean;
  319. begin
  320. if proctypeoption in [potype_proginit,potype_unitinit,potype_unitfinalize,
  321. potype_constructor,potype_destructor,
  322. potype_class_constructor,potype_class_destructor,
  323. potype_propgetter,potype_propsetter] then
  324. exit(true);
  325. if (procoptions*[po_classmethod,po_virtualmethod,po_abstractmethod,
  326. po_finalmethod,po_staticmethod,po_overridingmethod,
  327. po_external,po_public,po_interrupt])<>[] then
  328. exit(true);
  329. if is_methodpointer then
  330. exit(true);
  331. result:=not (visibility in [vis_private,vis_hidden]);
  332. end;
  333. procedure tcpuprocdef.Setinterfacedef(AValue: boolean);
  334. begin
  335. inherited;
  336. if (current_settings.x86memorymodel in x86_far_code_models) and AValue then
  337. include(procoptions,po_far);
  338. end;
  339. function tcpuprocdef.is_far: boolean;
  340. begin
  341. result:=(po_exports in procoptions) or
  342. (po_far in procoptions) or
  343. ((current_settings.x86memorymodel in x86_far_code_models) and default_far);
  344. end;
  345. {****************************************************************************
  346. tcpuprocvardef
  347. ****************************************************************************}
  348. constructor tcpuprocvardef.create(level: byte);
  349. begin
  350. inherited create(level);
  351. if current_settings.x86memorymodel in x86_far_code_models then
  352. procoptions:=procoptions+[po_far];
  353. end;
  354. function tcpuprocvardef.getcopyas(newtyp:tdeftyp;copytyp:tproccopytyp;const paraprefix:string):tstoreddef;
  355. begin
  356. result:=inherited;
  357. handle_procdef_copyas(self,is_far,copytyp,tabstractprocdef(result));
  358. end;
  359. function tcpuprocvardef.address_type:tdef;
  360. begin
  361. if is_addressonly then
  362. if is_far then
  363. result:=voidfarpointertype
  364. else
  365. begin
  366. { near }
  367. if current_settings.x86memorymodel=mm_tiny then
  368. result:=voidnearpointertype
  369. else
  370. result:=voidnearcspointertype;
  371. end
  372. else
  373. result:=inherited;
  374. end;
  375. function tcpuprocvardef.ofs_address_type:tdef;
  376. begin
  377. result:=voidnearpointertype;
  378. end;
  379. function tcpuprocvardef.size:asizeint;
  380. begin
  381. if is_addressonly then
  382. if is_far then
  383. result:=4
  384. else
  385. result:=2
  386. else
  387. result:=inherited;
  388. end;
  389. procedure tcpuprocvardef.declared_far;
  390. begin
  391. if is_addressonly then
  392. begin
  393. include(procoptions,po_far);
  394. include(procoptions,po_hasnearfarcallmodel);
  395. end
  396. else
  397. inherited;
  398. end;
  399. procedure tcpuprocvardef.declared_near;
  400. begin
  401. if is_addressonly then
  402. begin
  403. exclude(procoptions,po_far);
  404. include(procoptions,po_hasnearfarcallmodel);
  405. end
  406. else
  407. inherited;
  408. end;
  409. function tcpuprocvardef.is_far: boolean;
  410. begin
  411. if is_addressonly then
  412. result:=po_far in procoptions
  413. else
  414. result:=current_settings.x86memorymodel in x86_far_code_models;
  415. end;
  416. {****************************************************************************
  417. tcpupointerdef
  418. ****************************************************************************}
  419. class function tcpupointerdef.default_x86_data_pointer_type: tx86pointertyp;
  420. begin
  421. if current_settings.x86memorymodel in x86_far_data_models then
  422. result:=x86pt_far
  423. else
  424. result:=inherited;
  425. end;
  426. function tcpupointerdef.alignment:shortint;
  427. begin
  428. { on i8086, we use 16-bit alignment for all pointer types, even far and
  429. huge (which are 4 bytes long) }
  430. result:=2;
  431. end;
  432. function tcpupointerdef.pointer_arithmetic_int_type:tdef;
  433. begin
  434. case x86pointertyp of
  435. x86pt_huge:
  436. result:=s32inttype;
  437. x86pt_far,
  438. x86pt_near,
  439. x86pt_near_cs,
  440. x86pt_near_ds,
  441. x86pt_near_ss,
  442. x86pt_near_es,
  443. x86pt_near_fs,
  444. x86pt_near_gs:
  445. result:=s16inttype;
  446. end;
  447. end;
  448. function tcpupointerdef.pointer_arithmetic_uint_type:tdef;
  449. begin
  450. case x86pointertyp of
  451. x86pt_huge:
  452. result:=u32inttype;
  453. x86pt_far,
  454. x86pt_near,
  455. x86pt_near_cs,
  456. x86pt_near_ds,
  457. x86pt_near_ss,
  458. x86pt_near_es,
  459. x86pt_near_fs,
  460. x86pt_near_gs:
  461. result:=u16inttype;
  462. end;
  463. end;
  464. function tcpupointerdef.pointer_subtraction_result_type:tdef;
  465. begin
  466. case x86pointertyp of
  467. x86pt_huge:
  468. result:=s32inttype;
  469. x86pt_far:
  470. result:=u16inttype;
  471. x86pt_near,
  472. x86pt_near_cs,
  473. x86pt_near_ds,
  474. x86pt_near_ss,
  475. x86pt_near_es,
  476. x86pt_near_fs,
  477. x86pt_near_gs:
  478. result:=s16inttype;
  479. end;
  480. end;
  481. function tcpupointerdef.converted_pointer_to_array_range_type: tdef;
  482. begin
  483. case x86pointertyp of
  484. x86pt_huge:
  485. result:=s32inttype;
  486. x86pt_far,
  487. x86pt_near,
  488. x86pt_near_cs,
  489. x86pt_near_ds,
  490. x86pt_near_ss,
  491. x86pt_near_es,
  492. x86pt_near_fs,
  493. x86pt_near_gs:
  494. result:=s16inttype;
  495. end;
  496. end;
  497. {****************************************************************************
  498. tcpuabsolutevarsym
  499. ****************************************************************************}
  500. procedure tcpuabsolutevarsym.ppuload_platform(ppufile: tcompilerppufile);
  501. begin
  502. inherited;
  503. if absseg then
  504. addrsegment:=ppufile.getaword;
  505. end;
  506. procedure tcpuabsolutevarsym.ppuwrite_platform(ppufile: tcompilerppufile);
  507. begin
  508. inherited;
  509. if absseg then
  510. ppufile.putaword(addrsegment);
  511. end;
  512. begin
  513. { used tdef classes }
  514. cfiledef:=tcpufiledef;
  515. cvariantdef:=tcpuvariantdef;
  516. cformaldef:=tcpuformaldef;
  517. cforwarddef:=tcpuforwarddef;
  518. cundefineddef:=tcpuundefineddef;
  519. cerrordef:=tcpuerrordef;
  520. cpointerdef:=tcpupointerdef;
  521. crecorddef:=tcpurecorddef;
  522. cimplementedinterface:=tcpuimplementedinterface;
  523. cobjectdef:=tcpuobjectdef;
  524. cclassrefdef:=tcpuclassrefdef;
  525. carraydef:=tcpuarraydef;
  526. corddef:=tcpuorddef;
  527. cfloatdef:=tcpufloatdef;
  528. cprocvardef:=tcpuprocvardef;
  529. cprocdef:=tcpuprocdef;
  530. cstringdef:=tcpustringdef;
  531. cenumdef:=tcpuenumdef;
  532. csetdef:=tcpusetdef;
  533. { used tsym classes }
  534. clabelsym:=tcpulabelsym;
  535. cunitsym:=tcpuunitsym;
  536. cprogramparasym:=tcpuprogramparasym;
  537. cnamespacesym:=tcpunamespacesym;
  538. cprocsym:=tcpuprocsym;
  539. ctypesym:=tcputypesym;
  540. cfieldvarsym:=tcpufieldvarsym;
  541. clocalvarsym:=tcpulocalvarsym;
  542. cparavarsym:=tcpuparavarsym;
  543. cstaticvarsym:=tcpustaticvarsym;
  544. cabsolutevarsym:=tcpuabsolutevarsym;
  545. cpropertysym:=tcpupropertysym;
  546. cconstsym:=tcpuconstsym;
  547. cenumsym:=tcpuenumsym;
  548. csyssym:=tcpusyssym;
  549. end.